最近项目[Silverlight]中的需要实现国际化,需要对所有控件进行一个处理。由于使用了Telerik的控件,只需要去掉原有的Label或者Header属性,然后添加一个资源Key即可。但是在项目已经完全成熟的情况下,对大量的查询条件,数据列进行处理也是一个非常耗时的方案,因此对XAML文件进行处理能够节省大量的工作量,避免错误信息。

关键的步骤可以通过VS的向导自动完成。接下来只需在Conect.cs文件中对部分数据进行处理。

    /// <summary>实现 IDTCommandTarget 接口的 Exec 方法。此方法在调用该命令时调用。</summary>
/// <param term='commandName'>要执行的命令的名称。</param>
/// <param term='executeOption'>描述该命令应如何运行。</param>
/// <param term='varIn'>从调用方传递到命令处理程序的参数。</param>
/// <param term='varOut'>从命令处理程序传递到调用方的参数。</param>
/// <param term='handled'>通知调用方此命令是否已被处理。</param>
/// <seealso class='Exec' />
public void Exec(string commandName, vsCommandExecOption executeOption, ref object varIn, ref object varOut, ref bool handled)
{
handled = false;
if (executeOption == vsCommandExecOption.vsCommandExecOptionDoDefault)
{
if (_applicationObject.ActiveDocument == null)
return;
// 获取当前画面名称
string pageName = _applicationObject.ActiveDocument.Name;
pageName = pageName.Substring(, pageName.IndexOf(".")); TextDocument doc = _applicationObject.ActiveDocument.Object() as TextDocument;
if (commandName == "Aladdin.Connect.Aladdin")
{
try
{
// 获取当前的内容,并进行替换
var startPoint = doc.CreateEditPoint(doc.StartPoint);
var text = startPoint.GetText(doc.EndPoint);
string newContent = I18nHelper.DoReplace(pageName, text);
// 先删除
startPoint.Delete(doc.EndPoint);
// 重新写入
startPoint.Insert(newContent); handled = true;
return;
}
catch (Exception ex)
{
MessageBox.Show("格式化错误。" + ex.Message);
}
}
}
}

主要代码就是通过获取当前的TextDocument进行处理。

在编辑器文档中中获取一段内容的方法就是定义一个起始位置,然后从起始位置开始通过GetText获取到指定结束位置。

然后通过Delete和Insert进行处理。

NOTE:之前尝试了使用SelectionText进行处理,发现替换效率比较慢,应该是SelectionText这个对象是内容更新时一直在变化的原因。

对文本进行分析有多种办法,比如正则表达式、字符串查找替换等,这里由于处理的是XAML,因此使用了XML的处理方式。

在分析之前有几点需要注意的地方:

  • XAML是带有命名空间的,需要预先分析命名空间,并在带有命名空间的SelectNodes等地方使用;
  • 在添加(或者修改,此处有待确认)节点或者属性时,需要注意
  • XML内容在输出为字符串,并将内容传给编辑器时,需要使用XmlWriterSettings进行处理缩进等问题。
        public static string DoReplace(string pageName, string content)
{
if (String.IsNullOrEmpty(content)) return content; XmlDocument doc = new XmlDocument();
doc.LoadXml(content); XmlNamespaceManager nsManager = new XmlNamespaceManager(doc.NameTable);
var docRoot = doc.FirstChild; // 根据根节点信息,获取命名空间列表
foreach (var ns in GetNamespaces(docRoot))
nsManager.AddNamespace(ns.Key, ns.Value); // 遍历所有子节点,进行处理
foreach (XmlNode root in docRoot.ChildNodes)
{
Process(doc, root, pageName, nsManager);
} // 数据输出,格式设定
StringBuilder strBuilder = new StringBuilder();
XmlWriterSettings settings = new XmlWriterSettings();
settings.Indent = true;
settings.NewLineHandling = NewLineHandling.Replace;
settings.OmitXmlDeclaration = true;
settings.ConformanceLevel = ConformanceLevel.Fragment; XmlWriter writer = XmlWriter.Create(strBuilder, settings);
doc.WriteTo(writer);
writer.Flush();
writer.Close(); // 替换根节点内容
string newRootAttributes = ReplaceHelper.ProcessRoot(docRoot);
string newContent = strBuilder.ToString(); string newRawAttributes = newContent.Substring(, newContent.IndexOf(">"));
return newContent.Replace(newRawAttributes, newRootAttributes);
} private static void Process(XmlDocument doc, XmlNode root, string pageName, XmlNamespaceManager nsManager)
{
// TODO:节点处理和递归
}

简单的添加属性的方法类似于,即必须制定其NamespaceURI,否则会自动在结点后添加命名空间属性:

 foreach (XmlNode field in root.SelectNodes("EF:EFDataFieldGroup.DataFields", nsManager))
{
if (field.Attributes == null) continue; if (field.Attributes["EF:I18nManager.ResourceKey"] == null)
{
string fieldValue = field.OuterXml;
// 数据进行处理
var startIdx = fieldValue.IndexOf("inqu_status-0-");
var endIdx = fieldValue.IndexOf("\"", startIdx); if (startIdx + >= endIdx) continue; string itemId = fieldValue.Substring(startIdx + , endIdx - startIdx - );
string bindingName = string.Format("{0}U_DATAFIELD_{1}", pageName, itemId); //XmlAttribute attr = doc.CreateAttribute("EF:I18nManager.ResourceKey", bindingName);
//field.Attributes.Append(attr); //XmlElement element = field as XmlElement;
var attr = doc.CreateAttribute("EF:I18nManager.ResourceKey", field.NamespaceURI);
attr.Value = bindingName;
field.Attributes.Append(attr); // 同时删除原有的标签
var labelAttr = field.Attributes["eLabel"];
if (null != labelAttr)
field.Attributes.Remove(labelAttr);
//element.SetAttribute("EF:I18nManager.ResourceKey",field.NamespaceURI, bindingName);
}
}

分析XAML文档的命名空间列表的简单实现如下:

        private static IDictionary<string, string> GetNamespaces(XmlNode root)
{
IDictionary<string, string> dicNS = new Dictionary<string, string>(); // 获取第一个节点中的所有 foreach (XmlNode node in root.Attributes)
{
var item = node.OuterXml; if (item.IndexOf("xmlns") > -)
{
int nameStart = item.IndexOf(":");
int nameEnd = item.IndexOf("=");
if (nameStart >= nameEnd) continue; string key = item.Substring(nameStart + , nameEnd - nameStart - ).Trim(); if (String.IsNullOrEmpty(key)) continue; int contentStart = item.IndexOf("\"", nameEnd);
int contentEnd = item.IndexOf("\"", contentStart + );
string val = item.Substring(contentStart + , contentEnd - - contentStart); dicNS.Add(key, val);
}
} return dicNS;
}
}

【Visual Studio2010】创建XAML分析插件的更多相关文章

  1. Xamarin XAML语言教程使用Visual Studio创建XAML

    Xamarin XAML语言教程使用Visual Studio创建XAML Xamarin.Forms允许开发人员通过XAML语法对程序的所有用户界面元素进行详细的定制,如文本.按钮.图像和列表框等. ...

  2. Xamarin XAML语言教程使用Xamarin Studio创建XAML(二)

    Xamarin XAML语言教程使用Xamarin Studio创建XAML(二) 使用Xamarin Studio创建XAML Xamarin Studio和Visual Studio创建XAML文 ...

  3. Visual Studio 2017创建XAML文件

    Visual Studio 2017创建XAML文件   在Visual Stuido 2015中,在已经创建好的项目中添加XAML文件,只要右击项目,单击“添加”|“新建项”命令,然后从“添加新项” ...

  4. 11个Visual Studio代码性能分析工具

    软件开发中的性能优化对程序员来说是一个非常重要的问题.一个小问题可能成为一个大的系统的瓶颈.但是对于程序员来说,通过自身去优化代码是十分困难的.幸运的是,有一些非常棒的工具可以帮助程序员进行代码分析和 ...

  5. 【转】怎样创建一个Xcode插件(Part 1)

      原文:How To Create an Xcode Plugin: Part 1/3 原作者:Derek Selander 译者:@yohunl 译者注:原文使用的是xcode6.3.2,我翻译的 ...

  6. Visual Studio2015 Community一些必备插件

    Visual Studio2015 Community一些必备插件 是不是感觉虽然VS2015的代码编辑能力已经很强大了,但是总感觉差了那么一些呢?不用担心,它有很多非常强大的插件,能够让你打代码事半 ...

  7. MyBatis 源码分析 - 插件机制

    1.简介 一般情况下,开源框架都会提供插件或其他形式的拓展点,供开发者自行拓展.这样的好处是显而易见的,一是增加了框架的灵活性.二是开发者可以结合实际需求,对框架进行拓展,使其能够更好的工作.以 My ...

  8. .NET 11 个 Visual Studio 代码性能分析工具

    原文地址 软件开发中的性能优化对程序员来说是一个非常重要的问题.一个小问题可能成为一个大的系统的瓶颈.但是对于程序员来说,通过自身去优化代码是十分困难的.幸运的是,有一些非常棒的工具可以帮助程序员进行 ...

  9. mapbox-gl空间分析插件turf.js使用介绍

    mapbox-gl能够方便地显示地图,做一些交互,但是缺少空间分析功能,比如绘制缓冲区.判断点和面相交等等. turf.js是一个丰富的用于浏览器和node.js空间分析库,官网 http://tur ...

随机推荐

  1. Shadow mapping

    http://www.cnblogs.com/cxrs/archive/2009/10/17/1585038.html 1.什么是Shadow Maping?      Shadow Mapping是 ...

  2. FireMonkey 平台初探

    最为第一个本地化跨平台的框架:FireMonkey需要处理以及融合不同平台的技术非常之多,所以目前的测试仅仅在于表面现象,至于效率问题还不得而知. 从一个程序员的角度来看这个框架,我觉得有以下这些方面 ...

  3. HDU 2610 (自己完全找不到思路) Sequence one

    搜索虐我千百遍,我待搜索...好吧,我还木有初恋 题意: 我开始理解题意就理解偏了,Orz 题中有n个元素构成的序列,求出前p个非递减子序列.子序列是先按长度排序的,然后按原序列先后位置排序的. 这里 ...

  4. UVa 821 Page Hopping【Floyd】

    题意:给出一个n个点的有向图,任意两个点之间都相互到达,求任意两点间最短距离的平均值 因为n很小,所以可以用floyd 建立出图,然后用floyd,统计d[][]不为0且不为INF的边的和及条数,就可 ...

  5. [转] 判断html页是否滚动停止

    原文链接:http://www.phpernote.com/javascript-function/754.html 最近有个项目中遇到这样一个问题: 有一个用于展示数据的带滚动条的DIV块,业务需求 ...

  6. tableView的设置

    TableView的设置 1.设置table头部和底部的view // 底部(宽度固定是320) tableView.tableFooterView = footer; // 头部(宽度固定是320) ...

  7. *ecshop 限制文章帮助文章显示条数

    1.打开 /themes/default/library/help.lbi 文件 <!-- {foreach from=$help_cat.article item=item} --> & ...

  8. 15个必须知道的chrome开发者技巧(转)

    15个必须知道的chrome开发者技巧 在Web开发者中,Google Chrome是使用最广泛的浏览器.六周一次的发布周期和一套强大的不断扩大开发功能,使其成为了web开发者必备的工具.你可能已经熟 ...

  9. #define XXX do{...}while(0)

    <ol> <li>函数式宏定义的参数没有类型,预处理器只负责做形式上的替换,而不做参数类型检查,所以传参时要格外小心.</li> <li>调用真正函数的 ...

  10. A. Puzzles CodeForces Round #196 (Div.2)

    题目的大意是,给你 m 个数字,让你从中选 n 个,使得选出的数字的极差最小. 好吧,超级大水题.因为要极差最小,所以当然想到要排个序咯,然后去连续的 n 个数字,因为数据不大,所以排完序之后直接暴力 ...