从零开始编写属于我的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设计模式系列文章>.卡奴达摩的<设计模式>还有其他一些零散 ...
随机推荐
- Jexus 支持PHP的三种方式
Jexus不仅支持ASP.NET,而且能够通个自带的PHP-FCGI服务以及PHP-FPM等方式灵活支持PHP而且还可以以.NET(Phalanger)方式支持PHP. PHP-FCGI服务支持PHP ...
- Kosaraju 算法查找强连通分支
有向图 G = (V, E) 的一个强连通分支(SCC:Strongly Connected Components)是一个最大的顶点集合 C,C 是 V 的子集,对于 C 中的每一对顶点 u 和 v, ...
- iOS开发系列--让你的应用“动”起来
--iOS核心动画 概览 在iOS中随处都可以看到绚丽的动画效果,实现这些动画的过程并不复杂,今天将带大家一窥iOS动画全貌.在这里你可以看到iOS中如何使用图层精简非交互式绘图,如何通过核心动画创建 ...
- mysql定义和调用存储过程
/*定义delimiter为 // */ delimiter // CREATE procedure sp_add3(a int, b int,out c int) begin set c=a+ b; ...
- .NET项目版本号的小随笔
[题外话] 一直以来都对.NET项目中的几个版本号(AssemblyVersion.AssemblyFileVersion.AssemblyInformationalVersion)以及版本号中的Re ...
- Hystrix框架5--请求缓存和collapser
简介 在Hystrix中有个Request的概念,有一些操作需要在request中进行 缓存 在Hystrix调用服务时,如果只是查询接口,可以使用缓存进行优化,从而跳过真实访问请求. 应用 需要启用 ...
- 使用批处理设置JDK环境变量(Win7可用,新版本)
欢迎探讨,如有错误敬请指正 如需转载,请注明出处http://www.cnblogs.com/nullzx/ 1. JDK环境的设置 一般情况下来说按照网上大多数的教程设置JDK的环境变量即可.但对于 ...
- Sql Server系列:自定义函数
用户自定义函数可以像系统函数一样在查询或存储过程中调用,可以接受参数.执行操作并将操作结果以值的形式返回.返回值可以是单个标量或结果集. 1. 标量函数 标量函数返回一个确定类型的标量值,对于多语句的 ...
- OpenCascade Eigenvalues and Eigenvectors of Square Matrix
OpenCascade Eigenvalues and Eigenvectors of Square Matrix eryar@163.com Abstract. OpenCascade use th ...
- 为 Neutron 准备物理基础设施(II) - 每天5分钟玩转 OpenStack(76)
本节将按照上一节的规划安装配置控制节点和计算节点. 控制节点 devstack-controller 步骤如下 安装 Ubuntu 14.04 此处省略 256 个字 配置网卡 编辑 /etc/net ...