Я борюсь с использованием SaveFileDialog в MVVM.
Я использую класс RelayCommand и запускаю SaveAsFileCommand. В SaveAsFileCommand, используя лямбда-выражение, я разделяю два аргумента:
Экземпляр управления RichTextBox и целевой путь (Путь файла)
Затем я вызываю DataIO.SaveAsFile (arguments [0], arguments [1]), используя приведенные выше аргументы.
Чтобы создать SaveDialogBox на слое просмотра, я использую 3 класса: Чат, FileDialogBox и SaveFileDialogBox
В XAML я создаю SaveDialogBox и пытаюсь вызвать SaveAsFileCommand, используя MultiBinding, чтобы передать эти два параметра команды.
Чтобы показать SaveDialogBox, я использую кнопку, привязанную к SaveDialogBox
Проблема в: здесь компилятор жалуется, что он не может выполнить мульти-привязку для моего SaveDialogBox из-за того, что он не является DependencyObject и не-DependencyProperty.
Как я могу решить эту проблему и правильно сохранить файл, используя этот DialogBox
, соответствующий MVVM в моем случае ???
Части кода XAML:
<Button Command = "{Binding ElementName=SaveFileDB, Path=Show}" >
<Button.ToolTip>
<ToolTip Style = "{StaticResource styleToolTip}" >
<TextBlock Text = "Save" Style = "{StaticResource styleTextBlockTP}" />
</ToolTip>
</Button.ToolTip>
<Image Source = "Icon\Save.png"/>
</Button>
<local:SaveFileDialogBox x:Name = "SaveFileDB" Caption = "Save content to file..."
Filter = "Text files (*.txt)|*.txt|All files(*.*)|*.*"
FilterIndex = "0" DefaultExt = "txt"
CommandFileOK = "{Binding SaveAsFileCommand}" >
<local:SaveFileDialogBox.CommandParemeter>
<MultiBinding Converter = "{StaticResource parametersConverter}">
<Binding ElementName = "MainRichTextBox" />
<Binding ElementName = "SaveFileDB" Path = "Show" />
</MultiBinding>
</local:SaveFileDialogBox.CommandParemeter>
</local:SaveFileDialogBox>
SaveAsFileCommand:
private ICommand _SaveAsFileCommand;
public ICommand SaveAsFileCommand
{
get
{
if (_SaveAsFileCommand == null)
{
_SaveAsFileCommand = new RelayCommand(
argument =>
{
var arguments = (object[])argument;
DataIO.SaveAsFile(arguments[0], arguments[1]);
}
);
}
return _SaveAsFileCommand;
}
}
DataIO.SaveAsFile метод:
public static void SaveAsFile(object argument0, object argument1)
{
System.Windows.Controls.RichTextBox richTB = argument0 as System.Windows.Controls.RichTextBox;
string path = (string)argument1;
using (FileStream fileStream = new FileStream(path, FileMode.Create))
{
TextRange textRange = new TextRange(richTB.Document.ContentStart, richTB.Document.ContentEnd);
textRange.Save(fileStream, DataFormats.Text);
}
}
RelayCommand класс:
class RelayCommand : ICommand
{
private readonly Action<object> _Execute;
private readonly Func<object, bool> _CanExecute;
public RelayCommand(Action<object> execute, Func<object, bool> canExecute)
{
if (execute == null) throw new ArgumentNullException("execute");
_Execute = execute;
_CanExecute = canExecute;
}
public RelayCommand(Action<object> execute)
{
if (execute == null) throw new ArgumentNullException("execute");
_Execute = execute;
}
public bool CanExecute(object parameter)
{
return _CanExecute == null ? true : _CanExecute(parameter);
}
public event EventHandler CanExecuteChanged
{
add
{
if (_CanExecute != null) CommandManager.RequerySuggested += value;
}
remove
{
if (_CanExecute != null) CommandManager.RequerySuggested -= value;
}
}
public void Execute(object parameter)
{
_Execute(parameter);
}
}
Класс DialogBox:
public abstract class DialogBox : FrameworkElement, INotifyPropertyChanged
{
public event PropertyChangedEventHandler PropertyChanged;
protected void OnPropertyChanged(string parameter)
{
if (PropertyChanged != null)
PropertyChanged(this, new PropertyChangedEventArgs(parameter));
}
protected Action<object> execute = null;
public string Caption { get; set; }
protected ICommand show;
public virtual ICommand Show
{
get
{
if (show == null)
show = new RelayCommand(execute);
return show;
}
}
}
FileDialogBox класс:
public abstract class FileDialogBox : CommandDialogBox
{
public bool? FileDialogResult { get; protected set; }
public string FilePath { get; set; }
public string Filter { get; set; }
public int FilterIndex { get; set; }
public string DefaultExt { get; set; }
protected Microsoft.Win32.FileDialog fileDialog = null;
protected FileDialogBox()
{
execute =
o =>
{
var values = (object[])o;
RelayCommand relCom1 = (RelayCommand)values[1];
fileDialog.Title = Caption;
fileDialog.Filter = Filter;
fileDialog.FilterIndex = FilterIndex;
fileDialog.DefaultExt = DefaultExt;
string filePath = "";
if (FilePath != null) filePath = FilePath; else FilePath = "";
//if (o != null) filePath = (string)o;
//if (o != null) filePath = (string)values[1];
if (o != null) filePath = relCom1.ToString();
if (!String.IsNullOrWhiteSpace(filePath))
{
fileDialog.InitialDirectory = System.IO.Path.GetDirectoryName(filePath);
fileDialog.FileName = System.IO.Path.GetFileName(filePath);
}
FileDialogResult = fileDialog.ShowDialog();
OnPropertyChanged("FileDialogResult");
if (FileDialogResult.HasValue && FileDialogResult != null)
{
FilePath = fileDialog.FileName;
OnPropertyChanged("FilePath");
ExecuteCommand(CommandFileOK, FilePath);
};
};
}
public static DependencyProperty CommandFileOKProperty =
DependencyProperty.Register("CommandFileOK", typeof(ICommand), typeof(FileDialogBox));
public ICommand CommandFileOK
{
get { return (ICommand)GetValue(CommandFileOKProperty); }
set { SetValue(CommandFileOKProperty, value); }
}
}
SaveFileDialogBox класс:
public class SaveFileDialogBox : FileDialogBox
{
public SaveFileDialogBox()
{
fileDialog = new SaveFileDialog();
}
}
В этом случае мне нужно реализовать встроенные диалоговые окна (именно те, что из пространства имен Microsoft.Win32), но я хочу вызывать их из уровня просмотра, а не из ViewModel или Model - и это вызывает проблемы
Я обрабатываю требование ввода данных пользователем в диалоговом окне, используя элемент управления, который отображается в представлении, но не имеет пользовательского интерфейса. Я разделил команду на две части. По сути, они показывают диалог и вызывают команду, когда вы закончите. Элемент управления показывает диалоговое окно, которое захватывает данные и затем вызывает команду, которую вы даете ему через привязку. Поскольку это элемент управления, вы можете выполнить привязку в порядке, и он находится в визуальном дереве, поэтому он может получить ссылку на окно.
Пожалуйста, смотрите подтверждение запроса в этом:
https://gallery.technet.microsoft.com/WPF-User-Notification-MVVM-98940828
Это предназначено для подтверждения удаления записи, но тот же принцип может быть распространен на средство выбора файлов с небольшими изменениями.
Исходя из этого, команда, вызываемая после того, как пользователь щелкает и закрывает диалог, может захватывать любые переменные, которые вам нужны. Если связать их:
private RelayCommand _confirmCommand;
public RelayCommand ConfirmCommand
{
get
{
return _confirmCommand
?? (_confirmCommand = new RelayCommand(
() =>
{
confirmer.Caption = "Please Confirm";
confirmer.Message = "Are you SURE you want to delete this record?";
confirmer.MsgBoxButton = MessageBoxButton.YesNo;
confirmer.MsgBoxImage = MessageBoxImage.Warning;
OkCommand = new RelayCommand(
() =>
{
// You would do some actual deletion or something here
UserNotificationMessage msg = new UserNotificationMessage { Message = "OK.\rDeleted it.\rYour data is consigned to oblivion.", SecondsToShow = 5 };
Messenger.Default.Send<UserNotificationMessage>(msg);
});
RaisePropertyChanged("OkCommand");
ShowConfirmation = true;
}));
}
}
От запрашивающего подтверждения, вызывая эту команду:
public static readonly DependencyProperty ShowConfirmDialogProperty =
DependencyProperty.Register("ShowConfirmDialog",
typeof(bool?),
typeof(ConfirmationRequestor),
new FrameworkPropertyMetadata(null
, new PropertyChangedCallback(ConfirmDialogChanged)
)
{ BindsTwoWayByDefault = true }
);
private static void ConfirmDialogChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
if ((bool?)e.NewValue != true)
{
return;
}
ConfirmationRequestor cr = (ConfirmationRequestor)d;
Window parent = Window.GetWindow(cr) as Window;
MessageBoxResult result = MessageBox.Show(parent, cr.Message, cr.Caption, cr.MsgBoxButton, cr.MsgBoxImage);
if (result == MessageBoxResult.OK || result == MessageBoxResult.Yes)
{
if (cr.Command != null)
{
cr.Command.Execute(cr.CommandParameter);
}
}
cr.SetValue(ShowConfirmDialogProperty, false);
}
Я полагаю, вы хотите реализовать диалоги самостоятельно? Потому что, если вы будете довольны встроенными системными диалогами и при этом будете дружелюбны к MVVM, вы всегда сможете использовать MvvmDialogs.