CheckinDialog.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383
  1. using System;
  2. using System.Collections.Generic;
  3. using System.IO;
  4. using Codice.Client.Common;
  5. using Codice.CM.Common;
  6. using GluonGui;
  7. using PlasticGui;
  8. using PlasticGui.Gluon;
  9. using Unity.PlasticSCM.Editor.AssetsOverlays;
  10. using Unity.PlasticSCM.Editor.AssetsOverlays.Cache;
  11. using Unity.PlasticSCM.Editor.AssetUtils;
  12. using Unity.PlasticSCM.Editor.UI;
  13. using Unity.PlasticSCM.Editor.UI.Progress;
  14. using Unity.PlasticSCM.Editor.UI.Tree;
  15. using UnityEditor;
  16. using UnityEngine;
  17. namespace Unity.PlasticSCM.Editor.AssetMenu.Dialogs
  18. {
  19. internal class CheckinDialog : PlasticDialog
  20. {
  21. protected override Rect DefaultRect
  22. {
  23. get
  24. {
  25. var baseRect = base.DefaultRect;
  26. return new Rect(baseRect.x, baseRect.y, 700, 450);
  27. }
  28. }
  29. protected override string GetTitle()
  30. {
  31. return PlasticLocalization.GetString(
  32. PlasticLocalization.Name.CheckinChanges);
  33. }
  34. internal static bool CheckinPaths(
  35. WorkspaceInfo wkInfo,
  36. List<string> paths,
  37. IAssetStatusCache assetStatusCache,
  38. bool isGluonMode,
  39. EditorWindow parentWindow,
  40. IWorkspaceWindow workspaceWindow,
  41. ViewHost viewHost,
  42. GuiMessage.IGuiMessage guiMessage,
  43. IMergeViewLauncher mergeViewLauncher,
  44. IGluonViewSwitcher gluonViewSwitcher)
  45. {
  46. MetaCache metaCache = new MetaCache();
  47. metaCache.Build(paths);
  48. CheckinDialog dialog = Create(
  49. wkInfo,
  50. paths,
  51. assetStatusCache,
  52. metaCache,
  53. isGluonMode,
  54. new ProgressControlsForDialogs(),
  55. workspaceWindow,
  56. viewHost,
  57. guiMessage,
  58. mergeViewLauncher,
  59. gluonViewSwitcher);
  60. return dialog.RunModal(parentWindow) == ResponseType.Ok;
  61. }
  62. protected override void OnModalGUI()
  63. {
  64. Title(PlasticLocalization.GetString(
  65. PlasticLocalization.Name.CheckinComment));
  66. GUI.SetNextControlName(CHECKIN_TEXTAREA_NAME);
  67. mComment = GUILayout.TextArea(
  68. mComment,
  69. EditorStyles.textArea,
  70. GUILayout.MinHeight(120));
  71. if (!mTextAreaFocused)
  72. {
  73. EditorGUI.FocusTextInControl(CHECKIN_TEXTAREA_NAME);
  74. mTextAreaFocused = true;
  75. }
  76. Title(PlasticLocalization.GetString(PlasticLocalization.Name.Files));
  77. DoFileList(
  78. mWkInfo,
  79. mPaths,
  80. mAssetStatusCache,
  81. mMetaCache);
  82. DrawProgressForDialogs.For(
  83. mProgressControls.ProgressData);
  84. DoButtonsArea();
  85. mProgressControls.ForcedUpdateProgress(this);
  86. }
  87. void DoFileList(
  88. WorkspaceInfo wkInfo,
  89. List<string> paths,
  90. IAssetStatusCache assetStatusCache,
  91. MetaCache metaCache)
  92. {
  93. mFileListScrollPosition = GUILayout.BeginScrollView(
  94. mFileListScrollPosition,
  95. EditorStyles.helpBox,
  96. GUILayout.ExpandHeight(true));
  97. foreach (string path in paths)
  98. {
  99. if (MetaPath.IsMetaPath(path))
  100. continue;
  101. Texture fileIcon = Directory.Exists(path) ?
  102. Images.GetDirectoryIcon() :
  103. Images.GetFileIcon(path);
  104. string label = WorkspacePath.GetWorkspaceRelativePath(
  105. wkInfo.ClientPath, path);
  106. if (metaCache.HasMeta(path))
  107. label = string.Concat(label, UnityConstants.TREEVIEW_META_LABEL);
  108. GUIContent content = new GUIContent(
  109. label, fileIcon);
  110. GUILayout.Label(
  111. content,
  112. GUILayout.Height(UnityConstants.TREEVIEW_ROW_HEIGHT));
  113. Rect iconRect = GUILayoutUtility.GetLastRect();
  114. DoStatusOverlays(
  115. iconRect,
  116. assetStatusCache,
  117. path);
  118. }
  119. GUILayout.EndScrollView();
  120. }
  121. static void DoStatusOverlays(
  122. Rect iconRect,
  123. IAssetStatusCache assetStatusCache,
  124. string path)
  125. {
  126. AssetsOverlays.AssetStatus statusesToDraw = DrawAssetOverlay.GetStatusesToDraw(
  127. assetStatusCache.GetStatusForPath(path));
  128. foreach (AssetsOverlays.AssetStatus status in Enum.GetValues(typeof(AssetsOverlays.AssetStatus)))
  129. {
  130. if (status == AssetsOverlays.AssetStatus.None)
  131. continue;
  132. if (!statusesToDraw.HasFlag(status))
  133. continue;
  134. GetChangesOverlayIcon.Data overlayIconData =
  135. GetChangesOverlayIcon.ForAssetStatus(status);
  136. if (overlayIconData != null)
  137. {
  138. Rect overlayIconRect = new Rect(
  139. iconRect.x + overlayIconData.XOffset,
  140. iconRect.y + overlayIconData.YOffset,
  141. overlayIconData.Size, overlayIconData.Size);
  142. GUI.DrawTexture(
  143. overlayIconRect, overlayIconData.Texture,
  144. ScaleMode.ScaleToFit);
  145. }
  146. }
  147. }
  148. void DoButtonsArea()
  149. {
  150. using (new EditorGUILayout.HorizontalScope())
  151. {
  152. GUILayout.FlexibleSpace();
  153. if (Application.platform == RuntimePlatform.WindowsEditor)
  154. {
  155. DoCheckinButton();
  156. DoCancelButton();
  157. return;
  158. }
  159. DoCancelButton();
  160. DoCheckinButton();
  161. }
  162. }
  163. void DoCheckinButton()
  164. {
  165. GUI.enabled = !string.IsNullOrEmpty(mComment) && !mIsRunningCheckin;
  166. try
  167. {
  168. if (!AcceptButton(PlasticLocalization.GetString(
  169. PlasticLocalization.Name.CheckinButton)))
  170. return;
  171. }
  172. finally
  173. {
  174. GUI.enabled = true;
  175. }
  176. OkButtonWithCheckinAction();
  177. }
  178. void DoCancelButton()
  179. {
  180. if (!NormalButton(PlasticLocalization.GetString(
  181. PlasticLocalization.Name.CancelButton)))
  182. return;
  183. CancelButtonAction();
  184. }
  185. void OkButtonWithCheckinAction()
  186. {
  187. bool isCancelled;
  188. SaveAssets.ForPathsWithConfirmation(mPaths, out isCancelled);
  189. if (isCancelled)
  190. return;
  191. mIsRunningCheckin = true;
  192. mPaths.AddRange(mMetaCache.GetExistingMeta(mPaths));
  193. if (mIsGluonMode)
  194. {
  195. CheckinDialogOperations.CheckinPathsPartial(
  196. mWkInfo,
  197. mPaths,
  198. mComment,
  199. mViewHost,
  200. this,
  201. mGuiMessage,
  202. mProgressControls,
  203. mGluonViewSwitcher);
  204. return;
  205. }
  206. CheckinDialogOperations.CheckinPaths(
  207. mWkInfo,
  208. mPaths,
  209. mComment,
  210. mWorkspaceWindow,
  211. this,
  212. mGuiMessage,
  213. mProgressControls,
  214. mMergeViewLauncher);
  215. }
  216. static CheckinDialog Create(
  217. WorkspaceInfo wkInfo,
  218. List<string> paths,
  219. IAssetStatusCache assetStatusCache,
  220. MetaCache metaCache,
  221. bool isGluonMode,
  222. ProgressControlsForDialogs progressControls,
  223. IWorkspaceWindow workspaceWindow,
  224. ViewHost viewHost,
  225. GuiMessage.IGuiMessage guiMessage,
  226. IMergeViewLauncher mergeViewLauncher,
  227. IGluonViewSwitcher gluonViewSwitcher)
  228. {
  229. var instance = CreateInstance<CheckinDialog>();
  230. instance.IsResizable = true;
  231. instance.minSize = new Vector2(520, 370);
  232. instance.mWkInfo = wkInfo;
  233. instance.mPaths = paths;
  234. instance.mAssetStatusCache = assetStatusCache;
  235. instance.mMetaCache = metaCache;
  236. instance.mIsGluonMode = isGluonMode;
  237. instance.mProgressControls = progressControls;
  238. instance.mWorkspaceWindow = workspaceWindow;
  239. instance.mViewHost = viewHost;
  240. instance.mGuiMessage = guiMessage;
  241. instance.mMergeViewLauncher = mergeViewLauncher;
  242. instance.mGluonViewSwitcher = gluonViewSwitcher;
  243. instance.mEnterKeyAction = instance.OkButtonAction;
  244. instance.mEscapeKeyAction = instance.CancelButtonAction;
  245. return instance;
  246. }
  247. WorkspaceInfo mWkInfo;
  248. List<string> mPaths;
  249. IAssetStatusCache mAssetStatusCache;
  250. MetaCache mMetaCache;
  251. bool mIsGluonMode;
  252. bool mTextAreaFocused;
  253. string mComment;
  254. bool mIsRunningCheckin;
  255. Vector2 mFileListScrollPosition;
  256. ProgressControlsForDialogs mProgressControls;
  257. IWorkspaceWindow mWorkspaceWindow;
  258. ViewHost mViewHost;
  259. IMergeViewLauncher mMergeViewLauncher;
  260. IGluonViewSwitcher mGluonViewSwitcher;
  261. GuiMessage.IGuiMessage mGuiMessage;
  262. const string CHECKIN_TEXTAREA_NAME = "checkin_textarea";
  263. class MetaCache
  264. {
  265. internal bool HasMeta(string path)
  266. {
  267. return mCache.Contains(MetaPath.GetMetaPath(path));
  268. }
  269. internal List<string> GetExistingMeta(List<string> paths)
  270. {
  271. List<string> result = new List<string>();
  272. foreach (string path in paths)
  273. {
  274. string metaPath = MetaPath.GetMetaPath(path);
  275. if (!mCache.Contains(metaPath))
  276. continue;
  277. result.Add(metaPath);
  278. }
  279. return result;
  280. }
  281. internal void Build(List<string> paths)
  282. {
  283. HashSet<string> indexedKeys = BuildIndexedKeys(paths);
  284. for (int i = paths.Count - 1; i >= 0; i--)
  285. {
  286. string currentPath = paths[i];
  287. if (!MetaPath.IsMetaPath(currentPath))
  288. continue;
  289. string realPath = MetaPath.GetPathFromMetaPath(currentPath);
  290. if (!indexedKeys.Contains(realPath))
  291. continue;
  292. // found foo.c and foo.c.meta
  293. // with the same chage types - move .meta to cache
  294. mCache.Add(currentPath);
  295. paths.RemoveAt(i);
  296. }
  297. }
  298. static HashSet<string> BuildIndexedKeys(List<string> paths)
  299. {
  300. HashSet<string> result = new HashSet<string>();
  301. foreach (string path in paths)
  302. {
  303. if (MetaPath.IsMetaPath(path))
  304. continue;
  305. result.Add(path);
  306. }
  307. return result;
  308. }
  309. HashSet<string> mCache =
  310. new HashSet<string>();
  311. }
  312. }
  313. }