2

在我的应用程序中,我有许多帮助页面,它们都有类似的布局 - 标题和正文。

我目前的结构方式是,对于每个帮助页面,我都有一个 View、ViewModel、.xaml 和 .json 文件。

问题是我的应用程序已经膨胀到我有大约 20 个帮助页面(不要问)的程度,并且为每个帮助文件复制和粘贴这些显然效率低下且容易出错。我想要的是一个HelpViewModel、HelpView、Page_Help.xml,然后为每个帮助页面创建一个包含相关文本资源的不同.json 文件。

尽管查看了现有的 MvvmCross 主题,但我无法弄清楚如何实现这一目标。

所以我的 XAML 有一个如下所示的标题:

<TextView
  style="@style/HelpTextTitle"
  local:MvxBind="{'Text':'Path':'TextSource','Converter':'Language','ConverterParameter':'Heading1'}}"
/>

与 ViewModel 同名的 json 文件(并通过一些自动魔术链接)如下所示:

{
  "Heading1":"Sample Help Heading",
}

我有一个可以接受参数的 HelpViewModel - 说 json 文件的名称

public string HeaderFile { get; private set; }
public HelpViewModel (string headerFile)
{
  HeaderFile = headerFile;
}

json 文件当前与我认为通过此类的 ViewModel 相关联:

public class TextProviderBuilder : MvxTextProviderBuilder
...
    protected override IDictionary<string, string> ResourceFiles
    {
        get
        {
            var dictionary = this.GetType()
                .Assembly
                .GetTypes()
                .Where(t => t.Name.EndsWith("ViewModel"))
                .Where(t => !t.Name.StartsWith("Base"))
                .ToDictionary(t => t.Name, t => t.Name);

            dictionary[Constants.Shared] = Constants.Shared;
            return dictionary;
        }
    }         

并且主应用程序将其注册为文本提供程序(因此促进了我假设的 ViewModel-Json 文件自动),如下所示:

var builder = new TextProviderBuilder();
this.RegisterServiceInstance<IMvxTextProvider>(builder.TextProvider);

所以很明显,在这里我想以某种方式使用相同的 HelpPage 资源,但能够将该帮助页面上的控件与(最好)与参数(Header1)具有相同名称的不同 json 文件绑定,或者在必要时在一个单个 json 文件。

非常感谢!

马修


编辑:斯图尔特下面的解决方案完美无缺。当我通过 ViewModel 的构造函数调用 RequestNavigate 时,我只需要传递 json 文件的名称:

return new MvxRelayCommand(() => this.RequestNavigate<GeneralHelpViewModel>(new { helpKey = "DescribeAudioCrimeStatementHelpViewModel" }));

我唯一需要做的就是创建我自己的 MvxLanguageBinder,因为在我的 MvvmCross(不是最近)的确切构建中,MvxLanguageBinder 是私有的且不可访问。

我只是从 Stuart 的链接中复制并粘贴了该版本并使用了它:

https://github.com/slodge/MvvmCross/blob/vnext/Cirrious/Cirrious.MvvmCross.Localisation/MvxLanguageBinder.cs

4

1 回答 1

1

有几种方法可以实现这种效果。

如果您想继续使用单独的 json 文件,我可能会采用类似...的方法

更改您的绑定以使用新的 TextSource -CustomTextSource

<TextView
       style="@style/HelpTextTitle"
       local:MvxBind="{'Text':'Path':'CustomTextSource','Converter':'Language','ConverterParameter':'Heading1'}}"
 />

创建您的HelpViewModel,以便它CustomTextSource使用导航参数加载它:

 public HelpViewModel (string helpKey)
 {
       CustomTextSource = new MvxLanguageBinder(Constants.GeneralNamespace, helpKey);
 }

 public IMvxLanguageBinder CustomTextSource { get; private set; }

更改执行文本资源加载的代码以包含您的帮助文件。

protected override IDictionary<string, string> ResourceFiles
{
    get
    {
        var dictionary = this.GetType()
            .Assembly
            .GetTypes()
            .Where(t => t.Name.EndsWith("ViewModel"))
            .Where(t => !t.Name.StartsWith("Base"))
            .ToDictionary(t => t.Name, t => t.Name);

        dictionary[Constants.Shared] = Constants.Shared;

        foreach (var additional in HelpFileList)
        {
             dictionary(additional) = additional;
        } 

        return dictionary;
    }
}  

(我在这里假设这HelpFileList是一个静态列表,但您也可以使用反射来做这件事。)


或者,如果您想将帮助 json 重新组织到一个文件中,您还可以考虑编写一个IMvxLanguageBinder使用您的帮助键的自定义。假设您使用的是 vNext,那么 for 的代码MvxLanguageBinder应该很容易适应——来自 axml 的调用进入第 58 行


或者,如果这真的只是帮助文本,那么您可以考虑放弃这种 xml 和 json 方法 - 您可以改用在 Web 浏览器小部件中显示的嵌入式 HTML - 这可能是最灵活的长期解决方案,它允许最简单的内容将来更新。

于 2013-05-10T06:56:15.360 回答