【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 ...
随机推荐
- 3D场景优化
一) 有效的性能评测 对于任何一个3D应用程序来说,追求场景画面真实感是一个无止尽的目标,其结果就是让我们的场景越来越复杂,模型更加精细,这必然给图形硬件带来极大的负荷以致于无法达到实时绘制帧率.因此 ...
- 齐次坐标概念&&透视投影变换推导
http://daehgib.blog.163.com/blog/static/1861071422011579551134/ 透视投影是3D固定流水线的重要组成部分,是将相机空间中的点从视锥体(fr ...
- how to check unsolved conflicts file list in git merge?
how to check unsolved conflicts file list in git merge?
- VS2015新功能
今天有幸参加了微软的 Visual Studio Dev Day,趁还没有忘记今天的学习内容. 先把这些内容记录下来,如果有其他人也参加此次交流活动,请补充完善. VS2015新功能 1,Roslyn ...
- 51nod1556 计算
ans[n]=ans[n-1]*3-m[n-2];YY一下可以懂的.减掉的就是往下走的情况不符合正整数的情况.m是默慈金数. #include<cstdio> #include<cs ...
- UVa 136 Ugly Numbers【优先队列】
题意:给出丑数的定义,不能被除2,3,5以外的素数整除的的数称为丑数. 和杭电的那一题丑数一样--这里学的紫书上的用优先队列来做. 用已知的丑数去生成新的丑数,利用优先队列的能够每次取出当前最小的丑数 ...
- Hibernate-Criteria Queries
1.实例 接口org.hibernate.Criteria针对特殊持久层类进行查询,Sesion是Criteria的工厂: Criteria crit = sess.createCriteria(Ca ...
- ORA-00257错误
Archiver error,connect internal only,until freed 表示归档日志目录已满,用户不能再连接数据库,现有用户可继续查询数据库,但是不能执行DML语句 插删 ...
- 通过外部接口 根据ip获取城市名
3种接口 淘宝/百度/不知名/ 推荐淘宝接口 ip自个去获取,下附带php 获取ip的示例 function getIP() { static $realip; if (isset($_SERVE ...
- JS面向对象组件(三)--面向对象中的常用属性和方法
图片的由来看下面: 1.hasOwnProperty:看是不是对象自身下面的属性 var arr = []; arr.num = ; Array.prototype.num2 = ; //所有的数组对 ...