C# 如何改进在PRISM 4中导航到新视图时对对象的传递

C# 如何改进在PRISM 4中导航到新视图时对对象的传递,c#,wpf,module,navigation,prism,C#,Wpf,Module,Navigation,Prism,我正在使用棱镜与国际奥委会。问题是通过导航传递对象(如集合)。我在看这篇帖子: 这就是解决办法 我提取对象的哈希代码并将其保存在字典中,哈希代码作为密钥,对象作为对的值 然后,我将哈希代码附加到UriQuery 之后,我只需要获取来自目标视图上Uri的哈希代码,并使用它从字典请求原始对象 一些示例代码: 参数存储库类: public class Parameters { private static Dictionary<int, object> paramList =

我正在使用棱镜与国际奥委会。问题是通过导航传递对象(如集合)。我在看这篇帖子:

这就是解决办法

我提取对象的哈希代码并将其保存在
字典中,哈希代码作为密钥,对象作为对的值

然后,我将哈希代码附加到
UriQuery

之后,我只需要获取来自目标视图上Uri的哈希代码,并使用它从
字典
请求原始对象

一些示例代码:

参数存储库类:

public class Parameters
{
    private static Dictionary<int, object> paramList =
        new Dictionary<int, object>();

    public static void save(int hash, object value)
    {
        if (!paramList.ContainsKey(hash))
            paramList.Add(hash, value);
    }

    public static object request(int hash)
    {
        return ((KeyValuePair<int, object>)paramList.
                    Where(x => x.Key == hash).FirstOrDefault()).Value;
    }
}
目标视图代码:

public partial class MyView : UserControl, INavigationAware
{
// some hidden code

    public void OnNavigatedTo(NavigationContext navigationContext)
    {
        int hash = int.Parse(navigationContext.Parameters["hash"]);
        Customer cust = (Customer)Parameters.request(hash);
    }
}
就这样


我不确定这个解决方案是否是传递对象的最佳方案。我想这可能是一种服务。是一个好方法还是有更好的方法呢?

我发布了一个更简单的方法。在此提及以供参考-

我将使用OnNavigatedTo和OnNavigatedFrom方法通过NavigationContext传递对象

首先从INavigationAware接口导出viewmodel-

 public class MyViewModel : INavigationAware
 { ...
然后,您可以实现OnNavigatedFrom,并将要传递的对象设置为导航上下文,如下所示-

void INavigationAware.OnNavigatedFrom(NavigationContext navigationContext)
{
     SharedData data = new SharedData();
     ...
     navigationContext.NavigationService.Region.Context = data;
}
当您想要接收数据时,在第二个视图模型中添加以下代码-

void INavigationAware.OnNavigatedTo(NavigationContext navigationContext)
{
    if (navigationContext.NavigationService.Region.Context != null)
    {
        if (navigationContext.NavigationService.Region.Context is SharedData)
        {
             SharedData data = (SharedData)navigationContext.NavigationService.Region.Context;
              ...
        }
    }
}

我刚开始使用Prism,这是我遇到的第一个限制之一。我用另一种方式解决了它。我首先创建了一个从Uri继承并实现IDictionary的类(加上一些更易于访问的泛型方法)


这是一个超级解决方案,也许prism 4.2也以同样的方式解决了这个问题,现在终于有了对通过对象的支持。但是我认为您只修复了90%的问题((:是一种避免每次要检索params.var值=((NavigationUri)NavigationContext.Uri)时将NavigationContext.Uri强制转换为NavigationUri的方法).Values;我想我不完全理解你的问题。在我们的应用程序中,视图创建了viewmodel。因此,如果我有一个MsgBoxView,并且在它的构造函数中,它创建了MsgBoxVM,MsgBoxVM构造函数接受另一个类,让我们调用该类MsgBoxOptions。我将创建一个var uri=new NavigationUri(“MsgBoxView”);如果我已经创建了MsgBoxOptions,并将其包含在名为MsgBoxOptions的变量中,那么我将执行uri.Add(MsgBoxOptions)。在viewmodel的构造函数中,MsgBoxOptions参数将使用相同的obj填充。我们所做的增强之一是在BaseRegionNavigationContentLoader中保存Uri时也保存UriQuery,并自动将其添加到NavigationUri。因此,如果我想要UriQuery,我只需一个简单的步骤就可以在构造函数中访问它将其作为参数添加到视图模型中。与在已定义的导航事件中捕获它相比,我更喜欢这种方式。这就是您要寻找的吗?如果是,我可以更新代码,使其更符合我们现在的要求。
void INavigationAware.OnNavigatedTo(NavigationContext navigationContext)
{
    if (navigationContext.NavigationService.Region.Context != null)
    {
        if (navigationContext.NavigationService.Region.Context is SharedData)
        {
             SharedData data = (SharedData)navigationContext.NavigationService.Region.Context;
              ...
        }
    }
}
public class NavigationUri : Uri, IDictionary<Type, object>
{
    private IDictionary<Type, object> _internalDictionary = new Dictionary<Type, object>();

    public NavigationUri(string uriString) : base(uriString, UriKind.Relative)
    {
    }
    public NavigationUri(string uriString, UriKind uriKind) :  base(uriString, uriKind)
    {
    }

    public void Add<T>(T value)
    {
        Add(typeof(T), value);
    }

    public void Add(Type key, object value)
    {
        _internalDictionary.Add(key, value);
    }

    public bool ContainsKey<T>()
    {
        return ContainsKey(typeof (T));
    }

    public bool ContainsKey(Type key)
    {
        return _internalDictionary.ContainsKey(key);
    }

    public ICollection<Type> Keys
    {
        get { return _internalDictionary.Keys; }
    }

    public bool Remove<T>()
    {
        return Remove(typeof (T));
    }

    public bool Remove(Type key)
    {
        return _internalDictionary.Remove(key);
    }

    public bool TryGetValue<T>(out object value)
    {
        return TryGetValue(typeof (T), out value);
    }

    public bool TryGetValue(Type key, out object value)
    {
        return _internalDictionary.TryGetValue(key, out value);
    }

    public ICollection<object> Values
    {
        get { return _internalDictionary.Values; }
    }

    public object this[Type key]
    {
        get { return _internalDictionary[key]; }
        set { _internalDictionary[key] = value; }
    }

    public void Add(KeyValuePair<Type, object> item)
    {
        _internalDictionary.Add(item);
    }

    public void Clear()
    {
        _internalDictionary.Clear();
    }

    public bool Contains(KeyValuePair<Type, object> item)
    {
        return _internalDictionary.Contains(item);
    }

    public void CopyTo(KeyValuePair<Type, object>[] array, int arrayIndex)
    {
        _internalDictionary.CopyTo(array, arrayIndex);
    }

    public int Count
    {
        get { return _internalDictionary.Count; }
    }

    public bool IsReadOnly
    {
        get { return _internalDictionary.IsReadOnly; }
    }

    public bool Remove(KeyValuePair<Type, object> item)
    {
        return _internalDictionary.Remove(item);
    }

    public IEnumerator<KeyValuePair<Type, object>> GetEnumerator()
    {
        return _internalDictionary.GetEnumerator();
    }

    System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
    {
        return _internalDictionary.GetEnumerator();
    }
}
public class BaseRegionNavigationContentLoader : RegionNavigationContentLoader
{
    private Uri _uri;
    private IServiceLocator _serviceLocator;
    private IUnityContainer _unityContainer;

    public BaseRegionNavigationContentLoader(IServiceLocator serviceLocator, IUnityContainer unityContainer) : base(serviceLocator)
    {
        _serviceLocator = serviceLocator;
        _unityContainer = unityContainer;
    }

    protected override string GetContractFromNavigationContext(NavigationContext navigationContext)
    {
        _uri = navigationContext.Uri;
        return base.GetContractFromNavigationContext(navigationContext);
    }

    protected override object CreateNewRegionItem(string candidateTargetContract)
    {
        object instance;
        try
        {
            var uri = _uri as NavigationUri;
            if (uri == null)
            {
                instance = _serviceLocator.GetInstance<object>(candidateTargetContract);
            }
            else
            {
                // Create injection overrides for all the types in the uri
                var depoverride = new DependencyOverrides();
                foreach (var supplant in uri)
                {
                    depoverride.Add(supplant.Key, supplant.Value);
                }

                instance = _unityContainer.Resolve<object>(candidateTargetContract, depoverride);
            }

        }
        catch (ActivationException exception)
        {
            throw new InvalidOperationException(string.Format(System.Globalization.CultureInfo.CurrentCulture, "CannotCreateNavigationTarget", new object[] { candidateTargetContract }), exception);
        }
        return instance;
    }
}
var uri = new NavigationUri("MessageBoxView");
uri.Add(messageBoxEventArgs);
regionManager.RequestNavigate(RegionNames.MainRegion, uri);