COMCMS v0.9 版本发布,带前后端的一个响应式企业站
前言:踏入十二月,人生也即将进入下一个阶段。
最近忙于其他,代码也是偶尔更新。目前算是0.9的版本,就是基本上可以完成一个简单的企业站/博客的功能。
主要特点:前台完整演示:文章、产品、留言。界面响应式:响应PC和手机
预计:1.0版本的话,带有app端(预计只有安卓版本,苹果版本没有证书-_-')演示,也算是可以应用于app、小程序(微信小程序、支付宝、百度小程序)。当然只是预告而已ORZ
github地址:https://github.com/hogenwang/comcms_core
讨论Q群:1600800
本次最重要的是,加上前台的完整前端。一方面,让整个小cms系统更加完整。另外一方面,也是演示了,如何搭配前端view进行开发。还有就是XCode如是使用。
之前有人提到,希望有多一点的二次开发文档。这个我尽量抽时间写写,其实主要还是XCode的用法。
话不多说,先看整体的效果吧:

以下就简单说一下,前台的一些特点和二次开发注意的内容:
1、前台控制器继承:HomeBaseController 主要是:
提供了后台配置,和前后端交互的json基础类JsonTip
所有需要展示网站完整页面的需要加入:
ViewBag.cfg = cfg;
2、ServerController 为数据交互提交方法,可以将所有需要交互的数据,方法都写在本控制器,当然,你也可以写在其他的。
而一般,提交的方法,都使用POST外加:AutoValidateAntiforgeryToken 保证数据提交的安全,最后,统一都返回一个json:
/// <summary>
/// 系统JSON提示实体类
/// </summary>
[Serializable]
public class JsonTip
{
public JsonTip() { }
/// <summary>
/// 成功状态
/// </summary>
public static readonly string SUCCESS = "success";
/// <summary>
/// 失败状态
/// </summary>
public static readonly string ERROR = "error";
/// <summary>
/// 请求返回状态 默认 error(错误);成功:success
/// </summary>
public string Status { get; set; } = ERROR;
/// <summary>
/// 提示信息
/// </summary>
public string Message { get; set; }
/// <summary>
/// 其他信息
/// </summary>
public string Other { get; set; }
/// <summary>
/// Id 可选
/// </summary>
public int Id { get; set; } = ;
/// <summary>
/// 返回URL
/// </summary>
public string ReturnUrl { get; set; } /// <summary>
/// 获取JSON字符串
/// </summary>
/// <param name="json">JsonTip实体类</param>
/// <returns></returns>
public static string GetJsonString(JsonTip json)
{
if (json != null)
return JsonConvert.SerializeObject(json);
else
return string.Empty;
}
}
如演示里面的提交留言:
#region 提交前台底部留言
[HttpPost]
[AutoValidateAntiforgeryToken]
public IActionResult DoPostMessage(string name,string phone,string content)
{
if (string.IsNullOrWhiteSpace(name))
{
tip.Message = "请填写您的姓名!";
return Json(tip);
}
if (!Utils.IsTel(phone))
{
tip.Message = "请填写正确的手机号码!";
return Json(tip);
}
if (string.IsNullOrWhiteSpace(content))
{
tip.Message = "请填写您的留言内容!";
return Json(tip);
} Guestbook entity = new Guestbook()
{
AddTime = DateTime.Now,
IP = Utils.GetIP(),
KId = ,
Nickname = name,
Content = Utils.NoHTML(content),
Tel = phone
};
entity.Insert(); tip.Message = "留言成功,感谢您的留言!";
tip.Status = JsonTip.SUCCESS;
return Json(tip);
}
#endregion
可以看到,涉及到XCode的用法也是非常简单的,new一个对象后,最后调用Insert();就可以插入数据了。值得一提的是,如果插入这个数据后,想获取主键Id的值:entity.Id就可以了。不要用Insert() 返回值。
3、为了方便演示,我大部分获取数据都写在views里面,建议还是写在Controller里面。如首页模板
IList<Link> listLinks = Link.FindAll(Link._.KId == , Link._.Sequence.Asc(), null, , );
//处理bannr广告
var bannerPC = Ads.GetSlideAds();
var bannerMobile = Ads.GetSlideAds();
4、后台文章,可以配置相应的模板,可以根据不同情况配置不同模板。如演示,关于我们、联系我们和新闻中心模板就不一样。
因为编译了模板,所以选择模板使用反射实现:
//获取模板 模板规则,以Index_开头的,为栏目列表,以Detial_开头的为文章详情
List<string> listTpls = new List<string>();
var asms = AppDomain.CurrentDomain.GetAssemblies();
foreach (var asmItem in asms)
{
var types = asmItem.GetTypes().Where(e => e.Name.StartsWith("Views_Article")).ToList();
if (types.Count == ) continue;
foreach (var type in types)
{
string viewName = type.Name.Replace("Views_Article_", "") + ".cshtml";
listTpls.Add(viewName);
}
}
ViewBag.ListTpl = listTpls;
注意看注释部分内容哦。
5、列表分页的,这里我使用了手动计算分页的形式:
IList<Article> list = new List<Article>();
int numPerPage, currentPage, startRowIndex;
numPerPage = model.PageSize;
if (page > 0)
currentPage = page;
else
currentPage = 1; startRowIndex = (currentPage - 1) * numPerPage; var ex = Article._.IsHide != 1 & Article._.IsDel != 1; //如果显示下级栏目文章
if (model.IsShowSubDetail == 1)
{
//获取下级所有栏目
List<int> allsubkids = new List<int>();
allsubkids.Add(model.Id); IList<ArticleCategory> allkind = ArticleCategory.FindAllSubKinds(model.Id);
if (allkind != null && allkind.Count > 0)
{
foreach (var s in allkind)
{
allsubkids.Add(s.Id);
}
}
ex &= Article._.KId.In(allsubkids);
}
else
ex &= Article._.KId == model.Id;
long totalCount = Article.FindCount(ex, Article._.Sequence.Asc().And(Article._.Id.Desc()), null, startRowIndex, numPerPage);
int pageCount = (int)totalCount / numPerPage;//总页数
if (totalCount % numPerPage > 0)
{
pageCount += 1;
}
list = Article.FindAll(ex, Article._.Sequence.Asc().And(Article._.Id.Desc()), null, startRowIndex, numPerPage);
ViewBag.list = list;//列表
//分页信息
ViewBag.totalCount = totalCount;
ViewBag.pageCount = pageCount;
ViewBag.page = page;
ViewBag.pagesize = numPerPage;
而views里面调用分页代码:
@Html.Raw(Pages.GetPageStr(ViewBag.pagesize, (int)ViewBag.totalCount, ViewBag.page, $"/Article/Index/{Model.Id}" + "?page={0}", , , ""))
6、后台权限使用方法
由于上面提到统一执行post返回一个标准的json。所以后台权限,目前设计验证两种,一种是普通的验证,错误返回提示页面,另外一种是提交验证,错误返回一个json。也就是POST提交的时候。
用法也是很简单,给方法名加上:
[MyAuthorize( "viewlist", "articlecategory")]
[MyAuthorize( "add", "articlecategory", "JSON")]
其中第一个参数为事件权限,在后台:后台权限》事件权限管理管理

第二个参数为页面的key。这个在后台栏目权限管理中设置:

这样,在二次开发中,就可以很快速的添加功能和权限控制了。
其他方面暂不一一列出。如果有什么不明白的,可以给我留言。
也欢迎大家点个star。
最后再次提醒,讨论Q群:1600800
COMCMS v0.9 版本发布,带前后端的一个响应式企业站的更多相关文章
- Jeecg-Boot 2.0.1 版本发布,前后端分离快速开发平台
Jeecg-Boot项目简介 Jeecg-boot 是一款基于代码生成器的快速开发平台! 采用前后端分离技术:SpringBoot,Mybatis,Shiro,JWT,Vue & Ant De ...
- 由浅入深,带你用JavaScript实现响应式原理(Vue2、Vue3响应式原理)
由浅入深,带你用JavaScript实现响应式原理 前言 为什么前端框架Vue能够做到响应式?当依赖数据发生变化时,会对页面进行自动更新,其原理还是在于对响应式数据的获取和设置进行了监听,一旦监听到数 ...
- django开发最完美手机购物商城APP带前后端源码
后端和数据接口,全采用django开发 从0到大神的进阶之路 一句话,放弃单文件引用vue.js练手的学习方式 马上从vue-cli4练手,要不然,学几年,你也不懂组件式开发,不懂VUEX,不懂路由, ...
- ajax向Django前后端提交请求和CSRF跨站请求伪造
1.ajax登录示例 urls.py from django.conf.urls import url from django.contrib import admin from app01 impo ...
- (转)也谈基于NodeJS的全栈式开发(基于NodeJS的前后端分离)
原文链接:http://ued.taobao.org/blog/2014/04/full-stack-development-with-nodejs/ 随着不同终端(pad/mobile/pc)的兴起 ...
- 也谈基于NodeJS的全栈式开发(基于NodeJS的前后端分离)
前言 为了解决传统Web开发模式带来的各种问题,我们进行了许多尝试,但由于前/后端的物理鸿沟,尝试的方案都大同小异.痛定思痛,今天我们重新思考了“前后端”的定义,引入前端同学都熟悉的NodeJS,试图 ...
- vue菜鸟从业记:公司项目里如何进行前后端接口联调
最近我的朋友王小闰进入一家新的公司,正好公司项目采用的是前后端分离架构,技术栈是王小闰非常熟悉的vue全家桶,后端用的是Java语言. 在前后端开发人员碰面之后,协商确定好了前端需要的数据接口(扯那么 ...
- 基于NodeJS的全栈式开发(基于NodeJS的前后端分离)
也谈基于NodeJS的全栈式开发(基于NodeJS的前后端分离) 前言 为了解决传统Web开发模式带来的各种问题,我们进行了许多尝试,但由于前/后端的物理鸿沟,尝试的方案都大同小异.痛定思痛,今天我们 ...
- SSM框架中的前后端分离
认识前后端分离 在传统的web应用开发中,大多数的程序员会将浏览器作为前后端的分界线.将浏览器中为用户进行页面展示的部分称之为前端,而将运行在服务器,为前端提供业务逻辑和数据准备的所有代码统称为后端. ...
随机推荐
- springboot部分常用注解
目录:[持续更新.....] spring 部分常用注解 spring boot 学习之路1(简单入门) spring boot 学习之路2(注解介绍) spring boot 学习之路3( 集成my ...
- jQuery如何判断input元素是否获得焦点(点击编辑时)
问题提出 如果你要判断input元素是否获得焦点,或者是否处在活动编辑状态,使用jQuery的 hasFocus() 方法或 is(':focus') 方法貌似都无效!搜索网上给出的办法,几乎净是采用 ...
- OID的编解码(即在报文中的体现)
先上干货: 我们常见到OID的地方是SNMP和MIB,实际上理论上所有对象都可以有自己的ID.已存在的ID可以在http://www.oid-info.com/查到.这些ID在报文里并非字符串或直接的 ...
- Win10更新
Turn: https://m.uczzd.cn/webview/news?app=meizubrw-iflow&aid=11529477703533248224&cid=100&am ...
- MyBatis笔记----报错:Exception in thread "main" org.apache.ibatis.binding.BindingException: Invalid bound statement (not found)解决方法
报错 Exception in thread "main" org.apache.ibatis.binding.BindingException: Invalid bound st ...
- Oracle EBS FORM lov
存在一种情况: 一个LOV的值当前有效,因此填入保存.但突然无效后,当查询该界面时就会弹出LOV框使其修改. 解决方案: 1. 非常粗暴,不设置校验,在LOV对应的item强行将校验设置为NO. 2. ...
- web前端(11)—— 页面布局1
要说页面布局的话,那就必须说说margin,padding,和background.这三个属性其实都是前面讲过的,这里还是再次讲解以下,为什么呢?因为是这样的,光靠前面的css样式来设置,你很可能会遇 ...
- web前端(9)—— CSS属性
属性 终于到css属性,前面就零零散散的用了什么color,font-size之类,本篇博文就专项的介绍它了 字体属性 font-family 此属性是设置字体样式的,比如微软雅黑,方正书体,华文宋体 ...
- 洗礼灵魂,修炼python(81)--全栈项目实战篇(9)—— 购物商城登录验证系统
都在线购物过吧?那么你应该体验过,当没有登录账户时,点开购物车,个人中心,收藏物品等的操作时,都会直接跳转到登录账户的界面,然后如果登录一次后就不用再登录,直到用户登出. 是的,本次项目就是做一个登录 ...
- SQL SERVER启动步骤
第一步 从注册表读取SQL SERVER启动信息 (1)Audit Level:设置SQL SERVER是否记录用户登陆信息 Login Mode:设置SQL SERVER登陆类型是只接受windo ...