C#开发word(wps)插件(com加载项)作为一个web前端开发,不搞点后端的东西玩玩也说不过去。除了常见的node

关于word(wps)插件,要实现的最终效果,无非如下两种:

因为这些基于microsoftoffice的上做的二次开发,必须要基于window系统,参考我微软的技术栈开发。微软的技术主要的无非两种:

两者无所谓好坏,但我个人对vb不太感冒,而且气语法跟java的有点点差异的,因此技术选型选择C#。

建议安装完MicrosoftVisualStudio后以管理员身份运行启动

选中解决方案下的项目---【右键】---【属性】,开始如下设置:

【生成】---勾选【为com互操作注册】

【调试】---选中【启动外部程序】---选择wps安装路径

我这里是wps文字,因此选中后的完成路径是:

C:\ProgramFiles(x86)\Kingsoft\WPSOffice\10.8.0.6370\office6\wps.exe

如果你们是基于wps表格

C:\ProgramFiles(x86)\Kingsoft\WPSOffice\10.8.0.6370\office6\et.exe

wpsppt的话

C:\ProgramFiles(x86)\Kingsoft\WPSOffice\10.8.0.6370\office6\wpp.exe

【签名】--【新建签名】

图中的xx_officialDocument.pfx就是此处下拉框选则【新建】后当场生成的。

至此,新建项目完毕,

UpgradeWPSOffice3.0ObjectLibrary对应的是WPS文字、UpgradeWPSSpreadsheets3.0ObjectLibrary对应的是WPS表格。

提示:若添加后提示引用出错,可以尝试的解决方法有:

引用添加完毕后,我们可以看到我们总共依赖了哪些引用

在Properties上右键→添加→新建项→找到“资源文件”→输入或保持默认名称→添加

打开Resource1.resx后--【添加资源】---【添加新文本文件】---【输入MyRibbon】

这个MyRibbon.txt文件就是一个xml菜单文件,配置你在wps中的菜单的显示

双击MyRibbon.txt后进入编辑,输入如下文本

这里的主类,就是我们一开始把Class1.cs该成了OfficialDocument.cs这个文件。当然你可以命名为其他。

打开了OfficialDocument.cs:文件最上方就是我们引用的类,都写进来

usingSystem;usingSystem.Collections.Generic;usingSystem.Linq;usingSystem.Text;usingSystem.Threading.Tasks;usingAddInDesignerObjects;usingOffice;usingSystem.Windows.Forms;继续往下写

usingSystem;usingSystem.Collections.Generic;usingSystem.Linq;usingSystem.Text;usingSystem.Threading.Tasks;usingAddInDesignerObjects;usingOffice;usingSystem.Windows.Forms;//namespace[yourProjectName]视具体情况改成你的项目名namespaceyourProjectName{publicclassClass1:IDTExtensibility2,IRibbonExtensibility{{}}

有红色的波浪线,提示我们有错,不过没关系,鼠标悬浮在上面,点击黄色灯泡图标的下拉箭头---【实现接口】,visualstudio会自动实现其接口。IRibbonExtensibility这个也一样这么操作一下。

得到修复后的代码如下:

usingSystem.Collections.Generic;usingSystem.Linq;usingSystem.Text;usingSystem.Threading.Tasks;usingAddInDesignerObjects;usingOffice;usingSystem.Windows.Forms;usingSystem;namespaceyourProjectName{publicclassClass1:IDTExtensibility2,IRibbonExtensibility{publicvoidOnConnection(objectApplication,ext_ConnectModeConnectMode,objectAddInInst,refArraycustom){thrownewNotImplementedException();}publicvoidOnDisconnection(ext_DisconnectModeRemoveMode,refArraycustom){thrownewNotImplementedException();}publicvoidOnAddInsUpdate(refArraycustom){thrownewNotImplementedException();}publicvoidOnStartupComplete(refArraycustom){thrownewNotImplementedException();}publicvoidOnBeginShutdown(refArraycustom){thrownewNotImplementedException();}publicstringGetCustomUI(stringRibbonID){thrownewNotImplementedException();}}}在OnConnection事件中初始化app和wordDoc对象

publicclassClass1:IDTExtensibility2,IRibbonExtensibility{publicstaticWord.Applicationapp=null;publicstaticobjectwps;publicstaticWord.DocumentwordDoc;ObjectNothing=System.Reflection.Missing.Value;publicvoidOnConnection(objectApplication,ext_ConnectModeConnectMode,objectAddInInst,refArraycustom){wps=Application;app=wpsasWord.Application;wordDoc=app.Documents.Add(refNothing,refNothing,refNothing,refNothing);//wordDoc=app.ActiveDocument;wordDoc.PageSetup.PaperSize=Word.WdPaperSize.wdPaperA4;wordDoc.PageSetup.TopMargin=app.CentimetersToPoints(3.7f);//37mm,对应104.9磅(1磅约等于0.3572mm)wordDoc.PageSetup.BottomMargin=app.CentimetersToPoints(3.5f);wordDoc.PageSetup.LeftMargin=app.CentimetersToPoints(2.8f);wordDoc.PageSetup.RightMargin=app.CentimetersToPoints(2.7f);}...}在GetCustomUI事件中调用在Resource1.resx中添加的MyRibbon.txt,获取菜单配置文件

publicstringGetCustomUI(stringRibbonID){returnProperties.Resource1.MyRibbon;}添加事件试试。我们在MyRibbon.txt文件中的每一项按钮里面,都有一个onAction,这个就是绑定的事件。需要我们在Class1.cs文件中去实现

publicvoidreduceFontSpace(IRibbonControlctrl){MessageBox.Show("youclickreduceFontSpace")//floatformer=app.Selection.Font.Spacing;//app.Selection.Font.Spacing=former-0.3f;}

提示:MessageBox需要引用System.Windows.Forms。引用方法如下:

在第三部中,我们实现了一个自定义的菜单,并期望其加到wps的菜单栏中,具体展示结果如下

怎么样才能展示这个菜单呢?

在【开发工具】---【com加载项】选中我们的实现的com加载项(wps专业版才有【开发工具】这个菜单选项):

新增一个空的txt记事本文件→修改后缀名,另存为install.reg→右键此文件编辑→拷贝如下代码:

WindowsRegistryEditorVersion5.00[HKEY_CURRENT_USER\SOFTWARE\Microsoft\Office\Word\Addins\Zhjk.OfficialDocument]"FriendlyName"="Zhjk.OfficialDocument""Description"="【自定义中文描述方便他人理解】""LoadBehavior"=dword:00000003"CommandLineSafe"=dword:00000001[HKEY_CURRENT_USER\Software\Kingsoft\Office\WPS\AddinsWL]"Zhjk.OfficialDocument"=""需要注意的是:

[HKEY_CURRENT_USER\SOFTWARE\Microsoft\Office\Word\Addins\Zhjk.OfficialDocument]这里[HKEY_CURRENT_USER\SOFTWARE\Microsoft\Office\Word\Addins固定不变,后面的xxx1.xxx2,表示xxx1是你项目的名,xxx2是你cs入口文件的类名,本例中就是yourProjectName.Class1

以上,写入注册表信息install.reg文件完成。

但是,如果你想清理掉这个注册表信息,你可以手动在命令行输入regedt,打开操作界面,你也可以专门生成一个unistall.reg文件,双击实行即可。unistall.reg内容如下:

WindowsRegistryEditorVersion5.00[-HKEY_CURRENT_USER\SOFTWARE\Microsoft\Office\Word\Addins\Zhjk.OfficialDocument][HKEY_CURRENT_USER\Software\Kingsoft\Office\WPS\AddinsWL]"Zhjk.OfficialDocument"=-注意减号(横杠)“-”,还有xxx1.xxx2(本例子中是Zhjk.OfficialDocument)要与你安装的install.reg里面的保持一致。若开发的是WPS文字、WPS表格的外接程序,则注册路径中HKEY_CURRENT_USER\SOFTWARE\Microsoft后面:WPS文字对应的是Word和WPS;WPS表格对应的是Excel和ET

某些电脑可能要用管理权权限来运行这个reg文件

命令行输入regedit

沿着这个HKEY_CURRENT_USER\SOFTWARE\Microsoft\Office\Word\Addins路径一直打开目录,如果看到如下,即说明写入注册表成功

之前做了那么多准备工作,现在可以启动调试,看看我们的wps文字到底有没有按照预定的情况加载自定义菜单。打开visualstudio界面,在degug模式下,点击启动按钮:

然后自动自动wps文字(如果此前你打开了wps文字应用,或者word,请先关掉)

看到此界面,说明我们的自定义com加载项本地调试成功。

部署之前,我们要先对我们的dll签名,保证系统信任此文件

Office.dllVBIDE.dllWord.dll

后面我们需要这三个dll。

若用户是WPS专业版则不需要,为了兼容性,推荐进行强签名。另外,如果项目中引用了第三方未签名的dll,则必须进行强签名才能引用。

ildasmWord.dll/out:Word.ililasm/dll/res:Word.res/key:zhjk.pfxWord.il/out:Word.dll

可以看到强签名后的文件如下:

Word.dll、Office.dll这两个强签名的,就是我们后面需要的文件。

在此前的步骤中,我们拿到了强签名的dll和Release模式下应用生曾的dll,

Zhjk.dllZhjk.pdbZhjk.tlbWord.dllOffice.dll现在开始对window系统进行操作了。

这个文件夹路径

C:\ProgramFiles(x86)\MicrosoftSDKs\Windows\v10.0A\bin

你可以在命令行,一行行输入命令来实现。这里我们推荐用批处理命令。把所有的命令都放在一个xxx.bat文件里。

这里我们的在当前Release目录下新增一个install.bat,里面内容如下:

@echooff@setbaseDir=%~dp0regedit/s%baseDir%\reg\Zhjk.OfficialDocument.regC:\Windows\Microsoft.NET\Framework\v4.0.30319\RegAsm%baseDir%\Zhjk.dll/tlb:%baseDir%\Zhjk.tlb@SETGACUTIL="%baseDir%\NETFX4.8Tools\gacutil.exe"%GACUTIL%-i%baseDir%\Zhjk.dll%GACUTIL%-i%baseDir%\Word.dll%GACUTIL%-i%baseDir%\Office.dllpause同样,再建一个卸载的bat批处理命令文件uninstall.bat:

@echooff@setbaseDir=%~dp0Echo1.从缓存中移除程序集@SETGACUTIL="%baseDir%\NETFX4.8Tools\gacutil.exe"rd/s/QC:\Windows\Microsoft.NET\assembly\GAC_MSIL\Zhjkrd/s/QC:\Windows\Microsoft.NET\assembly\GAC_MSIL\Wordrd/s/QC:\Windows\Microsoft.NET\assembly\GAC_MSIL\OfficeEcho2.注销类型C:\Windows\Microsoft.NET\Framework\v4.0.30319\RegAsm/u%baseDir%\Zhjk.dll/tlb:%baseDir%\Zhjk.tlbEcho.Echo3.清除注册表regedit/s%baseDir%\reg\Zhjk.OfficialDocument-delete.regpause%~dp0是bat文件当前文件夹路径;RegAsm是程序集注册工具,需要用户先安装Microsoft.NetFramework4.0;gacutil.exe是全局程序集缓存工具,用户电脑一般没有,所以需要我们将NETFX4.8Tools文件夹附带在安装包里

THE END
1.合法C标识符给定一个不包含空白符的字符串,请判断是否是C语言合法的标识符号(注:题目保证这些字符串一定不是C语言的保留字)。 C语言标识符要求: 非保留字; 只包含字母、数字及下划线(“_”)。 不以数字开头。 输入 一行,包含一个字符串,字符串中不包含任何空白字符,且长度不大于20。 https://blog.csdn.net/qq_41840843/article/details/144174625
2.一个让代码可读性暴增的现代C++特性,同事看了都说好!// 老方法1:用指针来搞定User*findUser(conststring&name){// 找到了就返回指针,找不到就返回 nullptr// 但是等等这个指针谁来删啊?记不住删除的话就内存泄漏啦!}// 老方法2:用 pair 大法pair<User,bool>findUser(conststring&name){// 返回一个值和一个标志位// 但是就算没找https://www.51cto.com/article/803487.html
3.c#中方法签名指的是?任飞儿方法签名由方法名称和一个参数列表(方法的参数顺序和类型)组成。 注意:方法的签名并不包括方法的返回值。虽然每个重载方法可以有不同的返回类型,单返回类型并不足以区分所条用的是哪个方法。 在C#中,同一个类中的两个或两个以上的方法可以相同的名字,只要他们的参数声明不同即可。在这种情况下,该方法就被称为重https://www.cnblogs.com/mmbbflyer/archive/2009/12/14/1623577.html
4.C#中的方法签名是什么?B ( int q1,int q2){}的签名相同 而public int C (int m1,int m2){}则和方法A签名不同https://www.imooc.com/wap/wenda/id/446787
5.C#创建自签名认证文件的方法C#教程这篇文章主要介绍了C#创建自签名认证文件的方法,实例分析了C#自签名认证文件的实现技巧,具有一定参考借鉴价值,需要的朋友可以参考下GPT4.0+Midjourney绘画+国内大模型 会员永久免费使用!【 如果你想靠AI翻身,你先需要一个靠谱的工具!】 本文实例讲述了C#创建自签名认证文件的方法。分享给大家供大家参考。具体如下: 1https://www.jb51.net/article/69514.htm
6.方法(C#编程指南)MicrosoftLearn方法签名 方法访问 方法形参和实参 通过引用传递与通过值传递 显示另外 3 个 更新:2007 年 11 月 “方法”是包含一系列语句的代码块。程序通过“调用”方法并指定所需的任何方法参数来执行语句。在 C# 中,每个执行指令都是在方法的上下文中执行的。Main 方法是每个 C# 应用程序的入口点,在启动程序时由公共语言https://msdn.microsoft.com/zh-cn/library/ms173114(v=vs.90).aspx
7.C#Notizen3理解C#类和对象您可能利用不同的返回类型进行重载,虽然这可能是合法的 C#代码,但是由于方法签名不包含返回类型,因此这可能导致混乱。为最大限度地减少混乱,应避免这样做。 在需要提供多种执行动作的方式时,方法重载很有用,但是可供选择的空间太大时,可能难以应付。 如下是一个方法重载示例: https://www.jianshu.com/p/01e3320194a7
8.C#调用c++的dll执行带参数的函数时请检查PInvoke签名的调用其他信息: 对 PInvoke 函数“test!test.FUNC1::Invoke”的调用导致堆栈不对称。原因可能是托管的 PInvoke 签名与非托管的目标签名不匹配。请检查 PInvoke 签名的调用约定和参数与非托管的目标签名是否匹配。 解决方法: C#默认是stdcall调用约定,你可以把委托改成下面写法 https://www.feelsight.cn/post/140.html
9.c#signature最佳实践有哪些问答C#签名是定义方法、属性、事件等成员的方式,良好的签名可以使代码更易于理解、维护和扩展。以下是一些C#签名的最佳实践:1. 使用有意义的名称:方法、属性和事件的名称应该清晰地表达其功能和用途。避免https://www.yisu.com/ask/38523568.html
10.干货C#自定义特性(Attribute)讲解与实际应用特性可以与方法和属性相同的方式接受参数。 程序可以使用反射检查自己的元数据或其他程序内的元数据。 特性的用处 我们直接从实战中来了解特性意义,针对前两天我们发的一个通讯签名的问题《C#根据类生成签名字符串》和《Android根据类生成签名字符串》,当我们C#后端和Android前端类是一致的时候,根据类型动态生成签名那没https://cloud.tencent.com/developer/article/1471809
11.C#方法详解:定义调用与最佳实践二、定义方法 1. 返回类型 方法可以有返回类型,也可以没有返回类型。如果没有返回类型,则使用void关键字。 2. 方法签名 方法签名由方法名和参数列表组成。参数列表可以为空,也可以包含一个或多个参数。 3. 参数 C# 中的方法可以接受零个或多个参数。参数可以是基本类型,也可以是引用类型。 https://maimai.cn/article/detail?fid=1843011637&efid=zOE53CY6JSasNqG_JPuo1g