layui/src/modules/util.js

327 lines
11 KiB
JavaScript
Raw Normal View History

2022-05-18 22:35:13 +08:00
/**
2021-05-08 06:31:19 +08:00
* util 工具组件
2022-05-18 22:35:13 +08:00
*/
2017-08-21 08:51:13 +08:00
layui.define('jquery', function(exports){
"use strict";
2022-08-29 12:30:46 +08:00
var $ = layui.$;
var hint = layui.hint();
2017-08-21 08:51:13 +08:00
2022-08-29 12:30:46 +08:00
// 外部接口
var util = {
// 固定块
2017-08-21 08:51:13 +08:00
fixbar: function(options){
2022-08-29 12:30:46 +08:00
var ELEM = 'layui-fixbar';
var $doc = $(document);
2017-08-21 08:51:13 +08:00
2022-08-29 12:30:46 +08:00
// 默认可选项
options = $.extend(true, {
target: 'body', // fixbar 的插入目标选择器
bars: [], // bar 信息
default: true, // 是否显示默认 bar
showHeight: 200, // 出现 top bar 的滚动条高度临界值
duration: 200 // top bar 等动画时长(毫秒)
2017-08-21 08:51:13 +08:00
}, options);
2022-08-29 12:30:46 +08:00
// 目标元素对象
2022-08-29 12:30:46 +08:00
var $target = $(options.target);
// 滚动条所在元素对象
var $scroll = options.scroll
? $(options.scroll)
: $(options.target === 'body' ? $doc : $target)
2022-08-29 12:30:46 +08:00
// 是否提供默认图标
if(options.default){
// 兼容旧版本的一些属性
if(options.bar1){
options.bars.push({
type: 'bar1',
icon: 'layui-icon-chat'
});
2017-08-21 08:51:13 +08:00
}
2022-08-29 12:30:46 +08:00
if(options.bar2){
options.bars.push({
type: 'bar2',
icon: 'layui-icon-help'
});
}
// 默认 top bar
options.bars.push({
type: 'top',
icon: 'layui-icon-top'
});
}
var elem = $('<ul>').addClass(ELEM);
var elemTopBar;
// 遍历生成 bars 节点
layui.each(options.bars, function(i, item){
var elemBar = $('<li class="layui-icon">');
// 设置 bar 相关属性
elemBar.addClass(item.icon).attr({
'lay-type': item.type,
'style': item.style || ('background-color: '+ options.bgcolor)
}).html(item.content);
// bar 点击事件
elemBar.on('click', function(){
var type = $(this).attr('lay-type');
if(type === 'top'){
(
options.target === 'body'
? $('html,body')
: $scroll
).animate({
2022-08-29 12:30:46 +08:00
scrollTop : 0
}, options.duration);
2022-08-29 12:30:46 +08:00
}
typeof options.click === 'function' && options.click.call(this, type);
});
// 自定义任意事件
if(layui.type(options.on) === 'object'){
layui.each(options.on, function(eventName, callback){
elemBar.on(eventName, function(){
var type = $(this).attr('lay-type');
typeof callback === 'function' && callback.call(this, type);
});
})
2017-08-21 08:51:13 +08:00
}
2022-08-29 12:30:46 +08:00
// 获得 top bar 节点
if(item.type === 'top'){
elemBar.addClass('layui-fixbar-top');
elemTopBar = elemBar;
}
elem.append(elemBar); // 插入 bar 节点
2017-08-21 08:51:13 +08:00
});
2022-08-29 12:30:46 +08:00
// 若目标元素已存在 fixbar则移除旧的节点
$target.find('.'+ ELEM).remove();
// 向目标元素插入 fixbar 节点
typeof options.css === 'object' && elem.css(options.css);
$target.append(elem);
// top bar 的显示隐藏
if(elemTopBar){
var lock;
var setTopBar = (function setTopBar(){
var top = $scroll.scrollTop();
2022-08-29 12:30:46 +08:00
if(top >= options.showHeight){
lock || (elemTopBar.show(), lock = 1);
} else {
lock && (elemTopBar.hide(), lock = 0);
}
return setTopBar;
})();
}
// 根据 scrollbar 设置 fixbar 相关状态
var timer;
$scroll.on('scroll', function(){
2022-08-29 12:30:46 +08:00
if(!setTopBar) return;
2017-08-21 08:51:13 +08:00
clearTimeout(timer);
timer = setTimeout(function(){
2022-08-29 12:30:46 +08:00
setTopBar();
2017-08-21 08:51:13 +08:00
}, 100);
});
}
//倒计时
,countdown: function(endTime, serverTime, callback){
var that = this
,type = typeof serverTime === 'function'
,end = new Date(endTime).getTime()
,now = new Date((!serverTime || type) ? new Date().getTime() : serverTime).getTime()
,count = end - now
,time = [
Math.floor(count/(1000*60*60*24)) //天
,Math.floor(count/(1000*60*60)) % 24 //时
,Math.floor(count/(1000*60)) % 60 //分
,Math.floor(count/1000) % 60 //秒
];
if(type) callback = serverTime;
var timer = setTimeout(function(){
that.countdown(endTime, now + 1000, callback);
}, 1000);
callback && callback(count > 0 ? time : [0,0,0,0], serverTime, timer);
if(count <= 0) clearTimeout(timer);
return timer;
}
//某个时间在当前时间的多久前
,timeAgo: function(time, onlyDate){
2017-11-15 11:57:51 +08:00
var that = this
,arr = [[], []]
2017-10-30 15:03:16 +08:00
,stamp = new Date().getTime() - new Date(time).getTime();
2017-08-21 08:51:13 +08:00
2017-10-30 15:03:16 +08:00
//返回具体日期
2020-01-15 06:30:00 +08:00
if(stamp > 1000*60*60*24*31){
2017-10-30 15:03:16 +08:00
stamp = new Date(time);
2017-11-15 11:57:51 +08:00
arr[0][0] = that.digit(stamp.getFullYear(), 4);
arr[0][1] = that.digit(stamp.getMonth() + 1);
arr[0][2] = that.digit(stamp.getDate());
2017-10-30 15:03:16 +08:00
//是否输出时间
if(!onlyDate){
2017-11-15 11:57:51 +08:00
arr[1][0] = that.digit(stamp.getHours());
arr[1][1] = that.digit(stamp.getMinutes());
arr[1][2] = that.digit(stamp.getSeconds());
2017-10-30 15:03:16 +08:00
}
return arr[0].join('-') + ' ' + arr[1].join(':');
2017-08-21 08:51:13 +08:00
}
//30天以内返回“多久前”
if(stamp >= 1000*60*60*24){
return ((stamp/1000/60/60/24)|0) + '天前';
} else if(stamp >= 1000*60*60){
return ((stamp/1000/60/60)|0) + '小时前';
2020-01-15 06:30:00 +08:00
} else if(stamp >= 1000*60*3){ //3分钟以内为刚刚
2017-08-21 08:51:13 +08:00
return ((stamp/1000/60)|0) + '分钟前';
} else if(stamp < 0){
return '未来';
} else {
return '刚刚';
}
}
2017-11-15 11:57:51 +08:00
//数字前置补零
2017-11-27 23:44:54 +08:00
,digit: function(num, length){
2017-11-15 11:57:51 +08:00
var str = '';
num = String(num);
length = length || 2;
for(var i = num.length; i < length; i++){
str += '0';
}
return num < Math.pow(10, length) ? str + (num|0) : num;
}
2017-11-27 23:44:54 +08:00
//转化为日期格式字符
,toDateString: function(time, format){
2021-05-08 06:31:19 +08:00
//若 null 或空字符,则返回空字符
if(time === null || time === '') return '';
2017-11-27 23:44:54 +08:00
var that = this
2021-05-08 06:31:19 +08:00
,date = new Date(function(){
if(!time) return;
return isNaN(time) ? time : (typeof time === 'string' ? parseInt(time) : time)
}() || new Date())
2017-11-27 23:44:54 +08:00
,ymd = [
that.digit(date.getFullYear(), 4)
,that.digit(date.getMonth() + 1)
,that.digit(date.getDate())
]
,hms = [
that.digit(date.getHours())
,that.digit(date.getMinutes())
,that.digit(date.getSeconds())
];
2021-05-08 06:31:19 +08:00
if(!date.getDate()) return hint.error('Invalid Msec for "util.toDateString(Msec)"'), '';
2017-11-27 23:44:54 +08:00
format = format || 'yyyy-MM-dd HH:mm:ss';
return format.replace(/yyyy/g, ymd[0])
.replace(/MM/g, ymd[1])
.replace(/dd/g, ymd[2])
.replace(/HH/g, hms[0])
.replace(/mm/g, hms[1])
.replace(/ss/g, hms[2]);
}
2018-05-28 18:50:37 +08:00
2022-05-27 07:28:25 +08:00
//转义 html
2018-05-28 18:50:37 +08:00
,escape: function(html){
2022-05-27 07:28:25 +08:00
var exp = /[<"'>]|&(?=#[a-zA-Z0-9]+)/g;
if(html === undefined || html === null) return '';
2022-05-18 22:35:13 +08:00
html += '';
2022-05-27 07:28:25 +08:00
if(!exp.test(html)) return html;
2022-05-18 22:35:13 +08:00
return html.replace(/&(?!#?[a-zA-Z0-9]+;)/g, '&amp;')
2018-05-28 18:50:37 +08:00
.replace(/</g, '&lt;').replace(/>/g, '&gt;')
.replace(/'/g, '&#39;').replace(/"/g, '&quot;');
}
2019-05-31 14:36:29 +08:00
2021-05-08 06:31:19 +08:00
//还原转义的 html
2022-05-22 08:17:44 +08:00
,unescape: function(html){
2022-05-18 22:35:13 +08:00
if(html === undefined || html === null) html = '';
html += '';
2022-05-27 07:28:25 +08:00
2022-05-18 22:35:13 +08:00
return html.replace(/\&amp;/g, '&')
2021-03-31 14:07:23 +08:00
.replace(/\&lt;/g, '<').replace(/\&gt;/g, '>')
2022-06-05 23:55:31 +08:00
.replace(/\&#39;/g, '\'').replace(/\&quot;/g, '"');
2021-03-31 14:07:23 +08:00
}
2021-05-08 06:31:19 +08:00
//让指定的元素保持在可视区域
,toVisibleArea: function(options){
options = $.extend({
margin: 160 //触发动作的边界值
,duration: 200 //动画持续毫秒数
,type: 'y' //触发方向x 水平、y 垂直
}, options);
if(!options.scrollElem[0] || !options.thisElem[0]) return;
var scrollElem = options.scrollElem //滚动元素
,thisElem = options.thisElem //目标元素
,vertical = options.type === 'y' //是否垂直方向
,SCROLL_NAME = vertical ? 'scrollTop' : 'scrollLeft' //滚动方法
,OFFSET_NAME = vertical ? 'top' : 'left' //坐标方式
,scrollValue = scrollElem[SCROLL_NAME]() //当前滚动距离
,size = scrollElem[vertical ? 'height' : 'width']() //滚动元素的尺寸
,scrollOffet = scrollElem.offset()[OFFSET_NAME] //滚动元素所处位置
,thisOffset = thisElem.offset()[OFFSET_NAME] - scrollOffet //目标元素当前的所在位置
,obj = {};
//边界满足条件
if(thisOffset > size - options.margin || thisOffset < options.margin){
obj[SCROLL_NAME] = thisOffset - size/2 + scrollValue
scrollElem.animate(obj, options.duration);
}
}
2019-05-31 14:36:29 +08:00
//批量事件
,event: function(attr, obj, eventType){
2020-01-15 06:30:00 +08:00
var _body = $('body');
eventType = eventType || 'click';
//记录事件回调集合
2019-05-31 14:36:29 +08:00
obj = util.event[attr] = $.extend(true, util.event[attr], obj) || {};
2020-01-15 06:30:00 +08:00
//清除委托事件
util.event.UTIL_EVENT_CALLBACK = util.event.UTIL_EVENT_CALLBACK || {};
_body.off(eventType, '*['+ attr +']', util.event.UTIL_EVENT_CALLBACK[attr])
//绑定委托事件
util.event.UTIL_EVENT_CALLBACK[attr] = function(){
2019-05-31 14:36:29 +08:00
var othis = $(this)
,key = othis.attr(attr);
2020-01-15 06:30:00 +08:00
(typeof obj[key] === 'function') && obj[key].call(this, othis);
};
//清除旧事件,绑定新事件
_body.on(eventType, '*['+ attr +']', util.event.UTIL_EVENT_CALLBACK[attr]);
return obj;
2019-05-31 14:36:29 +08:00
}
2017-08-21 08:51:13 +08:00
};
2022-05-18 22:35:13 +08:00
util.on = util.event;
2017-08-21 08:51:13 +08:00
2021-03-31 14:07:23 +08:00
// DOM 尺寸变化该创意来自http://benalman.com/projects/jquery-resize-plugin/
/*
2018-08-18 20:46:14 +08:00
!function(a,b,c){"$:nomunge";function l(){f=b[g](function(){d.each(function(){var b=a(this),c=b.width(),d=b.height(),e=a.data(this,i);(c!==e.w||d!==e.h)&&b.trigger(h,[e.w=c,e.h=d])}),l()},e[j])}var f,d=a([]),e=a.resize=a.extend(a.resize,{}),g="setTimeout",h="resize",i=h+"-special-event",j="delay",k="throttleWindow";e[j]=250,e[k]=!0,a.event.special[h]={setup:function(){if(!e[k]&&this[g])return!1;var b=a(this);d=d.add(b),a.data(this,i,{w:b.width(),h:b.height()}),1===d.length&&l()},teardown:function(){if(!e[k]&&this[g])return!1;var b=a(this);d=d.not(b),b.removeData(i),d.length||clearTimeout(f)},add:function(b){function f(b,e,f){var g=a(this),h=a.data(this,i)||{};h.w=e!==c?e:g.width(),h.h=f!==c?f:g.height(),d.apply(this,arguments)}if(!e[k]&&this[g])return!1;var d;return a.isFunction(b)?(d=b,f):(d=b.handler,b.handler=f,void 0)}}}($,window);
2021-03-31 14:07:23 +08:00
*/
2018-08-18 20:46:14 +08:00
2022-05-27 07:28:25 +08:00
// 输出接口
2017-08-21 08:51:13 +08:00
exports('util', util);
});