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

MVVM и свойство TextBox SelectedText

У меня есть TextBox с ContextMenu. Когда пользователь щелкнет правой кнопкой мыши внутри TextBox и выберет соответствующий MenuItem, я хотел бы захватить SelectedText в моей модели viewmodel. Я не нашел хорошего способа сделать это "MVVM".

До сих пор у меня есть мое приложение, использующее способ Джошуса Смита MVVM. Я ищу, чтобы перебраться в Чинч. Не уверен, что Cinch Framework будет обрабатывать такие проблемы. Мысли?

4b9b3361

Ответ 1

Нет простого способа привязать SelectedText к источнику данных, потому что это не DependencyProperty... однако довольно легко создать прикрепленное свойство, которое вы могли бы привязать вместо этого.

Вот базовая реализация:

public static class TextBoxHelper
{

    public static string GetSelectedText(DependencyObject obj)
    {
        return (string)obj.GetValue(SelectedTextProperty);
    }

    public static void SetSelectedText(DependencyObject obj, string value)
    {
        obj.SetValue(SelectedTextProperty, value);
    }

    // Using a DependencyProperty as the backing store for SelectedText.  This enables animation, styling, binding, etc...
    public static readonly DependencyProperty SelectedTextProperty =
        DependencyProperty.RegisterAttached(
            "SelectedText",
            typeof(string),
            typeof(TextBoxHelper),
            new FrameworkPropertyMetadata(null, FrameworkPropertyMetadataOptions.BindsTwoWayByDefault, SelectedTextChanged));

    private static void SelectedTextChanged(DependencyObject obj, DependencyPropertyChangedEventArgs e)
    {
        TextBox tb = obj as TextBox;
        if (tb != null)
        {
            if (e.OldValue == null && e.NewValue != null)
            {
                tb.SelectionChanged += tb_SelectionChanged;
            }
            else if (e.OldValue != null && e.NewValue == null)
            {
                tb.SelectionChanged -= tb_SelectionChanged;
            }

            string newValue = e.NewValue as string;

            if (newValue != null && newValue != tb.SelectedText)
            {
                tb.SelectedText = newValue as string;
            }
        }
    }

    static void tb_SelectionChanged(object sender, RoutedEventArgs e)
    {
        TextBox tb = sender as TextBox;
        if (tb != null)
        {
            SetSelectedText(tb, tb.SelectedText);
        }
    }

}

Затем вы можете использовать его в XAML:

<TextBox Text="{Binding Message}" u:TextBoxHelper.SelectedText="{Binding SelectedText}" />

Ответ 2

Примеры приложений в WPF Application Framework (WAF) выбрали другой способ решить эту проблему. Там ViewModel разрешен доступ к представлению через интерфейс (IView), и поэтому он может запрашивать текущий SelectedText.

Я считаю, что привязка не должна использоваться в каждом сценарии. Иногда писать несколько строк в коде намного проще, чем использовать высокоразвитые вспомогательные классы. Но это только мое мнение: -)

JBE

Ответ 3

Я знаю, что он был принят и принят, но я думал, что добавлю свое решение. Я использую Поведение для соединения между моделью просмотра и TextBox. Поведение имеет свойство зависимости (CaretPositionProperty), которое может быть связано двумя способами с моделью представления. Внутренне поведение связано с обновлениями/из TextBox.

public class SetCaretIndexBehavior : Behavior<TextBox>
    {
        public static readonly DependencyProperty CaretPositionProperty;
        private bool _internalChange;

    static SetCaretIndexBehavior()
    {

    CaretPositionProperty = DependencyProperty.Register("CaretPosition", typeof(int), typeof(SetCaretIndexBehavior), new PropertyMetadata(0, OnCaretPositionChanged));
}

public int CaretPosition
{
    get { return Convert.ToInt32(GetValue(CaretPositionProperty)); }
    set { SetValue(CaretPositionProperty, value); }
}

protected override void OnAttached()
{
    base.OnAttached();
    AssociatedObject.KeyUp += OnKeyUp;
}

private static void OnCaretPositionChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
    var behavior = (SetCaretIndexBehavior)d;
    if (!behavior._internalChange)
    {
        behavior.AssociatedObject.CaretIndex = Convert.ToInt32(e.NewValue);
    }
}

    private void OnKeyUp(object sender, KeyEventArgs e)
    {
        _internalChange = true;
        CaretPosition = AssociatedObject.CaretIndex;
        _internalChange = false;
    }
}