/VsIntegration/Nemerle.VisualStudio/Project/References/Base/ProjectReferenceNode.cs

http://github.com/xxVisorxx/nemerle · C# · 542 lines · 362 code · 97 blank · 83 comment · 74 complexity · 3d77d9e291e00c895211910a03890d81 MD5 · raw file

  1. /// Copyright (c) Microsoft Corporation. All rights reserved.
  2. using System;
  3. using System.Diagnostics;
  4. using System.Globalization;
  5. using System.IO;
  6. using System.Runtime.InteropServices;
  7. using Microsoft.VisualStudio;
  8. using Microsoft.VisualStudio.Shell;
  9. using Microsoft.VisualStudio.Shell.Interop;
  10. using System.Collections.Generic;
  11. using System.Linq;
  12. using Microsoft.VisualStudio.Project.Automation;
  13. using EnvDTE80;
  14. namespace Microsoft.VisualStudio.Project
  15. {
  16. [CLSCompliant(false), ComVisible(true)]
  17. public class ProjectReferenceNode : ReferenceNode
  18. {
  19. #region fieds
  20. /// <summary>
  21. /// The name of the assembly this refernce represents
  22. /// </summary>
  23. private Guid referencedProjectGuid;
  24. private string referencedProjectName = String.Empty;
  25. private string referencedProjectRelativePath = String.Empty;
  26. private string referencedProjectFullPath = String.Empty;
  27. protected BuildDependency buildDependency;
  28. /// <summary>
  29. /// This is a reference to the automation object for the referenced project.
  30. /// </summary>
  31. private EnvDTE.Project referencedProject;
  32. /// <summary>
  33. /// This state is controlled by the solution events.
  34. /// The state is set to false by OnBeforeUnloadProject.
  35. /// The state is set to true by OnBeforeCloseProject event.
  36. /// </summary>
  37. private bool canRemoveReference = true;
  38. /// <summary>
  39. /// Possibility for solution listener to update the state on the dangling reference.
  40. /// It will be set in OnBeforeUnloadProject then the nopde is invalidated then it is reset to false.
  41. /// </summary>
  42. private bool isNodeValid;
  43. #endregion
  44. #region properties
  45. public override string Url
  46. {
  47. get
  48. {
  49. return this.referencedProjectFullPath;
  50. }
  51. }
  52. public override string Caption
  53. {
  54. get
  55. {
  56. return this.referencedProjectName;
  57. }
  58. }
  59. internal Guid ReferencedProjectGuid
  60. {
  61. get
  62. {
  63. return this.referencedProjectGuid;
  64. }
  65. }
  66. /// <summary>
  67. /// Possiblity to shortcut and set the dangling project reference icon.
  68. /// It is ussually manipulated by solution listsneres who handle reference updates.
  69. /// </summary>
  70. internal protected bool IsNodeValid
  71. {
  72. get
  73. {
  74. return this.isNodeValid;
  75. }
  76. set
  77. {
  78. this.isNodeValid = value;
  79. }
  80. }
  81. /// <summary>
  82. /// Controls the state whether this reference can be removed or not. Think of the project unload scenario where the project reference should not be deleted.
  83. /// </summary>
  84. internal bool CanRemoveReference
  85. {
  86. get
  87. {
  88. return this.canRemoveReference;
  89. }
  90. set
  91. {
  92. this.canRemoveReference = value;
  93. }
  94. }
  95. internal string ReferencedProjectName
  96. {
  97. get { return this.referencedProjectName; }
  98. }
  99. private EnvDTE.Property GetProperty(EnvDTE.Project project, string propertyName)
  100. {
  101. EnvDTE.Property result = null;
  102. try
  103. {
  104. if (project.Properties == null)
  105. return null;
  106. result = project.Properties.Item(propertyName);
  107. }
  108. catch { }
  109. return result;
  110. }
  111. private EnvDTE.Project FindProjectByPath(IEnumerable<EnvDTE.Project> projects)
  112. {
  113. foreach(EnvDTE.Project prj in projects)
  114. {
  115. //Skip this project if it is an umodeled project (unloaded)
  116. if(string.Compare(EnvDTE.Constants.vsProjectKindUnmodeled, prj.Kind, StringComparison.OrdinalIgnoreCase) == 0)
  117. continue;
  118. //SolutionFolder
  119. if (string.Compare(EnvDTE.Constants.vsProjectKindSolutionItems, prj.Kind, StringComparison.OrdinalIgnoreCase) == 0)
  120. {
  121. var subProjects = new List<EnvDTE.Project>();
  122. foreach (EnvDTE.ProjectItem item in prj.ProjectItems)
  123. {
  124. try
  125. {
  126. EnvDTE.Project subProject = item.SubProject as EnvDTE.Project;
  127. if (subProject != null)
  128. subProjects.Add(subProject);
  129. }
  130. catch { }
  131. }
  132. if (subProjects.Count > 0)
  133. {
  134. var result = FindProjectByPath(subProjects);
  135. if (result != null)
  136. return result;
  137. }
  138. }
  139. // Get the full path of the current project.
  140. EnvDTE.Property pathProperty = GetProperty(prj, "FullPath");
  141. if (pathProperty == null)
  142. // The full path should alway be availabe, but if this is not the
  143. // case then we have to skip it.
  144. continue;
  145. string prjPath = pathProperty.Value.ToString();
  146. // Get the name of the project file.
  147. EnvDTE.Property fileNameProperty = GetProperty(prj, "FileName");
  148. if (fileNameProperty == null)
  149. // Again, this should never be the case, but we handle it anyway.
  150. continue;
  151. prjPath = System.IO.Path.Combine(prjPath, fileNameProperty.Value.ToString());
  152. // If the full path of this project is the same as the one of this
  153. // reference, then we have found the right project.
  154. if(NativeMethods.IsSamePath(prjPath, referencedProjectFullPath))
  155. return prj;
  156. }
  157. return null;
  158. }
  159. /// <summary>
  160. /// Gets the automation object for the referenced project.
  161. /// </summary>
  162. internal EnvDTE.Project ReferencedProjectObject
  163. {
  164. get
  165. {
  166. // If the referenced project is null then re-read.
  167. if(this.referencedProject == null)
  168. {
  169. // Search for the project in the collection of the projects in the
  170. // current solution.
  171. EnvDTE.DTE dte = (EnvDTE.DTE)this.ProjectMgr.GetService(typeof(EnvDTE.DTE));
  172. if(null == dte || null == dte.Solution)
  173. return null;
  174. this.referencedProject = FindProjectByPath(dte.Solution.Projects.OfType<EnvDTE.Project>());
  175. }
  176. return this.referencedProject;
  177. }
  178. set
  179. {
  180. this.referencedProject = value;
  181. }
  182. }
  183. /// <summary>
  184. /// Gets the full path to the assembly generated by this project.
  185. /// </summary>
  186. internal string ReferencedProjectOutputPath
  187. {
  188. get
  189. {
  190. // Make sure that the referenced project implements the automation object.
  191. if (null == this.ReferencedProjectObject)
  192. return null;
  193. // Get the configuration manager from the project.
  194. EnvDTE.ConfigurationManager confManager = this.ReferencedProjectObject.ConfigurationManager;
  195. if (null == confManager)
  196. return null;
  197. // Get the active configuration.
  198. EnvDTE.Configuration config = confManager.ActiveConfiguration;
  199. if (null == config)
  200. return null;
  201. // Get the output path for the current configuration.
  202. EnvDTE.Property outputPathProperty = config.Properties.Item("OutputPath");
  203. if (null == outputPathProperty)
  204. return null;
  205. string outputPath = outputPathProperty.Value.ToString();
  206. // Ususally the output path is relative to the project path, but it is possible
  207. // to set it as an absolute path. If it is not absolute, then evaluate its value
  208. // based on the project directory.
  209. if (!Path.IsPathRooted(outputPath))
  210. {
  211. string projectDir = Path.GetDirectoryName(referencedProjectFullPath);
  212. outputPath = Path.Combine(projectDir, outputPath);
  213. }
  214. // Now get the name of the assembly from the project.
  215. // Some project system throw if the property does not exist. We expect an ArgumentException.
  216. EnvDTE.Property assemblyNameProperty = null;
  217. try
  218. {
  219. assemblyNameProperty = this.ReferencedProjectObject.Properties.Item("OutputFileName");
  220. }
  221. catch (ArgumentException)
  222. {
  223. }
  224. if (null == assemblyNameProperty)
  225. return null;
  226. // build the full path adding the name of the assembly to the output path.
  227. outputPath = Path.Combine(outputPath, assemblyNameProperty.Value.ToString());
  228. return outputPath;
  229. }
  230. }
  231. private Automation.OAProjectReference projectReference;
  232. internal override object Object
  233. {
  234. get
  235. {
  236. if(null == projectReference)
  237. {
  238. projectReference = new Automation.OAProjectReference(this);
  239. }
  240. return projectReference;
  241. }
  242. }
  243. #endregion
  244. #region ctors
  245. /// <summary>
  246. /// Constructor for the ReferenceNode. It is called when the project is reloaded, when the project element representing the refernce exists.
  247. /// </summary>
  248. [System.Diagnostics.CodeAnalysis.SuppressMessage("Microsoft.Usage", "CA2234:PassSystemUriObjectsInsteadOfStrings")]
  249. public ProjectReferenceNode(ProjectNode root, ProjectElement element)
  250. : base(root, element)
  251. {
  252. this.referencedProjectRelativePath = this.ItemNode.GetMetadata(ProjectFileConstants.Include);
  253. Debug.Assert(!String.IsNullOrEmpty(this.referencedProjectRelativePath), "Could not retrive referenced project path form project file");
  254. string guidString = this.ItemNode.GetMetadata(ProjectFileConstants.Project);
  255. // Continue even if project setttings cannot be read.
  256. try
  257. {
  258. this.referencedProjectGuid = new Guid(guidString);
  259. this.buildDependency = new BuildDependency(this.ProjectMgr, this.referencedProjectGuid);
  260. this.ProjectMgr.AddBuildDependency(this.buildDependency);
  261. }
  262. finally
  263. {
  264. Debug.Assert(this.referencedProjectGuid != Guid.Empty, "Could not retrive referenced project guidproject file");
  265. this.referencedProjectName = this.ItemNode.GetMetadata(ProjectFileConstants.Name);
  266. Debug.Assert(!String.IsNullOrEmpty(this.referencedProjectName), "Could not retrive referenced project name form project file");
  267. }
  268. Uri uri = new Uri(this.ProjectMgr.BaseURI.Uri, this.referencedProjectRelativePath);
  269. if(uri != null)
  270. {
  271. this.referencedProjectFullPath = Microsoft.VisualStudio.Shell.Url.Unescape(uri.LocalPath, true);
  272. }
  273. }
  274. /// <summary>
  275. /// constructor for the ProjectReferenceNode
  276. /// </summary>
  277. public ProjectReferenceNode(ProjectNode root, string referencedProjectName, string projectPath, string projectReference)
  278. : base(root)
  279. {
  280. Debug.Assert(root != null && !String.IsNullOrEmpty(referencedProjectName) && !String.IsNullOrEmpty(projectReference)
  281. && !String.IsNullOrEmpty(projectPath), "Can not add a reference because the input for adding one is invalid.");
  282. this.referencedProjectName = referencedProjectName;
  283. int indexOfSeparator = projectReference.IndexOf('|');
  284. string fileName = String.Empty;
  285. // Unfortunately we cannot use the path part of the projectReference string since it is not resolving correctly relative pathes.
  286. if(indexOfSeparator != -1)
  287. {
  288. string projectGuid = projectReference.Substring(0, indexOfSeparator);
  289. this.referencedProjectGuid = new Guid(projectGuid);
  290. if(indexOfSeparator + 1 < projectReference.Length)
  291. {
  292. string remaining = projectReference.Substring(indexOfSeparator + 1);
  293. indexOfSeparator = remaining.IndexOf('|');
  294. if(indexOfSeparator == -1)
  295. {
  296. fileName = remaining;
  297. }
  298. else
  299. {
  300. fileName = remaining.Substring(0, indexOfSeparator);
  301. }
  302. }
  303. }
  304. Debug.Assert(!String.IsNullOrEmpty(fileName), "Can not add a project reference because the input for adding one is invalid.");
  305. // Did we get just a file or a relative path?
  306. Uri uri = new Uri(projectPath);
  307. string referenceDir = PackageUtilities.GetPathDistance(this.ProjectMgr.BaseURI.Uri, uri);
  308. Debug.Assert(!String.IsNullOrEmpty(referenceDir), "Can not add a project reference because the input for adding one is invalid.");
  309. string justTheFileName = Path.GetFileName(fileName);
  310. this.referencedProjectRelativePath = Path.Combine(referenceDir, justTheFileName);
  311. this.referencedProjectFullPath = Path.Combine(projectPath, justTheFileName);
  312. this.buildDependency = new BuildDependency(this.ProjectMgr, this.referencedProjectGuid);
  313. }
  314. #endregion
  315. #region methods
  316. protected override NodeProperties CreatePropertiesObject()
  317. {
  318. return new ProjectReferencesProperties(this);
  319. }
  320. /// <summary>
  321. /// The node is added to the hierarchy and then updates the build dependency list.
  322. /// </summary>
  323. public override void AddReference()
  324. {
  325. if(this.ProjectMgr == null)
  326. {
  327. return;
  328. }
  329. base.AddReference();
  330. this.ProjectMgr.AddBuildDependency(this.buildDependency);
  331. return;
  332. }
  333. /// <summary>
  334. /// Overridden method. The method updates the build dependency list before removing the node from the hierarchy.
  335. /// </summary>
  336. public override void Remove(bool removeFromStorage)
  337. {
  338. if(this.ProjectMgr == null || !this.CanRemoveReference)
  339. {
  340. return;
  341. }
  342. this.ProjectMgr.RemoveBuildDependency(this.buildDependency);
  343. base.Remove(removeFromStorage);
  344. return;
  345. }
  346. /// <summary>
  347. /// Links a reference node to the project file.
  348. /// </summary>
  349. protected override void BindReferenceData()
  350. {
  351. Debug.Assert(!String.IsNullOrEmpty(this.referencedProjectName), "The referencedProjectName field has not been initialized");
  352. Debug.Assert(this.referencedProjectGuid != Guid.Empty, "The referencedProjectName field has not been initialized");
  353. this.ItemNode = new ProjectElement(this.ProjectMgr, this.referencedProjectRelativePath, ProjectFileConstants.ProjectReference);
  354. this.ItemNode.SetMetadata(ProjectFileConstants.Name, this.referencedProjectName);
  355. this.ItemNode.SetMetadata(ProjectFileConstants.Project, this.referencedProjectGuid.ToString("B"));
  356. this.ItemNode.SetMetadata(ProjectFileConstants.Private, true.ToString());
  357. }
  358. /// <summary>
  359. /// Defines whether this node is valid node for painting the refererence icon.
  360. /// </summary>
  361. /// <returns></returns>
  362. protected override bool CanShowDefaultIcon()
  363. {
  364. if(this.referencedProjectGuid == Guid.Empty || this.ProjectMgr == null || this.ProjectMgr.IsClosed || this.isNodeValid)
  365. {
  366. return false;
  367. }
  368. IVsHierarchy hierarchy = null;
  369. hierarchy = VsShellUtilities.GetHierarchy(this.ProjectMgr.Site, this.referencedProjectGuid);
  370. if(hierarchy == null)
  371. {
  372. return false;
  373. }
  374. //If the Project is unloaded return false
  375. if(this.ReferencedProjectObject == null)
  376. {
  377. return false;
  378. }
  379. return (!String.IsNullOrEmpty(this.referencedProjectFullPath) && File.Exists(this.referencedProjectFullPath));
  380. }
  381. /// <summary>
  382. /// Checks if a project reference can be added to the hierarchy. It calls base to see if the reference is not already there, then checks for circular references.
  383. /// </summary>
  384. /// <param name="errorHandler">The error handler delegate to return</param>
  385. /// <returns></returns>
  386. protected override bool CanAddReference(out CannotAddReferenceErrorMessage errorHandler)
  387. {
  388. // When this method is called this refererence has not yet been added to the hierarchy, only instantiated.
  389. if(!base.CanAddReference(out errorHandler))
  390. {
  391. return false;
  392. }
  393. errorHandler = null;
  394. if(this.IsThisProjectReferenceInCycle())
  395. {
  396. errorHandler = new CannotAddReferenceErrorMessage(ShowCircularReferenceErrorMessage);
  397. return false;
  398. }
  399. return true;
  400. }
  401. private bool IsThisProjectReferenceInCycle()
  402. {
  403. return IsReferenceInCycle(this.referencedProjectGuid);
  404. }
  405. private void ShowCircularReferenceErrorMessage()
  406. {
  407. string message = String.Format(CultureInfo.CurrentCulture, SR.GetString(SR.ProjectContainsCircularReferences, CultureInfo.CurrentUICulture), this.referencedProjectName);
  408. string title = string.Empty;
  409. OLEMSGICON icon = OLEMSGICON.OLEMSGICON_CRITICAL;
  410. OLEMSGBUTTON buttons = OLEMSGBUTTON.OLEMSGBUTTON_OK;
  411. OLEMSGDEFBUTTON defaultButton = OLEMSGDEFBUTTON.OLEMSGDEFBUTTON_FIRST;
  412. VsShellUtilities.ShowMessageBox(this.ProjectMgr.Site, title, message, icon, buttons, defaultButton);
  413. }
  414. /// <summary>
  415. /// Recursively search if this project reference guid is in cycle.
  416. /// </summary>
  417. private bool IsReferenceInCycle(Guid projectGuid)
  418. {
  419. IVsHierarchy hierarchy = VsShellUtilities.GetHierarchy(this.ProjectMgr.Site, projectGuid);
  420. IReferenceContainerProvider provider = hierarchy as IReferenceContainerProvider;
  421. if(provider != null)
  422. {
  423. IReferenceContainer referenceContainer = provider.GetReferenceContainer();
  424. Debug.Assert(referenceContainer != null, "Could not found the References virtual node");
  425. foreach(ReferenceNode refNode in referenceContainer.EnumReferences())
  426. {
  427. ProjectReferenceNode projRefNode = refNode as ProjectReferenceNode;
  428. if(projRefNode != null)
  429. {
  430. if(projRefNode.ReferencedProjectGuid == this.ProjectMgr.ProjectIDGuid)
  431. {
  432. return true;
  433. }
  434. if(this.IsReferenceInCycle(projRefNode.ReferencedProjectGuid))
  435. {
  436. return true;
  437. }
  438. }
  439. }
  440. }
  441. return false;
  442. }
  443. #endregion
  444. }
  445. }