【Visual Studio2010】创建XAML分析插件
最近项目[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分析插件的更多相关文章
- Xamarin XAML语言教程使用Visual Studio创建XAML
Xamarin XAML语言教程使用Visual Studio创建XAML Xamarin.Forms允许开发人员通过XAML语法对程序的所有用户界面元素进行详细的定制,如文本.按钮.图像和列表框等. ...
- Xamarin XAML语言教程使用Xamarin Studio创建XAML(二)
Xamarin XAML语言教程使用Xamarin Studio创建XAML(二) 使用Xamarin Studio创建XAML Xamarin Studio和Visual Studio创建XAML文 ...
- Visual Studio 2017创建XAML文件
Visual Studio 2017创建XAML文件 在Visual Stuido 2015中,在已经创建好的项目中添加XAML文件,只要右击项目,单击“添加”|“新建项”命令,然后从“添加新项” ...
- 11个Visual Studio代码性能分析工具
软件开发中的性能优化对程序员来说是一个非常重要的问题.一个小问题可能成为一个大的系统的瓶颈.但是对于程序员来说,通过自身去优化代码是十分困难的.幸运的是,有一些非常棒的工具可以帮助程序员进行代码分析和 ...
- 【转】怎样创建一个Xcode插件(Part 1)
原文:How To Create an Xcode Plugin: Part 1/3 原作者:Derek Selander 译者:@yohunl 译者注:原文使用的是xcode6.3.2,我翻译的 ...
- Visual Studio2015 Community一些必备插件
Visual Studio2015 Community一些必备插件 是不是感觉虽然VS2015的代码编辑能力已经很强大了,但是总感觉差了那么一些呢?不用担心,它有很多非常强大的插件,能够让你打代码事半 ...
- MyBatis 源码分析 - 插件机制
1.简介 一般情况下,开源框架都会提供插件或其他形式的拓展点,供开发者自行拓展.这样的好处是显而易见的,一是增加了框架的灵活性.二是开发者可以结合实际需求,对框架进行拓展,使其能够更好的工作.以 My ...
- .NET 11 个 Visual Studio 代码性能分析工具
原文地址 软件开发中的性能优化对程序员来说是一个非常重要的问题.一个小问题可能成为一个大的系统的瓶颈.但是对于程序员来说,通过自身去优化代码是十分困难的.幸运的是,有一些非常棒的工具可以帮助程序员进行 ...
- mapbox-gl空间分析插件turf.js使用介绍
mapbox-gl能够方便地显示地图,做一些交互,但是缺少空间分析功能,比如绘制缓冲区.判断点和面相交等等. turf.js是一个丰富的用于浏览器和node.js空间分析库,官网 http://tur ...
随机推荐
- Linux同步机制 - 多线程开发总结
1 对于CPU开销大的场景,能利用多核,就尽量利用多核(常常自以为某需求的运算量不大,且CPU足够快,就偷懒写个单线程,结果效率很低) 2 使用多线程的时候,默认是加锁的.在加锁保证业务正常的条件下, ...
- Datagridview的某些属性again
private void button1_Click(object sender, EventArgs e) //这里是不用泛型集合的情况,用的是Datatable { SqlConnection c ...
- Elasticsearch学习笔记
Why Elasticsearch? 由于需要提升项目的搜索质量,最近研究了一下Elasticsearch,一款非常优秀的分布式搜索程序.最开始的一些笔记放到github,这里只是归纳总结一下. 首先 ...
- 通知(NSNotification)
通知 一个完整的通知一般包含3个属性: - (NSString *)name; // 通知的名称 - (id)object; // 通知发布者(是谁要发布通知) - (NSDictionary *)u ...
- Jave 鼠标点击画太极 PaintTaiji (整理)
package demo; /** * Jave 鼠标点击画太极 PaintTaiji (整理) * 声明: * 又是一份没有注释的代码,而且时间已经久远了,不过代码很短,解读起来应该 * 不会很麻烦 ...
- MySQL的性能调优工具:比mysqlreport更方便的tuning-primer.sh
年初的时候收藏过一篇关于mysqlreport的报表解读,和内置的show status,和show variables相比mysqlreport输出一个可读性更好的报表:但Sundry MySQL提 ...
- 流程引擎的API和服务基础
RepositoryService : 管理和控制 发布包 和 流程定义(包含了一个流程每个环节的结构和行为) 的操作 除此之外,服务可以 查询引擎中的发布包和流程定义. 暂停或激活发布包,对应全部 ...
- HDU1010 Tempter of the Bone
解题思路:相当经典的一题,回溯,具体细节处理见代码. #include<cstdio> #include<cstring> #include<algorithm> ...
- Android Studio 学习 - HelloWorld
今天是学习Android Studio的第2天,加油! 1. 首先要记录下使用Android Studio的一个代码自动完成的功能.平常基本上用Delphi,乍一换工具,各种不习惯,或者说不熟悉.按照 ...
- 修改Chrome默认搜索引擎为Google.com
在使用Chrome的时候,Google为增强本地化搜索,或将默认的Google搜索引擎转换为本地语言,如在中国会自动转到google.com.hk,日本会会自动转到google.co.jp,如果你是一 ...