MVVM从VM访问Richtextbox flowdocument

时间:2017-06-02 03:11:42

标签: c# mvvm richtextbox flowdocument

我正在尝试使用FlowDocument创建一个RichTextBox,我可以在插入位置插入文本。我可以在文档的末尾添加文本。我想我在设置中遗漏了一些东西,允许我的VM访问Flowocument或者我设置错了。如果我在我的VM中创建一个FlowDocument并尝试将我的RichTextBox设置为它,我得到一个错误,我的MyEditor(RichTextBox)不存在。我可以使用ListBox中的AddItemBtn向RichTextBox添加文本,所以至少可以使用它。

我的问题是“我该如何设置我的RichTextBox / FlowDocument?

XAML代码

<Window x:Class="Scripter.Views.MainWindow"
    xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
    xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
    xmlns:local="clr-namespace:Scripter.ViewModels"
    xmlns:wpftoolkit="clr-namespace:Xceed.Wpf.Toolkit;assembly=Xceed.Wpf.Toolkit"
    Title="MainWindow" Height="350" Width="725">


<Grid HorizontalAlignment="Stretch">

    <Grid HorizontalAlignment="Stretch" Height="72" Margin="10,14,0,0" VerticalAlignment="Top" Width="auto">
        <WrapPanel HorizontalAlignment="Left" Height="50" Margin="10,0,0,0" VerticalAlignment="Top">


        </WrapPanel>
        <Button x:Name="OpenFilesBtn" Content="Open" HorizontalAlignment="Left" Margin="15,10,0,0" VerticalAlignment="Top" Width="75" Command="{Binding OpenFileBtn}"/>
        <Button x:Name="SavefilesBtn" Content="Save" HorizontalAlignment="Left" Margin="104,10,0,0" VerticalAlignment="Top" Width="75" Command="{Binding SaveFileBtn}"/>
        <TextBlock x:Name="OpenFile" Text="{Binding OpenFile,Mode=OneWay, UpdateSourceTrigger=PropertyChanged}" HorizontalAlignment="Left" Margin="15,37,0,0" VerticalAlignment="Top" Width="353"/>
        <ComboBox x:Name="TipsBtn" SelectedIndex="0" ItemsSource="{Binding Path=Tabs, UpdateSourceTrigger=PropertyChanged}" SelectedItem="{Binding Path=SelectedOption}"  HorizontalAlignment="Left" Margin="538,10,0,0" VerticalAlignment="Top" Width="120"/>
        <Button x:Name="AddItemBtn" Content="Add Item" HorizontalAlignment="Left" Margin="417,10,0,0" VerticalAlignment="Top" Width="100" Command="{Binding AddItemBtn}" CommandParameter="{Binding ElementName=AddItemList,Path=SelectedItem}"/>

    </Grid>
    <Grid Margin="10,100,10,0">
        <Grid.ColumnDefinitions>
            <ColumnDefinition Width="2*"/>
            <ColumnDefinition Width="*"/>
        </Grid.ColumnDefinitions>
        <RichTextBox  Grid.Column="0" x:Name="MyEditor" SelectionChanged="MyEditor_SelectionChanged" ScrollViewer.VerticalScrollBarVisibility="Auto" Margin="0" Height="Auto" HorizontalAlignment="Stretch"  VerticalAlignment="Stretch" Width="Auto" IsDocumentEnabled="True" AcceptsTab="True" AcceptsReturn="True" >
            <RichTextBox.Resources>
                <Style TargetType="{x:Type Paragraph}">
                    <Setter Property="Margin" Value="0" ></Setter>
                    <Setter Property="FontSize" Value="15"></Setter>
                </Style>
            </RichTextBox.Resources>
            <FlowDocument >
                <Paragraph >
                    <Run Text="{Binding TestText}" ></Run>
                </Paragraph>
            </FlowDocument>
        </RichTextBox>
        <ListBox x:Name="AddItemList" Grid.Column="1" Width="Auto" Height="Auto" ItemsSource="{Binding Path=OptionsToChoose}" SelectedItem="ItemSelected">
            <ListBox.ItemTemplate>
                <DataTemplate>
                    <TextBlock x:Name="TextSelected" Text="{Binding Description}" />
                </DataTemplate>
            </ListBox.ItemTemplate>
        </ListBox>

    </Grid>
</Grid>

有问题的VM代码

public ScripterViewModel()
    {
        ScripterModel scripterModel = new ScripterModel();

        ObservableCollection<string> tabsChoice = new ObservableCollection<string>();
        tabsChoice.Add("Tabs");
        tabsChoice.Add("Buttons");
        Tabs = tabsChoice;

        this.OpenFileBtn = new DelegateCommand(chooseFile, canChooseFile).ObservesProperty(() => OpenFile);
        this.SaveFileBtn = new DelegateCommand(saveFile, canSaveFile).ObservesProperty(() => SaveFile);
        this.AddItemBtn = new DelegateCommand<Tabbed>(addItem);

        FlowDocument flowDoc = new FlowDocument();

        Paragraph p = new Paragraph(new Run("new paragraph"));
        flowDoc.Blocks.Add(new Paragraph(new Run("Paragraph 1")));
        flowDoc.Blocks.Add(p);

        //MyEditor = flowDoc;

    }

    public void MyEditor_SelectionChanged(object sender, RoutedEventArgs e)
    {
      // TextRange tempRange = new TextRange(MyEditor.Document.ContentStart, MyEditor.Selection.Start);
        MessageBox.Show("Selection Changed");

    }

    private string _testText;
    public string TestText
    {
        get
        {
            return _testText;
        }

        set
        {
            string _temp;
            _temp = _testText + value;
            SetProperty(ref _testText, value);
        }

    }

1 个答案:

答案 0 :(得分:1)

嘿,我是WPF和MVVM的新手,但我会尽力帮助你。所以,如果我错了,不要怪我。

1。设置Window.DataContext

首先,您必须告诉 View 从哪里获取数据。 这可以通过将此代码添加到 View.xaml

来完成
<Window.DataContext>
    <local:MainViewModel/>
</Window.DataContext>

但请确保您的命名空间变量(此处为“local”)指向您的ViewModel。

xmlns:local="clr-namespace:Client.ViewModel"

这是前者。指向我的ViewModel文件夹。

2。定义OnPropertyChanged方法

您的查看不知道您是否修改了变量。因此,您需要一种方法来通知您的查看有关更改的信息。 首先将接口INotifyPropertyChanged实现到 ViewModel

public class MainViewModel : ViewModelBase, INotifyPropertyChanged

现在添加以下代码:

    public event PropertyChangedEventHandler PropertyChanged;
    public void OnPropertyChanged([CallerMemberName]string propertyName = null)
    {
        PropertyChanged?.Invoke(this, new PropertyChangedEventArgs(propertyName));
    }

3。使用OnPropertyChanged

所以现在你有一种方法告诉你的View变量已经改变但是你如何使用它? 为了向您解释这一点,我将使用您的FlowDocument flowDoc

让我们首先定义您的FlowDocument设置:

private FlowDocument _flowDoc;

现在让我们写一个吸气剂&amp; flowDoc 的setter:

    public FlowDocument FlowDoc
    {
        get
        {
            return _flowDoc;
        }
        set
        {
           _flowDoc = value;
        }
    }

现在是时候使用我们在 2中创建的 OnPropertyChanged 方法。 在setter部分中,您要添加以下代码:

OnPropertyChanged("variable");

您的结果现在应如下所示:

public FlowDocument FlowDoc
    {
        get
        {
            return _flowDoc;
        }
        set
        {
           _flowDoc = value;
           OnPropertyChanged("FlowDoc");
        }
    }

重要提示:请记住将其应用于所有变量!

4。正确使用MVVM模式

MVVM 中,您有模型 a 查看 ViewModel 。 该模型适用于您的数据,因此如果可能,请不要在ViewModel中存储数据,而是使用数据类,例如

您可以查看 this 和/或 this

正如我在开始时说的那样,我对这一切都是新手,但我希望它有所帮助 您。随意问。