Avvio di una compilazione all'interno dell'IDE
I sistemi di progetto personalizzati devono utilizzare IVsBuildManagerAccessor per avviare le compilazioni. In questo argomento vengono descritti i motivi e viene delineata la procedura.
Thread e compilazioni paralleli
Visual Studio consente compilazioni parallele che richiedono una mediazione per l'accesso alle risorse comuni. I sistemi di progetto consentono di eseguire le compilazioni in modo asincrono, ma tali sistemi non devono chiamare funzioni di compilazione dall'interno di callback forniti al gestore di compilazione.
Se il sistema di progetto modifica le variabili di ambiente, deve impostare il valore NodeAffinity della compilazione su OutOfProc. Questo vuol dire che non è possibile utilizzare oggetti host, poiché richiedono il nodo in-process.
Utilizzo di IVSBuildManagerAccessor
Nel codice seguente viene delineato un metodo che un sistema di progetto può utilizzare per avviare una compilazione:
public bool Build(Project project, bool isDesignTimeBuild)
{
// Get the accessor from the IServiceProvider interface for the
// project system
IVsBuildManagerAccessor accessor =
serviceProvider.GetService(typeof(SVsBuildManagerAccessor)) as
IVsBuildManagerAccessor;
bool releaseUIThread = false;
try
{
if(accessor != null)
{
// Claim the UI thread under the following conditions:
// 1. The build must use a resource that uses the UI thread
// or,
// 2. The build requires the in-proc node AND waits on the
// UI thread for the build to complete
if(NeedsUIThread)
{
int result = accessor.ClaimUIThreadForBuild();
if(result != S_OK)
{
// Not allowed to claim the UI thread right now
return false;
}
releaseUIThread = true;
}
if(isDesignTimeBuild)
{
// Start the design time build
int result = accessor.BeginDesignTimeBuild();
if(result != S_OK)
{
// Not allowed to begin a design-time build at
// this time. Try again later.
return false;
}
}
}
bool buildSucceeded = false;
// perform project-system specific build set up tasks
// Create your BuildRequestData
// This assumes a IHostServices variable (hostServices) set
// to your host services. If you don't use a project instance
// (you build from a file for example) then use another
// constructor.
BuildRequestData requestData = new
BuildRequestData(project.CreateProjectInstance(),
"myTarget", hostServices,
BuildRequestData.BuildRequestDataFlags.None);
// Mark your your submission as Pending
BuildSubmission submission =
BuildManager.DefaultBuildManager.
PendBuildRequest(requestData);
// Register the loggers in BuildLoggers
if (accessor != null)
{
foreach (ILogger logger in BuildLoggers)
{
accessor.RegisterLogger(submission.SubmissionId,
logger);
}
}
BuildResult buildResult = submission.Execute();
return buildResult;
}
// Clean up resources
finally
{
if(accessor != null)
{
// Unregister the loggers, if necessary.
accessor.UnregisterLoggers(submission.SubmissionId);
// Release the UI thread, if used
if(releaseUIThread)
{
accessor.ReleaseUIThreadForBuild();
}
// End the design time build, if used
if(isDesignTimeBuild)
{
accessor.EndDesignTimeBuild();
}
}
}
}