- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我正在尝试为 VS2017(在 C# 中)创建一个扩展 VSPackage,它将二进制数据转换为 XML,在默认的 VS XML 编辑器和 XML 语言服务中打开它,然后在保存时将其转换回二进制。
但是,我很难列出为此需要哪些步骤。在编辑器工厂中创建新编辑器时,我暂时想到了以下几点:
现在我的尝试是这样的:
private MyPackage _package; // Filled via constructor
private IServiceProvider _serviceProvider; // Filled via SetSite
public int CreateEditorInstance(uint grfCreateDoc, string pszMkDocument, string pszPhysicalView,
IVsHierarchy pvHier, uint itemid, IntPtr punkDocDataExisting, out IntPtr ppunkDocView,
out IntPtr ppunkDocData, out string pbstrEditorCaption, out Guid pguidCmdUI, out int pgrfCDW)
{
// Initialize and validate parameters.
ppunkDocView = IntPtr.Zero;
ppunkDocData = IntPtr.Zero;
pbstrEditorCaption = String.Empty;
pguidCmdUI = Guid.Empty;
pgrfCDW = 0;
VSConstants.CEF createDocFlags = (VSConstants.CEF)grfCreateDoc;
if (!createDocFlags.HasFlag(VSConstants.CEF.OpenFile) && !createDocFlags.HasFlag(VSConstants.CEF.Silent))
return VSConstants.E_INVALIDARG;
if (punkDocDataExisting != IntPtr.Zero)
return VSConstants.VS_E_INCOMPATIBLEDOCDATA;
// Create a sited IVsTextBuffer storing the converted data with the XML data and language service set.
IVsTextLines textLines = _package.CreateComInstance<VsTextBufferClass, IVsTextLines>();
SiteObject(textLines);
string xmlText = BinaryXmlData.GetXmlString(pszMkDocument);
textLines.InitializeContent(xmlText, xmlText.Length);
ErrorHandler.ThrowOnFailure(textLines.SetLanguageServiceID(ref Guids.XmlLanguageServiceGuid));
// Instantiate a sited IVsCodeWindow and feed it with the text buffer.
IVsCodeWindow codeWindow = _package.CreateComInstance<VsCodeWindowClass, IVsCodeWindow>();
SiteObject(codeWindow);
codeWindow.SetBuffer(textLines);
// Return the created instances to the caller.
ppunkDocView = Marshal.GetIUnknownForObject(codeWindow);
ppunkDocData = Marshal.GetIUnknownForObject(textLines);
return VSConstants.S_OK;
}
private void SiteObject(object obj)
{
(obj as IObjectWithSite)?.SetSite(_serviceProvider);
}
// --- CreateComInstance is a method on my package ----
internal TInterface CreateComInstance<TClass, TInterface>()
{
Guid guidT = typeof(TClass).GUID;
Guid guidInterface = typeof(TInterface).GUID;
TInterface instance = (TInterface)CreateInstance(ref guidT, ref guidInterface, typeof(TInterface));
if (instance == null)
throw new COMException($"Could not instantiate {typeof(TClass).Name} / {typeof(TInterface).Name}.");
return instance;
}
当我尝试使用我的编辑器明确打开文件时,它显示“无法使用所选编辑器打开该文件。请选择另一个编辑器。”该消息对我来说没有意义,我尝试使用 XML 编辑器打开 XML 数据,但它仍以某种方式尝试使用二进制数据打开文本编辑器。
我被困在这里,我尽我所能来提供转换后的数据。显然这种方式是不正确的。
如果这些问题需要冗长的回答,我很抱歉;如果我能指出正确的方向或一些已经开源的扩展做类似的事情(在 VS 代码编辑器中显示它之前转换文件数据),我会很高兴。
最佳答案
总体思路是让 Xml 编辑器执行它通常执行的操作:打开一个文档。
对于您的情况,如果我理解正确的话,您没有物理 Xml 文档,因此您必须创建一个。文档是在 Visual Studio 的 Running Object Table 中注册的东西(不一定是物理文件)。 .
一旦你有了一个文档,你就可以让 shell 打开它。您可以再次使用 ROT 来处理 BeforeSave and AfterSave events .下面是一些应该执行所有这些操作的示例代码:
public int CreateEditorInstance(uint grfCreateDoc, string pszMkDocument, string pszPhysicalView, IVsHierarchy pvHier, uint itemid, IntPtr punkDocDataExisting, out IntPtr ppunkDocView, out IntPtr ppunkDocData, out string pbstrEditorCaption, out Guid pguidCmdUI, out int pgrfCDW)
{
ppunkDocView = IntPtr.Zero;
ppunkDocData = IntPtr.Zero;
pbstrEditorCaption = null;
pguidCmdUI = Guid.Empty;
pgrfCDW = 0;
// create your virtual Xml buffer
var data = Package.CreateComInstance<VsTextBufferClass, IVsTextLines>();
SiteObject(data);
// this is where you're supposed to build your virtual Xml content from your binary data
string myXml = "<root>blah</root>";
data.InitializeContent(myXml, myXml.Length);
var dataPtr = Marshal.GetIUnknownForObject(data);
// build a document and register it in the Running Object Table
// this document has no hierarchy (it will be handled by the 'Miscellaneous Files' fallback project)
var rotFlags = _VSRDTFLAGS.RDT_ReadLock | _VSRDTFLAGS.RDT_VirtualDocument;
// come up with a moniker (which will be used as the caption also by the Xml editor)
// Note I presume the moniker is a file path, wich may not always be ok depending on your context
var virtualMk = Path.ChangeExtension(pszMkDocument, ".xml");
var rot = (IVsRunningDocumentTable)_sp.GetService(typeof(SVsRunningDocumentTable));
int hr = rot.RegisterAndLockDocument((uint)rotFlags, virtualMk, null, VSConstants.VSITEMID_NIL, dataPtr, out uint docCookie);
if (hr != 0)
return hr;
try
{
// ask Visual Studio to open that document
var opener = (IVsUIShellOpenDocument)_sp.GetService(typeof(SVsUIShellOpenDocument));
var view = VSConstants.LOGVIEWID_Primary;
opener.OpenDocumentViaProject(virtualMk, ref view,
out Microsoft.VisualStudio.OLE.Interop.IServiceProvider psp,
out IVsUIHierarchy uiHier,
out uint id,
out IVsWindowFrame frame);
if (frame != null)
{
// Hmm.. the dirty bit (the star after the caption) is not updated by the Xml Editor...
// If you close the document (or close VS), it does update it, but it does not react when we type in the editor.
// This is unexpected, so, let's do the "dirty" work ourselves
// hook on text line events from the buffer
var textLineEvents = new TextLineEvents((IConnectionPointContainer)data);
// we want to know when to unadvise, to hook frame events too
((IVsWindowFrame2)frame).Advise(textLineEvents, out uint frameCookie);
textLineEvents.LineTextChanged += (sender, e) =>
{
// get the dirty bit and override the frame's dirty state
((IVsPersistDocData)data).IsDocDataDirty(out int dirty);
frame.SetProperty((int)__VSFPROPID2.VSFPROPID_OverrideDirtyState, dirty != 0 ? true : false);
};
// now handle save events using the rot
var docEventHandler = new RotDocumentEvents(docCookie);
docEventHandler.Saving += (sender, e) =>
{
// this is where you can get the content of the data and save your binary data back
// you can use Saved or Saving
};
docEventHandler.Saved += (sender, e) =>
{
// manual reset of dirty bit...
frame.SetProperty((int)__VSFPROPID2.VSFPROPID_OverrideDirtyState, false);
};
rot.AdviseRunningDocTableEvents(docEventHandler, out uint rootCookie);
frame.Show();
}
}
finally
{
rot.UnlockDocument((uint)_VSRDTFLAGS.RDT_ReadLock, docCookie);
}
return VSConstants.S_OK;
}
private class TextLineEvents : IVsTextLinesEvents, IVsWindowFrameNotify, IVsWindowFrameNotify2
{
public event EventHandler LineTextChanged;
private uint _cookie;
private IConnectionPoint _cp;
public TextLineEvents(IConnectionPointContainer cpc)
{
var textLineEventsGuid = typeof(IVsTextLinesEvents).GUID;
cpc.FindConnectionPoint(ref textLineEventsGuid, out _cp);
_cp.Advise(this, out _cookie);
}
public void OnChangeLineText(TextLineChange[] pTextLineChange, int fLast) => LineTextChanged?.Invoke(this, EventArgs.Empty);
public int OnClose(ref uint pgrfSaveOptions)
{
_cp.Unadvise(_cookie);
return VSConstants.S_OK;
}
public void OnChangeLineAttributes(int iFirstLine, int iLastLine) { }
public int OnShow(int fShow) => VSConstants.S_OK;
public int OnMove() => VSConstants.S_OK;
public int OnSize() => VSConstants.S_OK;
public int OnDockableChange(int fDockable) => VSConstants.S_OK;
}
private class RotDocumentEvents : IVsRunningDocTableEvents3
{
public event EventHandler Saved;
public event EventHandler Saving;
public RotDocumentEvents(uint docCookie)
{
DocCookie = docCookie;
}
public uint DocCookie { get; }
public int OnBeforeSave(uint docCookie)
{
if (docCookie == DocCookie)
{
Saving?.Invoke(this, EventArgs.Empty);
}
return VSConstants.S_OK;
}
public int OnAfterSave(uint docCookie)
{
if (docCookie == DocCookie)
{
Saved?.Invoke(this, EventArgs.Empty);
}
return VSConstants.S_OK;
}
public int OnAfterFirstDocumentLock(uint docCookie, uint dwRDTLockType, uint dwReadLocksRemaining, uint dwEditLocksRemaining) => VSConstants.S_OK;
public int OnBeforeLastDocumentUnlock(uint docCookie, uint dwRDTLockType, uint dwReadLocksRemaining, uint dwEditLocksRemaining) => VSConstants.S_OK;
public int OnAfterAttributeChange(uint docCookie, uint grfAttribs) => VSConstants.S_OK;
public int OnBeforeDocumentWindowShow(uint docCookie, int fFirstShow, IVsWindowFrame pFrame) => VSConstants.S_OK;
public int OnAfterDocumentWindowHide(uint docCookie, IVsWindowFrame pFrame) => VSConstants.S_OK;
public int OnAfterAttributeChangeEx(uint docCookie, uint grfAttribs, IVsHierarchy pHierOld, uint itemidOld, string pszMkDocumentOld, IVsHierarchy pHierNew, uint itemidNew, string pszMkDocumentNew) => VSConstants.S_OK;
}
关于c# - VSX:如何重用现有的 XML 编辑器来处理转换为 XML 的二进制文件?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/46707307/
将已完成的 MPI_Request 重新用于另一个请求是否安全?我一直在使用 MPI_Request 池来提高性能并且没有错误。但肯定知道会很好。 最佳答案 MPI_Request 类型的变量不是请求
我注意到 Qt 文档在翻译的某些方面不是很冗长。我一直在玩弄它,试图通过反复试验来弄清楚他们的行为。最终目标是在运行时更改翻译,但我很困惑 QTranslator 对象在多大程度上可以重用。 考虑一下
我有一个 UIImageView 对象,它只是一个纯黑色矩形。这是我用来选择 View 中的按钮的方法。 问题是,我的 View 中有 49 个这样的按钮,并且所有这些按钮都可以同时选择。 我用来向按
在 R 中构建模型时,如何保存模型规范以便可以在新数据上重用它?假设我根据历史数据建立逻辑回归,但直到下个月才会有新的观察结果。最好的方法是什么? 我考虑过的事情: 保存模型对象并在新 session
我是 React/Redux 的初学者。 我已经完成了一个基本组件在我的应用程序中,其操作/ reducer /商店运行良好。 我将渲染另一个 具有不同的设置( Prop )。 我想做的是分离这两个组
我正在开发 GUI 纸牌游戏,我想知道是否有办法改进我的代码。这是我的情况。 我有三张牌:A、B 和 C。玩家可以通过分别单击三个按钮之一来更换牌:分别是按钮 1、按钮 2 或按钮 3。 class
每个文本框旁边有 2 个文本框和 2 个按钮 [...]。是否可以使用一个 OpenFileDialog 并将 FilePath 传递到相应的文本框,基于单击哪个按钮?即...如果我单击第一个按钮并打
我有两个场景:第一个场景,渲染纹理平面,第二个场景,应该渲染为纹理。该纹理应用作主场景中平面的贴图。 出于某种原因,所有 THREE.WebGLRenderTarget 示例每帧都会重新绘制两个场景,
我知道 concat、StringBuffer 和 StringBuilder 之间的区别。我知道 StringBuffer.toString 支持数组的内存问题可能会导致内存爆炸。我什至知道 JDK
我有 2 个 Activity 。 A 和 B。A 有一个包含 4 个项目的操作栏。每个项目显示不同的电影列表。 B extends A 因为我希望能够使用操作栏来更改电影列表。 所以我的问题是,当我
我有一个查询,用于检查从搜索文本框中输入的每个关键字,并且必须返回最匹配的关键字。 问题是,我想排除返回行中所有值为 0 的 KW_MATCHED。 SELECT A1.*, (
当方法重用时,是否有像这样的代码可以与 UICollectionViewCell 一起使用? - (UITableViewCell *)tableView:(UITableView *)tableVi
在我的项目中,我想在可 ScrollView 中以zig-zag 模式显示图像。所以我使用 uiscrollview 子查看其中的图像。它工作正常,但它占用了太多内存,因为我将所有图像加载到 Scro
如果我有 UIViewController1 并且我让它以模态方式显示 UIViewController2,但我希望 UIViewController2 显示 UIViewController1 模式
我想在所有 CCMenuItem 中使用完全相同的标签。如果我创建相同的 CCLabelTTF 一次,那么我无法将其添加到多个 CCMenuItem 中,因为它会给出有关已添加标签的运行时错误。但是,
我正在做一个项目,我们需要显示列表与用户位置的距离。为了显示距离,当在输入中给出纬度/经度时,我们使用名为“distance”的脚本字段计算距离 "script_fields" : {
我正在尝试重用我的 UITableViewCells。目前我的应用程序运行良好,在 tableView 中显示内容。然而,当我尝试实现 - 重用 UITableViewCells 时,我的应用程序崩溃
假设我在外部样式表中定义了几个类 .b {font-weight:bold;} .c {text-align:center;} 现在我想要另一个类,它是 b 和 c 的组合 .bc 是否可以使用类 b
我目前经常分配新的协程实例(请参阅我的回答中的代码 here)。 这样做的开销并不小。 我猜想是否有某种方法可以通过重用之前分配的协程来降低成本? 虽然不确定如何实现这一点? 我可以为协程 Alloc
在我的应用程序中,我使用如下代码下载多张图片。这样做是高性能还是我可以以某种方式重用连接? for(int i = 0; i < 100; i++){ URL url = new UR
我是一名优秀的程序员,十分优秀!