// +----------------------------------------------------------------------
// | ThinkAdmin
// +----------------------------------------------------------------------
// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
// +----------------------------------------------------------------------
// | 官方网站: http://demo.thinkadmin.top
// +----------------------------------------------------------------------
// | 开源协议 ( https://mit-license.org )
// +----------------------------------------------------------------------
// | gitee 代码仓库:https://gitee.com/zoujingli/ThinkAdmin
// | github 代码仓库:https://github.com/zoujingli/ThinkAdmin
// +----------------------------------------------------------------------
// Layui & jQuery
if (typeof jQuery === 'undefined') window.$ = window.jQuery = layui.$;
window.form = layui.form, window.layer = layui.layer, window.laydate = layui.laydate;
// 资源URL目录
window.baseRoot = (function (src) {
src = document.scripts[document.scripts.length - 1].src;
return src.substring(0, src.lastIndexOf("/") + 1);
})();
// require 配置参数
require.config({
waitSeconds: 60,
baseUrl: baseRoot,
map: {'*': {css: baseRoot + 'plugs/require/css.js'}},
paths: {
'md5': ['plugs/jquery/md5.min'],
'json': ['plugs/jquery/json.min'],
'michat': ['plugs/michat/michat'],
'upload': ['plugs/jquery/uploader'],
'base64': ['plugs/jquery/base64.min'],
'echarts': ['plugs/echarts/echarts.min'],
'angular': ['plugs/angular/angular.min'],
'ckeditor': ['plugs/ckeditor/ckeditor'],
'websocket': ['plugs/socket/websocket'],
'pcasunzips': ['plugs/jquery/pcasunzips'],
'jquery.ztree': ['plugs/ztree/ztree.all.min'],
'jquery.masonry': ['plugs/jquery/masonry.min'],
'jquery.autocompleter': ['plugs/jquery/autocompleter.min'],
},
shim: {
'websocket': {deps: [baseRoot + 'plugs/socket/swfobject.min.js']},
'jquery.ztree': {deps: ['jquery', 'css!' + baseRoot + 'plugs/ztree/zTreeStyle/zTreeStyle.css']},
'jquery.autocompleter': {deps: ['jquery', 'css!' + baseRoot + 'plugs/jquery/autocompleter.css']},
}
});
// 注册jquery到require模块
define('jquery', [], function () {
return layui.$;
});
$(function () {
window.$body = $('body');
/*! 消息组件实例 */
$.msg = new function (that) {
that = this;
this.idx = [];
this.shade = [0.02, '#000'];
// 关闭消息框
this.close = function (index) {
return layer.close(index);
};
// 弹出警告框
this.alert = function (msg, callback) {
var index = layer.alert(msg, {end: callback, scrollbar: false});
return this.idx.push(index), index;
};
// 确认对话框
this.confirm = function (msg, ok, no) {
var index = layer.confirm(msg, {title: '操作确认', btn: ['确认', '取消']}, function () {
typeof ok === 'function' && ok.call(this, index);
}, function () {
typeof no === 'function' && no.call(this, index);
that.close(index);
});
return index;
};
// 显示成功类型的消息
this.success = function (msg, time, callback) {
var index = layer.msg(msg, {icon: 1, shade: this.shade, scrollbar: false, end: callback, time: (time || 2) * 1000, shadeClose: true});
return this.idx.push(index), index;
};
// 显示失败类型的消息
this.error = function (msg, time, callback) {
var index = layer.msg(msg, {icon: 2, shade: this.shade, scrollbar: false, time: (time || 3) * 1000, end: callback, shadeClose: true});
return this.idx.push(index), index;
};
// 状态消息提示
this.tips = function (msg, time, callback) {
var index = layer.msg(msg, {time: (time || 3) * 1000, shade: this.shade, end: callback, shadeClose: true});
return this.idx.push(index), index;
};
// 显示正在加载中的提示
this.loading = function (msg, callback) {
var index = msg ? layer.msg(msg, {icon: 16, scrollbar: false, shade: this.shade, time: 0, end: callback}) : layer.load(2, {time: 0, scrollbar: false, shade: this.shade, end: callback});
return this.idx.push(index), index;
};
// 自动处理显示Think返回的Json数据
this.auto = function (ret, time) {
var url = ret.url || (typeof ret.data === 'string' ? ret.data : '');
var msg = ret.msg || (typeof ret.info === 'string' ? ret.info : '');
if (parseInt(ret.code) === 1 && time === 'false') {
return url ? (window.location.href = url) : $.form.reload();
}
return (parseInt(ret.code) === 1) ? this.success(msg, time, function () {
url ? (window.location.href = url) : $.form.reload();
for (var i in that.idx) layer.close(that.idx[i]);
that.idx = [];
}) : this.error(msg, 3, function () {
url ? window.location.href = url : '';
});
};
};
/*! 表单自动化组件 */
$.form = new function (that) {
that = this;
// 内容区选择器
this.selecter = '.layui-layout-admin>.layui-body';
// 刷新当前页面
this.reload = function () {
window.onhashchange.call(this);
};
// 内容区域动态加载后初始化
this.reInit = function ($dom) {
$.vali.listen(this);
$dom = $dom || $(this.selecter);
$dom.find('[required]').map(function ($parent) {
if (($parent = $(this).parent()) && $parent.is('label')) {
$parent.addClass('label-required-prev');
} else {
$parent.prevAll('label').addClass('label-required-next');
}
});
$dom.find('input[data-date-range]').map(function () {
this.setAttribute('autocomplete', 'off');
laydate.render({
type: this.getAttribute('data-date-range') || 'date',
range: true, elem: this, done: function (value) {
$(this.elem).val(value).trigger('change');
}
});
});
$dom.find('input[data-date-input]').map(function () {
this.setAttribute('autocomplete', 'off');
laydate.render({
type: this.getAttribute('data-date-input') || 'date',
range: false, elem: this, done: function (value) {
$(this.elem).val(value).trigger('change');
}
});
});
$dom.find('[data-file]:not([data-inited])').map(function (index, elem, $this, field) {
$this = $(elem), field = $this.attr('data-field') || 'file';
if (!$this.data('input')) $this.data('input', $('[name="' + field + '"]').get(0));
$this.uploadFile(function (url) {
$($this.data('input')).val(url).trigger('change');
});
});
};
// 在内容区显示视图
this.show = function (html) {
$(this.selecter).html(html);
this.reInit($(this.selecter));
setTimeout(function () {
that.reInit($(that.selecter));
}, 1000);
};
// 以HASH打开新网页
this.href = function (url, obj) {
if (url !== '#') {
window.location.href = '#' + $.menu.parseUri(url, obj);
} else if (obj && obj.getAttribute('data-menu-node')) {
$('[data-menu-node^="' + obj.getAttribute('data-menu-node') + '-"][data-open!="#"]:first').trigger('click');
}
};
// 异步加载的数据
this.load = function (url, data, method, callback, loading, tips, time, headers) {
var index = loading !== false ? $.msg.loading(tips) : 0;
if (typeof data === 'object' && typeof data['_csrf_'] === 'string') {
headers = headers || {};
headers['User-Token-Csrf'] = data['_csrf_'];
delete data['_csrf_'];
}
$.ajax({
data: data || {}, type: method || 'GET', url: $.menu.parseUri(url), beforeSend: function (xhr) {
if (typeof Pace === 'object') Pace.restart();
if (typeof headers === 'object') for (var i in headers) xhr.setRequestHeader(i, headers[i]);
}, error: function (XMLHttpRequest) {
if (XMLHttpRequest.responseText.indexOf('exception') > -1) layer.open({
title: XMLHttpRequest.status + ' - ' + XMLHttpRequest.statusText, type: 2,
area: '800px', content: 'javascript:void(0)', success: function ($element, index) {
try {
layer.full(index);
$element.find('iframe')[0].contentWindow.document.write(XMLHttpRequest.responseText);
$element.find('.layui-layer-setwin').css({right: '35px', top: '28px'}).find('a').css({marginLeft: 0});
$element.find('.layui-layer-title').css({color: 'red', height: '70px', lineHeight: '70px', fontSize: '22px', textAlign: 'center', fontWeight: 700});
} catch (e) {
layer.close(index);
}
}
});
if (parseInt(XMLHttpRequest.status) === 200) {
this.success(XMLHttpRequest.responseText);
} else {
$.msg.tips('E' + XMLHttpRequest.status + ' - 服务器繁忙,请稍候再试!');
}
}, success: function (ret) {
if (typeof callback === 'function' && callback.call(that, ret) === false) return false;
return typeof ret === 'object' ? $.msg.auto(ret, time || ret.wait || undefined) : that.show(ret);
}, complete: function () {
$.msg.close(index);
}
});
};
// 加载HTML到目标位置
this.open = function (url, data, callback, loading, tips) {
this.load(url, data, 'get', function (ret) {
return (typeof ret === 'object' ? $.msg.auto(ret) : that.show(ret)), false;
}, loading, tips);
};
// 打开一个iframe窗口
this.iframe = function (url, title, area) {
return layer.open({title: title || '窗口', type: 2, area: area || ['800px', '580px'], fix: true, maxmin: false, content: url});
};
// 加载HTML到弹出层
this.modal = function (url, data, title, callback, loading, tips) {
this.load(url, data, 'GET', function (res, index) {
if (typeof (res) === 'object') return $.msg.auto(res), false;
index = layer.open({
type: 1, btn: false, area: "800px", content: res, title: title || '', success: function (dom, index) {
$(dom).find('[data-close]').off('click').on('click', function () {
if ($(this).attr('data-confirm')) return $.msg.confirm($(this).attr('data-confirm'), function (_index) {
layer.close(_index), layer.close(index);
}), false;
layer.close(index);
});
$.form.reInit($(dom));
}
});
$.msg.idx.push(index);
return (typeof callback === 'function') && callback.call(that);
}, loading, tips);
};
};
/*! 后台菜单辅助插件 */
$.menu = new function (that) {
that = this;
// 计算URL地址中有效的URI
this.getUri = function (uri) {
uri = uri || window.location.href;
uri = (uri.indexOf(window.location.host) > -1 ? uri.split(window.location.host)[1] : uri);
return (uri.indexOf('#') > -1 ? uri.split('#')[1] : uri).split('?')[0];
};
// 通过URI查询最有可能的菜单NODE
this.queryNode = function (url, node) {
node = node || location.href.replace(/.*spm=([\d\-m]+).*/ig, '$1');
if (!/^m-/.test(node)) {
var $menu = $('[data-menu-node][data-open*="' + url.replace(/\.html$/ig, '') + '"]');
return $menu.size() ? $menu.get(0).getAttribute('data-menu-node') : '';
}
return node;
};
// URL转URI
this.parseUri = function (uri, obj) {
var params = {};
if (uri.indexOf('?') > -1) {
var attrs = uri.split('?')[1].split('&');
for (var i in attrs) if (attrs[i].indexOf('=') > -1) {
var tmp = attrs[i].split('=').slice();
if (typeof tmp[0] === 'string' && tmp[0].length > 0) {
params[tmp[0]] = decodeURIComponent(tmp[1].replace(/%2B/ig, '%20'));
}
}
}
uri = this.getUri(uri);
if (typeof params.spm !== 'string') {
params.spm = obj && obj.getAttribute('data-menu-node') || this.queryNode(uri);
}
if (typeof params.spm !== 'string' || params.spm.length < 1) delete params.spm;
// 生成新的 URL 参数
var attrs = [];
for (var i in params) attrs.push([i, params[i]].join('='));
var query = '?' + attrs.join('&');
return uri + (query === '?' ? '' : query);
};
// 后台菜单动作初始化
this.listen = function () {
// 菜单模式切换
(function ($menu, miniClass) {
// Mini 菜单模式切换及显示
if (layui.data('admin-menu-type')['type-min']) $menu.addClass(miniClass);
$body.on('click', '[data-target-menu-type]', function () {
$menu.toggleClass(miniClass);
layui.data('admin-menu-type', {key: 'type-min', value: $menu.hasClass(miniClass)});
}).on('resize', function () {
if ($body.width() > 1000) {
layui.data('admin-menu-type')['type-min'] ? $menu.addClass(miniClass) : $menu.removeClass(miniClass);
} else {
$menu.addClass(miniClass);
}
}).trigger('resize');
// Mini 菜单模式时TIPS文字显示
$('[data-target-tips]').mouseenter(function () {
if ($menu.hasClass(miniClass)) {
$(this).attr('index', layer.tips($(this).attr('data-target-tips') || '', this));
}
}).mouseleave(function () {
layer.close($(this).attr('index'));
});
})($('.layui-layout-admin'), 'layui-layout-left-mini');
// 左则二级菜单展示
$('[data-submenu-layout]>a').on('click', function () {
that.syncOpenStatus(1);
});
// 同步二级菜单展示状态
this.syncOpenStatus = function (mode) {
$('[data-submenu-layout]').map(function (node) {
node = $(this).attr('data-submenu-layout');
if (mode === 1) layui.data('menu', {key: node, value: $(this).hasClass('layui-nav-itemed') ? 2 : 1});
else if ((layui.data('menu')[node] || 2) === 2) $(this).addClass('layui-nav-itemed');
});
};
window.onhashchange = function (hash, node) {
hash = window.location.hash || '';
if (hash.length < 1) return $('[data-menu-node][data-open!="#"]:first').trigger('click');
$.form.load(hash), that.syncOpenStatus(2);
// 菜单选择切换
node = that.queryNode(that.getUri());
if (/^m-/.test(node)) {
var $all = $('a[data-menu-node]').parent(), tmp = node.split('-'), tmpNode = tmp.shift();
while (tmp.length > 0) {
tmpNode = tmpNode + '-' + tmp.shift();
$all = $all.not($('a[data-menu-node="' + tmpNode + '"]').parent().addClass('layui-this'));
}
$all.removeClass('layui-this');
// 菜单模式切换
if (node.split('-').length > 2) {
var _tmp = node.split('-'), _node = _tmp.shift() + '-' + _tmp.shift();
$('[data-menu-layout]').not($('[data-menu-layout="' + _node + '"]').removeClass('layui-hide')).addClass('layui-hide');
$('[data-menu-node="' + node + '"]').parent().parent().parent().addClass('layui-nav-itemed');
$('.layui-layout-admin').removeClass('layui-layout-left-hide');
} else $('.layui-layout-admin').addClass('layui-layout-left-hide');
that.syncOpenStatus(1);
}
};
// URI初始化动作
window.onhashchange.call(this);
};
};
/*! 注册对象到Jq */
$.vali = function (form, callback, options) {
return (new function (that) {
that = this;
// 表单元素
this.tags = 'input,textarea,select';
// 检测元素事件
this.checkEvent = {change: true, blur: true, keyup: false};
// 去除字符串两头的空格
this.trim = function (str) {
return str.replace(/(^\s*)|(\s*$)/g, '');
};
// 标签元素是否可见
this.isVisible = function (ele) {
return $(ele).is(':visible');
};
// 检测属性是否有定义
this.hasProp = function (ele, prop) {
if (typeof prop !== "string") return false;
var attrProp = ele.getAttribute(prop);
return (typeof attrProp !== 'undefined' && attrProp !== null && attrProp !== false);
};
// 判断表单元素是否为空
this.isEmpty = function (ele, value) {
var trim = this.trim(ele.value);
value = value || ele.getAttribute('placeholder');
return (trim === "" || trim === value);
};
// 正则验证表单元素
this.isRegex = function (ele, regex, params) {
var input = $(ele).val(), real = this.trim(input);
regex = regex || ele.getAttribute('pattern');
if (real === "" || !regex) return true;
return new RegExp(regex, params || 'i').test(real);
};
// 检侧所的表单元素
this.checkAllInput = function () {
var isPass = true;
$(form).find(this.tags).each(function () {
if (that.checkInput(this) === false) return $(this).focus(), isPass = false;
});
return isPass;
};
// 检测表单单元
this.checkInput = function (input) {
var tag = input.tagName.toLowerCase(), need = this.hasProp(input, "required");
var type = (input.getAttribute("type") || '').replace(/\W+/, "").toLowerCase();
if (this.hasProp(input, 'data-auto-none')) return true;
var ingoreTags = ['select'], ingoreType = ['radio', 'checkbox', 'submit', 'reset', 'image', 'file', 'hidden'];
for (var i in ingoreTags) if (tag === ingoreTags[i]) return true;
for (var i in ingoreType) if (type === ingoreType[i]) return true;
if (need && this.isEmpty(input)) return this.remind(input);
return this.isRegex(input) ? (this.hideError(input), true) : this.remind(input);
};
// 验证标志
this.remind = function (input) {
if (!this.isVisible(input)) return true;
this.showError(input, input.getAttribute('title') || input.getAttribute('placeholder') || '输入错误');
return false;
};
// 错误消息显示
this.showError = function (ele, content) {
$(ele).addClass('validate-error'), this.insertError(ele);
$($(ele).data('input-info')).addClass('layui-anim layui-anim-fadein').css({width: 'auto'}).html(content);
};
// 错误消息消除
this.hideError = function (ele) {
$(ele).removeClass('validate-error'), this.insertError(ele);
$($(ele).data('input-info')).removeClass('layui-anim-fadein').css({width: '30px'}).html('');
};
// 错误消息标签插入
this.insertError = function (ele) {
var $html = $('');
$html.css({top: $(ele).position().top + 'px', paddingBottom: $(ele).css('paddingBottom'), lineHeight: $(ele).css('height')});
$(ele).data('input-info') || $(ele).data('input-info', $html.insertAfter(ele));
};
// 表单验证入口
this.check = function (form, callback) {
$(form).attr("novalidate", "novalidate");
$(form).find(that.tags).map(function () {
this.bindEventMethod = function () {
that.checkInput(this);
};
for (var e in that.checkEvent) if (that.checkEvent[e] === true) {
$(this).off(e, this.bindEventMethod).on(e, this.bindEventMethod);
}
});
$(form).bind("submit", function (event) {
if (that.checkAllInput() && typeof callback === 'function') {
if (typeof CKEDITOR === 'object' && typeof CKEDITOR.instances === 'object') {
for (var i in CKEDITOR.instances) CKEDITOR.instances[i].updateElement();
}
callback.call(this, $(form).formToJson());
}
return event.preventDefault(), false;
});
$(form).find('[data-form-loaded]').map(function () {
$(this).html(this.getAttribute('data-form-loaded') || this.innerHTML);
$(this).removeAttr('data-form-loaded').removeClass('layui-disabled');
});
return $(form).data('validate', this);
};
}).check(form, callback, options);
};
/*! 自动监听规则内表单 */
$.vali.listen = function () {
$('form[data-auto]').map(function () {
if ($(this).attr('data-listen') !== 'true') $(this).attr('data-listen', 'true').vali(function (data) {
var call = $(this).attr('data-callback') || '_default_callback';
var type = this.getAttribute('method') || 'POST', tips = this.getAttribute('data-tips') || undefined;
var time = this.getAttribute('data-time') || undefined, href = this.getAttribute('action') || window.location.href;
$.form.load(href, data, type, window[call] || undefined, true, tips, time);
});
});
};
/*! 注册对象到JqFn */
$.fn.vali = function (callback, options) {
return $.vali(this, callback, options);
};
/*! 表单转JSON */
$.fn.formToJson = function () {
var self = this, data = {}, pushCounters = {};
var patterns = {"key": /[a-zA-Z0-9_]+|(?=\[\])/g, "push": /^$/, "fixed": /^\d+$/, "named": /^[a-zA-Z0-9_]+$/};
this.build = function (base, key, value) {
base[key] = value;
return base;
};
this.pushCounter = function (name) {
if (pushCounters[name] === undefined) pushCounters[name] = 0;
return pushCounters[name]++;
};
$.each($(this).serializeArray(), function () {
var key, keys = this.name.match(patterns.key), merge = this.value, name = this.name;
while ((key = keys.pop()) !== undefined) {
name = name.replace(new RegExp("\\[" + key + "\\]$"), '');
if (key.match(patterns.push)) { // push
merge = self.build([], self.pushCounter(name), merge);
} else if (key.match(patterns.fixed)) { // fixed
merge = self.build([], key, merge);
} else if (key.match(patterns.named)) { // named
merge = self.build({}, key, merge);
}
}
data = $.extend(true, data, merge);
});
return data;
};
/*! 全局文件上传入口 */
$.fn.uploadFile = function (callback) {
if (this.attr('data-inited')) return false;
var that = this, mode = $(this).attr('data-file') || 'one';
this.attr('data-inited', true).attr('data-multiple', (mode !== 'btn' && mode !== 'one') ? 1 : 0);
setTimeout(function () {
require(['upload'], function (apply) {
apply(that, null, callback);
});
},500)
};
/*! 上传单个图视频 */
$.fn.uploadOneVideo = function () {
var name = $(this).attr('name') || 'video_link', type = $(this).data('type') || 'mp4';
if(type == 'mp3'){
var $tpl = $('