406 lines
14 KiB
JavaScript
406 lines
14 KiB
JavaScript
|
||
/**
|
||
* 判断是否是JSON对象
|
||
* @param {*} obj
|
||
* @returns
|
||
*/
|
||
function isPlainObject(obj){
|
||
if (typeof obj !== 'object' || obj === null) return false;
|
||
var proto = Object.getPrototypeOf(obj);
|
||
if (proto === null) return true;
|
||
var baseProto = proto;
|
||
|
||
while (Object.getPrototypeOf(baseProto) !== null) {
|
||
baseProto = Object.getPrototypeOf(baseProto);
|
||
}
|
||
return proto === baseProto;
|
||
}
|
||
|
||
/**
|
||
* 判断值是否是一个数字
|
||
* @param {*} value
|
||
* @returns
|
||
*/
|
||
function isNumber(value){
|
||
if(value==undefined) return false
|
||
if(typeof(value)=='number') return true
|
||
if(typeof(value)!='string') return false
|
||
try{
|
||
if(value.includes(".")){
|
||
let v = parseFloat(value)
|
||
if(value.endsWith(".")){
|
||
return !isNaN(v) && String(v).length===value.length-1
|
||
}else{
|
||
return !isNaN(v) && String(v).length===value.length
|
||
}
|
||
}else{
|
||
let v = parseInt(value)
|
||
return !isNaN(v) && String(v).length===value.length
|
||
}
|
||
}catch{
|
||
return false
|
||
}
|
||
}
|
||
function isFunction(fn){
|
||
return typeof fn === "function"
|
||
}
|
||
/**
|
||
* 当value= null || undefined || "" || [] || {} 时返回true
|
||
* @param {*} value
|
||
* @returns
|
||
*/
|
||
function isNothing(value){
|
||
if(["boolean","function"].includes(typeof(value))) return false
|
||
if(value=="") return true
|
||
if(value==undefined) return true
|
||
if(Array.isArray(value) && value.length==0) return true
|
||
if(typeof(value)=="object" && Object.keys(value).length==0) return true
|
||
return false
|
||
}
|
||
|
||
/**
|
||
* 深度合并对象
|
||
*
|
||
* 注意:
|
||
* - 不会对数组成员进行再次遍历
|
||
* - 不能处理循环引入
|
||
*
|
||
* @param {*} toObj
|
||
* @param {*} formObj
|
||
* @param {*} options
|
||
* array : 数组合并策略,0-替换,1-合并,2-去重合并
|
||
* mixin : 是否采用混入方式来,=false, 则会创建新对象并返回
|
||
*/
|
||
function deepMerge(toObj,formObj,options={}){
|
||
let results = options.mixin ? toObj : Object.assign({},toObj)
|
||
Object.entries(formObj).forEach(([key,value])=>{
|
||
if(key in results){
|
||
if(typeof value === "object" && value !== null){
|
||
if(Array.isArray(value)){
|
||
if(options.array === 1){//合并
|
||
results[key] = [...results[key],...value]
|
||
}else if(options.array === 2){//去重合并
|
||
results[key] = [...new Set([...results[key],...value])]
|
||
}else{ //默认: 替换
|
||
results[key] = value
|
||
}
|
||
}else{
|
||
results[key] = deepMerge(results[key],value,options)
|
||
}
|
||
}else{
|
||
results[key] = value
|
||
}
|
||
}else{
|
||
results[key] = value
|
||
}
|
||
})
|
||
return results
|
||
}
|
||
|
||
function deepMixin(toObj,formObj,options={}){
|
||
return deepMerge(toObj,formObj,{...options,mixin:true})
|
||
}
|
||
|
||
/**
|
||
* 获取指定变量类型名称
|
||
* getDataTypeName(1) == Number
|
||
* getDataTypeName("") == String
|
||
* getDataTypeName(null) == Null
|
||
* getDataTypeName(undefined) == Undefined
|
||
* getDataTypeName(new Date()) == Date
|
||
* getDataTypeName(new Error()) == Error
|
||
*
|
||
* @param {*} v
|
||
* @returns
|
||
*/
|
||
function getDataTypeName(v){
|
||
if (v === null) return 'Null'
|
||
if (v === undefined) return 'Undefined'
|
||
if(typeof(v)==="function") return "Function"
|
||
return v.constructor && v.constructor.name;
|
||
};
|
||
/**
|
||
* 格式化日期
|
||
* 将值转换为Date类型
|
||
* @param {*} value
|
||
*/
|
||
function toDate(value) {
|
||
try {
|
||
return value instanceof Date ? value : new Date(value)
|
||
} catch {
|
||
return parseInt(value)
|
||
}
|
||
}
|
||
/**
|
||
* 转换为数字类型
|
||
*/
|
||
function toNumber(value,defualt=0) {
|
||
try {
|
||
if (isNumber(value)) {
|
||
return parseFloat(value)
|
||
} else {
|
||
return defualt
|
||
}
|
||
} catch {
|
||
return value
|
||
}
|
||
}
|
||
/**
|
||
* 转换为货币格式
|
||
*
|
||
* @param {*} value 可以是数字也可以是字符串
|
||
* @param {*} division 分割符号位数,3代表每3个数字添加一个,号
|
||
* @param {*} prefix 前缀
|
||
* @param {*} suffix 后缀
|
||
* @param {*} precision 小数点精确到几位,0-自动
|
||
* @param {*} format 格式模块板字符串
|
||
* @returns
|
||
*/
|
||
function toCurrency(value,params={}){
|
||
let {symbol="",division=3,prefix="",precision=2,suffix="",unit=0,unitName="",radix=3,format="{symbol}{value}{unit}"} = params
|
||
|
||
// 1. 分离出整数和小数部分
|
||
let [wholeDigits,decimalDigits] = String(value).split(".")
|
||
// 2. 转换数制单位 比如将元转换到万元单位
|
||
// 如果指定了unit单位,0-代表默认,1-N代表将小数点字向后移动radix*unit位
|
||
// 比如 123456789.88
|
||
// 当unit=1,radix=3时, == [123456,78988] // [整数,小数]
|
||
// 当unit=2,radix=3时, == [123,45678988] // [整数,小数]
|
||
if(unit>0 && radix>0){
|
||
// 不足位数时补零
|
||
if(wholeDigits.length<radix*unit) wholeDigits = new Array(radix*unit-wholeDigits.length+1).fill(0).join("")+ wholeDigits
|
||
// 将整数的最后radix*unit字符移到小数部分前面
|
||
decimalDigits=wholeDigits.substring(wholeDigits,wholeDigits.length-radix*unit)+decimalDigits
|
||
wholeDigits = wholeDigits.substring(0,wholeDigits.length-radix*unit)
|
||
if(wholeDigits=="") wholeDigits = "0"
|
||
}
|
||
|
||
// 3. 添加分割符号
|
||
let result = []
|
||
for(let i=0;i<wholeDigits.length;i++){
|
||
if(((wholeDigits.length - i) % division)==0 && i>0) result.push(",")
|
||
result.push(wholeDigits[i])
|
||
}
|
||
// 4. 处理保留小数位数,即精度
|
||
if(decimalDigits){
|
||
// 如果precision是一个数字,则进行四舍五入处理
|
||
if(isNumber(precision) && precision>0){// 四舍五入处理
|
||
let finalBits = decimalDigits.length // 四舍五入前的位数
|
||
decimalDigits = String(parseFloat(`0.${decimalDigits}`).toFixed(precision)).split(".")[1]
|
||
//如果经过四舍五入处理后的位数小于,代表精度进行舍去,则未尾显示+符号
|
||
if(finalBits > decimalDigits.length) decimalDigits+="+"
|
||
}
|
||
result.push(`.${decimalDigits}`)
|
||
}
|
||
result = result.join("")
|
||
// 5. 模板替换
|
||
result = format.replace("{value}",result)
|
||
.replace("{symbol}",symbol)
|
||
.replace("{prefix}",prefix)
|
||
.replace("{suffix}",suffix)
|
||
.replace("{unit}",unitName)
|
||
return result
|
||
}
|
||
|
||
/**
|
||
* 根据路径获取指定值
|
||
* 只支持简单的.分割路径
|
||
* getByPath({a:{b:1}},"a.b") == 1
|
||
* getByPath({a:{b:1}},"a.c",2) == 2
|
||
*
|
||
* @param {*} obj
|
||
* @param {*} path 使用.分割的路径
|
||
* @param {*} defaultValue 默认值
|
||
* @returns
|
||
*/
|
||
function getByPath(obj,path,defaultValue){
|
||
if(typeof(obj)!="object" || typeof(path)!="string") return defaultValue
|
||
let paths = path.split(".")
|
||
let cur = obj
|
||
for(let key of paths){
|
||
if(typeof(cur)=="object" && (key in cur) ){
|
||
cur = cur[key]
|
||
}else{
|
||
return defaultValue
|
||
}
|
||
}
|
||
return cur
|
||
}
|
||
function deepClone(obj){
|
||
if(obj==undefined) return obj
|
||
if (['string',"number","boolean","function","undefined"].includes(typeof(obj))){
|
||
return obj
|
||
}else if(Array.isArray(obj)){
|
||
return obj.map(item => deepClone(item))
|
||
}else if(typeof(obj)=="object"){
|
||
let results = {}
|
||
Object.entries(obj).forEach(([key,value])=>{
|
||
results[key] = deepClone(value)
|
||
})
|
||
return results
|
||
}else{
|
||
return obj
|
||
}
|
||
}
|
||
|
||
|
||
// YY 18 年,两位数
|
||
// YYYY 2018 年,四位数
|
||
// M 1-12 月,从1开始
|
||
// MM 01-12 月,两位数字
|
||
// MMM Jan-Dec 月,英文缩写
|
||
// D 1-31 日
|
||
// DD 01-31 日,两位数
|
||
// H 0-23 24小时
|
||
// HH 00-23 24小时,两位数
|
||
// h 1-12 12小时
|
||
// hh 01-12 12小时,两位数
|
||
// m 0-59 分钟
|
||
// mm 00-59 分钟,两位数
|
||
// s 0-59 秒
|
||
// ss 00-59 秒,两位数
|
||
// S 0-9 毫秒(百),一位数
|
||
// SS 00-99 毫秒(十),两位数
|
||
// SSS 000-999 毫秒,三位数
|
||
// Z -05:00 UTC偏移
|
||
// ZZ -0500 UTC偏移,两位数
|
||
// A AM / PM 上/下午,大写
|
||
// a am / pm 上/下午,小写
|
||
// Do 1st... 31st 月份的日期与序号
|
||
|
||
function formatDatetime(value,templ="YYYY/MM/DD HH:mm:ss"){
|
||
const v = toDate(value)
|
||
const year =String(v.getFullYear()),month = String(v.getMonth()+1),weekday=String(v.getDay()),day=String(v.getDate())
|
||
const hourNum = v.getHours()
|
||
const hour = String(hourNum), minute = String(v.getMinutes()),second = String(v.getSeconds()),millisecond=String(v.getMilliseconds())
|
||
const vars = [
|
||
["YYYY", year], // 2018 年,四位数
|
||
["YY", year.substring(year.length - 2, year.length)], // 18 年,两位数
|
||
["MMM", ""], // Jan-Dec 月,英文缩写
|
||
["MM", month.padStart(2, "0")], // 01-12 月,两位数字
|
||
["M", month], // 1-12 月,从1开始
|
||
["DD", day.padStart(2, "0")], // 01-31 日,两位数
|
||
["D", day], // 1-31 日
|
||
["HH", hour.padStart(2, "0")], // 00-23 24小时,两位数
|
||
["H", hour], // 0-23 24小时
|
||
["hh", String(hourNum > 12 ? hourNum - 12 : hourNum).padStart(2, "0")], // 01-12 12小时,两位数
|
||
["h", String(hourNum > 12 ? hourNum - 12 : hourNum)], // 1-12 12小时
|
||
["mm", minute.padStart(2, "0")], // 00-59 分钟,两位数
|
||
["m", minute], // 0-59 分钟
|
||
["ss", second.padStart(2, "0")], // 00-59 秒,两位数
|
||
["s", second], // 0-59 秒
|
||
["SSS", millisecond], // 000-999 毫秒,三位数
|
||
["A", hour > 12 ? "PM" : "AM"], // AM / PM 上/下午,大写
|
||
["a", hour > 12 ? "pm" : "am"], // am / pm 上/下午,小写
|
||
]
|
||
let result = templ
|
||
vars.forEach(([k,v])=>result = replaceAll(result,k,v))
|
||
return result
|
||
}
|
||
|
||
function formatTime(value,templ="HH:mm:ss"){
|
||
const v = toDate(value)
|
||
const hourNum = v.getHours()
|
||
const hour = String(hourNum),minute = String(v.getMinutes()),second = String(v.getSeconds()),millisecond=String(v.getMilliseconds())
|
||
let result = templ
|
||
const vars = [
|
||
["HH", hour.padStart(2, "0")], // 00-23 24小时,两位数
|
||
["H", hour], // 0-23 24小时
|
||
["hh", String(hour > 12 ? hour - 12 : hour).padStart(2, "0")], // 01-12 12小时,两位数
|
||
["h", String(hour > 12 ? hour - 12 : hour)], // 1-12 12小时
|
||
["mm", minute.padStart(2, "0")], // 00-59 分钟,两位数
|
||
["m", minute], // 0-59 分钟
|
||
["ss", second.padStart(2, "0")], // 00-59 秒,两位数
|
||
["s", second], // 0-59 秒
|
||
["SSS", millisecond], // 000-999 毫秒,三位数
|
||
["A", hour > 12 ? "PM" : "AM"], // AM / PM 上/下午,大写
|
||
["a", hour > 12 ? "pm" : "am"] // am / pm 上/下午,小写
|
||
]
|
||
vars.forEach(([k,v])=>result = replaceAll(result,k,v))
|
||
return result
|
||
}
|
||
/**
|
||
* 替换所有字符串
|
||
* 低版本ES未提供replaceAll,此函数用来替代
|
||
* @param {*} str
|
||
* @param {*} findValue
|
||
* @param {*} replaceValue
|
||
*/
|
||
function replaceAll(str,findValue,replaceValue){
|
||
if(typeof(str)!=="string" || findValue=="" || findValue==replaceValue) return str
|
||
try{
|
||
return str.replace(new RegExp(escapeRegexpStr(findValue),"g"),replaceValue)
|
||
}catch{}
|
||
return str
|
||
}
|
||
/**
|
||
* 使用正则表达式解析非标JOSN
|
||
*
|
||
*/
|
||
|
||
const bastardJsonKeyRegex = /(?<value>(?<=:\s*)(\'.*?\')+)|(?<key>(([\w\u4e00-\u9fa5])|(\'.*?\'))+(?=\s*\:))/g
|
||
|
||
/**
|
||
* 当需要采用正则表达式进行字符串替换时,需要对字符串进行转义
|
||
*
|
||
* 比如 str = "I am {username}"
|
||
* replace(new RegExp(str),"Tom") !=== I am Tom
|
||
*
|
||
* 因为{}是正则表达式元字符,需要转义成 "\{username\}"
|
||
*
|
||
* replace(new RegExp(escapeRegexpStr(str)),"Tom")
|
||
*
|
||
*
|
||
* @param {*} str
|
||
* @returns
|
||
*/
|
||
function escapeRegexpStr(str){
|
||
return str.replace(/([.?*+^$[\]\\(){}|-])/g, "\\$1")
|
||
}
|
||
/**
|
||
* 解析非标的JSON字符串为{}
|
||
* 非标的JSON字符串指的是:
|
||
* - key没有使用使用""包裹
|
||
* - 字符串value没有使用""包裹
|
||
*
|
||
* @param {*} str
|
||
* @returns
|
||
*/
|
||
function safeParseJson(str){
|
||
let matched;
|
||
while ((matched = bastardJsonKeyRegex.exec(str)) !== null) {
|
||
if (matched.index === bastardJsonKeyRegex.lastIndex) {
|
||
bastardJsonKeyRegex.lastIndex++;
|
||
}
|
||
let item = matched[0]
|
||
if(item.startsWith("'") && item.endsWith("'")){
|
||
item = item.substring(1,item.length-1)
|
||
}
|
||
const findValue = matched.groups.key ? new RegExp( escapeRegexpStr(matched[0]) + "\s*:") : new RegExp(":\s*" + escapeRegexpStr(matched[0]))
|
||
const replaceTo = matched.groups.key ? `"${item}":` : `: "${item}"`
|
||
str = str.replace(findValue,replaceTo)
|
||
}
|
||
return JSON.parse(str)
|
||
}
|
||
const DataTypes = ["String","Number","Boolean","Object","Array","Function","Error","Symbol","RegExp","Date","Null","Undefined","Set","Map","WeakSet","WeakMap"]
|
||
|
||
module.exports ={
|
||
DataTypes,
|
||
isPlainObject,
|
||
isFunction,
|
||
isNumber,
|
||
isNothing,
|
||
deepClone,
|
||
deepMerge,
|
||
deepMixin,
|
||
replaceAll,
|
||
getByPath,
|
||
getDataTypeName,
|
||
formatDatetime,
|
||
formatTime,
|
||
toDate,
|
||
toNumber,
|
||
toCurrency,
|
||
escapeRegexpStr,
|
||
safeParseJson
|
||
} |