mirror of
https://github.com/microsoft/PowerToys.git
synced 2026-02-24 04:00:02 +01:00
Functionally, no differences. - Removed Core projects. - Core.Common => Microsoft.CmdPal.Common - Core.ViewModels => Microsoft.CmdPal.UI.ViewModels --------- Co-authored-by: Jiří Polášek <me@jiripolasek.com>
163 lines
4.8 KiB
C#
163 lines
4.8 KiB
C#
// Copyright (c) Microsoft Corporation
|
|
// The Microsoft Corporation licenses this file to you under the MIT license.
|
|
// See the LICENSE file in the project root for more information.
|
|
|
|
using System.Collections.ObjectModel;
|
|
using Microsoft.CmdPal.UI.ViewModels;
|
|
using Microsoft.CmdPal.UI.ViewModels.Models;
|
|
using Microsoft.CommandPalette.Extensions;
|
|
using Microsoft.CommandPalette.Extensions.Toolkit;
|
|
|
|
namespace Microsoft.CmdPal.UI.ViewModels;
|
|
|
|
public partial class ContentTreeViewModel(ITreeContent _tree, WeakReference<IPageContext> context) :
|
|
ContentViewModel(context)
|
|
{
|
|
public ExtensionObject<ITreeContent> Model { get; } = new(_tree);
|
|
|
|
// Remember - "observable" properties from the model (via PropChanged)
|
|
// cannot be marked [ObservableProperty]
|
|
public ContentViewModel? RootContent { get; protected set; }
|
|
|
|
public ObservableCollection<ContentViewModel> Children { get; } = [];
|
|
|
|
public bool HasChildren => Children.Count > 0;
|
|
|
|
// This is the content that's actually bound in XAML. We needed a
|
|
// collection, even if the collection is just a single item.
|
|
public ObservableCollection<ContentViewModel> Root => RootContent is not null ? [RootContent] : [];
|
|
|
|
public override void InitializeProperties()
|
|
{
|
|
var model = Model.Unsafe;
|
|
if (model is null)
|
|
{
|
|
return;
|
|
}
|
|
|
|
var root = model.RootContent;
|
|
if (root is not null)
|
|
{
|
|
RootContent = ViewModelFromContent(root, PageContext);
|
|
RootContent?.InitializeProperties();
|
|
UpdateProperty(nameof(RootContent));
|
|
UpdateProperty(nameof(Root));
|
|
}
|
|
|
|
FetchContent();
|
|
model.PropChanged += Model_PropChanged;
|
|
model.ItemsChanged += Model_ItemsChanged;
|
|
}
|
|
|
|
// Theoretically, we should unify this with the one in CommandPalettePageViewModelFactory
|
|
// and maybe just have a ContentViewModelFactory or something
|
|
public ContentViewModel? ViewModelFromContent(IContent content, WeakReference<IPageContext> context)
|
|
{
|
|
ContentViewModel? viewModel = content switch
|
|
{
|
|
IFormContent form => new ContentFormViewModel(form, context),
|
|
IMarkdownContent markdown => new ContentMarkdownViewModel(markdown, context),
|
|
ITreeContent tree => new ContentTreeViewModel(tree, context),
|
|
_ => null,
|
|
};
|
|
return viewModel;
|
|
}
|
|
|
|
// TODO: Does this need to hop to a _different_ thread, so that we don't block the extension while we're fetching?
|
|
private void Model_ItemsChanged(object sender, IItemsChangedEventArgs args) => FetchContent();
|
|
|
|
private void Model_PropChanged(object sender, IPropChangedEventArgs args)
|
|
{
|
|
try
|
|
{
|
|
var propName = args.PropertyName;
|
|
FetchProperty(propName);
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
ShowException(ex);
|
|
}
|
|
}
|
|
|
|
protected void FetchProperty(string propertyName)
|
|
{
|
|
var model = Model.Unsafe;
|
|
if (model is null)
|
|
{
|
|
return; // throw?
|
|
}
|
|
|
|
switch (propertyName)
|
|
{
|
|
case nameof(RootContent):
|
|
var root = model.RootContent;
|
|
if (root is not null)
|
|
{
|
|
RootContent = ViewModelFromContent(root, PageContext);
|
|
}
|
|
else
|
|
{
|
|
root = null;
|
|
}
|
|
|
|
UpdateProperty(nameof(Root));
|
|
|
|
break;
|
|
}
|
|
|
|
UpdateProperty(propertyName);
|
|
}
|
|
|
|
//// Run on background thread, from InitializeAsync or Model_ItemsChanged
|
|
private void FetchContent()
|
|
{
|
|
List<ContentViewModel> newContent = [];
|
|
try
|
|
{
|
|
var newItems = Model.Unsafe!.GetChildren();
|
|
|
|
foreach (var item in newItems)
|
|
{
|
|
var viewModel = ViewModelFromContent(item, PageContext);
|
|
if (viewModel is not null)
|
|
{
|
|
viewModel.InitializeProperties();
|
|
newContent.Add((ContentViewModel)viewModel);
|
|
}
|
|
}
|
|
}
|
|
catch (Exception ex)
|
|
{
|
|
ShowException(ex);
|
|
throw;
|
|
}
|
|
|
|
// Now, back to a UI thread to update the observable collection
|
|
DoOnUiThread(
|
|
() =>
|
|
{
|
|
ListHelpers.InPlaceUpdateList(Children, newContent);
|
|
});
|
|
|
|
UpdateProperty(nameof(HasChildren));
|
|
}
|
|
|
|
protected override void UnsafeCleanup()
|
|
{
|
|
base.UnsafeCleanup();
|
|
RootContent?.SafeCleanup();
|
|
foreach (var item in Children)
|
|
{
|
|
item.SafeCleanup();
|
|
}
|
|
|
|
Children.Clear();
|
|
var model = Model.Unsafe;
|
|
if (model is not null)
|
|
{
|
|
model.PropChanged -= Model_PropChanged;
|
|
model.ItemsChanged -= Model_ItemsChanged;
|
|
}
|
|
}
|
|
}
|