No Description
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

CheckinDialog.cs 12KB

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