MVVM - पदानुक्रम और नेविगेशन
एमवीवीएम अनुप्रयोगों का निर्माण करते समय, आप आम तौर पर जानकारी के जटिल स्क्रीन को माता-पिता और बच्चे के विचारों के एक समूह में विघटित करते हैं, जहां बच्चे के विचारों को पैनल या कंटेनर नियंत्रण में माता-पिता के विचारों में शामिल किया जाता है, और खुद के उपयोग का एक पदानुक्रम बनाता है।
जटिल दृश्यों को विघटित करने के बाद इसका मतलब यह नहीं है कि प्रत्येक बाल सामग्री का प्रत्येक टुकड़ा जिसे आप अपनी स्वयं की XAML फाइल में अलग करते हैं, आवश्यक रूप से एक MVVM दृश्य होना चाहिए।
सामग्री का हिस्सा केवल स्क्रीन को कुछ प्रदान करने के लिए संरचना प्रदान करता है और उस सामग्री के लिए उपयोगकर्ता द्वारा किसी भी इनपुट या हेरफेर का समर्थन नहीं करता है।
यह एक अलग ViewModel की आवश्यकता नहीं हो सकती है, लेकिन यह सिर्फ एक XAML हो सकता है जो माता-पिता ViewModel द्वारा उजागर गुणों के आधार पर प्रस्तुत करता है।
अंत में, यदि आपके पास दृश्य और दृश्यमॉडल का एक पदानुक्रम है, तो माता-पिता ViewModel संचार के लिए एक केंद्र बन सकते हैं, ताकि प्रत्येक बच्चा ViewModel दूसरे बच्चे ViewModels से और उनके माता-पिता से जितना संभव हो सके, अलग रह सके।
आइए एक उदाहरण देखें, जिसमें हम विभिन्न विचारों के बीच एक सरल पदानुक्रम को परिभाषित करेंगे। एक नया WPF एप्लिकेशन प्रोजेक्ट बनाएंMVVMHierarchiesDemo
Step 1 - अपनी परियोजना में तीन फ़ोल्डरों (मॉडल, ViewModel, और दृश्य) जोड़ें।
Step 2 - मॉडल फोल्डर में कस्टमर और ऑर्डर क्लासेस, व्यू फोल्डर में कस्टमर लिस्ट व्यू और आर्डर व्यू और कस्टममिस्ट व्यूमॉडल और आर्डर व्यूमॉडल को व्यूमॉडल फोल्डर में जोड़ें जैसा कि निम्नलिखित इमेज में दिखाया गया है।
Step 3- CustomerListView और OrderView दोनों में टेक्स्टब्लॉक जोड़ें। यहाँ CustomerListView.xaml फ़ाइल है।
<UserControl x:Class="MVVMHierarchiesDemo.Views.CustomerListView"
xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:d = "http://schemas.microsoft.com/expression/blend/2008"
xmlns:local = "clr-namespace:MVVMHierarchiesDemo.Views"
mc:Ignorable = "d"
d:DesignHeight = "300" d:DesignWidth = "300">
<Grid>
<TextBlock Text = "Customer List View"/>
</Grid>
</UserControl>
निम्नलिखित OrderView.xaml फ़ाइल है।
<UserControl x:Class = "MVVMHierarchiesDemo.Views.OrderView"
xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x ="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:mc ="http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:d ="http://schemas.microsoft.com/expression/blend/2008"
xmlns:local = "clr-namespace:MVVMHierarchiesDemo.Views" mc:Ignorable = "d"
d:DesignHeight = "300" d:DesignWidth = "300">
<Grid>
<TextBlock Text = "Order View"/>
</Grid>
</UserControl>
अब हमें इन दृश्यों को होस्ट करने के लिए कुछ चाहिए, और हमारे मेनविंडो में इसके लिए एक अच्छी जगह है क्योंकि यह एक सरल अनुप्रयोग है। हमें एक कंटेनर नियंत्रण की आवश्यकता है जिसे हम अपने विचार रख सकते हैं और उन्हें नेविगेशन फैशन में बदल सकते हैं। इस उद्देश्य के लिए, हमें अपनी MainWindow.xaml फ़ाइल में ContentControl को जोड़ना होगा और हम इसकी सामग्री संपत्ति का उपयोग करेंगे और ViewModel संदर्भ में इसे बांधेंगे।
अब संसाधन शब्दकोश में प्रत्येक दृश्य के लिए डेटा टेम्प्लेट परिभाषित करें। निम्नलिखित MainWindow.xaml फ़ाइल है। ध्यान दें कि प्रत्येक डेटा टेम्प्लेट किसी डेटा प्रकार (ViewModel प्रकार) को संबंधित दृश्य में कैसे मैप करता है।
<Window x:Class = "MVVMHierarchiesDemo.MainWindow"
xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d = "http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:local = "clr-namespace:MVVMHierarchiesDemo"
xmlns:views = "clr-namespace:MVVMHierarchiesDemo.Views"
xmlns:viewModels = "clr-namespace:MVVMHierarchiesDemo.ViewModel"
mc:Ignorable = "d"
Title = "MainWindow" Height = "350" Width = "525">
<Window.DataContext>
<local:MainWindowViewModel/>
</Window.DataContext>
<Window.Resources>
<DataTemplate DataType = "{x:Type viewModels:CustomerListViewModel}">
<views:CustomerListView/>
</DataTemplate>
<DataTemplate DataType = "{x:Type viewModels:OrderViewModel}">
<views:OrderView/>
</DataTemplate>
</Window.Resources>
<Grid>
<ContentControl Content = "{Binding CurrentView}"/>
</Grid>
</Window>
कभी भी वर्तमान दृश्य मॉडल एक CustomerListViewModel के उदाहरण के लिए सेट है, यह ViewModel के साथ एक CustomerListView को झुका देगा। यह एक आदेश ViewModel है, यह आर्डर व्यू इत्यादि प्रस्तुत करेगा।
अब हमें एक ViewModel की आवश्यकता है जिसमें एक CurrentViewModel संपत्ति और कुछ तर्क और कमांडिंग हैं जो संपत्ति के अंदर ViewModel के वर्तमान संदर्भ को स्विच करने में सक्षम हों।
चलिए इस MainWindow के लिए एक ViewModel बनाते हैं जिसे MainWindowViewModel कहा जाता है। हम सिर्फ XAML से हमारे ViewModel का एक उदाहरण बना सकते हैं और इसका उपयोग करके विंडो की DataContext संपत्ति सेट कर सकते हैं। इसके लिए, हमें अपने ViewModels के लिए INotifyPropertyChanged के कार्यान्वयन को छोटा करने के लिए एक आधार वर्ग बनाने की आवश्यकता है।
इस वर्ग के पीछे मुख्य विचार INotifyPropertyChanged कार्यान्वयन को अतिक्रमण करना और व्युत्पन्न वर्ग को सहायक तरीके प्रदान करना है ताकि वे आसानी से उचित सूचनाओं को ट्रिगर कर सकें। निम्नलिखित BindableBase वर्ग का कार्यान्वयन है।
using System;
using System.Collections.Generic;
using System.ComponentModel;
using System.Linq;
using System.Runtime.CompilerServices;
using System.Text;
using System.Threading.Tasks;
namespace MVVMHierarchiesDemo {
class BindableBase : INotifyPropertyChanged {
protected virtual void SetProperty<T>(ref T member, T val,
[CallerMemberName] string propertyName = null) {
if (object.Equals(member, val)) return;
member = val;
PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
}
protected virtual void OnPropertyChanged(string propertyName) {
PropertyChanged(this, new PropertyChangedEventArgs(propertyName));
}
public event PropertyChangedEventHandler PropertyChanged = delegate { };
}
}
अब यह वास्तव में हमारी करंट व्यू संपत्ति का उपयोग करके कुछ दृश्य स्विच करने का समय है। हमें इस संपत्ति की सेटिंग को चलाने के लिए कुछ तरीके की आवश्यकता है। और हम इसे बनाने जा रहे हैं ताकि अंतिम उपयोगकर्ता ग्राहक सूची में या ऑर्डर दृश्य पर जा सके। पहले अपने प्रोजेक्ट में एक नया वर्ग जोड़ें जो ICommand इंटरफ़ेस को लागू करेगा। निम्नलिखित ICommand इंटरफ़ेस का कार्यान्वयन है।
using System;
using System.Windows.Input;
namespace MVVMHierarchiesDemo {
public class MyICommand<T> : ICommand {
Action<T> _TargetExecuteMethod;
Func<T, bool> _TargetCanExecuteMethod;
public MyICommand(Action<T> executeMethod) {
_TargetExecuteMethod = executeMethod;
}
public MyICommand(Action<T> executeMethod, Func<T, bool> canExecuteMethod) {
_TargetExecuteMethod = executeMethod;
_TargetCanExecuteMethod = canExecuteMethod;
}
public void RaiseCanExecuteChanged() {
CanExecuteChanged(this, EventArgs.Empty);
}
#region ICommand Members
bool ICommand.CanExecute(object parameter) {
if (_TargetCanExecuteMethod != null) {
T tparm = (T)parameter;
return _TargetCanExecuteMethod(tparm);
}
if (_TargetExecuteMethod != null) {
return true;
}
return false;
}
// Beware - should use weak references if command instance lifetime is
longer than lifetime of UI objects that get hooked up to command
// Prism commands solve this in their implementation
public event EventHandler CanExecuteChanged = delegate { };
void ICommand.Execute(object parameter) {
if (_TargetExecuteMethod != null) {
_TargetExecuteMethod((T)parameter);
}
}
#endregion
}
}
अब हमें ViewModels के लिए इन पर कुछ शीर्ष स्तर के नेविगेशन सेट करने की आवश्यकता है और उस स्विचिंग के लिए तर्क MainWindowViewModel के अंदर होना चाहिए। इसके लिए हम नेविगेट पर एक विधि का उपयोग करने जा रहे हैं जो एक स्ट्रिंग गंतव्य लेता है और CurrentViewModel संपत्ति लौटाता है।
private void OnNav(string destination) {
switch (destination) {
case "orders":
CurrentViewModel = orderViewModelModel;
break;
case "customers":
default:
CurrentViewModel = custListViewModel;
break;
}
}
इन विभिन्न दृश्यों के नेविगेशन के लिए, हमें अपनी MainWindow.xaml फ़ाइल में दो बटन जोड़ने होंगे। निम्नलिखित पूर्ण XAML फ़ाइल कार्यान्वयन है।
<Window x:Class = "MVVMHierarchiesDemo.MainWindow"
xmlns = "http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x = "http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:d = "http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc = "http://schemas.openxmlformats.org/markup-compatibility/2006"
xmlns:local = "clr-namespace:MVVMHierarchiesDemo"
xmlns:views = "clr-namespace:MVVMHierarchiesDemo.Views"
xmlns:viewModels = "clr-namespace:MVVMHierarchiesDemo.ViewModel"
mc:Ignorable = "d"
Title = "MainWindow" Height = "350" Width = "525">
<Window.DataContext>
<local:MainWindowViewModel/>
</Window.DataContext>
<Window.Resources>
<DataTemplate DataType = "{x:Type viewModels:CustomerListViewModel}">
<views:CustomerListView/>
</DataTemplate>
<DataTemplate DataType = "{x:Type viewModels:OrderViewModel}">
<views:OrderView/>
</DataTemplate>
</Window.Resources>
<Grid>
<Grid.RowDefinitions>
<RowDefinition Height = "Auto" />
<RowDefinition Height = "*" />
</Grid.RowDefinitions>
<Grid x:Name = "NavBar">
<Grid.ColumnDefinitions>
<ColumnDefinition Width = "*" />
<ColumnDefinition Width = "*" />
<ColumnDefinition Width = "*" />
</Grid.ColumnDefinitions>
<Button Content = "Customers"
Command = "{Binding NavCommand}"
CommandParameter = "customers"
Grid.Column = "0" />
<Button Content = "Order"
Command = "{Binding NavCommand}"
CommandParameter = "orders"
Grid.Column = "2" />
</Grid>
<Grid x:Name = "MainContent" Grid.Row = "1">
<ContentControl Content = "{Binding CurrentViewModel}" />
</Grid>
</Grid>
</Window>
निम्नलिखित MainWindowViewModel कार्यान्वयन है।
using MVVMHierarchiesDemo.ViewModel;
using MVVMHierarchiesDemo.Views;
using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
namespace MVVMHierarchiesDemo {
class MainWindowViewModel : BindableBase {
public MainWindowViewModel() {
NavCommand = new MyICommand<string>(OnNav);
}
private CustomerListViewModel custListViewModel = new CustomerListViewModel();
private OrderViewModel orderViewModelModel = new OrderViewModel();
private BindableBase _CurrentViewModel;
public BindableBase CurrentViewModel {
get {return _CurrentViewModel;}
set {SetProperty(ref _CurrentViewModel, value);}
}
public MyICommand<string> NavCommand { get; private set; }
private void OnNav(string destination) {
switch (destination) {
case "orders":
CurrentViewModel = orderViewModelModel;
break;
case "customers":
default:
CurrentViewModel = custListViewModel;
break;
}
}
}
}
BindableBase वर्ग से अपने सभी ViewModels को वितरित करें। जब उपरोक्त कोड संकलित और निष्पादित किया जाता है, तो आपको निम्न आउटपुट दिखाई देगा।
जैसा कि आप देख सकते हैं, हमने अपने MainWindow पर केवल दो बटन और एक CurrentViewModel जोड़ा है। यदि आप किसी भी बटन पर क्लिक करते हैं तो यह उस विशेष दृश्य में नेविगेट हो जाएगा। ग्राहक बटन पर क्लिक करें और आप देखेंगे कि CustomerListView प्रदर्शित है।
हम आपको बेहतर समझ के लिए चरण-दर-चरण तरीके से उपरोक्त उदाहरण को निष्पादित करने की सलाह देते हैं।