Подтвердить что ты не робот

Как подписаться на решения и события проекта из VSPackage

Я разрабатываю языковой сервис для Visual Studio через VSPackage. Мне нужно обновить данные анализа, когда файлы будут добавлены/удалены из проектов решений.

Я хочу подписаться на решения и события проекта.

Я попытался сделать следующее, но ни одно из этих событий не срабатывает, когда я добавляю/удаляю проекты в решение или добавляю/удаляю элементы в проекты.

DTE dte = (DTE)languageService.GetService(typeof(DTE));
if (dte == null)
    return;

((Events2)dte.Events).SolutionEvents.ProjectAdded += SolutionEvents_ProjectAdded;
((Events2)dte.Events).SolutionEvents.ProjectRemoved += SolutionEvents_ProjectRemoved;
((Events2)dte.Events).ProjectItemsEvents.ItemAdded += ProjectItemsEvents_ItemAdded;
((Events2)dte.Events).ProjectItemsEvents.ItemRemoved += ProjectItemsEvents_ItemRemoved;

Какой лучший способ подписаться на эти события из VSPackage? Любая помощь ценится!

4b9b3361

Ответ 1

В качестве альтернативы вы можете использовать IVsSolutionEvents3, который имеет гораздо лучшие события

[PackageRegistration( UseManagedResourcesOnly = true )]
[InstalledProductRegistration( "#110", "#112", "1.0", IconResourceID = 400 )]
// add these 2 Annotations to execute Initialize() immediately when a project is loaded
[ProvideAutoLoad( VSConstants.UICONTEXT.SolutionHasSingleProject_string )]
[ProvideAutoLoad( VSConstants.UICONTEXT.SolutionHasMultipleProjects_string )]
[Guid( GuidList.XYZ )]
public sealed class UnityProjectUpdateHandlerPackage : Package, IVsSolutionEvents3
{
    private DTE _dte;
    private IVsSolution solution = null;
    private uint _hSolutionEvents = uint.MaxValue;

    protected override void Initialize()
    {
        Trace.WriteLine( string.Format( CultureInfo.CurrentCulture, "Entering Initialize() of: {0}", this.ToString() ) );
        base.Initialize();

        this._dte = (DTE) this.GetService( typeof( DTE ) );

        AdviseSolutionEvents();
    }

    protected override void Dispose( bool disposing )
    {
        UnadviseSolutionEvents();

        base.Dispose( disposing );
    }

    private void AdviseSolutionEvents()
    {
        UnadviseSolutionEvents();

        solution = this.GetService( typeof( SVsSolution ) ) as IVsSolution;

        if ( solution != null )
        {
            solution.AdviseSolutionEvents( this, out _hSolutionEvents );
        }
    }

    private void UnadviseSolutionEvents()
    {
        if ( solution != null )
        {
            if ( _hSolutionEvents != uint.MaxValue )
            {
                solution.UnadviseSolutionEvents( _hSolutionEvents );
                _hSolutionEvents = uint.MaxValue;
            }

            solution = null;
        }
    }

    private Project[] GetProjects()
    {
        return _dte.Solution.Projects
            .Cast<Project>()
            .Select( x => ( (VSProject) x.Object ).Project )
            .ToArray();
    }

    public int OnAfterLoadProject( IVsHierarchy pStubHierarchy, IVsHierarchy pRealHierarchy )
    {
        // Do something
        return VSConstants.S_OK;
    }

    public int OnAfterOpenSolution( object pUnkReserved, int fNewSolution )
    {
        foreach ( var project in GetProjects() )
            ; // Do something

        return VSConstants.S_OK;
    }

    public int OnBeforeUnloadProject( IVsHierarchy pRealHierarchy, IVsHierarchy pStubHierarchy )
    {
        // Do something
        return VSConstants.S_OK;
    }

    public int OnAfterCloseSolution( object pUnkReserved )
    { return VSConstants.S_OK; }

    public int OnAfterClosingChildren( IVsHierarchy pHierarchy )
    { return VSConstants.S_OK; }

    public int OnAfterMergeSolution( object pUnkReserved )
    { return VSConstants.S_OK; }

    public int OnAfterOpenProject( IVsHierarchy pHierarchy, int fAdded )
    { return VSConstants.S_OK; }

    public int OnAfterOpeningChildren( IVsHierarchy pHierarchy )
    { return VSConstants.S_OK; }

    public int OnBeforeCloseProject( IVsHierarchy pHierarchy, int fRemoved )
    { return VSConstants.S_OK; }

    public int OnBeforeClosingChildren( IVsHierarchy pHierarchy )
    { return VSConstants.S_OK; }

    public int OnBeforeOpeningChildren( IVsHierarchy pHierarchy )
    { return VSConstants.S_OK; }

    public int OnBeforeCloseSolution( object pUnkReserved )
    { return VSConstants.S_OK; }

    public int OnQueryCloseProject( IVsHierarchy pHierarchy, int fRemoving, ref int pfCancel )
    { return VSConstants.S_OK; }

    public int OnQueryCloseSolution( object pUnkReserved, ref int pfCancel )
    { return VSConstants.S_OK; }

    public int OnQueryUnloadProject( IVsHierarchy pRealHierarchy, ref int pfCancel )
    { return VSConstants.S_OK; }
}

Ответ 2

События DTE немного странны, вам нужно кэшировать объект источника события (SolutionEvents и ProjectItemEvents в вашем случае), чтобы COM Interop знал, чтобы сохранить их в живых.

public class MyClass
{
    SolutionEvents solutionEvents; 

    public void ConnectToEvents()
    {
        solutionEvents = ((Events2)dte.Events).SolutionEvents; 
        solutionEvents.ProjectAdded += OnProjectAdded; 
        // Etc 
    }
}

Подробнее об этом @http://msdn.microsoft.com/en-us/library/ms165650(v=vs.80).aspx

Ответ 3

Давайте сосредоточимся на событии ProjectAdded (хотя описанная проблема точно такая же для остальных событий).

Выбранный образец кода пытается зарегистрировать обработчик SolutionEvents_ProjectAdded для события ProjectAdded. Тем не менее, объект SolutionEvents, отображающий событие, имеет ограниченный срок действия для закрытия его метода упаковки (вы не указали его подпись - позвоните ему Connect). После того, как поток управления оставил эту область видимости, локальный объект уже был собран мусором, поэтому его событие никогда не вызывается:

Сломанный код:

public class Connector
{
    public void Connect()
    {
        ((Events2)dte.Events).SolutionEvents.ProjectAdded += SolutionEvents_ProjectAdded;
    }
    void SolutionEvents_ProjectAdded() 
    { 
        // callback is dead
    }
}

Чтобы исправить это, вам нужно назначить объект SolutionEvents некоторой переменной, чье жизненное время распространяется на обработчик SolutionEvents_ProjectAdded - например. по всему классу упаковки. В приведенном ниже примере область действия распространяется на весь тип (пусть называется Connector) и гарантирует, что обработчик доступен в течение всего времени жизни этого типа:

Фиксированный код:

public class Connector
{
    SolutionEvents _solutionEvents;
    public void Connect()
    {
        _solutionEvents = ((Events2)dte.Events).SolutionEvents; 
        _solutionEvents.ProjectAdded += SolutionEvents_ProjectAdded;
    }
    void SolutionEvents_ProjectAdded() 
    { 
        // callback works
    }
}

Чтобы быть более точным, проверьте эту ссылку в MSDN - Переменные для области видимости Соответственно в обработчиках событий:

Общей ошибкой в ​​обработчиках событий события является подключение события обработчик объекта, который был объявлен с ограниченным объемом с целью обработки события. Объект должен иметь жизнь который охватывает не только функцию, которая соединяет обратный вызов метод как обработчик события объекта, но также и над обратным вызовом самого метода, где событие фактически обрабатывается. В противном случае, если объект выходит за рамки и больше не определяется в обратном вызове метод, метод обратного вызова не вызывается и событие не обрабатывается по желанию.