从零开始编写属于我的CMS:(六)插件
二三四五还没写,先写六吧(有道友说想看看插件部分)。
这里是一 从零开始编写属于我的CMS:(一)前言
一,首先预定义接口
新建类库,WangCms.PluginInterface

新建两个类,一个实体Models.cs,一个接口IPlugin.cs
Models是插件所用到的实体集合类;IPlugin是为第三方预定义接口,所有插件必须实现该接口。
实体Models代码如下
namespace WangCms.PluginInterface
{
public class PluginInfo
{
/// <summary>
/// Code
/// </summary>
public string Code { get; set; }
/// <summary>
/// 插件名称
/// </summary>
public string Name { get; set; }
/// <summary>
/// 作者
/// </summary>
public string Author { get; set; }
/// <summary>
/// 插件版本
/// </summary>
public string Version { get; set; }
/// <summary>
/// 适用cms版本
/// </summary>
public string ApplyVersion { get; set; }
/// <summary>
/// 描述信息
/// </summary>
public string Description { get; set; }
//附加信息
public string Area { get; set; }
public string AdminController { get; set; }
public string AdminAction { get; set; }
public string AdminQueryString { get; set; }
}
public class ResultOptin<T>
{
public bool State { get; set; }
public string Msg { get; set; }
public T Result { get; set; }
}
}
接口IPlufin代码如下
namespace WangCms.PluginInterface
{
public interface IPlugin
{
/// <summary>
/// 插件注册
/// </summary>
/// <returns></returns>
ResultOptin<PluginInfo> Register();
/// <summary>
/// 插件安装
/// </summary>
/// <param name="model"></param>
/// <returns></returns>
ResultOptin<string> Install();
/// <summary>
/// 插件卸载
/// </summary>
/// <returns></returns>
ResultOptin<string> Uninstall();
}
}
二,留言插件
新建一个插件项目(类库),WangCms.Plugin.LiuYan

然后添加相关引用(MVC、WangCms.PluginInterface)等。
首先实现插件接口,新建PluginRegister.cs继承IPlugin,主要为了实现插件的注册、安装以及卸载功能的实现。
PluginRegister代码如下
namespace WangCms.Plugin.LiuYan
{
public class PluginRegister : IPlugin
{
#region 实现接口
public ResultOptin<PluginInfo> Register()
{
ResultOptin<PluginInfo> result = new ResultOptin<PluginInfo>();
try
{
PluginInfo model = new PluginInfo();
//后台管理入口
model.Area = "LiuYan";
model.AdminController = "LiuYanAdmin";
model.AdminAction = "Index";
model.AdminQueryString = null;
//插件基本信息
model.Code = "48a3619327c64a9aa68645007037b451";
model.Name = "在线留言";
model.Author = "千年";
model.Version = "1.0.0";
model.ApplyVersion = "1.0.0";
model.Description = "";
//状态 特别重要
result.State = true;
result.Result = model;
}
catch(Exception ex)
{
result.State = false;
result.Msg = ex.Message;
}
return result;
}
public ResultOptin<string> Install()
{
ResultOptin<string> result = new ResultOptin<string>();
try
{
//安装插件
//比如执行sql(创建表,插入数据等),创建目录,创建文件等
string sql =
@"create table LiuYan(
Id varchar(50) primary key,
Title varchar(50),
Name varchar(50),
Contact varchar(50),
Content text
);";
LiuYanService.Instance.Excute(sql);
result.State = true;
}
catch(Exception ex)
{
result.State = false;
result.Msg = ex.Message;
}
return result;
}
public ResultOptin<string> Uninstall()
{
ResultOptin<string> result = new ResultOptin<string>();
try
{
//卸载插件
//比如执行sql(删除表,删除数据等),删除目录,删除文件等
string sql = @"drop table LiuYan;";
LiuYanService.Instance.Excute(sql);
result.State = true;
}
catch(Exception ex)
{
result.State = false;
result.Msg = ex.Message;
}
return result;
}
#endregion
}
}
其实,只要实现了接口插件就算完成了,只是该插件还不具备任何实用功能,根据插件名字我想大家知道这个是干嘛的了吧,这就是在线留言的插件,下面就是很简单的留言业务逻辑以及功能的实现了。
插件区域,就是MVC的区域,我们用不同区域来区分和管理插件。

MVC区域,不熟悉的同学可以在园子里搜搜相关文章。
在区域里面可以建Controller和View,还有比较重要的就是区域注册,就是区域的路由吧。
LiuYanAreaRegistration.cs
namespace WangCms.Plugin.LiuYan
{
public class LiuYanAreaRegistration : AreaRegistration
{
public override string AreaName
{
get
{
return "LiuYan";
}
}
public override void RegisterArea(AreaRegistrationContext context)
{
context.MapRoute(
"LiuYan_default",
"LiuYan/{controller}/{action}/{id}",
new { action = "Index", id = UrlParameter.Optional },
new string[] { "WangCms.Plugin.LiuYan.Controllers" }
);
}
}
}
然后,就是功能实现了,这部分就略过吧。
在线留言后台管理,新建一个控制器AdminController.cs以及相应的视图。
在线留言前台功能,新建一个控制器PageController.cs以及相应的视图。
三,插件使用
主项目如何使用插件呢?
首先将编译好的插件,上传至主项目下,结构如图。

视图文件和插件时路径一致,WangCms.Plugin.LiuYan.dll上传至主项目bin下。
3.1 获取插件列表
private List<Plugin> GetPluginList()
{
List<Plugin> list = new List<Plugin>();
string path = Server.MapPath("/bin/");
FileInfo[] files = (new DirectoryInfo(path)).GetFiles("*.dll");
foreach (var item in files)
{
try
{
if (!item.Name.StartsWith("WangCms.Plugin.")) continue;
Plugin model = new Plugin();
Assembly ass = Assembly.LoadFile(item.FullName);
Type tp = ass.GetType(item.Name.Replace(".dll", "") + "." + "PluginRegister"); //获取类名,必须 命名空间+类名
Object obj = Activator.CreateInstance(tp); //建立实例
MethodInfo meth = tp.GetMethod("Register"); //获取方法
object t = meth.Invoke(obj, null); //Invoke调用方法
PluginInterface.ResultOptin<PluginInterface.PluginInfo> result = (PluginInterface.ResultOptin<PluginInterface.PluginInfo>)t;
if (result.State)//插件注册成功
{
ToPlugin(result.Result, ref model);
model.Type = tp;
list.Add(model);
}
}
catch { }
}
return list;
}
这里有一个插件类转换的方法。
private void ToPlugin(PluginInterface.PluginInfo t, ref Plugin p)
{
if (t != null)
{
p.Code = t.Code;
p.Name = t.Name;
p.Author = t.Author;
p.Version = t.Version;
p.ApplyVersion = t.ApplyVersion;
p.Description = t.Description;
p.AdminController = t.AdminController;
p.AdminAction = t.AdminAction;
p.AdminQueryString = t.AdminQueryString;
}
}
3.2 安装插件
public ActionResult plugin_install(string code)
{
var list = GetPluginList();
var o = list.FirstOrDefault(op => op.Code == code);
if (o != null)
{
//执行安装方法
Object obj = Activator.CreateInstance(o.Type); //建立实例
MethodInfo meth = o.Type.GetMethod("Install"); //获取方法
object t = meth.Invoke(obj, null); //Invoke调用方法
PluginInterface.ResultOptin<string> result = (PluginInterface.ResultOptin<string>)t;
if (result.State)
{
//记录数据
PluginService.Instance.UpdateOrInsert(o);
}
else
{
return Content(result.Msg);
}
}
return RedirectToAction("plugin_list");
}
3.3 卸载插件
public ActionResult plugin_uninstall(string code)
{
var list = GetPluginList();
var o = list.FirstOrDefault(op => op.Code == code);
if (o != null)
{
//执行安装方法
Object obj = Activator.CreateInstance(o.Type); //建立实例
MethodInfo meth = o.Type.GetMethod("Uninstall"); //获取方法
object t = meth.Invoke(obj, null); //Invoke调用方法
PluginInterface.ResultOptin<string> result = (PluginInterface.ResultOptin<string>)t;
if (result.State)
{
//删除数据
PluginService.Instance.DeleteByCode(o.Code);
}
else
{
return Content(result.Msg);
}
}
return RedirectToAction("plugin_list");
}
插件源码下载
从零开始编写属于我的CMS:(六)插件的更多相关文章
- 从零开始编写属于我的CMS:(一)前言
一,项目背景 记得大学毕业课题,我就是选择做个CMS,不过当时虽然做了个,不过感觉不是很好,所以现在又重做了,顺便发上来供大家讨论的.虽然CMS不是什么特别的项目,但是还是想从一个普通项目学到更多的东 ...
- 从零开始实现ASP.NET Core MVC的插件式开发(六) - 如何加载插件引用
标题:从零开始实现ASP.NET Core MVC的插件式开发(六) - 如何加载插件引用. 作者:Lamond Lu 地址:https://www.cnblogs.com/lwqlun/p/1171 ...
- 从零开始编写一个vue插件
title: 从零开始编写一个vue插件 toc: true date: 2018-12-17 10:54:29 categories: Web tags: vue mathjax 写毕设的时候需要一 ...
- 从零开始编写IntelliJ IDEA插件
写Java代码的时候,经常会涉及到重复性的操作,这个时候就会想要是有这样一个插件就好了,如果是大家都会遇到的场景,IDE或许已经提供了,再不然也有可能有人编写了相关的插件.要是这个操作是你们的编码环境 ...
- 从零开始实现ASP.NET Core MVC的插件式开发(一) - 使用ApplicationPart动态加载控制器和视图
标题:从零开始实现ASP.NET Core MVC的插件式开发(一) - 使用Application Part动态加载控制器和视图 作者:Lamond Lu 地址:http://www.cnblogs ...
- 从零开始实现ASP.NET Core MVC的插件式开发(八) - Razor视图相关问题及解决方案
标题:从零开始实现ASP.NET Core MVC的插件式开发(八) - Razor视图相关问题及解决方案 作者:Lamond Lu 地址:https://www.cnblogs.com/lwqlun ...
- 从零开始编写自己的C#框架(1)——前言
记得十五年前自学编程时,拿着C语言厚厚的书,想要上机都不知道要用什么编译器来执行书中的例子.十二年前在大学自学ASP时,由于身边没有一位同学和朋友学习这种语言,也只能整天混在图收馆里拼命的啃书.而再后 ...
- 从零开始编写自己的C#框架(20)——框架异常处理及日志记录
最近很忙,杂事也多,所以开发本框架也是断断续续的,终于在前两天将前面设定的功能都基本完成了,剩下一些小功能遗漏的以后发现再补上.接下来的章节主要都是讲解在本框架的基础上进行开发的小巧. 本框架主要有四 ...
- 从零开始编写自己的C#框架(11)——创建解决方案
这段时间一直在充电,拜读了园子中大神们的博文(wayfarer的<设计之道>.TerryLee的<.NET设计模式系列文章>.卡奴达摩的<设计模式>还有其他一些零散 ...
随机推荐
- HTML和CSS经典布局6
如下图: 需求: 1. 如图 2. 可以从任意div标签开始. <!DOCTYPE html> <html xmlns="http://www.w3.org/1999/xh ...
- angularjs 指令详解 - template, restrict, replace
通过指令机制,angularjs 提供了一个强大的扩展系统,我们可以通过自定义指令来扩展自己的指令系统. 怎样定义自己的指令呢? 我们通过 Bootstrap UI来学习吧.这个项目使用 angula ...
- 关于新书《修炼之道:.NET开发要点精讲》的各种说明
索引 新书介绍 新书封面 新书目录 试读章节 原稿试读 网购地址 规格参数 反馈方式 一些感谢 附加说明 1.新书介绍 从2013年年底到2014年9月,历时将近10个月,这本书终于看到了“出版发行” ...
- Go语言实战 - revel框架教程之用户注册
用户注册.登录和注销是任何一个网站都必然会有的功能,可以说,这是重新造轮子做多的领域,每个做网站的人应该都做过很多遍.见微知著,从这么一个小功能其实就可以看到所使用的web框架中的大部分东西. 今天就 ...
- Chrome开发者工具不完全指南(四、性能进阶篇)
前言 Profiles面板功能的作用主要是监控网页中各种方法执行时间和内存的变化,简单来说它就是Timeline的数字化版本.它的功能选项卡不是很多(只有三个),操作起来比较前面的几块功能版本来说简单 ...
- webBrowser 加载网页
事件 webBrowser_DocumentCompleted private void webBrowser_DocumentCompleted(object sender, WebBrowserD ...
- 【转】oracle中in和exists的区别
原文地址:http://blog.itpub.net/7478833/viewspace-441043/ 感谢作者 in 和 exists区别 in 是把外表和内表作hash join,而ex ...
- Notes: select选择框
HTML选择框通过select标签创建,该元素是HTMLSelectElement的实例,拥有以下属性和方法: selectedIndex:选中项的索引 options:选择框的所有选项 add:向选 ...
- Android随笔之——按键长按事件onKeyLongPress
现在安卓手机实体键是越来越少了,但还是有的,恰好自己就碰上了:按键的长按事件...百度了一些博客,内容都基本上是完全一样的,虽然可以捕获到长按事件,但却会和正常的单击冲突.幸好最近开个VPN,goog ...
- [c++] Class
也是醉了,一个.h文件就有这么多细节问题: 初始化列表,使用{} 也可以. 类中的引用和const变量,必须立即在初始化列表中提前初始化. 常成员函数,const 放在函数后, 常成员函数即不能改变成 ...