最近项目[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. 3D场景优化

    一) 有效的性能评测 对于任何一个3D应用程序来说,追求场景画面真实感是一个无止尽的目标,其结果就是让我们的场景越来越复杂,模型更加精细,这必然给图形硬件带来极大的负荷以致于无法达到实时绘制帧率.因此 ...

  2. 齐次坐标概念&&透视投影变换推导

    http://daehgib.blog.163.com/blog/static/1861071422011579551134/ 透视投影是3D固定流水线的重要组成部分,是将相机空间中的点从视锥体(fr ...

  3. how to check unsolved conflicts file list in git merge?

    how to check unsolved conflicts file list in git merge?

  4. VS2015新功能

    今天有幸参加了微软的 Visual Studio Dev Day,趁还没有忘记今天的学习内容. 先把这些内容记录下来,如果有其他人也参加此次交流活动,请补充完善. VS2015新功能 1,Roslyn ...

  5. 51nod1556 计算

    ans[n]=ans[n-1]*3-m[n-2];YY一下可以懂的.减掉的就是往下走的情况不符合正整数的情况.m是默慈金数. #include<cstdio> #include<cs ...

  6. UVa 136 Ugly Numbers【优先队列】

    题意:给出丑数的定义,不能被除2,3,5以外的素数整除的的数称为丑数. 和杭电的那一题丑数一样--这里学的紫书上的用优先队列来做. 用已知的丑数去生成新的丑数,利用优先队列的能够每次取出当前最小的丑数 ...

  7. Hibernate-Criteria Queries

    1.实例 接口org.hibernate.Criteria针对特殊持久层类进行查询,Sesion是Criteria的工厂: Criteria crit = sess.createCriteria(Ca ...

  8. ORA-00257错误

    Archiver error,connect internal only,until freed   表示归档日志目录已满,用户不能再连接数据库,现有用户可继续查询数据库,但是不能执行DML语句 插删 ...

  9. 通过外部接口 根据ip获取城市名

    3种接口 淘宝/百度/不知名/   推荐淘宝接口 ip自个去获取,下附带php 获取ip的示例 function getIP() { static $realip; if (isset($_SERVE ...

  10. JS面向对象组件(三)--面向对象中的常用属性和方法

    图片的由来看下面: 1.hasOwnProperty:看是不是对象自身下面的属性 var arr = []; arr.num = ; Array.prototype.num2 = ; //所有的数组对 ...