在很多场合,我们需要在已有软件程序上增加一些新的功能,几乎所有原因是因为原有软件功能不能满足我们的需要,我们平时做的插件就属于这种情况,最常见的是VS IDE的插件开发,网上老外写的一篇关于插件开发的文章,很详细(网址)。如果我们要给一个已有软件扩展新的功能,一般我们必须知道原有软件提供给二次开发人员的接口,也就是说,如果原有软件在设计的时候,压根儿就没有考虑到后续可能存在的二次开发,也不提供任何接口,那么通常情况下,是很难在它的基础上扩展出新功能的(除非是原有软件开发者)。

还有一种可以扩展已有程序功能的方式,网址,利用windows消息、windows hook技术,理论上可以给任何一个桌面应用程序扩展出新功能,而不需要任何接口,但是,这种方式很有局限性,扩展出来的功能几乎停留在操作系统级别上,比如UI外观样式等,并不能真正的去与已有软件程序进行交互。老外这篇文章其实重点是在讲Windows hooks和Windows Message。

这篇文章不是讲怎么去开发VS插件,更不是谈哪个具体软件比如CAD、PROE的二次开发,我只是想将看似复杂的东西简单化地解释一下,看看“给已有软件扩展新功能”到底是怎么回事。以插件为例:

首先,宿主程序和插件之间一定要有交互的,不然的话,插件是不会知道什么时候该干什么事情;其次,宿主程序一定会传递某些数据信息给插件,否则你叫插件拿什么原材料干活?最后,宿主程序一定是要有所准备的,什么叫有所准备?也就是说,在开发宿主程序的时候,一定要为以后的功能扩展留有接口,所有插件必须遵守这个接口给出的规范,知道应该在什么时候跟插件通讯,了解插件的任何一个行为将会导致什么样的结果,并且作出相应的反应。综上所述,给已有程序扩展新功能,关键还是在这个“已有程序”身上,如果一个程序出生的时候就没想着将来别人要给自己增加功能,那你不用再想着去给它扩展功能了。也就是我文章刚开始说到的,并不是你可以在任何一个程序基础上扩展新功能。

图1

如上图所示,宿主程序与插件之间通过某一协议进行通信,这个跟上一篇最后讲到的“框架和客户端代码之间的关系”很相似,你可以把宿主程序看做是框架,而插件则是客户端代码(参见上一篇文章图6)。

图2

如上图,在宿主程序中应该提前设计好该在什么时候与插件通信,以及给它传递对应数据信息,接着返回交互结果。宿主程序应该考虑所有与插件交互的地方和时间,然而插件不一定处处都会有所反应,也就是说,一个宿主程序设计好100个与插件交互的地方(插件最多可以在这100个地方大做文章),但是你开发一个插件时,根据具体需要,完全可以只响应其中的某几个。

图3

文章后面我附上一个简单的画图Demo,实现简单的画板、保存(默认可以保存JPG图片格式和PIC可编辑格式)等功能,然后自己又做了一个插件,插件主要新增了以下功能:

  • 增加一个“关于”菜单,点击弹出关于对话框;
  • 已有画板程序只能绘制圆形和正方形,增加了一个三角形图形;
  • 将画图保存成JPG格式时,在图片上添加水印;
  • 增加一种全新的文件格式(newpic格式),可以将画图保存为newpic格式的文件,这个有点类似photoshop的ico插件,安装后,PS可以将图片保存为ico格式。

整个项目源码分为以下三个部分:

  1. PluginDemo:宿主程序,在它的基础上扩展新的功能;
  2. PluginHelper:扩展功能时必须遵守的规范(接口),随宿主程序一起开发,通常就是我们常说的“二次开发包”,理论上应该还有二次开发说明文档之类的东西;
  3. Plugin:我自己开发的一个插件。

正常情况下,1和2由已有软件开发商提供,3由二次开发人员开发。

下面主要说明一下PluginHelper中的两个接口,其余的源码诸位可以自己下下来看看。

IPlugin接口:

     /// <summary>
/// 插件接口 所有插件必须实现该接口
/// </summary>
public interface IPlugin
{
void ApplicationLoaded(PluginApplication pluginApplication); //应用加载后
void FileSavingAsJPG(Bitmap bitmap,string filepath); //文件保存为JPG
void FileSavingAsPIC(PluginApplication pluginApplication); //文件保存为PIC
void BeforeSave(Dictionary<string,SaveFileHandler> extensions); //保存文件之前
void BeforeOpen(Dictionary<string,OpenFileHandler> extensions); //打开文件之前
void ApplicationExiting(); //应用退出时
}

如接口代码所示,在固定时候固定地方,宿主程序都会调用对应方法与插件通信。

IObject接口:

    /// <summary>
/// 图形接口 所有的图形都必须实现该接口
/// </summary>
public interface IObject
{
int X
{
get;
set;
}
int Y
{
get;
set;
}
void Draw(Graphics g);
}

所有新扩展图形都必须实现该接口。

如果想要开发自己的插件,只需要知道二次开发包(PluginHelper.dll),定义一个类实现IPlugin接口就行。

图4

最后上几张效果图,没插件之前的宿主程序:

图5

安装插件后的宿主程序:

图6

如上图所示,安装插件后,菜单多了“关于”菜单项,工具栏多了“三角形”按钮,可以保存另外一种“newpic”格式的文件,另外,在保存为JPG格式图片时,已有软件保存图片为:

图7

安装插件后,保存为JPG格式文件如下:

图8

如上图,安装插件后,保存的JPG图有水印。

下载源码:http://files.cnblogs.com/xiaozhi_5638/PluginDemo.rar

将开发的插件放在宿主程序的plugins目录下,重启宿主程序就可以。希望对各位有帮助!

.Net开发笔记(十七) 应用程序扩展的更多相关文章

  1. Java开发笔记(八十七)随机访问文件的读写

    前面介绍了字符流读写文件的两种方式,包括文件字符流和缓存字符流,但是它们的写操作都存在一个问题:不管是write方法还是append方法,都只能从文件开头写入,而不能追加到文件末尾或者在文件中间某个位 ...

  2. Java开发笔记(二十七)数值包装类型

    方法的出现缘起优化代码结构,但它的意义并不局限于此,正因为有了方法定义,编程语言才更像一门能解决实际问题的工具,而不仅仅是只能用于加减乘除的计算器.在数学的发展过程中,为了表示四则运算,人们创造了加减 ...

  3. Java开发笔记(三十七)利用正则串分割字符串

    前面介绍了处理字符串的常用方法,还有一种分割字符串的场景也很常见,也就是按照某个规则将字符串切割为若干子串.分割规则通常是指定某个分隔符,根据字符串内部的分隔符将字符串进行分割,例如逗号.空格等等都可 ...

  4. Java开发笔记(四十七)关键字this的用法

    前面介绍了类的基本定义,包括成员属性.成员方法.构造方法几个组成要素,可谓是具备了类的完整封装形态.不过在进行下一阶段的学习之前,有必要梳理一下前述的类定义代码,看看是否存在哪些需要优化的地方.首先观 ...

  5. Java开发笔记(五十七)因抽象方法而产生的抽象类

    前面介绍了类的常见用法,令人感叹面向对象的强大,几乎日常生活中的所有事物,都可以抽象成Java的基类及其子类.然而抽象操作也有副作用,就是某个抽象而来的行为可能是不确定的,比如半夜鸡叫,如果是公鸡则必 ...

  6. Java开发笔记(六十七)清单:ArrayList和LinkedList

    前面介绍了集合与映射两类容器,它们的共同特点是每个元素都是唯一的,并且采用二叉树方式的类型还自带有序性.然而这两个特点也存在弊端:其一,为啥内部元素必须是唯一的呢?像手机店卖出了两部Mate20,虽然 ...

  7. Java开发笔记(七十七)使用Optional规避空指针异常

    前面在介绍清单用法的时候,讲到了既能使用for循环遍历清单,也能通过stream流式加工清单.譬如从一个苹果清单中挑选出红苹果清单,采取for循环和流式处理都可以实现.下面是通过for循环挑出红苹果清 ...

  8. Java开发笔记(九十七)利用Runnable启动线程

    前面介绍了线程的基本用法,按理说足够一般的场合使用了,只是每次开辟新线程,都得单独定义专门的线程类,着实开销不小.注意到新线程内部真正需要开发者重写的仅有run方法,其实就是一段代码块,分线程启动之后 ...

  9. Java开发笔记(一百四十七)通过JDBC管理数据库

    前面介绍了如何通过JDBC获取数据库连接,可是Connection对象不能直接执行SQL语句,需要引入Statement报告对象才能操作SQL.Statement对象由Connection的creat ...

  10. Java开发笔记(一百一十七)AWT窗口

    前面介绍的所有Java代码,都只能通过日志观察运行情况,就算编译成class文件,也必须在命令行下面运行,这样的程序无疑只能给开发者做调试用,不能拿给一般人使用.因为普通用户早已习惯在窗口界面上操作, ...

随机推荐

  1. 一个简单的消息提示jquery插件

    最近在工作中写了一个jquery插件,效果如下: 就是一个简单的提示消息的一个东西,支持最大化.最小化.关闭.自定义速度.自定义点击事件,数据有ajax请求和本地数据两种形式.还有不完善的地方,只做了 ...

  2. OpenCV特征点检测------ORB特征

    OpenCV特征点检测------ORB特征 ORB是是ORiented Brief的简称.ORB的描述在下面文章中: Ethan Rublee and Vincent Rabaud and Kurt ...

  3. 动态生成一个设定好特殊样式的Tlabel,快速生成代码

    动态生成一个设定好特殊样式的Tlabel,快速生成代码: 1.自己先在可视化界面设定一个Label,像这样: 2.选择label,快捷键ctrl+C 复制,粘贴带代码编辑器去,会生成一段这样的窗体代码 ...

  4. iOS 链式编程探索(Masonry)

    看了几篇关于链式编程的文章,还是理解的不透彻,我想这可能是因为我自己对block掌握的不熟练. 我已经明白了,所以,和大家分享一下我的理解!如有问题,麻烦大家指出! 直接看代码吧!关键的注释都有. 我 ...

  5. linpe包-让发送和接收数据分析更快和更容易

    1.简介 通常在R中从来进行分析和展现的数据都是以基本的格式保存的,如.csv或者.Rdata,然后使用.Rmd文件来进行分析的呈现.通过这个方式,分析师不仅可以呈现他们的统计分析的结果,还可以直接生 ...

  6. 1_MVC+EF+Autofac(dbfirst)轻型项目框架_core层(以登陆为例)

    前言 在上一篇0_MVC+EF+Autofac(dbfirst)轻型项目框架_基本框架中,我已经介绍了这个轻型框架的层次结构,在下面的这篇文章中,我将以教师登陆功能为例,具体来扩充下我的core层的代 ...

  7. 使用TextUtils.isEmpty简单化代码

    如果让你判断一个文本框是否为空(null)或者没有任何值(length=0),你会怎么怎样去写代码,很多初学者可能会这样写: if(text==null || text.length==0) {... ...

  8. 微信小程序火车票查询 直取12306数据

    最终效果图: 样式丑哭了,我毕竟不是前端,宗旨就是练练手,体验微信小程序的开发,以最直接的方式获取12306数据查询火车票. 目录结构: search1是出发站列表,search2是目的站列表,命名没 ...

  9. ASP.NET图形验证码的生成

    效果: 调用方法: int[] r = QAPI.VerifImage.RandomList();//取得随机数种子列 );//产生验证码字符 pictureBox1.Image = QAPI.Ver ...

  10. postman发送带cookie的http请求

    1:需求:测试接口的访问权限,对于某些接口A可以访问,B不能访问. 2:问题:对于get请求很简单,登录之后,直接使用浏览器访问就可以: 对于post请求的怎么测试呢?前提是需要登录态,才能访问接口. ...