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

发布于 2025-01-06 21:11:34 字数 1673 浏览 1 评论 0原文

我将 Prism 与 IoC 一起使用。问题是通过导航传递对象(如集合)。我正在看这篇文章: 如何在 PRISM 4 中导航到新视图时传递对象

这是

我提取对象的哈希代码并将其与哈希一起保存在字典中 的解决方案代码作为密钥和对象作为该对的值。

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

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

一些示例代码:

参数存储库类:

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;
    }
}

调用者代码:

UriQuery q = null;
Customer customer = new Customer();
q = new UriQuery();
Parameters.save(customer.GetHashCode(), customer);
q.Add("hash", customer.GetHashCode().ToString());

Uri viewUri = new Uri("MyView" + q.ToString(), UriKind.Relative);
regionManager.RequestNavigate(region, viewUri);

目标视图代码:

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);
    }
}

就是这样。

我不确定这个解决方案是否是传递对象的最佳解决方案。我想这可能是一项服务。有一个好的方法可以做到这一点或者有更好的方法吗?

I'm using Prism with IoC. The problem is to pass an object (like collections) through navigation. I was watching this post: How to Pass an object when navigating to a new view in PRISM 4

And this is the solution

I extract the hash code of the object and save it in a Dictionary, with the hash code as the key and the object as the value of the pair.

Then, I attach the hash code to the UriQuery.

After, I only have to get the hash code that comes from the Uri on the target view and use it to request the original object from the Dictionary.

Some example code:

Parameter repository class:

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;
    }
}

The caller code:

UriQuery q = null;
Customer customer = new Customer();
q = new UriQuery();
Parameters.save(customer.GetHashCode(), customer);
q.Add("hash", customer.GetHashCode().ToString());

Uri viewUri = new Uri("MyView" + q.ToString(), UriKind.Relative);
regionManager.RequestNavigate(region, viewUri);

The target view code:

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);
    }
}

That's it.

I'm not sure if this solution is the best to pass objects. I guess this maybe would be a service. Is a good way to do this or is there a better way to do it?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(2

ぽ尐不点ル 2025-01-13 21:11:34

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

我将使用 OnNavieratedTo 和 OnNavieratedFrom 方法来使用 NavigationContext 传递对象。

首先从 INavigationAware 接口派生视图模型 -

 public class MyViewModel : INavigationAware
 { ...

然后您可以实现 OnNavieratedFrom 并将要传递的对象设置为导航上下文,如下所示 -

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;
              ...
        }
    }
}

I posted an easier way. Mentioning it here for reference -

I would use the OnNavigatedTo and OnNavigatedFrom methods to pass on the objects using the NavigationContext.

First derive the viewmodel from INavigationAware interface -

 public class MyViewModel : INavigationAware
 { ...

You can then implement OnNavigatedFrom and set the object you want to pass as navigation context as follows -

void INavigationAware.OnNavigatedFrom(NavigationContext navigationContext)
{
     SharedData data = new SharedData();
     ...
     navigationContext.NavigationService.Region.Context = data;
}

and when you want to receive the data, add the following piece of code in the second view model -

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;
              ...
        }
    }
}
素年丶 2025-01-13 21:11:34

我刚刚开始使用 Prism,这是我遇到的第一个限制。我用不同的方式解决了它。我首先创建了一个继承自 Uri 并实现 IDictionary 的类(加上一些通用方法以便于访问),

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();
    }
}

然后我创建了一个继承自 RegionNavigationContentLoader 的类。在 GetContractFromNavigationContext 上,我存储传入的 Uri,以便可以在 CreateNewRegionItem 方法中访问它。在该方法中,我检查 Uri 是否是 NavigationUri,如果是,我会循环添加所有依赖项注入覆盖。我正在使用 Unity,但我认为代码可以轻松转换为另一个 IOC 容器。

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;
    }
}

现在,在 prism Bootstrapper 中,您需要在 ConfigureServiceLocator 方法中将 BaseRegionNavigationContentLoader 注册为 IRegionNavigationContentLoader。 确保将其标记为 TransientLifetimeManager,以便它每次都会更新。 IRegionNavigationContentLoader 的默认注册是由容器控制的,这使得它的行为就像一个单例,但我们每次都需要一个新的注册,因为我们需要将 uri 从一个方法传递到属性中的下一个方法。

现在我可以编写如下代码,并且仍然使用构造函数注入。

var uri = new NavigationUri("MessageBoxView");
uri.Add(messageBoxEventArgs);
regionManager.RequestNavigate(RegionNames.MainRegion, uri);

I just started using Prism and this is one of the first limitations I ran into. I solved it a different way. I first created a class that inherits from Uri and implements IDictionary (plus some generic methods for easier access)

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();
    }
}

Then I created a class that inherits from RegionNavigationContentLoader. On GetContractFromNavigationContext I store the passed in Uri so I can access it in the CreateNewRegionItem method. In that method I check to see if the Uri is the NavigationUri and if so I loop though adding all the dependency injection overrides. I'm using Unity but I assume the code could easy be converted to another IOC container.

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;
    }
}

Now in the prism Bootstrapper you need to register the BaseRegionNavigationContentLoader as IRegionNavigationContentLoader in the ConfigureServiceLocator method. Make sure you mark it as TransientLifetimeManager so it gets newed up each time. The default registration for IRegionNavigationContentLoader is container controlled which makes it act like a singleton but we need a new one each time since we need to pass the uri from one method to the next in a property.

Now I can wright code like the following and still use constructor injection.

var uri = new NavigationUri("MessageBoxView");
uri.Add(messageBoxEventArgs);
regionManager.RequestNavigate(RegionNames.MainRegion, uri);
~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文