xe-utils.umd.js 103 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329233023312332233323342335233623372338233923402341234223432344234523462347234823492350235123522353235423552356235723582359236023612362236323642365236623672368236923702371237223732374237523762377237823792380238123822383238423852386238723882389239023912392239323942395239623972398239924002401240224032404240524062407240824092410241124122413241424152416241724182419242024212422242324242425242624272428242924302431243224332434243524362437243824392440244124422443244424452446244724482449245024512452245324542455245624572458245924602461246224632464246524662467246824692470247124722473247424752476247724782479248024812482248324842485248624872488248924902491249224932494249524962497249824992500250125022503250425052506250725082509251025112512251325142515251625172518251925202521252225232524252525262527252825292530253125322533253425352536253725382539254025412542254325442545254625472548254925502551255225532554255525562557255825592560256125622563256425652566256725682569257025712572257325742575257625772578257925802581258225832584258525862587258825892590259125922593259425952596259725982599260026012602260326042605260626072608260926102611261226132614261526162617261826192620262126222623262426252626262726282629263026312632263326342635263626372638263926402641264226432644264526462647264826492650265126522653265426552656265726582659266026612662266326642665266626672668266926702671267226732674267526762677267826792680268126822683268426852686268726882689269026912692269326942695269626972698269927002701270227032704270527062707270827092710271127122713271427152716271727182719272027212722272327242725272627272728272927302731273227332734273527362737273827392740274127422743274427452746274727482749275027512752275327542755275627572758275927602761276227632764276527662767276827692770277127722773277427752776277727782779278027812782278327842785278627872788278927902791279227932794279527962797279827992800280128022803280428052806280728082809281028112812281328142815281628172818281928202821282228232824282528262827282828292830283128322833283428352836283728382839284028412842284328442845284628472848284928502851285228532854285528562857285828592860286128622863286428652866286728682869287028712872287328742875287628772878287928802881288228832884288528862887288828892890289128922893289428952896289728982899290029012902290329042905290629072908290929102911291229132914291529162917291829192920292129222923292429252926292729282929293029312932293329342935293629372938293929402941294229432944294529462947294829492950295129522953295429552956295729582959296029612962296329642965296629672968296929702971297229732974297529762977297829792980298129822983298429852986298729882989299029912992299329942995299629972998299930003001300230033004300530063007300830093010301130123013301430153016301730183019302030213022302330243025302630273028302930303031303230333034303530363037303830393040304130423043304430453046304730483049305030513052305330543055305630573058305930603061306230633064306530663067306830693070307130723073307430753076307730783079308030813082308330843085308630873088308930903091309230933094309530963097309830993100310131023103310431053106310731083109311031113112311331143115311631173118311931203121312231233124312531263127312831293130313131323133313431353136313731383139314031413142314331443145314631473148314931503151315231533154315531563157315831593160316131623163316431653166316731683169317031713172317331743175317631773178317931803181318231833184318531863187318831893190319131923193319431953196319731983199320032013202320332043205320632073208320932103211321232133214321532163217321832193220322132223223322432253226322732283229323032313232323332343235323632373238323932403241324232433244324532463247324832493250325132523253325432553256325732583259326032613262326332643265326632673268326932703271327232733274327532763277327832793280328132823283328432853286328732883289329032913292329332943295329632973298329933003301330233033304330533063307330833093310331133123313331433153316331733183319332033213322332333243325332633273328332933303331333233333334333533363337333833393340334133423343334433453346334733483349335033513352335333543355335633573358335933603361336233633364336533663367336833693370337133723373337433753376337733783379338033813382338333843385338633873388338933903391339233933394339533963397339833993400340134023403340434053406340734083409341034113412341334143415341634173418341934203421342234233424342534263427342834293430343134323433343434353436343734383439344034413442344334443445344634473448344934503451345234533454345534563457345834593460346134623463346434653466346734683469347034713472347334743475347634773478347934803481348234833484348534863487348834893490349134923493349434953496349734983499350035013502350335043505350635073508350935103511351235133514351535163517351835193520352135223523352435253526352735283529353035313532353335343535353635373538353935403541354235433544354535463547354835493550355135523553355435553556355735583559356035613562356335643565356635673568356935703571357235733574357535763577357835793580358135823583358435853586358735883589359035913592359335943595359635973598359936003601360236033604360536063607360836093610361136123613361436153616361736183619362036213622362336243625362636273628362936303631363236333634363536363637363836393640364136423643364436453646364736483649365036513652365336543655365636573658365936603661366236633664366536663667366836693670367136723673367436753676367736783679368036813682368336843685368636873688368936903691369236933694369536963697369836993700370137023703370437053706370737083709371037113712371337143715371637173718371937203721372237233724372537263727372837293730373137323733373437353736373737383739374037413742374337443745374637473748374937503751375237533754375537563757375837593760376137623763376437653766376737683769377037713772377337743775377637773778377937803781378237833784378537863787378837893790379137923793379437953796379737983799380038013802380338043805380638073808380938103811381238133814381538163817381838193820382138223823382438253826382738283829383038313832383338343835383638373838383938403841384238433844384538463847384838493850385138523853385438553856385738583859386038613862386338643865386638673868386938703871387238733874387538763877387838793880388138823883388438853886388738883889389038913892389338943895389638973898389939003901390239033904390539063907390839093910391139123913391439153916391739183919392039213922392339243925392639273928392939303931393239333934393539363937393839393940394139423943394439453946394739483949395039513952395339543955395639573958395939603961396239633964396539663967396839693970397139723973397439753976397739783979398039813982398339843985398639873988398939903991399239933994399539963997399839994000400140024003400440054006400740084009401040114012401340144015401640174018401940204021402240234024402540264027402840294030403140324033403440354036403740384039404040414042404340444045404640474048404940504051405240534054405540564057405840594060406140624063406440654066406740684069407040714072407340744075407640774078407940804081408240834084408540864087408840894090409140924093409440954096409740984099410041014102410341044105410641074108410941104111411241134114411541164117411841194120412141224123412441254126412741284129413041314132413341344135413641374138413941404141414241434144414541464147414841494150415141524153415441554156415741584159416041614162416341644165416641674168416941704171417241734174417541764177417841794180418141824183418441854186418741884189419041914192419341944195419641974198419942004201420242034204420542064207420842094210421142124213421442154216421742184219422042214222422342244225422642274228422942304231423242334234423542364237423842394240424142424243424442454246424742484249425042514252425342544255425642574258425942604261426242634264426542664267426842694270427142724273427442754276427742784279428042814282428342844285428642874288428942904291429242934294429542964297429842994300430143024303430443054306430743084309431043114312431343144315431643174318431943204321432243234324432543264327432843294330433143324333433443354336433743384339434043414342434343444345434643474348434943504351435243534354435543564357435843594360436143624363436443654366436743684369437043714372437343744375437643774378437943804381438243834384438543864387438843894390439143924393439443954396439743984399440044014402440344044405440644074408440944104411441244134414441544164417441844194420442144224423442444254426442744284429443044314432443344344435443644374438443944404441444244434444444544464447444844494450445144524453445444554456445744584459446044614462446344644465446644674468446944704471447244734474447544764477447844794480448144824483448444854486448744884489449044914492449344944495449644974498449945004501450245034504450545064507450845094510451145124513451445154516451745184519452045214522452345244525452645274528452945304531453245334534453545364537453845394540454145424543454445454546454745484549455045514552455345544555455645574558455945604561456245634564456545664567456845694570457145724573457445754576457745784579458045814582458345844585458645874588458945904591459245934594459545964597
  1. /**
  2. * xe-utils.js v3.1.13
  3. * MIT License.
  4. * @preserve
  5. */
  6. (function (global, factory) {
  7. typeof exports === 'object' && typeof module !== 'undefined' ? module.exports = factory() :
  8. typeof define === 'function' && define.amd ? define(factory) :
  9. (global.XEUtils = factory());
  10. }(this, function () {'use strict'
  11. var formatString = 'yyyy-MM-dd HH:mm:ss'
  12. var setupDefaults = {
  13. treeOptions: {
  14. parentKey: 'parentId',
  15. key: 'id',
  16. children: 'children'
  17. },
  18. formatDate: formatString + '.SSSZ',
  19. formatString: formatString,
  20. dateDiffRules: [
  21. ['yyyy', 31536000000],
  22. ['MM', 2592000000],
  23. ['dd', 86400000],
  24. ['HH', 3600000],
  25. ['mm', 60000],
  26. ['ss', 1000],
  27. ['S', 0]
  28. ]
  29. }
  30. var XEUtils = function () {}
  31. function mixin () {
  32. arrayEach(arguments, function (methods) {
  33. each(methods, function (fn, name) {
  34. XEUtils[name] = isFunction(fn) ? function () {
  35. var result = fn.apply(XEUtils.$context, arguments)
  36. XEUtils.$context = null
  37. return result
  38. } : fn
  39. })
  40. })
  41. }
  42. function setup (options) {
  43. return assign(setupDefaults, options)
  44. }
  45. XEUtils.VERSION = '3.1.13'
  46. XEUtils.mixin = mixin
  47. XEUtils.setup = setup
  48. function helperCreateGetObjects (name, getIndex) {
  49. var proMethod = Object[name]
  50. return function (obj) {
  51. var result = []
  52. if (obj) {
  53. if (proMethod) {
  54. return proMethod(obj)
  55. }
  56. each(obj, getIndex > 1 ? function (key) {
  57. result.push(['' + key, obj[key]])
  58. } : function () {
  59. result.push(arguments[getIndex])
  60. })
  61. }
  62. return result
  63. }
  64. }
  65. function helperCreateIndexOf (name, callback) {
  66. return function (obj, val) {
  67. if (obj) {
  68. if (obj[name]) {
  69. return obj[name](val)
  70. }
  71. if (isString(obj) || isArray(obj)) {
  72. return callback(obj, val)
  73. }
  74. for (var key in obj) {
  75. if (hasOwnProp(obj, key)) {
  76. if (val === obj[key]) {
  77. return key
  78. }
  79. }
  80. }
  81. }
  82. return -1
  83. }
  84. }
  85. function helperCreateInInObjectString (type) {
  86. return function (obj) {
  87. return '[object ' + type + ']' === objectToString.call(obj)
  88. }
  89. }
  90. /* eslint-disable valid-typeof */
  91. function helperCreateInTypeof (type) {
  92. return function (obj) {
  93. return typeof obj === type
  94. }
  95. }
  96. function helperCreateIterateHandle (prop, useArray, restIndex, matchValue, defaultValue) {
  97. return function (obj, iterate, context) {
  98. if (obj && iterate) {
  99. if (prop && obj[prop]) {
  100. return obj[prop](iterate, context)
  101. } else {
  102. if (useArray && isArray(obj)) {
  103. for (var index = 0, len = obj.length; index < len; index++) {
  104. if (!!iterate.call(context, obj[index], index, obj) === matchValue) {
  105. return [true, false, index, obj[index]][restIndex]
  106. }
  107. }
  108. } else {
  109. for (var key in obj) {
  110. if (hasOwnProp(obj, key)) {
  111. if (!!iterate.call(context, obj[key], key, obj) === matchValue) {
  112. return [true, false, key, obj[key]][restIndex]
  113. }
  114. }
  115. }
  116. }
  117. }
  118. }
  119. return defaultValue
  120. }
  121. }
  122. function helperCreateiterateIndexOf (callback) {
  123. return function (obj, iterate, context) {
  124. if (obj && isFunction(iterate)) {
  125. if (isArray(obj) || isString(obj)) {
  126. return callback(obj, iterate, context)
  127. }
  128. for (var key in obj) {
  129. if (hasOwnProp(obj, key)) {
  130. if (iterate.call(context, obj[key], key, obj)) {
  131. return key
  132. }
  133. }
  134. }
  135. }
  136. return -1
  137. }
  138. }
  139. function helperCreateMathNumber(name) {
  140. return function (num, digits) {
  141. var numRest = toNumber(num)
  142. var rest = numRest
  143. if (numRest) {
  144. digits = digits >> 0
  145. var numStr = toNumberString(numRest)
  146. var nums = numStr.split('.')
  147. var intStr = nums[0]
  148. var floatStr = nums[1] || ''
  149. var fStr = floatStr.substring(0, digits + 1)
  150. var subRest = intStr + (fStr ? ('.' + fStr) : '')
  151. if (digits >= floatStr.length) {
  152. return toNumber(subRest)
  153. }
  154. subRest = numRest
  155. if (digits > 0) {
  156. var ratio = Math.pow(10, digits)
  157. rest = Math[name](subRest * ratio) / ratio
  158. } else {
  159. rest = Math[name](subRest)
  160. }
  161. }
  162. return rest
  163. }
  164. }
  165. function helperCreateMinMax (handle) {
  166. return function (arr, iterate) {
  167. if (arr && arr.length) {
  168. var rest, itemIndex
  169. arrayEach(arr, function (itemVal, index) {
  170. if (iterate) {
  171. itemVal = isFunction(iterate) ? iterate(itemVal, index, arr) : get(itemVal, iterate)
  172. }
  173. if (!eqNull(itemVal) && (eqNull(rest) || handle(rest, itemVal))) {
  174. itemIndex = index
  175. rest = itemVal
  176. }
  177. })
  178. return arr[itemIndex]
  179. }
  180. return rest
  181. }
  182. }
  183. function helperCreatePickOmit (case1, case2) {
  184. return function (obj, callback) {
  185. var item, index
  186. var rest = {}
  187. var result = []
  188. var context = this
  189. var args = arguments
  190. var len = args.length
  191. if (!isFunction(callback)) {
  192. for (index = 1; index < len; index++) {
  193. item = args[index]
  194. result.push.apply(result, isArray(item) ? item : [item])
  195. }
  196. callback = 0
  197. }
  198. each(obj, function (val, key) {
  199. if ((callback ? callback.call(context, val, key, obj) : findIndexOf(result, function (name) {
  200. return name === key
  201. }) > -1) ? case1 : case2) {
  202. rest[key] = val
  203. }
  204. })
  205. return rest
  206. }
  207. }
  208. function helperCreateToNumber (handle) {
  209. return function (str) {
  210. if (str) {
  211. var num = handle(str)
  212. if (!isNaN(num)) {
  213. return num
  214. }
  215. }
  216. return 0
  217. }
  218. }
  219. function helperCreateTreeFunc (handle) {
  220. return function (obj, iterate, options, context) {
  221. var opts = options || {}
  222. var optChildren = opts.children || 'children'
  223. return handle(null, obj, iterate, context, [], [], optChildren, opts)
  224. }
  225. }
  226. function helperDefaultCompare (v1, v2) {
  227. return v1 === v2
  228. }
  229. function helperDeleteProperty (obj, property) {
  230. try {
  231. delete obj[property]
  232. } catch (e) {
  233. obj[property] = undefined
  234. }
  235. }
  236. function helperEqualCompare (val1, val2, compare, func, key, obj1, obj2) {
  237. if (val1 === val2) {
  238. return true
  239. }
  240. if (val1 && val2 && !isNumber(val1) && !isNumber(val2) && !isString(val1) && !isString(val2)) {
  241. if (isRegExp(val1)) {
  242. return compare('' + val1, '' + val2, key, obj1, obj2)
  243. } if (isDate(val1) || isBoolean(val1)) {
  244. return compare(+val1, +val2, key, obj1, obj2)
  245. } else {
  246. var result, val1Keys, val2Keys
  247. var isObj1Arr = isArray(val1)
  248. var isObj2Arr = isArray(val2)
  249. if (isObj1Arr || isObj2Arr ? isObj1Arr && isObj2Arr : val1.constructor === val2.constructor) {
  250. val1Keys = keys(val1)
  251. val2Keys = keys(val2)
  252. if (func) {
  253. result = func(val1, val2, key)
  254. }
  255. if (val1Keys.length === val2Keys.length) {
  256. return isUndefined(result) ? every(val1Keys, function (key, index) {
  257. return key === val2Keys[index] && helperEqualCompare(val1[key], val2[val2Keys[index]], compare, func, isObj1Arr || isObj2Arr ? index : key, val1, val2)
  258. }) : !!result
  259. }
  260. return false
  261. }
  262. }
  263. }
  264. return compare(val1, val2, key, obj1, obj2)
  265. }
  266. function helperFormatEscaper (dataMap) {
  267. var replaceRegexp = new RegExp('(?:' + keys(dataMap).join('|') + ')', 'g')
  268. return function (str) {
  269. return toValueString(str).replace(replaceRegexp, function (match) {
  270. return dataMap[match]
  271. })
  272. }
  273. }
  274. function helperGetDateFullYear (date) {
  275. return date.getFullYear()
  276. }
  277. function helperGetDateMonth (date) {
  278. return date.getMonth()
  279. }
  280. function helperGetDateTime (date) {
  281. return date.getTime()
  282. }
  283. function helperGetHGSKeys (property) {
  284. // 以最快的方式判断数组,可忽略准确性
  285. return property ? (property.splice && property.join ? property : ('' + property).split('.')) : []
  286. }
  287. function helperGetLocatOrigin () {
  288. return staticLocation ? (staticLocation.origin || (staticLocation.protocol + '//' + staticLocation.host)) : ''
  289. }
  290. function helperGetUTCDateTime (dates) {
  291. return Date.UTC(dates[0], dates[1], dates[2], dates[3], dates[4], dates[5], dates[6])
  292. }
  293. function helperGetYMD (date) {
  294. return new Date(helperGetDateFullYear(date), helperGetDateMonth(date), date.getDate())
  295. }
  296. function helperGetYMDTime (date) {
  297. return helperGetDateTime(helperGetYMD(date))
  298. }
  299. function helperNewDate () {
  300. return new Date()
  301. }
  302. function helperNumberAdd (addend, augend) {
  303. var str1 = toNumberString(addend)
  304. var str2 = toNumberString(augend)
  305. var ratio = Math.pow(10, Math.max(helperNumberDecimal(str1), helperNumberDecimal(str2)))
  306. return (multiply(addend, ratio) + multiply(augend, ratio)) / ratio
  307. }
  308. function helperNumberDecimal (numStr) {
  309. return (numStr.split('.')[1] || '').length
  310. }
  311. function helperNumberDivide (divisor, dividend) {
  312. var str1 = toNumberString(divisor)
  313. var str2 = toNumberString(dividend)
  314. var divisorDecimal = helperNumberDecimal(str1)
  315. var dividendDecimal = helperNumberDecimal(str2)
  316. var powY = dividendDecimal - divisorDecimal
  317. var isMinus = powY < 0
  318. var multiplicand = Math.pow(10, isMinus ? Math.abs(powY) : powY)
  319. return multiply(str1.replace('.', '') / str2.replace('.', ''), isMinus ? 1 / multiplicand : multiplicand)
  320. }
  321. function helperNumberOffsetPoint (str, offsetIndex) {
  322. return str.substring(0, offsetIndex) + '.' + str.substring(offsetIndex, str.length)
  323. }
  324. function helperStringLowerCase (str) {
  325. return str.toLowerCase()
  326. }
  327. function helperStringRepeat (str, count) {
  328. if (str.repeat) {
  329. return str.repeat(count)
  330. }
  331. var list = isNaN(count) ? [] : new Array(staticParseInt(count))
  332. return list.join(str) + (list.length > 0 ? str : '')
  333. }
  334. function helperStringSubstring (str, start, end) {
  335. return str.substring(start, end)
  336. }
  337. function helperStringUpperCase (str) {
  338. return str.toUpperCase()
  339. }
  340. var staticStrUndefined = 'undefined'
  341. var staticStrLast = 'last'
  342. var staticStrFirst = 'first'
  343. var staticDayTime = 86400000
  344. var staticWeekTime = staticDayTime * 7
  345. /* eslint-disable valid-typeof */
  346. var staticLocation = typeof location === staticStrUndefined ? 0 : location
  347. /* eslint-disable valid-typeof */
  348. var staticWindow = typeof window === staticStrUndefined ? 0 : window
  349. /* eslint-disable valid-typeof */
  350. var staticDocument = typeof document === staticStrUndefined ? 0 : document
  351. var staticEncodeURIComponent = encodeURIComponent
  352. var staticDecodeURIComponent = decodeURIComponent
  353. var objectToString = Object.prototype.toString
  354. var staticParseInt = parseInt
  355. var staticEscapeMap = {
  356. '&': '&amp;',
  357. '<': '&lt;',
  358. '>': '&gt;',
  359. '"': '&quot;',
  360. "'": '&#x27;',
  361. '`': '&#x60;'
  362. }
  363. var staticHGKeyRE = /(.+)?\[(\d+)\]$/
  364. var objectAssignFns = Object.assign
  365. function handleAssign (destination, args, isClone) {
  366. var len = args.length
  367. for (var source, index = 1; index < len; index++) {
  368. source = args[index]
  369. arrayEach(keys(args[index]), isClone ? function (key) {
  370. destination[key] = clone(source[key], isClone)
  371. } : function (key) {
  372. destination[key] = source[key]
  373. })
  374. }
  375. return destination
  376. }
  377. /**
  378. * 将一个或多个源对象复制到目标对象中
  379. *
  380. * @param {Object} target 目标对象
  381. * @param {...Object}
  382. * @return {Boolean}
  383. */
  384. var assign = function (target) {
  385. if (target) {
  386. var args = arguments
  387. if (target === true) {
  388. if (args.length > 1) {
  389. target = isArray(target[1]) ? [] : {}
  390. return handleAssign(target, args, true)
  391. }
  392. } else {
  393. return objectAssignFns ? objectAssignFns.apply(Object, args) : handleAssign(target, args)
  394. }
  395. }
  396. return target
  397. }
  398. /**
  399. * 指定方法后的返回值组成的新对象
  400. *
  401. * @param {Object} obj 对象/数组
  402. * @param {Function} iterate(item, index, obj) 回调
  403. * @param {Object} context 上下文
  404. * @return {Object}
  405. */
  406. function objectMap (obj, iterate, context) {
  407. var result = {}
  408. if (obj) {
  409. if (iterate) {
  410. if (!isFunction(iterate)) {
  411. iterate = property(iterate)
  412. }
  413. each(obj, function (val, index) {
  414. result[index] = iterate.call(context, val, index, obj)
  415. })
  416. } else {
  417. return obj
  418. }
  419. }
  420. return result
  421. }
  422. function objectEach (obj, iterate, context) {
  423. if (obj) {
  424. for (var key in obj) {
  425. if (hasOwnProp(obj, key)) {
  426. iterate.call(context, obj[key], key, obj)
  427. }
  428. }
  429. }
  430. }
  431. function lastObjectEach (obj, iterate, context) {
  432. lastArrayEach(keys(obj), function (key) {
  433. iterate.call(context, obj[key], key, obj)
  434. })
  435. }
  436. function handleMerge (target, source) {
  437. if ((isPlainObject(target) && isPlainObject(source)) || (isArray(target) && isArray(source))) {
  438. each(source, function (obj, key) {
  439. target[key] = handleMerge(target[key], obj)
  440. })
  441. return target
  442. }
  443. return source
  444. }
  445. /**
  446. * 将一个或多个源对象合并到目标对象中
  447. *
  448. * @param {Object} target 目标对象
  449. * @param {...Object}
  450. * @return {Boolean}
  451. */
  452. var merge = function (target) {
  453. if (!target) {
  454. target = {}
  455. }
  456. var args = arguments
  457. var len = args.length
  458. for (var source, index = 1; index < len; index++) {
  459. source = args[index]
  460. if (source) {
  461. handleMerge(target, source)
  462. }
  463. }
  464. return target
  465. }
  466. /**
  467. * 数组去重
  468. *
  469. * @param {Array} array 数组
  470. * @return {Array}
  471. */
  472. function uniq (array) {
  473. var result = []
  474. each(array, function (value) {
  475. if (!includes(result, value)) {
  476. result.push(value)
  477. }
  478. })
  479. return result
  480. }
  481. /**
  482. * 将多个数的值返回唯一的并集数组
  483. *
  484. * @param {...Array} 数组
  485. * @return {Array}
  486. */
  487. function union () {
  488. var args = arguments
  489. var result = []
  490. var index = 0
  491. var len = args.length
  492. for (; index < len; index++) {
  493. result = result.concat(toArray(args[index]))
  494. }
  495. return uniq(result)
  496. }
  497. var sortBy = orderBy
  498. var ORDER_PROP_ASC = 'asc'
  499. var ORDER_PROP_DESC = 'desc'
  500. // function handleSort (v1, v2) {
  501. // return v1 > v2 ? 1 : -1
  502. // }
  503. // '' < 数字 < 字符 < null < undefined
  504. function handleSort (v1, v2) {
  505. if (isUndefined(v1)) {
  506. return 1
  507. }
  508. if (isNull(v1)) {
  509. return isUndefined(v2) ? -1 : 1
  510. }
  511. return v1 && v1.localeCompare ? v1.localeCompare(v2) : (v1 > v2 ? 1 : -1)
  512. }
  513. function buildMultiOrders (name, confs, compares) {
  514. return function (item1, item2) {
  515. var v1 = item1[name]
  516. var v2 = item2[name]
  517. if (v1 === v2) {
  518. return compares ? compares(item1, item2) : 0
  519. }
  520. return confs.order === ORDER_PROP_DESC ? handleSort(v2, v1) : handleSort(v1, v2)
  521. }
  522. }
  523. function getSortConfs (arr, list, fieldConfs, context) {
  524. var sortConfs = []
  525. fieldConfs = isArray(fieldConfs) ? fieldConfs : [fieldConfs]
  526. arrayEach(fieldConfs, function (handle, index) {
  527. if (handle) {
  528. var field = handle
  529. var order
  530. if (isArray(handle)) {
  531. field = handle[0]
  532. order = handle[1]
  533. } else if (isPlainObject(handle)) {
  534. field = handle.field
  535. order = handle.order
  536. }
  537. sortConfs.push({
  538. field: field,
  539. order: order || ORDER_PROP_ASC
  540. })
  541. arrayEach(list, isFunction(field) ? function (item, key) {
  542. item[index] = field.call(context, item.data, key, arr)
  543. } : function (item) {
  544. item[index] = field ? get(item.data, field) : item.data
  545. })
  546. }
  547. })
  548. return sortConfs
  549. }
  550. /**
  551. * 将数组进行排序
  552. *
  553. * @param {Array} arr 数组
  554. * @param {Function/String/Array} fieldConfs 方法或属性
  555. * @param {Object} context 上下文
  556. * @return {Array}
  557. */
  558. function orderBy (arr, fieldConfs, context) {
  559. if (arr) {
  560. if (eqNull(fieldConfs)) {
  561. return toArray(arr).sort(handleSort)
  562. }
  563. var compares
  564. var list = map(arr, function (item) {
  565. return { data: item }
  566. })
  567. var sortConfs = getSortConfs(arr, list, fieldConfs, context)
  568. var len = sortConfs.length - 1
  569. while (len >= 0) {
  570. compares = buildMultiOrders(len, sortConfs[len], compares)
  571. len--
  572. }
  573. if (compares) {
  574. list = list.sort(compares)
  575. }
  576. return map(list, property('data'))
  577. }
  578. return []
  579. }
  580. /**
  581. * 将一个数组随机打乱,返回一个新的数组
  582. *
  583. * @param {Array} array 数组
  584. * @return {Array}
  585. */
  586. function shuffle (array) {
  587. var index
  588. var result = []
  589. var list = values(array)
  590. var len = list.length - 1
  591. for (; len >= 0; len--) {
  592. index = len > 0 ? random(0, len) : 0
  593. result.push(list[index])
  594. list.splice(index, 1)
  595. }
  596. return result
  597. }
  598. /**
  599. * 从一个数组中随机返回几个元素
  600. *
  601. * @param {Array} array 数组
  602. * @param {Number} number 个数
  603. * @return {Array}
  604. */
  605. function sample (array, number) {
  606. var result = shuffle(array)
  607. if (arguments.length <= 1) {
  608. return result[0]
  609. }
  610. if (number < result.length) {
  611. result.length = number || 0
  612. }
  613. return result
  614. }
  615. /**
  616. * 对象中的值中的每一项运行给定函数,如果函数对任一项返回true,则返回true,否则返回false
  617. *
  618. * @param {Object} obj 对象/数组
  619. * @param {Function} iterate(item, index, obj) 回调
  620. * @param {Object} context 上下文
  621. * @return {Boolean}
  622. */
  623. var some = helperCreateIterateHandle('some', 1, 0, true, false)
  624. /**
  625. * 对象中的值中的每一项运行给定函数,如果该函数对每一项都返回true,则返回true,否则返回false
  626. *
  627. * @param {Object} obj 对象/数组
  628. * @param {Function} iterate(item, index, obj) 回调
  629. * @param {Object} context 上下文
  630. * @return {Boolean}
  631. */
  632. var every = helperCreateIterateHandle('every', 1, 1, false, true)
  633. /**
  634. * 裁剪 Arguments 或数组 array,从 start 位置开始到 end 结束,但不包括 end 本身的位置
  635. * @param {Array/Arguments} array 数组或Arguments
  636. * @param {Number} startIndex 开始索引
  637. * @param {Number} endIndex 结束索引
  638. */
  639. function slice (array, startIndex, endIndex) {
  640. var result = []
  641. var argsSize = arguments.length
  642. if (array) {
  643. startIndex = argsSize >= 2 ? toNumber(startIndex) : 0
  644. endIndex = argsSize >= 3 ? toNumber(endIndex) : array.length
  645. if (array.slice) {
  646. return array.slice(startIndex, endIndex)
  647. }
  648. for (; startIndex < endIndex; startIndex++) {
  649. result.push(array[startIndex])
  650. }
  651. }
  652. return result
  653. }
  654. /**
  655. * 根据回调过滤数据
  656. *
  657. * @param {Object} obj 对象/数组
  658. * @param {Function} iterate(item, index, obj) 回调
  659. * @param {Object} context 上下文
  660. * @return {Object}
  661. */
  662. function filter (obj, iterate, context) {
  663. var result = []
  664. if (obj && iterate) {
  665. if (obj.filter) {
  666. return obj.filter(iterate, context)
  667. }
  668. each(obj, function (val, key) {
  669. if (iterate.call(context, val, key, obj)) {
  670. result.push(val)
  671. }
  672. })
  673. }
  674. return result
  675. }
  676. /**
  677. * 从左至右遍历,匹配最近的一条数据
  678. *
  679. * @param {Object} obj 对象/数组
  680. * @param {Function} iterate(item, index, obj) 回调
  681. * @param {Object} context 上下文
  682. * @return {Object}
  683. */
  684. var find = helperCreateIterateHandle('find', 1, 3, true)
  685. /**
  686. * 从右至左遍历,匹配最近的一条数据
  687. *
  688. * @param {Object} obj 对象/数组
  689. * @param {Function} iterate(item, index, obj) 回调
  690. * @param {Object} context 上下文
  691. * @return {Object}
  692. */
  693. function findLast (obj, iterate, context) {
  694. if (obj) {
  695. if (!isArray(obj)) {
  696. obj = values(obj)
  697. }
  698. for (var len = obj.length - 1; len >= 0; len--) {
  699. if (iterate.call(context, obj[len], len, obj)) {
  700. return obj[len]
  701. }
  702. }
  703. }
  704. }
  705. /**
  706. * 查找匹配第一条数据的键
  707. *
  708. * @param {Object} obj 对象/数组
  709. * @param {Function} iterate(item, index, obj) 回调
  710. * @param {Object} context 上下文
  711. * @return {Object}
  712. */
  713. var findKey = helperCreateIterateHandle('', 0, 2, true)
  714. /**
  715. * 判断对象是否包含该值,成功返回true否则false
  716. *
  717. * @param {Object} obj 对象
  718. * @param {Object} val 值
  719. * @return {Boolean}
  720. */
  721. function includes (obj, val) {
  722. if (obj) {
  723. if (obj.includes) {
  724. return obj.includes(val)
  725. }
  726. for (var key in obj) {
  727. if (hasOwnProp(obj, key)) {
  728. if (val === obj[key]) {
  729. return true
  730. }
  731. }
  732. }
  733. }
  734. return false
  735. }
  736. function arrayIndexOf (list, val) {
  737. if (list.indexOf) {
  738. return list.indexOf(val)
  739. }
  740. for (var index = 0, len = list.length; index < len; index++) {
  741. if (val === list[index]) {
  742. return index
  743. }
  744. }
  745. }
  746. function arrayLastIndexOf (list, val) {
  747. if (list.lastIndexOf) {
  748. return list.lastIndexOf(val)
  749. }
  750. for (var len = list.length - 1; len >= 0; len--) {
  751. if (val === list[len]) {
  752. return len
  753. }
  754. }
  755. return -1
  756. }
  757. /**
  758. * 指定方法后的返回值组成的新数组
  759. *
  760. * @param {Object} obj 对象/数组
  761. * @param {Function} iterate(item, index, obj) 回调
  762. * @param {Object} context 上下文
  763. * @return {Array}
  764. */
  765. function map (obj, iterate, context) {
  766. var result = []
  767. if (obj && arguments.length > 1) {
  768. if (obj.map) {
  769. return obj.map(iterate, context)
  770. } else {
  771. each(obj, function () {
  772. result.push(iterate.apply(context, arguments))
  773. })
  774. }
  775. }
  776. return result
  777. }
  778. /**
  779. * 接收一个函数作为累加器,数组中的每个值(从左到右)开始合并,最终为一个值。
  780. *
  781. * @param {Array} array 数组
  782. * @param {Function} callback 方法
  783. * @param {Object} initialValue 初始值
  784. * @return {Number}
  785. */
  786. function reduce (array, callback, initialValue) {
  787. if (array) {
  788. var len, reduceMethod
  789. var index = 0
  790. var context = null
  791. var previous = initialValue
  792. var isInitialVal = arguments.length > 2
  793. var keyList = keys(array)
  794. if (array.length && array.reduce) {
  795. reduceMethod = function () {
  796. return callback.apply(context, arguments)
  797. }
  798. if (isInitialVal) {
  799. return array.reduce(reduceMethod, previous)
  800. }
  801. return array.reduce(reduceMethod)
  802. }
  803. if (isInitialVal) {
  804. index = 1
  805. previous = array[keyList[0]]
  806. }
  807. for (len = keyList.length; index < len; index++) {
  808. previous = callback.call(context, previous, array[keyList[index]], index, array)
  809. }
  810. return previous
  811. }
  812. }
  813. /**
  814. * 浅复制数组的一部分到同一数组中的另一个位置,数组大小不变
  815. *
  816. * @param {Array} array 数组
  817. * @param {Number} target 从该位置开始替换数据
  818. * @param {Number} start 从该位置开始读取数据,默认为 0 。如果为负值,表示倒数
  819. * @param {Number} end 到该位置前停止读取数据,默认等于数组长度。如果为负值,表示倒数
  820. * @return {Array}
  821. */
  822. function copyWithin (array, target, start, end) {
  823. if (isArray(array) && array.copyWithin) {
  824. return array.copyWithin(target, start, end)
  825. }
  826. var replaceIndex, replaceArray
  827. var targetIndex = target >> 0
  828. var startIndex = start >> 0
  829. var len = array.length
  830. var endIndex = arguments.length > 3 ? end >> 0 : len
  831. if (targetIndex < len) {
  832. targetIndex = targetIndex >= 0 ? targetIndex : len + targetIndex
  833. if (targetIndex >= 0) {
  834. startIndex = startIndex >= 0 ? startIndex : len + startIndex
  835. endIndex = endIndex >= 0 ? endIndex : len + endIndex
  836. if (startIndex < endIndex) {
  837. for (replaceIndex = 0, replaceArray = array.slice(startIndex, endIndex); targetIndex < len; targetIndex++) {
  838. if (replaceArray.length <= replaceIndex) {
  839. break
  840. }
  841. array[targetIndex] = replaceArray[replaceIndex++]
  842. }
  843. }
  844. }
  845. }
  846. return array
  847. }
  848. /**
  849. * 将一个数组分割成大小的组。如果数组不能被平均分配,那么最后一块将是剩下的元素
  850. *
  851. * @param {Array} array 数组
  852. * @param {Number} size 每组大小
  853. * @return {Array}
  854. */
  855. function chunk (array, size) {
  856. var index
  857. var result = []
  858. var arrLen = size >> 0 || 1
  859. if (isArray(array)) {
  860. if (arrLen >= 0 && array.length > arrLen) {
  861. index = 0
  862. while (index < array.length) {
  863. result.push(array.slice(index, index + arrLen))
  864. index += arrLen
  865. }
  866. } else {
  867. result = array.length ? [array] : array
  868. }
  869. }
  870. return result
  871. }
  872. /**
  873. * 将每个数组中相应位置的值合并在一起
  874. *
  875. * @param {Array*} array 数组
  876. */
  877. function zip () {
  878. return unzip(arguments)
  879. }
  880. /**
  881. * 与 zip 相反
  882. *
  883. * @param {Array} arrays 数组集合
  884. */
  885. function unzip (arrays) {
  886. var index, maxItem, len
  887. var result = []
  888. if (arrays && arrays.length) {
  889. index = 0
  890. maxItem = max(arrays, function (item) {
  891. return item ? item.length : 0
  892. })
  893. for (len = maxItem ? maxItem.length : 0; index < len; index++) {
  894. result.push(pluck(arrays, index))
  895. }
  896. }
  897. return result
  898. }
  899. /**
  900. * 根据键数组、值数组对转换为对象
  901. *
  902. * @param {Array} props 键数组
  903. * @param {Number} arr 值数组
  904. * @return {Object}
  905. */
  906. function zipObject (props, arr) {
  907. var result = {}
  908. arr = arr || []
  909. each(values(props), function (val, key) {
  910. result[val] = arr[key]
  911. })
  912. return result
  913. }
  914. function flattenDeep (array, deep) {
  915. var result = []
  916. arrayEach(array, function (vals) {
  917. result = result.concat(isArray(vals) ? (deep ? flattenDeep(vals, deep) : vals) : [vals])
  918. })
  919. return result
  920. }
  921. /**
  922. * 将一个多维数组铺平
  923. * @param {Array} array 数组
  924. * @param {Boolean} deep 是否深层
  925. * @return {Array}
  926. */
  927. function flatten (array, deep) {
  928. if (isArray(array)) {
  929. return flattenDeep(array, deep)
  930. }
  931. return []
  932. }
  933. /**
  934. * 将对象或者伪数组转为新数组
  935. *
  936. * @param {Array} list 数组
  937. * @return {Array}
  938. */
  939. function toArray (list) {
  940. return map(list, function (item) {
  941. return item
  942. })
  943. }
  944. /**
  945. * 判断数组是否包含另一数组
  946. *
  947. * @param {Array} array1 数组
  948. * @param {Array} array2 被包含数组
  949. * @return {Boolean}
  950. */
  951. function includeArrays (array1, array2) {
  952. var len
  953. var index = 0
  954. if (isArray(array1) && isArray(array2)) {
  955. for (len = array2.length; index < len; index++) {
  956. if (!includes(array1, array2[index])) {
  957. return false
  958. }
  959. }
  960. return true
  961. }
  962. return includes(array1, array2)
  963. }
  964. /**
  965. * 获取数组对象中某属性值,返回一个数组
  966. *
  967. * @param {Array} array 数组
  968. * @param {String} key 属性值
  969. * @return {Array}
  970. */
  971. function pluck (obj, key) {
  972. return map(obj, property(key))
  973. }
  974. function deepGetObj (obj, path) {
  975. var index = 0
  976. var len = path.length
  977. while (obj && index < len) {
  978. obj = obj[path[index++]]
  979. }
  980. return len && obj ? obj : 0
  981. }
  982. /**
  983. * 在list的每个元素上执行方法,任何传递的额外参数都会在调用方法的时候传递给它
  984. *
  985. * @param {Array} list
  986. * @param {Array/String/Function} path
  987. * @param {...Object} arguments
  988. * @return {Array}
  989. */
  990. function invoke (list, path) {
  991. var func
  992. var args = arguments
  993. var params = []
  994. var paths = []
  995. var index = 2
  996. var len = args.length
  997. for (; index < len; index++) {
  998. params.push(args[index])
  999. }
  1000. if (isArray(path)) {
  1001. len = path.length - 1
  1002. for (index = 0; index < len; index++) {
  1003. paths.push(path[index])
  1004. }
  1005. path = path[len]
  1006. }
  1007. return map(list, function (context) {
  1008. if (paths.length) {
  1009. context = deepGetObj(context, paths)
  1010. }
  1011. func = context[path] || path
  1012. if (func && func.apply) {
  1013. return func.apply(context, params)
  1014. }
  1015. })
  1016. }
  1017. function arrayEach (list, iterate, context) {
  1018. if (list) {
  1019. if (list.forEach) {
  1020. list.forEach(iterate, context)
  1021. } else {
  1022. for (var index = 0, len = list.length; index < len; index++) {
  1023. iterate.call(context, list[index], index, list)
  1024. }
  1025. }
  1026. }
  1027. }
  1028. function lastArrayEach (obj, iterate, context) {
  1029. for (var len = obj.length - 1; len >= 0; len--) {
  1030. iterate.call(context, obj[len], len, obj)
  1031. }
  1032. }
  1033. function strictTree (array, optChildren) {
  1034. each(array, function (item) {
  1035. if (item.children && !item.children.length) {
  1036. remove(item, optChildren)
  1037. }
  1038. })
  1039. }
  1040. /**
  1041. * 将一个带层级的数据列表转成树结构
  1042. *
  1043. * @param {Array} array 数组
  1044. * @param {Object} options {strict: false, parentKey: 'parentId', key: 'id', children: 'children', data: 'data'}
  1045. * @return {Array}
  1046. */
  1047. function toArrayTree (array, options) {
  1048. var opts = assign({}, setupDefaults.treeOptions, options)
  1049. var optStrict = opts.strict
  1050. var optKey = opts.key
  1051. var optParentKey = opts.parentKey
  1052. var optChildren = opts.children
  1053. var optSortKey = opts.sortKey
  1054. var optReverse = opts.reverse
  1055. var optData = opts.data
  1056. var result = []
  1057. var treeMap = {}
  1058. var idList, id, treeData, parentId
  1059. if (optSortKey) {
  1060. array = orderBy(clone(array), optSortKey)
  1061. if (optReverse) {
  1062. array = array.reverse()
  1063. }
  1064. }
  1065. idList = map(array, function (item) {
  1066. return item[optKey]
  1067. })
  1068. each(array, function (item) {
  1069. id = item[optKey]
  1070. if (optData) {
  1071. treeData = {}
  1072. treeData[optData] = item
  1073. } else {
  1074. treeData = item
  1075. }
  1076. parentId = item[optParentKey]
  1077. treeMap[id] = treeMap[id] || []
  1078. treeMap[parentId] = treeMap[parentId] || []
  1079. treeMap[parentId].push(treeData)
  1080. treeData[optKey] = id
  1081. treeData[optParentKey] = parentId
  1082. treeData[optChildren] = treeMap[id]
  1083. if (!optStrict || (optStrict && !parentId)) {
  1084. if (!includes(idList, parentId)) {
  1085. result.push(treeData)
  1086. }
  1087. }
  1088. })
  1089. if (optStrict) {
  1090. strictTree(array, optChildren)
  1091. }
  1092. return result
  1093. }
  1094. function unTreeList (result, array, opts) {
  1095. var optChildren = opts.children
  1096. var optData = opts.data
  1097. var optClear = opts.clear
  1098. each(array, function (item) {
  1099. var children = item[optChildren]
  1100. if (optData) {
  1101. item = item[optData]
  1102. }
  1103. result.push(item)
  1104. if (children && children.length) {
  1105. unTreeList(result, children, opts)
  1106. }
  1107. if (optClear) {
  1108. delete item[optChildren]
  1109. }
  1110. })
  1111. return result
  1112. }
  1113. /**
  1114. * 将一个树结构转成数组列表
  1115. *
  1116. * @param {Array} array 数组
  1117. * @param {Object} options { children: 'children', data: 'data', clear: false }
  1118. * @return {Array}
  1119. */
  1120. function toTreeArray (array, options) {
  1121. return unTreeList([], array, assign({}, setupDefaults.treeOptions, options))
  1122. }
  1123. function findTreeItem (parent, obj, iterate, context, path, node, parseChildren, opts) {
  1124. if (obj) {
  1125. var item, index, len, paths, nodes, match
  1126. for (index = 0, len = obj.length; index < len; index++) {
  1127. item = obj[index]
  1128. paths = path.concat(['' + index])
  1129. nodes = node.concat([item])
  1130. if (iterate.call(context, item, index, obj, paths, parent, nodes)) {
  1131. return { index: index, item: item, path: paths, items: obj, parent: parent, nodes: nodes }
  1132. }
  1133. if (parseChildren && item) {
  1134. match = findTreeItem(item, item[parseChildren], iterate, context, paths.concat([parseChildren]), nodes, parseChildren, opts)
  1135. if (match) {
  1136. return match
  1137. }
  1138. }
  1139. }
  1140. }
  1141. }
  1142. /**
  1143. * 从树结构中查找匹配第一条数据的键、值、路径
  1144. *
  1145. * @param {Object} obj 对象/数组
  1146. * @param {Function} iterate(item, index, items, path, parent, nodes) 回调
  1147. * @param {Object} options {children: 'children'}
  1148. * @param {Object} context 上下文
  1149. * @return {Object} { item, index, items, path, parent, nodes }
  1150. */
  1151. var findTree = helperCreateTreeFunc(findTreeItem)
  1152. function eachTreeItem (parent, obj, iterate, context, path, node, parseChildren, opts) {
  1153. var paths, nodes
  1154. each(obj, function (item, index) {
  1155. paths = path.concat(['' + index])
  1156. nodes = node.concat([item])
  1157. iterate.call(context, item, index, obj, paths, parent, nodes)
  1158. if (item && parseChildren) {
  1159. paths.push(parseChildren)
  1160. eachTreeItem(item, item[parseChildren], iterate, context, paths, nodes, parseChildren, opts)
  1161. }
  1162. })
  1163. }
  1164. /**
  1165. * 从树结构中遍历数据的键、值、路径
  1166. *
  1167. * @param {Object} obj 对象/数组
  1168. * @param {Function} iterate(item, index, items, path, parent, nodes) 回调
  1169. * @param {Object} options {children: 'children', mapChildren: 'children}
  1170. * @param {Object} context 上下文
  1171. */
  1172. var eachTree = helperCreateTreeFunc(eachTreeItem)
  1173. function mapTreeItem (parent, obj, iterate, context, path, node, parseChildren, opts) {
  1174. var paths, nodes, rest
  1175. var mapChildren = opts.mapChildren || parseChildren
  1176. return map(obj, function (item, index) {
  1177. paths = path.concat(['' + index])
  1178. nodes = node.concat([item])
  1179. rest = iterate.call(context, item, index, obj, paths, parent, nodes)
  1180. if (rest && item && parseChildren && item[parseChildren]) {
  1181. rest[mapChildren] = mapTreeItem(item, item[parseChildren], iterate, context, paths, nodes, parseChildren, opts)
  1182. }
  1183. return rest
  1184. })
  1185. }
  1186. /**
  1187. * 从树结构中指定方法后的返回值组成的新数组
  1188. *
  1189. * @param {Object} obj 对象/数组
  1190. * @param {Function} iterate(item, index, items, path, parent, nodes) 回调
  1191. * @param {Object} options {children: 'children'}
  1192. * @param {Object} context 上下文
  1193. * @return {Object/Array}
  1194. */
  1195. var mapTree = helperCreateTreeFunc(mapTreeItem)
  1196. /**
  1197. * 从树结构中根据回调过滤数据
  1198. *
  1199. * @param {Object} obj 对象/数组
  1200. * @param {Function} iterate(item, index, items, path, parent) 回调
  1201. * @param {Object} options {children: 'children'}
  1202. * @param {Object} context 上下文
  1203. * @return {Array}
  1204. */
  1205. function filterTree (obj, iterate, options, context) {
  1206. var result = []
  1207. if (obj && iterate) {
  1208. eachTree(obj, function (item, index, items, path, parent, nodes) {
  1209. if (iterate.call(context, item, index, items, path, parent, nodes)) {
  1210. result.push(item)
  1211. }
  1212. }, options)
  1213. }
  1214. return result
  1215. }
  1216. function searchTreeItem (parentAllow, parent, obj, iterate, context, path, node, parseChildren, opts) {
  1217. var paths, nodes, rest, isAllow, hasChild
  1218. var rests = []
  1219. var hasOriginal = opts.original
  1220. var sourceData = opts.data
  1221. var mapChildren = opts.mapChildren || parseChildren
  1222. arrayEach(obj, function (item, index) {
  1223. paths = path.concat(['' + index])
  1224. nodes = node.concat([item])
  1225. isAllow = parentAllow || iterate.call(context, item, index, obj, paths, parent, nodes)
  1226. hasChild = parseChildren && item[parseChildren]
  1227. if (isAllow || hasChild) {
  1228. if (hasOriginal) {
  1229. rest = item
  1230. } else {
  1231. rest = assign({}, item)
  1232. if (sourceData) {
  1233. rest[sourceData] = item
  1234. }
  1235. }
  1236. rest[mapChildren] = searchTreeItem(isAllow, item, item[parseChildren], iterate, context, paths, nodes, parseChildren, opts)
  1237. if (isAllow || rest[mapChildren].length) {
  1238. rests.push(rest)
  1239. }
  1240. } else if (isAllow) {
  1241. rests.push(rest)
  1242. }
  1243. })
  1244. return rests
  1245. }
  1246. /**
  1247. * 从树结构中根据回调查找数据
  1248. *
  1249. * @param {Object} obj 对象/数组
  1250. * @param {Function} iterate(item, index, items, path, parent, nodes) 回调
  1251. * @param {Object} options {children: 'children'}
  1252. * @param {Object} context 上下文
  1253. * @return {Array}
  1254. */
  1255. var searchTree = helperCreateTreeFunc(function (parent, obj, iterate, context, path, nodes, parseChildren, opts) {
  1256. return searchTreeItem(0, parent, obj, iterate, context, path, nodes, parseChildren, opts)
  1257. })
  1258. /**
  1259. * 判断对象自身属性中是否具有指定的属性
  1260. *
  1261. * @param {Object} obj 对象
  1262. * @param {String/Number} key 键值
  1263. * @return {Boolean}
  1264. */
  1265. function hasOwnProp (obj, key) {
  1266. return obj && obj.hasOwnProperty ? obj.hasOwnProperty(key) : false
  1267. }
  1268. /**
  1269. * 判断是否 undefined 和 null
  1270. * @param {Object} obj 对象
  1271. * @return {Boolean}
  1272. */
  1273. function eqNull (obj) {
  1274. return isNull(obj) || isUndefined(obj)
  1275. }
  1276. /* eslint-disable eqeqeq */
  1277. function isNumberNaN (obj) {
  1278. return isNumber(obj) && isNaN(obj)
  1279. }
  1280. function isNumberFinite (obj) {
  1281. return isNumber(obj) && isFinite(obj)
  1282. }
  1283. /**
  1284. * 判断是否Undefined
  1285. *
  1286. * @param {Object} obj 对象
  1287. * @return {Boolean}
  1288. */
  1289. var isUndefined = helperCreateInTypeof(staticStrUndefined)
  1290. /**
  1291. * 判断是否数组
  1292. *
  1293. * @param {Object} obj 对象
  1294. * @return {Boolean}
  1295. */
  1296. var isArray = Array.isArray || helperCreateInInObjectString('Array')
  1297. /**
  1298. * 判断是否小数
  1299. *
  1300. * @param {Number} obj 数值
  1301. * @return {Boolean}
  1302. */
  1303. function isFloat (obj) {
  1304. return !isNull(obj) && !isNaN(obj) && !isArray(obj) && !isInteger(obj)
  1305. }
  1306. /**
  1307. * 判断是否整数
  1308. *
  1309. * @param {Number, String} number 数值
  1310. * @return {Boolean}
  1311. */
  1312. var isInteger = function (obj) {
  1313. return !isNull(obj) && !isNaN(obj) && !isArray(obj) && obj % 1 === 0
  1314. }
  1315. /**
  1316. * 判断是否方法
  1317. *
  1318. * @param {Object} obj 对象
  1319. * @return {Boolean}
  1320. */
  1321. var isFunction = helperCreateInTypeof('function')
  1322. /**
  1323. * 判断是否Boolean对象
  1324. *
  1325. * @param {Object} obj 对象
  1326. * @return {Boolean}
  1327. */
  1328. var isBoolean = helperCreateInTypeof('boolean')
  1329. /**
  1330. * 判断是否String对象
  1331. *
  1332. * @param {Object} obj 对象
  1333. * @return {Boolean}
  1334. */
  1335. var isString = helperCreateInTypeof('string')
  1336. /**
  1337. * 判断是否Number对象
  1338. *
  1339. * @param {Object} obj 对象
  1340. * @return {Boolean}
  1341. */
  1342. var isNumber = helperCreateInTypeof('number')
  1343. /**
  1344. * 判断是否RegExp对象
  1345. *
  1346. * @param {Object} obj 对象
  1347. * @return {Boolean}
  1348. */
  1349. var isRegExp = helperCreateInInObjectString('RegExp')
  1350. /**
  1351. * 判断是否Object对象
  1352. *
  1353. * @param {Object} obj 对象
  1354. * @return {Boolean}
  1355. */
  1356. var isObject = helperCreateInTypeof('object')
  1357. /**
  1358. * 判断是否对象
  1359. *
  1360. * @param {Object} obj 对象
  1361. * @return {Boolean}
  1362. */
  1363. function isPlainObject (obj) {
  1364. return obj ? obj.constructor === Object : false
  1365. }
  1366. /**
  1367. * 判断是否Date对象
  1368. *
  1369. * @param {Object} obj 对象
  1370. * @return {Boolean}
  1371. */
  1372. var isDate = helperCreateInInObjectString('Date')
  1373. /**
  1374. * 判断是否Error对象
  1375. *
  1376. * @param {Object} obj 对象
  1377. * @return {Boolean}
  1378. */
  1379. var isError = helperCreateInInObjectString('Error')
  1380. /**
  1381. * 判断是否TypeError对象
  1382. *
  1383. * @param {Object} obj 对象
  1384. * @return {Boolean}
  1385. */
  1386. function isTypeError (obj) {
  1387. return obj ? obj.constructor === TypeError : false
  1388. }
  1389. /**
  1390. * 判断是否为空对象
  1391. *
  1392. * @param {Object} obj 对象
  1393. * @return {Boolean}
  1394. */
  1395. function isEmpty (obj) {
  1396. for (var key in obj) {
  1397. return false
  1398. }
  1399. return true
  1400. }
  1401. /**
  1402. * 判断是否为Null
  1403. *
  1404. * @param {Object} obj 对象
  1405. * @return {Boolean}
  1406. */
  1407. function isNull (obj) {
  1408. return obj === null
  1409. }
  1410. /* eslint-disable valid-typeof */
  1411. /**
  1412. * 判断是否Symbol对象
  1413. *
  1414. * @param {Object} obj 对象
  1415. * @return {Boolean}
  1416. */
  1417. var supportSymbol = typeof Symbol !== staticStrUndefined
  1418. function isSymbol (obj) {
  1419. return supportSymbol && Symbol.isSymbol ? Symbol.isSymbol(obj) : (typeof obj === 'symbol')
  1420. }
  1421. /**
  1422. * 判断是否Arguments对象
  1423. *
  1424. * @param {Object} obj 对象
  1425. * @return {Boolean}
  1426. */
  1427. var isArguments = helperCreateInInObjectString('Arguments')
  1428. /**
  1429. * 判断是否Element对象
  1430. *
  1431. * @param {Object} obj 对象
  1432. * @return {Boolean}
  1433. */
  1434. function isElement (obj) {
  1435. return !!(obj && isString(obj.nodeName) && isNumber(obj.nodeType))
  1436. }
  1437. /**
  1438. * 判断是否Document对象
  1439. *
  1440. * @param {Object} obj 对象
  1441. * @return {Boolean}
  1442. */
  1443. function isDocument (obj) {
  1444. return !!(obj && staticDocument && obj.nodeType === 9)
  1445. }
  1446. /**
  1447. * 判断是否Window对象
  1448. *
  1449. * @param {Object} obj 对象
  1450. * @return {Boolean}
  1451. */
  1452. function isWindow (obj) {
  1453. return staticWindow && !!(obj && obj === obj.window)
  1454. }
  1455. /* eslint-disable valid-typeof */
  1456. /**
  1457. * 判断是否FormData对象
  1458. *
  1459. * @param {Object} obj 对象
  1460. * @return {Boolean}
  1461. */
  1462. var supportFormData = typeof FormData !== staticStrUndefined
  1463. function isFormData (obj) {
  1464. return supportFormData && obj instanceof FormData
  1465. }
  1466. /* eslint-disable valid-typeof */
  1467. /**
  1468. * 判断是否Map对象
  1469. *
  1470. * @param {Object} obj 对象
  1471. * @return {Boolean}
  1472. */
  1473. var supportMap = typeof Map !== staticStrUndefined
  1474. function isMap (obj) {
  1475. return supportMap && obj instanceof Map
  1476. }
  1477. /* eslint-disable valid-typeof */
  1478. /**
  1479. * 判断是否WeakMap对象
  1480. *
  1481. * @param {Object} obj 对象
  1482. * @return {Boolean}
  1483. */
  1484. var supportWeakMap = typeof WeakMap !== staticStrUndefined
  1485. function isWeakMap (obj) {
  1486. return supportWeakMap && obj instanceof WeakMap
  1487. }
  1488. /* eslint-disable valid-typeof */
  1489. /**
  1490. * 判断是否Set对象
  1491. *
  1492. * @param {Object} obj 对象
  1493. * @return {Boolean}
  1494. */
  1495. var supportSet = typeof Set !== staticStrUndefined
  1496. function isSet (obj) {
  1497. return supportSet && obj instanceof Set
  1498. }
  1499. /* eslint-disable valid-typeof */
  1500. /**
  1501. * 判断是否WeakSet对象
  1502. *
  1503. * @param {Object} obj 对象
  1504. * @return {Boolean}
  1505. */
  1506. var supportWeakSet = typeof WeakSet !== staticStrUndefined
  1507. function isWeakSet (obj) {
  1508. return supportWeakSet && obj instanceof WeakSet
  1509. }
  1510. /**
  1511. * 判断是否闰年
  1512. *
  1513. * @param {Date} date 日期或数字
  1514. * @return {Boolean}
  1515. */
  1516. function isLeapYear (date) {
  1517. var year
  1518. var currentDate = date ? toStringDate(date) : helperNewDate()
  1519. if (isDate(currentDate)) {
  1520. year = currentDate.getFullYear()
  1521. return (year % 4 === 0) && (year % 100 !== 0 || year % 400 === 0)
  1522. }
  1523. return false
  1524. }
  1525. /**
  1526. * 判断属性中的键和值是否包含在对象中
  1527. *
  1528. * @param {Object/Array} obj 对象
  1529. * @param {Object} source 值
  1530. * @return {Boolean}
  1531. */
  1532. function isMatch (obj, source) {
  1533. var objKeys = keys(obj)
  1534. var sourceKeys = keys(source)
  1535. if (sourceKeys.length) {
  1536. if (includeArrays(objKeys, sourceKeys)) {
  1537. return some(sourceKeys, function (key2) {
  1538. return findIndexOf(objKeys, function (key1) {
  1539. return key1 === key2 && isEqual(obj[key1], source[key2])
  1540. }) > -1
  1541. })
  1542. }
  1543. } else {
  1544. return true
  1545. }
  1546. return isEqual(obj, source)
  1547. }
  1548. /**
  1549. * 深度比较两个对象之间的值是否相等
  1550. *
  1551. * @param {Object} obj1 值1
  1552. * @param {Object} obj2 值2
  1553. * @return {Boolean}
  1554. */
  1555. function isEqual (obj1, obj2) {
  1556. return helperEqualCompare(obj1, obj2, helperDefaultCompare)
  1557. }
  1558. /**
  1559. * 深度比较两个对象之间的值是否相等,使用自定义比较函数
  1560. *
  1561. * @param {Object} obj1 值1
  1562. * @param {Object} obj2 值2
  1563. * @param {Function} func 自定义函数
  1564. * @return {Boolean}
  1565. */
  1566. function isEqualWith (obj1, obj2, func) {
  1567. if (isFunction(func)) {
  1568. return helperEqualCompare(obj1, obj2, function (v1, v2, key, obj1, obj2) {
  1569. var result = func(v1, v2, key, obj1, obj2)
  1570. return isUndefined(result) ? helperDefaultCompare(v1, v2) : !!result
  1571. }, func)
  1572. }
  1573. return helperEqualCompare(obj1, obj2, helperDefaultCompare)
  1574. }
  1575. /**
  1576. * 获取对象类型
  1577. *
  1578. * @param {Object} obj 对象
  1579. * @return {String}
  1580. */
  1581. function getType (obj) {
  1582. if (isNull(obj)) {
  1583. return 'null'
  1584. }
  1585. if (isSymbol(obj)) {
  1586. return 'symbol'
  1587. }
  1588. if (isDate(obj)) {
  1589. return 'date'
  1590. }
  1591. if (isArray(obj)) {
  1592. return 'array'
  1593. }
  1594. if (isRegExp(obj)) {
  1595. return 'regexp'
  1596. }
  1597. if (isError(obj)) {
  1598. return 'error'
  1599. }
  1600. return typeof obj
  1601. }
  1602. /**
  1603. * 获取一个全局唯一标识
  1604. *
  1605. * @param {String} prefix 前缀
  1606. * @return {Number}
  1607. */
  1608. var __uniqueId = 0
  1609. function uniqueId (prefix) {
  1610. return [prefix, ++__uniqueId].join('')
  1611. }
  1612. /**
  1613. * 返回对象的长度
  1614. *
  1615. * @param {Object} obj 对象
  1616. * @return {Number}
  1617. */
  1618. function getSize (obj) {
  1619. var len = 0
  1620. if (isString(obj) || isArray(obj)) {
  1621. return obj.length
  1622. }
  1623. each(obj, function () {
  1624. len++
  1625. })
  1626. return len
  1627. }
  1628. /**
  1629. * 返回对象第一个索引值
  1630. *
  1631. * @param {Object} obj 对象
  1632. * @param {Object} val 值
  1633. * @return {Number}
  1634. */
  1635. var indexOf = helperCreateIndexOf('indexOf', arrayIndexOf)
  1636. /**
  1637. * 从最后开始的索引值,返回对象第一个索引值
  1638. *
  1639. * @param {Object} array 对象
  1640. * @param {Object} val 值
  1641. * @return {Number}
  1642. */
  1643. var lastIndexOf = helperCreateIndexOf('lastIndexOf', arrayLastIndexOf)
  1644. /**
  1645. * 返回对象第一个索引值
  1646. *
  1647. * @param {Object} obj 对象/数组
  1648. * @param {Function} iterate(item, index, obj) 回调
  1649. * @param {Object} context 上下文
  1650. * @return {Object}
  1651. */
  1652. var findIndexOf = helperCreateiterateIndexOf(function (obj, iterate, context) {
  1653. for (var index = 0, len = obj.length; index < len; index++) {
  1654. if (iterate.call(context, obj[index], index, obj)) {
  1655. return index
  1656. }
  1657. }
  1658. return -1
  1659. })
  1660. /**
  1661. * 从最后开始的索引值,返回对象第一个索引值
  1662. *
  1663. * @param {Object} obj 对象/数组
  1664. * @param {Function} iterate(item, index, obj) 回调
  1665. * @param {Object} context 上下文
  1666. * @return {Object}
  1667. */
  1668. var findLastIndexOf = helperCreateiterateIndexOf(function (obj, iterate, context) {
  1669. for (var len = obj.length - 1; len >= 0; len--) {
  1670. if (iterate.call(context, obj[len], len, obj)) {
  1671. return len
  1672. }
  1673. }
  1674. return -1
  1675. })
  1676. /**
  1677. * 字符串转JSON
  1678. *
  1679. * @param {String} str 字符串
  1680. * @return {Object} 返回转换后对象
  1681. */
  1682. function toStringJSON (str) {
  1683. if (isPlainObject(str)) {
  1684. return str
  1685. } else if (isString(str)) {
  1686. try {
  1687. return JSON.parse(str)
  1688. } catch (e) {}
  1689. }
  1690. return {}
  1691. }
  1692. /**
  1693. * JSON转字符串
  1694. *
  1695. * @param {Object} obj 对象
  1696. * @return {String} 返回字符串
  1697. */
  1698. function toJSONString (obj) {
  1699. return eqNull(obj) ? '' : JSON.stringify(obj)
  1700. }
  1701. /**
  1702. * 获取对象所有属性
  1703. *
  1704. * @param {Object} obj 对象/数组
  1705. * @return {Array}
  1706. */
  1707. var keys = helperCreateGetObjects('keys', 1)
  1708. /**
  1709. * 获取对象所有值
  1710. *
  1711. * @param {Object} obj 对象/数组
  1712. * @return {Array}
  1713. */
  1714. var values = helperCreateGetObjects('values', 0)
  1715. /**
  1716. * 获取对象所有属性、值
  1717. *
  1718. * @param {Object} obj 对象/数组
  1719. * @return {Array}
  1720. */
  1721. var entries = helperCreateGetObjects('entries', 2)
  1722. /**
  1723. * 根据 key 过滤指定的属性值,返回一个新的对象
  1724. *
  1725. * @param {Object} obj 对象
  1726. * @param {String/Array} key 键数组
  1727. * @return {Object}
  1728. */
  1729. var pick = helperCreatePickOmit(1, 0)
  1730. /**
  1731. * 根据 key 排除指定的属性值,返回一个新的对象
  1732. *
  1733. * @param {Object} obj 对象
  1734. * @param {String/Array} key 键数组
  1735. * @return {Object}
  1736. */
  1737. var omit = helperCreatePickOmit(0, 1)
  1738. /**
  1739. * 获取对象第一个值
  1740. *
  1741. * @param {Object} obj 对象/数组
  1742. * @return {Object}
  1743. */
  1744. function first (obj) {
  1745. return values(obj)[0]
  1746. }
  1747. /**
  1748. * 获取对象最后一个值
  1749. *
  1750. * @param {Object} obj 对象/数组
  1751. * @return {Object}
  1752. */
  1753. function last (obj) {
  1754. var list = values(obj)
  1755. return list[list.length - 1]
  1756. }
  1757. /**
  1758. * 迭代器
  1759. *
  1760. * @param {Object} obj 对象/数组
  1761. * @param {Function} iterate(item, index, obj) 回调
  1762. * @param {Object} context 上下文
  1763. * @return {Object}
  1764. */
  1765. function each (obj, iterate, context) {
  1766. if (obj) {
  1767. return (isArray(obj) ? arrayEach : objectEach)(obj, iterate, context)
  1768. }
  1769. return obj
  1770. }
  1771. /**
  1772. * 已废弃,被 some, every 替换
  1773. * @deprecated
  1774. */
  1775. function forOf (obj, iterate, context) {
  1776. if (obj) {
  1777. if (isArray(obj)) {
  1778. for (var index = 0, len = obj.length; index < len; index++) {
  1779. if (iterate.call(context, obj[index], index, obj) === false) {
  1780. break
  1781. }
  1782. }
  1783. } else {
  1784. for (var key in obj) {
  1785. if (hasOwnProp(obj, key)) {
  1786. if (iterate.call(context, obj[key], key, obj) === false) {
  1787. break
  1788. }
  1789. }
  1790. }
  1791. }
  1792. }
  1793. }
  1794. /**
  1795. * 已废弃
  1796. * @deprecated
  1797. */
  1798. function lastForOf (obj, iterate, context) {
  1799. if (obj) {
  1800. var len, list
  1801. if (isArray(obj)) {
  1802. for (len = obj.length - 1; len >= 0; len--) {
  1803. if (iterate.call(context, obj[len], len, obj) === false) {
  1804. break
  1805. }
  1806. }
  1807. } else {
  1808. list = keys(obj)
  1809. for (len = list.length - 1; len >= 0; len--) {
  1810. if (iterate.call(context, obj[list[len]], list[len], obj) === false) {
  1811. break
  1812. }
  1813. }
  1814. }
  1815. }
  1816. }
  1817. /**
  1818. * 迭代器,从最后开始迭代
  1819. *
  1820. * @param {Object} obj 对象/数组
  1821. * @param {Function} iterate(item, index, obj) 回调
  1822. * @param {Object} context 上下文
  1823. * @return {Object}
  1824. */
  1825. function lastEach (obj, iterate, context) {
  1826. if (obj) {
  1827. return (isArray(obj) ? lastArrayEach : lastObjectEach)(obj, iterate, context)
  1828. }
  1829. return obj
  1830. }
  1831. /**
  1832. * 检查键、路径是否是该对象的属性
  1833. *
  1834. * @param {Object/Array} data 对象
  1835. * @param {String/Function} property 键、路径
  1836. * @return {Boolean}
  1837. */
  1838. function has (obj, property) {
  1839. if (obj) {
  1840. if (hasOwnProp(obj, property)) {
  1841. return true
  1842. } else {
  1843. var prop, arrIndex, objProp, matchs, rest, isHas
  1844. var props = helperGetHGSKeys(property)
  1845. var index = 0
  1846. var len = props.length
  1847. for (rest = obj; index < len; index++) {
  1848. isHas = false
  1849. prop = props[index]
  1850. matchs = prop ? prop.match(staticHGKeyRE) : ''
  1851. if (matchs) {
  1852. arrIndex = matchs[1]
  1853. objProp = matchs[2]
  1854. if (arrIndex) {
  1855. if (rest[arrIndex]) {
  1856. if (hasOwnProp(rest[arrIndex], objProp)) {
  1857. isHas = true
  1858. rest = rest[arrIndex][objProp]
  1859. }
  1860. }
  1861. } else {
  1862. if (hasOwnProp(rest, objProp)) {
  1863. isHas = true
  1864. rest = rest[objProp]
  1865. }
  1866. }
  1867. } else {
  1868. if (hasOwnProp(rest, prop)) {
  1869. isHas = true
  1870. rest = rest[prop]
  1871. }
  1872. }
  1873. if (isHas) {
  1874. if (index === len - 1) {
  1875. return true
  1876. }
  1877. } else {
  1878. break
  1879. }
  1880. }
  1881. }
  1882. }
  1883. return false
  1884. }
  1885. /**
  1886. * 获取对象的属性的值,如果值为 undefined,则返回默认值
  1887. * @param {Object/Array} obj 对象
  1888. * @param {String/Function} property 键、路径
  1889. * @param {Object} defaultValue 默认值
  1890. * @return {Object}
  1891. */
  1892. function get (obj, property, defaultValue) {
  1893. if (eqNull(obj)) {
  1894. return defaultValue
  1895. }
  1896. var result = getValueByPath(obj, property)
  1897. return isUndefined(result) ? defaultValue : result
  1898. }
  1899. function getDeepProps (obj, key) {
  1900. var matchs = key ? key.match(staticHGKeyRE) : ''
  1901. return matchs ? (matchs[1] ? (obj[matchs[1]] ? obj[matchs[1]][matchs[2]] : undefined) : obj[matchs[2]]) : obj[key]
  1902. }
  1903. function getValueByPath (obj, property) {
  1904. if (obj) {
  1905. var rest, props, len
  1906. var index = 0
  1907. if (obj[property] || hasOwnProp(obj, property)) {
  1908. return obj[property]
  1909. } else {
  1910. props = helperGetHGSKeys(property)
  1911. len = props.length
  1912. if (len) {
  1913. for (rest = obj; index < len; index++) {
  1914. rest = getDeepProps(rest, props[index])
  1915. if (eqNull(rest)) {
  1916. if (index === len - 1) {
  1917. return rest
  1918. }
  1919. return
  1920. }
  1921. }
  1922. }
  1923. return rest
  1924. }
  1925. }
  1926. }
  1927. var sKeyRE = /(.+)\[(\d+)\]$/
  1928. function setDeepProps (obj, key, isSet, value) {
  1929. if (obj[key]) {
  1930. if (isSet) {
  1931. obj[key] = value
  1932. }
  1933. } else {
  1934. var index
  1935. var matchs = key ? key.match(sKeyRE) : null
  1936. var rest = isSet ? value : {}
  1937. if (matchs) {
  1938. index = staticParseInt(matchs[2])
  1939. if (obj[matchs[1]]) {
  1940. obj[matchs[1]][index] = rest
  1941. } else {
  1942. obj[matchs[1]] = new Array(index + 1)
  1943. obj[matchs[1]][index] = rest
  1944. }
  1945. } else {
  1946. obj[key] = rest
  1947. }
  1948. return rest
  1949. }
  1950. return obj[key]
  1951. }
  1952. /**
  1953. * 设置对象属性上的值。如果属性不存在则创建它
  1954. * @param {Object/Array} obj 对象
  1955. * @param {String/Function} property 键、路径
  1956. * @param {Object} value 值
  1957. */
  1958. function set (obj, property, value) {
  1959. if (obj) {
  1960. if ((obj[property] || hasOwnProp(obj, property)) && !isPrototypePolluted(property)) {
  1961. obj[property] = value
  1962. } else {
  1963. var rest = obj
  1964. var props = helperGetHGSKeys(property)
  1965. var len = props.length
  1966. for (var index = 0; index < len; index++) {
  1967. if (isPrototypePolluted(props[index])) continue
  1968. rest = setDeepProps(rest, props[index], index === len - 1, value)
  1969. }
  1970. }
  1971. }
  1972. return obj
  1973. }
  1974. /**
  1975. * Blacklist certain keys to prevent Prototype Pollution
  1976. * @param {string} key
  1977. */
  1978. function isPrototypePolluted(key) {
  1979. return key === '__proto__' || key === 'constructor' || key === 'prototype'
  1980. }
  1981. function createiterateEmpty (iterate) {
  1982. return function () {
  1983. return isEmpty(iterate)
  1984. }
  1985. }
  1986. /**
  1987. * 集合分组,默认使用键值分组,如果有iterate则使用结果进行分组
  1988. *
  1989. * @param {Array} obj 对象
  1990. * @param {Function} iterate 回调/对象属性
  1991. * @param {Object} context 上下文
  1992. * @return {Object}
  1993. */
  1994. function groupBy (obj, iterate, context) {
  1995. var groupKey
  1996. var result = {}
  1997. if (obj) {
  1998. if (iterate && isObject(iterate)) {
  1999. iterate = createiterateEmpty(iterate)
  2000. } else if (!isFunction(iterate)) {
  2001. iterate = property(iterate)
  2002. }
  2003. each(obj, function (val, key) {
  2004. groupKey = iterate ? iterate.call(context, val, key, obj) : val
  2005. if (result[groupKey]) {
  2006. result[groupKey].push(val)
  2007. } else {
  2008. result[groupKey] = [val]
  2009. }
  2010. })
  2011. }
  2012. return result
  2013. }
  2014. /**
  2015. * 集合分组统计,返回各组中对象的数量统计
  2016. *
  2017. * @param {Array} obj 对象
  2018. * @param {Function} iterate 回调/对象属性
  2019. * @param {Object} context 上下文
  2020. * @return {Object}
  2021. */
  2022. function countBy (obj, iterate, context) {
  2023. var result = groupBy(obj, iterate, context || this)
  2024. objectEach(result, function (item, key) {
  2025. result[key] = item.length
  2026. })
  2027. return result
  2028. }
  2029. function handleObjectAndArrayClone (func, obj, deep) {
  2030. return func(obj, deep ? function (val) {
  2031. return copyValue(val, deep)
  2032. } : function (val) {
  2033. return val
  2034. })
  2035. }
  2036. function handleValueClone (val, deep) {
  2037. if (deep && val) {
  2038. var Ctor = val.constructor;
  2039. switch(objectToString.call(val)) {
  2040. case "[object Date]":
  2041. case "[object RegExp]":
  2042. return new Ctor(val.valueOf())
  2043. case "[object Set]":
  2044. var set = new Ctor()
  2045. val.forEach(function (v) {
  2046. set.add(v)
  2047. })
  2048. return set
  2049. case "[object Map]":
  2050. var map = new Ctor()
  2051. val.forEach(function (v, k) {
  2052. map.set(k, v)
  2053. })
  2054. return map
  2055. }
  2056. }
  2057. return val
  2058. }
  2059. function copyValue (val, deep) {
  2060. if (isPlainObject(val)) {
  2061. return handleObjectAndArrayClone(objectMap, val, deep)
  2062. } else if (isArray(val)) {
  2063. return handleObjectAndArrayClone(map, val, deep)
  2064. }
  2065. return handleValueClone(val, deep)
  2066. }
  2067. /**
  2068. * 浅拷贝/深拷贝
  2069. *
  2070. * @param {Object} obj 对象/数组
  2071. * @param {Boolean} deep 是否深拷贝
  2072. * @return {Object}
  2073. */
  2074. function clone (obj, deep) {
  2075. if (obj) {
  2076. return copyValue(obj, deep)
  2077. }
  2078. return obj
  2079. }
  2080. /**
  2081. * 清空对象
  2082. *
  2083. * @param {Object} obj 对象
  2084. * @param {*} defs 默认值,如果不传(清空所有属性)、如果传对象(清空并继承)、如果传值(给所有赋值)
  2085. * @param {Object/Array} assigns 默认值
  2086. * @return {Object}
  2087. */
  2088. function clear (obj, defs, assigns) {
  2089. if (obj) {
  2090. var len
  2091. var isDefs = arguments.length > 1 && (isNull(defs) || !isObject(defs))
  2092. var extds = isDefs ? assigns : defs
  2093. if (isPlainObject(obj)) {
  2094. objectEach(obj, isDefs ? function (val, key) {
  2095. obj[key] = defs
  2096. } : function (val, key) {
  2097. helperDeleteProperty(obj, key)
  2098. })
  2099. if (extds) {
  2100. assign(obj, extds)
  2101. }
  2102. } else if (isArray(obj)) {
  2103. if (isDefs) {
  2104. len = obj.length
  2105. while (len > 0) {
  2106. len--
  2107. obj[len] = defs
  2108. }
  2109. } else {
  2110. obj.length = 0
  2111. }
  2112. if (extds) {
  2113. obj.push.apply(obj, extds)
  2114. }
  2115. }
  2116. }
  2117. return obj
  2118. }
  2119. function pluckProperty (name) {
  2120. return function (obj, key) {
  2121. return key === name
  2122. }
  2123. }
  2124. /**
  2125. * 移除对象属性
  2126. *
  2127. * @param {Object/Array} obj 对象/数组
  2128. * @param {Function/String} iterate 方法或属性
  2129. * @param {Object} context 上下文
  2130. * @return {Object/Array}
  2131. */
  2132. function remove (obj, iterate, context) {
  2133. if (obj) {
  2134. if (!eqNull(iterate)) {
  2135. var removeKeys = []
  2136. var rest = []
  2137. if (!isFunction(iterate)) {
  2138. iterate = pluckProperty(iterate)
  2139. }
  2140. each(obj, function (item, index, rest) {
  2141. if (iterate.call(context, item, index, rest)) {
  2142. removeKeys.push(index)
  2143. }
  2144. })
  2145. if (isArray(obj)) {
  2146. lastEach(removeKeys, function (item, key) {
  2147. rest.push(obj[item])
  2148. obj.splice(item, 1)
  2149. })
  2150. } else {
  2151. rest = {}
  2152. arrayEach(removeKeys, function (key) {
  2153. rest[key] = obj[key]
  2154. helperDeleteProperty(obj, key)
  2155. })
  2156. }
  2157. return rest
  2158. }
  2159. return clear(obj)
  2160. }
  2161. return obj
  2162. }
  2163. /**
  2164. * 序号列表生成函数
  2165. *
  2166. * @param {Number} start 起始值
  2167. * @param {Number} stop 结束值
  2168. * @param {Number} step 自增值
  2169. * @return {Object}
  2170. */
  2171. function range (start, stop, step) {
  2172. var index, len
  2173. var result = []
  2174. var args = arguments
  2175. if (args.length < 2) {
  2176. stop = args[0]
  2177. start = 0
  2178. }
  2179. index = start >> 0
  2180. len = stop >> 0
  2181. if (index < stop) {
  2182. step = step >> 0 || 1
  2183. for (; index < len; index += step) {
  2184. result.push(index)
  2185. }
  2186. }
  2187. return result
  2188. }
  2189. /**
  2190. * 将一个或者多个对象值解构到目标对象
  2191. *
  2192. * @param {Object} destination 目标对象
  2193. * @param {...Object}
  2194. * @return {Boolean}
  2195. */
  2196. function destructuring (destination, sources) {
  2197. if (destination && sources) {
  2198. var rest = assign.apply(this, [{}].concat(slice(arguments, 1)))
  2199. var restKeys = keys(rest)
  2200. arrayEach(keys(destination), function (key) {
  2201. if (includes(restKeys, key)) {
  2202. destination[key] = rest[key]
  2203. }
  2204. })
  2205. }
  2206. return destination
  2207. }
  2208. /**
  2209. * 获取一个指定范围内随机数
  2210. *
  2211. * @param {Number} minVal 最小值
  2212. * @param {Number} maxVal 最大值
  2213. * @return {Number}
  2214. */
  2215. function random (minVal, maxVal) {
  2216. return minVal >= maxVal ? minVal : ((minVal = minVal >> 0) + Math.round(Math.random() * ((maxVal || 9) - minVal)))
  2217. }
  2218. /**
  2219. * 获取最小值
  2220. *
  2221. * @param {Array} arr 数组
  2222. * @param {Function} iterate(item, index, obj) 回调
  2223. * @return {Number}
  2224. */
  2225. var min = helperCreateMinMax(function (rest, itemVal) {
  2226. return rest > itemVal
  2227. })
  2228. /**
  2229. * 获取最大值
  2230. *
  2231. * @param {Array} arr 数组
  2232. * @param {Function} iterate(item, index, obj) 回调
  2233. * @return {Number}
  2234. */
  2235. var max = helperCreateMinMax(function (rest, itemVal) {
  2236. return rest < itemVal
  2237. })
  2238. /**
  2239. * 千分位分隔符、小数点
  2240. *
  2241. * @param {String/Number} num 数值
  2242. * @param {CommafyOptions} options 参数
  2243. * @return {String}
  2244. */
  2245. function commafy(num, options) {
  2246. var opts = options || {}
  2247. var optDigits = opts.digits
  2248. var isNum = isNumber(num)
  2249. var rest, result, isNegative, intStr, floatStr
  2250. if (isNum) {
  2251. rest = (opts.ceil ? ceil : (opts.floor ? floor : round))(num, optDigits)
  2252. result = toNumberString(optDigits ? toFixed(rest, optDigits) : rest).split('.')
  2253. intStr = result[0]
  2254. floatStr = result[1]
  2255. isNegative = intStr && rest < 0
  2256. if (isNegative) {
  2257. intStr = intStr.substring(1, intStr.length)
  2258. }
  2259. } else {
  2260. rest = toValueString(num).replace(/,/g, '')
  2261. result = rest ? [rest] : []
  2262. intStr = result[0]
  2263. }
  2264. if (result.length) {
  2265. return (isNegative ? '-' : '') + intStr.replace(new RegExp('(?=(?!(\\b))(.{' + (opts.spaceNumber || 3) + '})+$)', 'g'), (opts.separator || ',')) + (floatStr ? ('.' + floatStr) : '')
  2266. }
  2267. return rest
  2268. }
  2269. /**
  2270. * 将数值四舍五入
  2271. *
  2272. * @param {string|number} num 数值
  2273. * @param {number} digits 小数保留位数
  2274. * @return {number}
  2275. */
  2276. var round = helperCreateMathNumber('round')
  2277. /**
  2278. * 将数值向上舍入
  2279. *
  2280. * @param {string|number} num 数值
  2281. * @param {number} digits 小数保留位数
  2282. * @return {number}
  2283. */
  2284. var ceil = helperCreateMathNumber('ceil')
  2285. /**
  2286. * 将数值向下舍入
  2287. *
  2288. * @param {string|number} num 数值
  2289. * @param {number} digits 小数保留位数
  2290. * @return {number}
  2291. */
  2292. var floor = helperCreateMathNumber('floor')
  2293. /**
  2294. * 将数值四舍五入并格式化为固定小数位的字符串
  2295. *
  2296. * @param {string|number} num 数值
  2297. * @param {number} digits 小数保留位数
  2298. * @return {String}
  2299. */
  2300. function toFixed (num, digits) {
  2301. digits = digits >> 0
  2302. var str = toValueString(round(num, digits))
  2303. var nums = str.split('.')
  2304. var intStr = nums[0]
  2305. var floatStr = nums[1] || ''
  2306. var digitOffsetIndex = digits - floatStr.length
  2307. if (digits) {
  2308. if (digitOffsetIndex > 0) {
  2309. return intStr + '.' + floatStr + helperStringRepeat('0', digitOffsetIndex)
  2310. }
  2311. return intStr + helperNumberOffsetPoint(floatStr, Math.abs(digitOffsetIndex))
  2312. }
  2313. return intStr
  2314. }
  2315. /**
  2316. * 转数值
  2317. * @param { String/Number } str 数值
  2318. *
  2319. * @return {Number}
  2320. */
  2321. var toNumber = helperCreateToNumber(parseFloat)
  2322. /**
  2323. * 数值转字符串,科学计数转字符串
  2324. * @param { Number } num 数值
  2325. *
  2326. * @return {Number}
  2327. */
  2328. function toNumberString(num) {
  2329. var rest = '' + num
  2330. var scienceMatchs = rest.match(/^([-+]?)((\d+)|((\d+)?[.](\d+)?))e([-+]{1})([0-9]+)$/)
  2331. if (scienceMatchs) {
  2332. var isNegative = num < 0
  2333. var absFlag = isNegative ? '-' : ''
  2334. var intNumStr = scienceMatchs[3] || ''
  2335. var dIntNumStr = scienceMatchs[5] || ''
  2336. var dFloatNumStr = scienceMatchs[6] || ''
  2337. var sciencFlag = scienceMatchs[7]
  2338. var scienceNumStr = scienceMatchs[8]
  2339. var floatOffsetIndex = scienceNumStr - dFloatNumStr.length
  2340. var intOffsetIndex = scienceNumStr - intNumStr.length
  2341. var dIntOffsetIndex = scienceNumStr - dIntNumStr.length
  2342. if (sciencFlag === '+') {
  2343. if (intNumStr) {
  2344. return absFlag + intNumStr + helperStringRepeat('0', scienceNumStr)
  2345. }
  2346. if (floatOffsetIndex > 0) {
  2347. return absFlag + dIntNumStr + dFloatNumStr + helperStringRepeat('0', floatOffsetIndex)
  2348. }
  2349. return absFlag + dIntNumStr + helperNumberOffsetPoint(dFloatNumStr, scienceNumStr)
  2350. }
  2351. if (intNumStr) {
  2352. if (intOffsetIndex > 0) {
  2353. return absFlag + '0.' + helperStringRepeat('0', Math.abs(intOffsetIndex)) + intNumStr
  2354. }
  2355. return absFlag + helperNumberOffsetPoint(intNumStr, intOffsetIndex)
  2356. }
  2357. if (dIntOffsetIndex > 0) {
  2358. return absFlag + '0.' + helperStringRepeat('0', Math.abs(dIntOffsetIndex)) + dIntNumStr + dFloatNumStr
  2359. }
  2360. return absFlag + helperNumberOffsetPoint(dIntNumStr, dIntOffsetIndex) + dFloatNumStr
  2361. }
  2362. return rest
  2363. }
  2364. /**
  2365. * 转整数
  2366. * @param { String/Number } str 数值
  2367. *
  2368. * @return {Number}
  2369. */
  2370. var toInteger = helperCreateToNumber(staticParseInt)
  2371. /**
  2372. * 加法运算
  2373. *
  2374. * @param { Number } num1 被加数
  2375. * @param { Number } num2 加数
  2376. * @return {Number}
  2377. */
  2378. function add (num1, num2) {
  2379. return helperNumberAdd(toNumber(num1), toNumber(num2))
  2380. }
  2381. /**
  2382. * 减法运算
  2383. *
  2384. * @param { Number } num1 被减数
  2385. * @param { Number } num2 减数
  2386. * @return {Number}
  2387. */
  2388. function subtract (num1, num2) {
  2389. var subtrahend = toNumber(num1)
  2390. var minuend = toNumber(num2)
  2391. var str1 = toNumberString(subtrahend)
  2392. var str2 = toNumberString(minuend)
  2393. var digit1 = helperNumberDecimal(str1)
  2394. var digit2 = helperNumberDecimal(str2)
  2395. var ratio = Math.pow(10, Math.max(digit1, digit2))
  2396. var precision = (digit1 >= digit2) ? digit1 : digit2
  2397. return parseFloat(toFixed((subtrahend * ratio - minuend * ratio) / ratio, precision))
  2398. }
  2399. /**
  2400. * 乘法运算
  2401. *
  2402. * @param { Number } num1 数值1
  2403. * @param { Number } num2 数值2
  2404. * @return {Number}
  2405. */
  2406. function multiply (num1, num2) {
  2407. var multiplier = toNumber(num1)
  2408. var multiplicand = toNumber(num2)
  2409. var str1 = toNumberString(multiplier)
  2410. var str2 = toNumberString(multiplicand)
  2411. return parseInt(str1.replace('.', '')) * parseInt(str2.replace('.', '')) / Math.pow(10, helperNumberDecimal(str1) + helperNumberDecimal(str2))
  2412. }
  2413. /**
  2414. * 除法运算
  2415. *
  2416. * @param { Number } num1 数值1
  2417. * @param { Number } num2 数值2
  2418. * @return {Number}
  2419. */
  2420. function divide (num1, num2) {
  2421. return helperNumberDivide(toNumber(num1), toNumber(num2))
  2422. }
  2423. /**
  2424. * 求和函数,将数值相加
  2425. *
  2426. * @param {Array} array 数组
  2427. * @param {Function/String} iterate 方法或属性
  2428. * @param {Object} context 上下文
  2429. * @return {Number}
  2430. */
  2431. function sum (array, iterate, context) {
  2432. var result = 0
  2433. each(array, iterate ? isFunction(iterate) ? function () {
  2434. result = helperNumberAdd(result, iterate.apply(context, arguments))
  2435. } : function (val) {
  2436. result = helperNumberAdd(result, get(val, iterate))
  2437. } : function (val) {
  2438. result = helperNumberAdd(result, val)
  2439. })
  2440. return result
  2441. }
  2442. /**
  2443. * 求平均值函数
  2444. *
  2445. * @param {Array} array 数组
  2446. * @param {Function/String} iterate 方法或属性
  2447. * @param {Object} context 上下文
  2448. * @return {Number}
  2449. */
  2450. function mean (array, iterate, context) {
  2451. return helperNumberDivide(sum(array, iterate, context), getSize(array))
  2452. }
  2453. /**
  2454. * 返回当前时间戳
  2455. *
  2456. * @returns Number
  2457. */
  2458. var now = Date.now || function () {
  2459. return helperGetDateTime(helperNewDate())
  2460. }
  2461. /**
  2462. * 将日期格式化为时间戳
  2463. *
  2464. * @param {String/Number/Date} str 日期或数字
  2465. * @param {String} format 解析日期格式
  2466. * @returns Number
  2467. */
  2468. var timestamp = function (str, format) {
  2469. if (str) {
  2470. var date = toStringDate(str, format)
  2471. return isDate(date) ? helperGetDateTime(date) : date
  2472. }
  2473. return now()
  2474. }
  2475. /**
  2476. * 判断是否有效的Date对象
  2477. *
  2478. * @param {any} val 对象
  2479. * @return {boolean}
  2480. */
  2481. function isValidDate (val) {
  2482. return isDate(val) && !isNaN(helperGetDateTime(val))
  2483. }
  2484. /**
  2485. * 比较两个日期
  2486. *
  2487. * @param {Number/String/Date} date1 日期
  2488. * @param {Number/String/Date} date2 日期
  2489. * @param {String} format 对比格式
  2490. */
  2491. function isDateSame (date1, date2, format) {
  2492. if (date1 && date2) {
  2493. date1 = toDateString(date1, format)
  2494. return date1 !== 'Invalid Date' && date1 === toDateString(date2, format)
  2495. }
  2496. return false
  2497. }
  2498. var dateFormatRules = [
  2499. { rules: [['yyyy', 4]] },
  2500. { rules: [['MM', 2], ['M', 1]], offset: -1 },
  2501. { rules: [['dd', 2], ['d', 1]] },
  2502. { rules: [['HH', 2], ['H', 1]] },
  2503. { rules: [['mm', 2], ['m', 1]] },
  2504. { rules: [['ss', 2], ['s', 1]] },
  2505. { rules: [['SSS', 3], ['S', 1]] },
  2506. { rules: [['ZZ', 5], ['Z', 6], ['Z', 5], ['Z', 1]] }
  2507. ]
  2508. function parseStringDate (str, format) {
  2509. var arr, sIndex, fIndex, fLen, fItem, rules, rIndex, rLen, tempMatch
  2510. var dates = [0, 0, 1, 0, 0, 0, 0]
  2511. for (fIndex = 0, fLen = dateFormatRules.length; fIndex < fLen; fIndex++) {
  2512. fItem = dateFormatRules[fIndex]
  2513. for (rIndex = 0, rules = fItem.rules, rLen = rules.length; rIndex < rLen; rIndex++) {
  2514. arr = rules[rIndex]
  2515. sIndex = format.indexOf(arr[0])
  2516. if (sIndex > -1) {
  2517. tempMatch = str.substring(sIndex, sIndex + arr[1])
  2518. if (tempMatch && tempMatch.length === arr[1]) {
  2519. if (fItem.offset) {
  2520. tempMatch = staticParseInt(tempMatch) + fItem.offset
  2521. }
  2522. dates[fIndex] = tempMatch
  2523. break
  2524. }
  2525. }
  2526. if (rIndex === rLen - 1) {
  2527. return dates
  2528. }
  2529. }
  2530. }
  2531. return dates
  2532. }
  2533. /**
  2534. * 字符串转为日期
  2535. *
  2536. * @param {String/Number/Date} str 日期或数字
  2537. * @param {String} format 解析日期格式(yyyy年份、MM月份、dd天、hh(12)HH(24)小时、mm分钟、ss秒、SSS毫秒、Z时区)
  2538. * @return {Date}
  2539. */
  2540. function toStringDate (str, format) {
  2541. var rest, isDType
  2542. if (str) {
  2543. isDType = isDate(str)
  2544. if (isDType || (!format && /^[0-9]{11,15}$/.test(str))) {
  2545. rest = new Date(isDType ? helperGetDateTime(str) : staticParseInt(str))
  2546. } else if (isString(str)) {
  2547. var tempMatch
  2548. var dates = parseStringDate(str, format || setupDefaults.formatDate)
  2549. var zStr = dates[7]
  2550. if (dates[0]) {
  2551. // 解析时区
  2552. if (zStr) {
  2553. // 如果为UTC 时间
  2554. if (zStr[0] === 'z' || zStr[0] === 'Z') {
  2555. rest = new Date(helperGetUTCDateTime(dates))
  2556. } else {
  2557. // 如果指定时区,时区转换
  2558. tempMatch = zStr.match(/([-+]{1})(\d{2}):?(\d{2})/)
  2559. if (tempMatch) {
  2560. rest = new Date(helperGetUTCDateTime(dates) - (tempMatch[1] === '-' ? -1 : 1) * staticParseInt(tempMatch[2]) * 3600000 + staticParseInt(tempMatch[3]) * 60000)
  2561. }
  2562. }
  2563. } else {
  2564. rest = new Date(dates[0], dates[1], dates[2], dates[3], dates[4], dates[5], dates[6])
  2565. }
  2566. }
  2567. }
  2568. }
  2569. return rest ? rest : new Date('')
  2570. }
  2571. function handleCustomTemplate (date, formats, match, value) {
  2572. var format = formats[match]
  2573. if (format) {
  2574. if (isFunction(format)) {
  2575. return format(value, match, date)
  2576. } else {
  2577. return format[value]
  2578. }
  2579. }
  2580. return value
  2581. }
  2582. function formatDayE (day) {
  2583. return day === 0 ? 7 : day
  2584. }
  2585. /**
  2586. * 日期格式化为字符串,转义符号 []
  2587. *
  2588. * @param {Date} date 日期或数字
  2589. * @param {String} format 输出日期格式(年份(yy|yyyy)、月份(M|MM自动补0)、天(d|dd自动补0)、12小时制(h|hh自动补0)、24小时制(H|HH自动补0)、分钟(m|mm自动补0)、秒(s|ss自动补0)、毫秒(S|SSS自动补0)、D当年的第几天、a/A上午下午、e/E星期几、w当年的第几周、W当月的第几周、q当年第几个季度、Z时区)
  2590. * @param {Object} options {formats: {q: ['日', '一', '二', '三', '四', '五', '六'], E: function (value, match, date) {return '三'}, }} 自定义格式化模板
  2591. * @return {String}
  2592. */
  2593. var dateFormatRE = /\[([^\]]+)]|y{2,4}|M{1,2}|d{1,2}|H{1,2}|h{1,2}|m{1,2}|s{1,2}|S{1,3}|Z{1,2}|W{1,2}|D{1,3}|[aAeEq]/g
  2594. function toDateString (date, format, options) {
  2595. if (date) {
  2596. date = toStringDate(date)
  2597. if (isValidDate(date)) {
  2598. var result = format || setupDefaults.formatString
  2599. var hours = date.getHours()
  2600. var apm = hours < 12 ? 'am' : 'pm'
  2601. var formats = assign({}, setupDefaults.formatStringMatchs, options ? options.formats : null)
  2602. var fy = function (match, length) {
  2603. return ('' + helperGetDateFullYear(date)).substr(4 - length)
  2604. }
  2605. var fM = function (match, length) {
  2606. return padStart(helperGetDateMonth(date) + 1, length, '0')
  2607. }
  2608. var fd = function (match, length) {
  2609. return padStart(date.getDate(), length, '0')
  2610. }
  2611. var fH = function (match, length) {
  2612. return padStart(hours, length, '0')
  2613. }
  2614. var fh = function (match, length) {
  2615. return padStart(hours <= 12 ? hours : hours - 12, length, '0')
  2616. }
  2617. var fm = function (match, length) {
  2618. return padStart(date.getMinutes(), length, '0')
  2619. }
  2620. var fs = function (match, length) {
  2621. return padStart(date.getSeconds(), length, '0')
  2622. }
  2623. var fS = function (match, length) {
  2624. return padStart(date.getMilliseconds(), length, '0')
  2625. }
  2626. var fZ = function (match, length) {
  2627. var zoneHours = date.getTimezoneOffset() / 60 * -1
  2628. return handleCustomTemplate(date, formats, match, (zoneHours >= 0 ? '+' : '-') + padStart(zoneHours, 2, '0') + (length === 1 ? ':' : '') + '00')
  2629. }
  2630. var fW = function (match, length) {
  2631. return padStart(handleCustomTemplate(date, formats, match, getYearWeek(date)), length, '0')
  2632. }
  2633. var fD = function (match, length) {
  2634. return padStart(handleCustomTemplate(date, formats, match, getYearDay(date)), length, '0')
  2635. }
  2636. var parseDates = {
  2637. yyyy: fy,
  2638. yy: fy,
  2639. MM: fM,
  2640. M: fM,
  2641. dd: fd,
  2642. d: fd,
  2643. HH: fH,
  2644. H: fH,
  2645. hh: fh,
  2646. h: fh,
  2647. mm: fm,
  2648. m: fm,
  2649. ss: fs,
  2650. s: fs,
  2651. SSS: fS,
  2652. S: fS,
  2653. ZZ: fZ,
  2654. Z: fZ,
  2655. WW: fW,
  2656. W: fW,
  2657. DDD: fD,
  2658. D: fD,
  2659. a: function (match) {
  2660. return handleCustomTemplate(date, formats, match, apm)
  2661. },
  2662. A: function (match) {
  2663. return handleCustomTemplate(date, formats, match, helperStringUpperCase(apm))
  2664. },
  2665. e: function (match) {
  2666. return handleCustomTemplate(date, formats, match, date.getDay())
  2667. },
  2668. E: function (match) {
  2669. return handleCustomTemplate(date, formats, match, formatDayE(date.getDay()))
  2670. },
  2671. q: function (match) {
  2672. return handleCustomTemplate(date, formats, match, Math.floor((helperGetDateMonth(date) + 3) / 3))
  2673. }
  2674. }
  2675. return result.replace(dateFormatRE, function (match, skip) {
  2676. return skip || (parseDates[match] ? parseDates[match](match, match.length) : match)
  2677. })
  2678. }
  2679. return 'Invalid Date'
  2680. }
  2681. return ''
  2682. }
  2683. /**
  2684. * 返回前几年或后几年的日期
  2685. *
  2686. * @param {Date} date 日期或数字
  2687. * @param {Number} year 年(默认当前年)、前几个年(数值)、后几个年(数值)
  2688. * @param {Number/String} month 获取哪月(null默认当前年)、年初(first)、年末(last)、指定月份(0-11)
  2689. * @return {Date}
  2690. */
  2691. function getWhatYear (date, year, month) {
  2692. var number
  2693. date = toStringDate(date)
  2694. if (isValidDate(date)) {
  2695. if (year) {
  2696. number = year && !isNaN(year) ? year : 0
  2697. date.setFullYear(helperGetDateFullYear(date) + number)
  2698. }
  2699. if (month || !isNaN(month)) {
  2700. if (month === staticStrFirst) {
  2701. return new Date(helperGetDateFullYear(date), 0, 1)
  2702. } else if (month === staticStrLast) {
  2703. date.setMonth(11)
  2704. return getWhatMonth(date, 0, staticStrLast)
  2705. } else {
  2706. date.setMonth(month)
  2707. }
  2708. }
  2709. }
  2710. return date
  2711. }
  2712. /**
  2713. * 返回前几月或后几月的日期
  2714. *
  2715. * @param {Date} date 日期或数字
  2716. * @param {Number} month 月(默认当前月)、前几个月、后几个月
  2717. * @param {Number/String} day 获取哪天:月初(first)、月末(last)、指定天数(数值),如果为空,但超过指定月份的天数时,则默认单月最后一天
  2718. * @return {Date}
  2719. */
  2720. function getWhatMonth (date, month, day) {
  2721. var monthOffset = month && !isNaN(month) ? month : 0
  2722. date = toStringDate(date)
  2723. if (isValidDate(date)) {
  2724. if (day === staticStrFirst) {
  2725. return new Date(helperGetDateFullYear(date), helperGetDateMonth(date) + monthOffset, 1)
  2726. } else if (day === staticStrLast) {
  2727. return new Date(helperGetDateTime(getWhatMonth(date, monthOffset + 1, staticStrFirst)) - 1)
  2728. } else if (isNumber(day)) {
  2729. date.setDate(day)
  2730. }
  2731. if (monthOffset) {
  2732. var currDate = date.getDate()
  2733. date.setMonth(helperGetDateMonth(date) + monthOffset)
  2734. if (currDate !== date.getDate()) {
  2735. // 当为指定天数,且被跨月了,则默认单月最后一天
  2736. date.setDate(1)
  2737. return new Date(helperGetDateTime(date) - staticDayTime)
  2738. }
  2739. }
  2740. }
  2741. return date
  2742. }
  2743. /**
  2744. * 返回前几周或后几周的星期几
  2745. *
  2746. * @param {Date} date 日期
  2747. * @param {Number} week 周(默认当前周)、前几周、后几周
  2748. * @param {Number} day 星期天(默认0)、星期一(1)、星期二(2)、星期三(3)、星期四(4)、星期五(5)、星期六(6)
  2749. * @return {Date}
  2750. */
  2751. function getWhatWeek (date, week, day) {
  2752. var time, whatDayTime, currentDay, customDay
  2753. date = toStringDate(date)
  2754. if (isValidDate(date)) {
  2755. customDay = staticParseInt(/^[0-7]$/.test(day) ? day : date.getDay())
  2756. currentDay = date.getDay()
  2757. time = helperGetDateTime(date)
  2758. whatDayTime = time + ((customDay === 0 ? 7 : customDay) - (currentDay === 0 ? 7 : currentDay)) * staticDayTime
  2759. if (week && !isNaN(week)) {
  2760. whatDayTime += week * staticWeekTime
  2761. }
  2762. return new Date(whatDayTime)
  2763. }
  2764. return date
  2765. }
  2766. /**
  2767. * 返回前几天或后几天的日期
  2768. *
  2769. * @param {Date} date 日期或数字
  2770. * @param {Number} day 天(默认当天)、前几天、后几天
  2771. * @param {String} mode 获取时分秒(null默认当前时分秒)、日初(first)、日末(last)
  2772. * @return {Date}
  2773. */
  2774. function getWhatDay (date, day, mode) {
  2775. date = toStringDate(date)
  2776. if (isValidDate(date) && !isNaN(day)) {
  2777. date.setDate(date.getDate() + staticParseInt(day))
  2778. if (mode === staticStrFirst) {
  2779. return new Date(helperGetDateFullYear(date), helperGetDateMonth(date), date.getDate())
  2780. } else if (mode === staticStrLast) {
  2781. return new Date(helperGetDateTime(getWhatDay(date, 1, staticStrFirst)) - 1)
  2782. }
  2783. }
  2784. return date
  2785. }
  2786. /**
  2787. * 返回某个年份的第几天
  2788. *
  2789. * @param {Date} date 日期或数字
  2790. * @return {Number}
  2791. */
  2792. function getYearDay (date) {
  2793. date = toStringDate(date)
  2794. if (isValidDate(date)) {
  2795. return Math.floor((helperGetYMDTime(date) - helperGetYMDTime(getWhatYear(date, 0, staticStrFirst))) / staticDayTime) + 1
  2796. }
  2797. return NaN
  2798. }
  2799. /**
  2800. * 返回某个年份的第几周
  2801. *
  2802. * @param {Date} date 日期或数字
  2803. * @return {Number}
  2804. */
  2805. function getYearWeek (date) {
  2806. date = toStringDate(date)
  2807. if (isValidDate(date)) {
  2808. date.setHours(0, 0, 0, 0)
  2809. date.setDate(date.getDate() + 3 - (date.getDay() + 6) % 7)
  2810. var week = new Date(date.getFullYear(), 0, 4)
  2811. return Math.round(((date.getTime() - week.getTime()) / staticDayTime + (week.getDay() + 6) % 7 - 3) / 7) + 1
  2812. }
  2813. return NaN
  2814. }
  2815. /**
  2816. * 返回某个月的第几周
  2817. *
  2818. * @param {Date} date 日期或数字
  2819. * @return {Number}
  2820. */
  2821. function getMonthWeek (date) {
  2822. var monthFirst, monthFirstWeek
  2823. var currentDate = toStringDate(date)
  2824. if (isValidDate(currentDate)) {
  2825. monthFirst = getWhatMonth(currentDate, 0, staticStrFirst)
  2826. monthFirstWeek = getWhatWeek(monthFirst, 0, 1)
  2827. if (monthFirstWeek < monthFirst) {
  2828. monthFirstWeek = getWhatWeek(monthFirst, 1, 1)
  2829. }
  2830. if (currentDate >= monthFirstWeek) {
  2831. return Math.floor((helperGetYMDTime(currentDate) - helperGetYMDTime(monthFirstWeek)) / staticWeekTime) + 1
  2832. }
  2833. return getMonthWeek(getWhatWeek(currentDate, 0, 1))
  2834. }
  2835. return NaN
  2836. }
  2837. /**
  2838. * 返回某个年份的天数
  2839. *
  2840. * @param {Date} date 日期或数字
  2841. * @param {Number} offset 年(默认当年)、前几个年、后几个年
  2842. * @return {Number}
  2843. */
  2844. function getDayOfYear (date, year) {
  2845. date = toStringDate(date)
  2846. if (isValidDate(date)) {
  2847. return isLeapYear(getWhatYear(date, year)) ? 366 : 365
  2848. }
  2849. return NaN
  2850. }
  2851. /**
  2852. * 返回某个月份的天数
  2853. *
  2854. * @param {Date} date 日期或数字
  2855. * @param {Number} offset 月(默认当月)、前几个月、后几个月
  2856. * @return {Number}
  2857. */
  2858. function getDayOfMonth (date, month) {
  2859. date = toStringDate(date)
  2860. if (isValidDate(date)) {
  2861. return Math.floor((helperGetDateTime(getWhatMonth(date, month, staticStrLast)) - helperGetDateTime(getWhatMonth(date, month, staticStrFirst))) / staticDayTime) + 1
  2862. }
  2863. return NaN
  2864. }
  2865. /**
  2866. * 返回两个日期之间差距,如果结束日期小于开始日期done为fasle
  2867. *
  2868. * @param {Date} startDate 开始日期
  2869. * @param {Date} endDate 结束日期或当期日期
  2870. * @param {Date} rule 自定义计算规则
  2871. * @return {Object}
  2872. */
  2873. function getDateDiff (startDate, endDate, rules) {
  2874. var startTime, endTime, item, diffTime, rule, len, index
  2875. var result = { done: false, time: 0 }
  2876. startDate = toStringDate(startDate)
  2877. endDate = endDate ? toStringDate(endDate) : helperNewDate()
  2878. if (isValidDate(startDate) && isValidDate(endDate)) {
  2879. startTime = helperGetDateTime(startDate)
  2880. endTime = helperGetDateTime(endDate)
  2881. if (startTime < endTime) {
  2882. diffTime = result.time = endTime - startTime
  2883. rule = rules && rules.length > 0 ? rules : setupDefaults.dateDiffRules
  2884. result.done = true
  2885. for (index = 0, len = rule.length; index < len; index++) {
  2886. item = rule[index]
  2887. if (diffTime >= item[1]) {
  2888. if (index === len - 1) {
  2889. result[item[0]] = diffTime || 0
  2890. } else {
  2891. result[item[0]] = Math.floor(diffTime / item[1])
  2892. diffTime -= result[item[0]] * item[1]
  2893. }
  2894. } else {
  2895. result[item[0]] = 0
  2896. }
  2897. }
  2898. }
  2899. }
  2900. return result
  2901. }
  2902. /**
  2903. * 去除字符串左右两边的空格
  2904. *
  2905. * @param {String} str 字符串
  2906. * @return {String}
  2907. */
  2908. function trim (str) {
  2909. return str && str.trim ? str.trim() : trimRight(trimLeft(str))
  2910. }
  2911. /**
  2912. * 去除字符串左边的空格
  2913. *
  2914. * @param {String} str 字符串
  2915. * @return {String}
  2916. */
  2917. function trimLeft (str) {
  2918. return str && str.trimLeft ? str.trimLeft() : toValueString(str).replace(/^[\s\uFEFF\xA0]+/g, '')
  2919. }
  2920. /**
  2921. * 去除字符串右边的空格
  2922. *
  2923. * @param {String} str 字符串
  2924. * @return {String}
  2925. */
  2926. function trimRight (str) {
  2927. return str && str.trimRight ? str.trimRight() : toValueString(str).replace(/[\s\uFEFF\xA0]+$/g, '')
  2928. }
  2929. /**
  2930. * 转义HTML字符串,替换&, <, >, ", ', `字符
  2931. *
  2932. * @param {String} str 字符串
  2933. * @return {String}
  2934. */
  2935. var escape = helperFormatEscaper(staticEscapeMap)
  2936. var unescapeMap = {}
  2937. each(staticEscapeMap, function (item, key) {
  2938. unescapeMap[staticEscapeMap[key]] = key
  2939. })
  2940. /**
  2941. * 反转escape
  2942. *
  2943. * @param {String} str 字符串
  2944. * @return {String}
  2945. */
  2946. var unescape = helperFormatEscaper(unescapeMap)
  2947. var camelCacheMaps = {}
  2948. /**
  2949. * 将带字符串转成驼峰字符串,例如: project-name 转为 projectName
  2950. *
  2951. * @param {String} str 字符串
  2952. * @return {String}
  2953. */
  2954. function camelCase (str) {
  2955. str = toValueString(str)
  2956. if (camelCacheMaps[str]) {
  2957. return camelCacheMaps[str]
  2958. }
  2959. var strLen = str.length
  2960. var rest = str.replace(/([-]+)/g, function (text, flag, index) {
  2961. return index && index + flag.length < strLen ? '-' : ''
  2962. })
  2963. strLen = rest.length
  2964. rest = rest.replace(/([A-Z]+)/g, function (text, upper, index) {
  2965. var upperLen = upper.length
  2966. upper = helperStringLowerCase(upper)
  2967. if (index) {
  2968. if (upperLen > 2 && index + upperLen < strLen) {
  2969. return helperStringUpperCase(helperStringSubstring(upper, 0, 1)) + helperStringSubstring(upper, 1, upperLen - 1) + helperStringUpperCase(helperStringSubstring(upper, upperLen - 1, upperLen))
  2970. }
  2971. return helperStringUpperCase(helperStringSubstring(upper, 0, 1)) + helperStringSubstring(upper, 1, upperLen)
  2972. } else {
  2973. if (upperLen > 1 && index + upperLen < strLen) {
  2974. return helperStringSubstring(upper, 0, upperLen - 1) + helperStringUpperCase(helperStringSubstring(upper, upperLen - 1, upperLen))
  2975. }
  2976. }
  2977. return upper
  2978. }).replace(/(-[a-zA-Z])/g, function (text, upper) {
  2979. return helperStringUpperCase(helperStringSubstring(upper, 1, upper.length))
  2980. })
  2981. camelCacheMaps[str] = rest
  2982. return rest
  2983. }
  2984. var kebabCacheMaps = {}
  2985. /**
  2986. * 将带驼峰字符串转成字符串,例如: projectName 转为 project-name
  2987. *
  2988. * @param {String} str 字符串
  2989. * @return {String}
  2990. */
  2991. function kebabCase (str) {
  2992. str = toValueString(str)
  2993. if (kebabCacheMaps[str]) {
  2994. return kebabCacheMaps[str]
  2995. }
  2996. if (/^[A-Z]+$/.test(str)) {
  2997. return helperStringLowerCase(str)
  2998. }
  2999. var rest = str.replace(/^([a-z])([A-Z]+)([a-z]+)$/, function (text, prevLower, upper, nextLower) {
  3000. var upperLen = upper.length
  3001. if (upperLen > 1) {
  3002. return prevLower + '-' + helperStringLowerCase(helperStringSubstring(upper, 0, upperLen - 1)) + '-' + helperStringLowerCase(helperStringSubstring(upper, upperLen - 1, upperLen)) + nextLower
  3003. }
  3004. return helperStringLowerCase(prevLower + '-' + upper + nextLower)
  3005. }).replace(/^([A-Z]+)([a-z]+)?$/, function (text, upper, nextLower) {
  3006. var upperLen = upper.length
  3007. return helperStringLowerCase(helperStringSubstring(upper, 0, upperLen - 1) + '-' + helperStringSubstring(upper, upperLen - 1, upperLen) + (nextLower || ''))
  3008. }).replace(/([a-z]?)([A-Z]+)([a-z]?)/g, function (text, prevLower, upper, nextLower, index) {
  3009. var upperLen = upper.length
  3010. if (upperLen > 1) {
  3011. if (prevLower) {
  3012. prevLower += '-'
  3013. }
  3014. if (nextLower) {
  3015. return (prevLower || '') + helperStringLowerCase(helperStringSubstring(upper, 0, upperLen - 1)) + '-' + helperStringLowerCase(helperStringSubstring(upper, upperLen - 1, upperLen)) + nextLower
  3016. }
  3017. }
  3018. return (prevLower || '') + (index ? '-' : '') + helperStringLowerCase(upper) + (nextLower || '')
  3019. })
  3020. rest = rest.replace(/([-]+)/g, function (text, flag, index) {
  3021. return index && index + flag.length < rest.length ? '-' : ''
  3022. })
  3023. kebabCacheMaps[str] = rest
  3024. return rest
  3025. }
  3026. /**
  3027. * 将字符串重复 n 次
  3028. *
  3029. * @param {String} str 字符串
  3030. * @param {Number} count 次数
  3031. * @return {String}
  3032. */
  3033. function repeat (str, count) {
  3034. return helperStringRepeat(toValueString(str), count)
  3035. }
  3036. /**
  3037. * 用指定字符从前面开始补全字符串
  3038. *
  3039. * @param {String} str 字符串
  3040. * @param {Number} targetLength 结果长度
  3041. * @param {Number} padString 补全字符
  3042. * @return {String}
  3043. */
  3044. function padStart (str, targetLength, padString) {
  3045. var rest = toValueString(str)
  3046. targetLength = targetLength >> 0
  3047. padString = isUndefined(padString) ? ' ' : '' + padString
  3048. if (rest.padStart) {
  3049. return rest.padStart(targetLength, padString)
  3050. }
  3051. if (targetLength > rest.length) {
  3052. targetLength -= rest.length
  3053. if (targetLength > padString.length) {
  3054. padString += helperStringRepeat(padString, targetLength / padString.length)
  3055. }
  3056. return padString.slice(0, targetLength) + rest
  3057. }
  3058. return rest
  3059. }
  3060. /**
  3061. * 用指定字符从后面开始补全字符串
  3062. *
  3063. * @param {String} str 字符串
  3064. * @param {Number} targetLength 结果长度
  3065. * @param {Number} padString 补全字符
  3066. * @return {String}
  3067. */
  3068. function padEnd (str, targetLength, padString) {
  3069. var rest = toValueString(str)
  3070. targetLength = targetLength >> 0
  3071. padString = isUndefined(padString) ? ' ' : '' + padString
  3072. if (rest.padEnd) {
  3073. return rest.padEnd(targetLength, padString)
  3074. }
  3075. if (targetLength > rest.length) {
  3076. targetLength -= rest.length
  3077. if (targetLength > padString.length) {
  3078. padString += helperStringRepeat(padString, targetLength / padString.length)
  3079. }
  3080. return rest + padString.slice(0, targetLength)
  3081. }
  3082. return rest
  3083. }
  3084. /**
  3085. * 判断字符串是否在源字符串的头部
  3086. *
  3087. * @param {String} str 字符串
  3088. * @param {String/Number} val 值
  3089. * @param {Number} startIndex 开始索引
  3090. * @return {String}
  3091. */
  3092. function startsWith (str, val, startIndex) {
  3093. var rest = toValueString(str)
  3094. return (arguments.length === 1 ? rest : rest.substring(startIndex)).indexOf(val) === 0
  3095. }
  3096. /**
  3097. * 判断字符串是否在源字符串的尾部
  3098. *
  3099. * @param {String} str 字符串
  3100. * @param {String/Number} val 值
  3101. * @param {Number} startIndex 开始索引
  3102. * @return {String}
  3103. */
  3104. function endsWith (str, val, startIndex) {
  3105. var rest = toValueString(str)
  3106. var argsLen = arguments.length
  3107. return argsLen > 1 && (argsLen > 2 ? rest.substring(0, startIndex).indexOf(val) === startIndex - 1 : rest.indexOf(val) === rest.length - 1)
  3108. }
  3109. /**
  3110. * 解析动态字符串模板
  3111. * @param {atring} str 字符串模板
  3112. * @param {any | any[]} args 对象
  3113. * @param {any} options
  3114. */
  3115. function template (str, args, options) {
  3116. return toValueString(str).replace((options || setupDefaults).tmplRE || /\{{2}([.\w[\]\s]+)\}{2}/g, function (match, key) {
  3117. return get(args, trim(key))
  3118. })
  3119. }
  3120. /**
  3121. * 字符串格式化占位符
  3122. * @param { string } str
  3123. * @param { object | any[] } obj
  3124. */
  3125. function toFormatString (str, obj) {
  3126. return template(str, obj,{ tmplRE: /\{([.\w[\]\s]+)\}/g })
  3127. }
  3128. function toValueString (obj) {
  3129. if (isNumber(obj)) {
  3130. return toNumberString(obj)
  3131. }
  3132. return '' + (eqNull(obj) ? '' : obj)
  3133. }
  3134. /**
  3135. * 一个空的方法,始终返回 undefined,可用于初始化值
  3136. */
  3137. function noop () {}
  3138. /**
  3139. * 返回一个获取对象属性的函数
  3140. *
  3141. * @param {String} name 属性名
  3142. * @param {Object} defs 空值
  3143. */
  3144. function property (name, defs) {
  3145. return function (obj) {
  3146. return isNull(obj) ? defs : obj[name]
  3147. }
  3148. }
  3149. /**
  3150. * 创建一个绑定上下文的函数
  3151. *
  3152. * @param {Function} callback 函数
  3153. * @param {Object} context 上下文
  3154. * @param {*} args 额外的参数
  3155. * @return {Object}
  3156. */
  3157. function bind (callback, context) {
  3158. var args = slice(arguments, 2)
  3159. return function () {
  3160. return callback.apply(context, slice(arguments).concat(args))
  3161. }
  3162. }
  3163. /**
  3164. * 创建一个只能调用一次的函数,只会返回第一次执行后的结果
  3165. *
  3166. * @param {Function} callback 函数
  3167. * @param {Object} context 上下文
  3168. * @param {*} args 额外的参数
  3169. * @return {Object}
  3170. */
  3171. function once (callback, context) {
  3172. var done = false
  3173. var rest = null
  3174. var args = slice(arguments, 2)
  3175. return function () {
  3176. if (done) {
  3177. return rest
  3178. }
  3179. rest = callback.apply(context, slice(arguments).concat(args))
  3180. done = true
  3181. return rest
  3182. }
  3183. }
  3184. /**
  3185. * 创建一个函数, 调用次数超过 count 次之后执行回调并将所有结果记住后返回
  3186. *
  3187. * @param {Number} count 调用次数
  3188. * @param {Function} callback 完成回调
  3189. * @return {Object}
  3190. */
  3191. function after (count, callback, context) {
  3192. var runCount = 0
  3193. var rests = []
  3194. return function () {
  3195. var args = arguments
  3196. runCount++
  3197. if (runCount <= count) {
  3198. rests.push(args[0])
  3199. }
  3200. if (runCount >= count) {
  3201. callback.apply(context, [rests].concat(slice(args)))
  3202. }
  3203. }
  3204. }
  3205. /**
  3206. * 创建一个函数, 调用次数不超过 count 次之前执行回调并将所有结果记住后返回
  3207. *
  3208. * @param {Number} count 调用次数
  3209. * @param {Function} callback 完成回调
  3210. * @return {Object}
  3211. */
  3212. function before (count, callback, context) {
  3213. var runCount = 0
  3214. var rests = []
  3215. context = context || this
  3216. return function () {
  3217. var args = arguments
  3218. runCount++
  3219. if (runCount < count) {
  3220. rests.push(args[0])
  3221. callback.apply(context, [rests].concat(slice(args)))
  3222. }
  3223. }
  3224. }
  3225. /**
  3226. * 节流函数;当被调用 n 毫秒后才会执行,如果在这时间内又被调用则至少每隔 n 秒毫秒调用一次该函数
  3227. *
  3228. * @param {Function} callback 回调
  3229. * @param {Number} wait 多少秒毫
  3230. * @param {Object} options 参数{leading: 是否在之前执行, trailing: 是否在之后执行}
  3231. * @return {Function}
  3232. */
  3233. function throttle (callback, wait, options) {
  3234. var args, context
  3235. var opts = options || {}
  3236. var runFlag = false
  3237. var timeout = 0
  3238. var optLeading = 'leading' in opts ? opts.leading : true
  3239. var optTrailing = 'trailing' in opts ? opts.trailing : false
  3240. var runFn = function () {
  3241. runFlag = true
  3242. callback.apply(context, args)
  3243. timeout = setTimeout(endFn, wait)
  3244. }
  3245. var endFn = function () {
  3246. timeout = 0
  3247. if (!runFlag && optTrailing === true) {
  3248. runFn()
  3249. }
  3250. }
  3251. var cancelFn = function () {
  3252. var rest = timeout !== 0
  3253. clearTimeout(timeout)
  3254. runFlag = false
  3255. timeout = 0
  3256. return rest
  3257. }
  3258. var throttled = function () {
  3259. args = arguments
  3260. context = this
  3261. runFlag = false
  3262. if (timeout === 0) {
  3263. if (optLeading === true) {
  3264. runFn()
  3265. } else if (optTrailing === true) {
  3266. timeout = setTimeout(endFn, wait)
  3267. }
  3268. }
  3269. }
  3270. throttled.cancel = cancelFn
  3271. return throttled
  3272. }
  3273. /**
  3274. * 函数去抖;当被调用 n 毫秒后才会执行,如果在这时间内又被调用则将重新计算执行时间
  3275. *
  3276. * @param {Function} callback 回调
  3277. * @param {Number} wait 多少秒毫
  3278. * @param {Object} options 参数{leading: 是否在之前执行, trailing: 是否在之后执行}
  3279. * @return {Function}
  3280. */
  3281. function debounce (callback, wait, options) {
  3282. var args, context
  3283. var opts = options || {}
  3284. var runFlag = false
  3285. var timeout = 0
  3286. var isLeading = typeof options === 'boolean'
  3287. var optLeading = 'leading' in opts ? opts.leading : isLeading
  3288. var optTrailing = 'trailing' in opts ? opts.trailing : !isLeading
  3289. var runFn = function () {
  3290. runFlag = true
  3291. timeout = 0
  3292. callback.apply(context, args)
  3293. }
  3294. var endFn = function () {
  3295. if (optLeading === true) {
  3296. timeout = 0
  3297. }
  3298. if (!runFlag && optTrailing === true) {
  3299. runFn()
  3300. }
  3301. }
  3302. var cancelFn = function () {
  3303. var rest = timeout !== 0
  3304. clearTimeout(timeout)
  3305. timeout = 0
  3306. return rest
  3307. }
  3308. var debounced = function () {
  3309. runFlag = false
  3310. args = arguments
  3311. context = this
  3312. if (timeout === 0) {
  3313. if (optLeading === true) {
  3314. runFn()
  3315. }
  3316. } else {
  3317. clearTimeout(timeout)
  3318. }
  3319. timeout = setTimeout(endFn, wait)
  3320. }
  3321. debounced.cancel = cancelFn
  3322. return debounced
  3323. }
  3324. /**
  3325. * 该方法和 setTimeout 一样的效果,区别就是支持上下文和额外参数
  3326. *
  3327. * @param {Function} callback 函数
  3328. * @param {Number} wait 延迟毫秒
  3329. * @param {*} args 额外的参数
  3330. * @return {Number}
  3331. */
  3332. function delay (callback, wait) {
  3333. var args = slice(arguments, 2)
  3334. var context = this
  3335. return setTimeout(function () {
  3336. callback.apply(context, args)
  3337. }, wait)
  3338. }
  3339. function parseURLQuery (uri) {
  3340. return unserialize(uri.split('?')[1] || '')
  3341. }
  3342. function parseUrl (url) {
  3343. var hashs, portText, searchs, parsed
  3344. var href = '' + url
  3345. if (href.indexOf('//') === 0) {
  3346. href = (staticLocation ? staticLocation.protocol : '') + href
  3347. } else if (href.indexOf('/') === 0) {
  3348. href = helperGetLocatOrigin() + href
  3349. }
  3350. searchs = href.replace(/#.*/, '').match(/(\?.*)/)
  3351. parsed = {
  3352. href: href,
  3353. hash: '',
  3354. host: '',
  3355. hostname: '',
  3356. protocol: '',
  3357. port: '',
  3358. search: searchs && searchs[1] && searchs[1].length > 1 ? searchs[1] : ''
  3359. }
  3360. parsed.path = href.replace(/^([a-z0-9.+-]*:)\/\//, function (text, protocol) {
  3361. parsed.protocol = protocol
  3362. return ''
  3363. }).replace(/^([a-z0-9.+-]*)(:\d+)?\/?/, function (text, hostname, port) {
  3364. portText = port || ''
  3365. parsed.port = portText.replace(':', '')
  3366. parsed.hostname = hostname
  3367. parsed.host = hostname + portText
  3368. return '/'
  3369. }).replace(/(#.*)/, function (text, hash) {
  3370. parsed.hash = hash.length > 1 ? hash : ''
  3371. return ''
  3372. })
  3373. hashs = parsed.hash.match(/#((.*)\?|(.*))/)
  3374. parsed.pathname = parsed.path.replace(/(\?|#.*).*/, '')
  3375. parsed.origin = parsed.protocol + '//' + parsed.host
  3376. parsed.hashKey = hashs ? (hashs[2] || hashs[1] || '') : ''
  3377. parsed.hashQuery = parseURLQuery(parsed.hash)
  3378. parsed.searchQuery = parseURLQuery(parsed.search)
  3379. return parsed
  3380. }
  3381. function stringifyParams (resultVal, resultKey, isArr) {
  3382. var _arr
  3383. var result = []
  3384. each(resultVal, function (item, key) {
  3385. _arr = isArray(item)
  3386. if (isPlainObject(item) || _arr) {
  3387. result = result.concat(stringifyParams(item, resultKey + '[' + key + ']', _arr))
  3388. } else {
  3389. result.push(staticEncodeURIComponent(resultKey + '[' + (isArr ? '' : key) + ']') + '=' + staticEncodeURIComponent(isNull(item) ? '' : item))
  3390. }
  3391. })
  3392. return result
  3393. }
  3394. /**
  3395. * 序列化查询参数
  3396. *
  3397. * @param {Object} query 查询参数
  3398. */
  3399. function serialize (query) {
  3400. var _arr
  3401. var params = []
  3402. each(query, function (item, key) {
  3403. if (!isUndefined(item)) {
  3404. _arr = isArray(item)
  3405. if (isPlainObject(item) || _arr) {
  3406. params = params.concat(stringifyParams(item, key, _arr))
  3407. } else {
  3408. params.push(staticEncodeURIComponent(key) + '=' + staticEncodeURIComponent(isNull(item) ? '' : item))
  3409. }
  3410. }
  3411. })
  3412. return params.join('&').replace(/%20/g, '+')
  3413. }
  3414. /**
  3415. * 反序列化查询参数
  3416. * @param {String} query 字符串
  3417. */
  3418. function unserialize (str) {
  3419. var items
  3420. var result = {}
  3421. if (str && isString(str)) {
  3422. arrayEach(str.split('&'), function (param) {
  3423. items = param.split('=')
  3424. result[staticDecodeURIComponent(items[0])] = staticDecodeURIComponent(items[1] || '')
  3425. })
  3426. }
  3427. return result
  3428. }
  3429. function getBaseURL () {
  3430. if (staticLocation) {
  3431. var pathname = staticLocation.pathname
  3432. var lastIndex = lastIndexOf(pathname, '/') + 1
  3433. return helperGetLocatOrigin() + (lastIndex === pathname.length ? pathname : pathname.substring(0, lastIndex))
  3434. }
  3435. return ''
  3436. }
  3437. /**
  3438. * 获取地址栏信息
  3439. *
  3440. * @return Object
  3441. */
  3442. function locat () {
  3443. return staticLocation ? parseUrl(staticLocation.href) : {}
  3444. }
  3445. /* eslint-disable valid-typeof */
  3446. function isBrowseStorage (storage) {
  3447. try {
  3448. var testKey = '__xe_t'
  3449. storage.setItem(testKey, 1)
  3450. storage.removeItem(testKey)
  3451. return true
  3452. } catch (e) {
  3453. return false
  3454. }
  3455. }
  3456. function isBrowseType (type) {
  3457. return navigator.userAgent.indexOf(type) > -1
  3458. }
  3459. /**
  3460. * 获取浏览器内核
  3461. * @return Object
  3462. */
  3463. function browse () {
  3464. var $body, isChrome, isEdge
  3465. var isMobile = false
  3466. var result = {
  3467. isNode: false,
  3468. isMobile: isMobile,
  3469. isPC: false,
  3470. isDoc: !!staticDocument
  3471. }
  3472. if (!staticWindow && typeof process !== staticStrUndefined) {
  3473. result.isNode = true
  3474. } else {
  3475. isEdge = isBrowseType('Edge')
  3476. isChrome = isBrowseType('Chrome')
  3477. isMobile = /(Android|webOS|iPhone|iPad|iPod|SymbianOS|BlackBerry|Windows Phone)/.test(navigator.userAgent)
  3478. if (result.isDoc) {
  3479. $body = staticDocument.body || staticDocument.documentElement
  3480. arrayEach(['webkit', 'khtml', 'moz', 'ms', 'o'], function (core) {
  3481. result['-' + core] = !!$body[core + 'MatchesSelector']
  3482. })
  3483. }
  3484. assign(result, {
  3485. edge: isEdge,
  3486. firefox: isBrowseType('Firefox'),
  3487. msie: !isEdge && result['-ms'],
  3488. safari: !isChrome && !isEdge && isBrowseType('Safari'),
  3489. isMobile: isMobile,
  3490. isPC: !isMobile,
  3491. isLocalStorage: isBrowseStorage(staticWindow.localStorage),
  3492. isSessionStorage: isBrowseStorage(staticWindow.sessionStorage)
  3493. })
  3494. }
  3495. return result
  3496. }
  3497. function toCookieUnitTime (unit, expires) {
  3498. var num = parseFloat(expires)
  3499. var nowdate = helperNewDate()
  3500. var time = helperGetDateTime(nowdate)
  3501. switch (unit) {
  3502. case 'y': return helperGetDateTime(getWhatYear(nowdate, num))
  3503. case 'M': return helperGetDateTime(getWhatMonth(nowdate, num))
  3504. case 'd': return helperGetDateTime(getWhatDay(nowdate, num))
  3505. case 'h':
  3506. case 'H': return time + num * 60 * 60 * 1000
  3507. case 'm': return time + num * 60 * 1000
  3508. case 's': return time + num * 1000
  3509. }
  3510. return time
  3511. }
  3512. function toCookieUTCString (date) {
  3513. return (isDate(date) ? date : new Date(date)).toUTCString()
  3514. }
  3515. /**
  3516. * cookie操作函数
  3517. * @param {String/Array/Object} name 键/数组/对象
  3518. * @param {String} value 值
  3519. * @param {Object} options 参数
  3520. * @param {String} name: 键
  3521. * @param {Object} value: 值
  3522. * @param {String} path: 路径
  3523. * @param {String} domain: 作用域
  3524. * @param {Boolean} secure: 设置为安全的,只能用https协议
  3525. * @param {Number} expires: 过期时间,可以指定日期或者字符串,默认天
  3526. */
  3527. function cookie (name, value, options) {
  3528. if (staticDocument) {
  3529. var opts, expires, values, result, cookies, keyIndex
  3530. var inserts = []
  3531. var args = arguments
  3532. if (isArray(name)) {
  3533. inserts = name
  3534. } else if (args.length > 1) {
  3535. inserts = [assign({ name: name, value: value }, options)]
  3536. } else if (isObject(name)) {
  3537. inserts = [name]
  3538. }
  3539. if (inserts.length > 0) {
  3540. arrayEach(inserts, function (obj) {
  3541. opts = assign({}, setupDefaults.cookies, obj)
  3542. values = []
  3543. if (opts.name) {
  3544. expires = opts.expires
  3545. values.push(staticEncodeURIComponent(opts.name) + '=' + staticEncodeURIComponent(isObject(opts.value) ? JSON.stringify(opts.value) : opts.value))
  3546. if (expires) {
  3547. if (isNaN(expires)) {
  3548. // UTCString || Unit
  3549. expires = expires.replace(/^([0-9]+)(y|M|d|H|h|m|s)$/, function (text, num, unit) {
  3550. return toCookieUTCString(toCookieUnitTime(unit, num))
  3551. })
  3552. } else if (/^[0-9]{11,13}$/.test(expires) || isDate(expires)) {
  3553. // Date || now
  3554. expires = toCookieUTCString(expires)
  3555. } else {
  3556. // day
  3557. expires = toCookieUTCString(toCookieUnitTime('d', expires))
  3558. }
  3559. opts.expires = expires
  3560. }
  3561. arrayEach(['expires', 'path', 'domain', 'secure'], function (key) {
  3562. if (!isUndefined(opts[key])) {
  3563. values.push(opts[key] && key === 'secure' ? key : (key + '=' + opts[key]))
  3564. }
  3565. })
  3566. }
  3567. staticDocument.cookie = values.join('; ')
  3568. })
  3569. return true
  3570. } else {
  3571. result = {}
  3572. cookies = staticDocument.cookie
  3573. if (cookies) {
  3574. arrayEach(cookies.split('; '), function (val) {
  3575. keyIndex = val.indexOf('=')
  3576. result[staticDecodeURIComponent(val.substring(0, keyIndex))] = staticDecodeURIComponent(val.substring(keyIndex + 1) || '')
  3577. })
  3578. }
  3579. return args.length === 1 ? result[name] : result
  3580. }
  3581. }
  3582. return false
  3583. }
  3584. function hasCookieItem (key) {
  3585. return includes(cookieKeys(), key)
  3586. }
  3587. function getCookieItem (name, key) {
  3588. return cookie(name, key)
  3589. }
  3590. function setCookieItem (name, key, options) {
  3591. cookie(name, key, options)
  3592. return cookie
  3593. }
  3594. function removeCookieItem (name, options) {
  3595. cookie(name, 0, assign({ expires: -1 }, setupDefaults.cookies, options))
  3596. }
  3597. function cookieKeys () {
  3598. return keys(cookie())
  3599. }
  3600. assign(cookie, {
  3601. has: hasCookieItem,
  3602. set: setCookieItem,
  3603. setItem: setCookieItem,
  3604. get: getCookieItem,
  3605. getItem: cookie,
  3606. remove: removeCookieItem,
  3607. removeItem: removeCookieItem,
  3608. keys: cookieKeys,
  3609. getJSON: cookie
  3610. })
  3611. // 核心
  3612. // 对象相关的方法
  3613. // 数组相关的方法
  3614. // 基础方法
  3615. // 数值相关方法
  3616. // 日期相关的方法
  3617. // 字符串相关的方法
  3618. // 函数相关的方法
  3619. // 地址相关的方法
  3620. // 浏览器相关的方法
  3621. assign(XEUtils, {
  3622. // object
  3623. assign: assign,
  3624. objectEach: objectEach,
  3625. lastObjectEach: lastObjectEach,
  3626. objectMap: objectMap,
  3627. merge: merge,
  3628. // array
  3629. uniq: uniq,
  3630. union: union,
  3631. sortBy: sortBy,
  3632. orderBy: orderBy,
  3633. shuffle: shuffle,
  3634. sample: sample,
  3635. some: some,
  3636. every: every,
  3637. slice: slice,
  3638. filter: filter,
  3639. find: find,
  3640. findLast: findLast,
  3641. findKey: findKey,
  3642. includes: includes,
  3643. arrayIndexOf: arrayIndexOf,
  3644. arrayLastIndexOf: arrayLastIndexOf,
  3645. map: map,
  3646. reduce: reduce,
  3647. copyWithin: copyWithin,
  3648. chunk: chunk,
  3649. zip: zip,
  3650. unzip: unzip,
  3651. zipObject: zipObject,
  3652. flatten: flatten,
  3653. toArray: toArray,
  3654. includeArrays: includeArrays,
  3655. pluck: pluck,
  3656. invoke: invoke,
  3657. arrayEach: arrayEach,
  3658. lastArrayEach: lastArrayEach,
  3659. toArrayTree: toArrayTree,
  3660. toTreeArray: toTreeArray,
  3661. findTree: findTree,
  3662. eachTree: eachTree,
  3663. mapTree: mapTree,
  3664. filterTree: filterTree,
  3665. searchTree: searchTree,
  3666. // base
  3667. hasOwnProp: hasOwnProp,
  3668. eqNull: eqNull,
  3669. isNaN: isNumberNaN,
  3670. isFinite: isNumberFinite,
  3671. isUndefined: isUndefined,
  3672. isArray: isArray,
  3673. isFloat: isFloat,
  3674. isInteger: isInteger,
  3675. isFunction: isFunction,
  3676. isBoolean: isBoolean,
  3677. isString: isString,
  3678. isNumber: isNumber,
  3679. isRegExp: isRegExp,
  3680. isObject: isObject,
  3681. isPlainObject: isPlainObject,
  3682. isDate: isDate,
  3683. isError: isError,
  3684. isTypeError: isTypeError,
  3685. isEmpty: isEmpty,
  3686. isNull: isNull,
  3687. isSymbol: isSymbol,
  3688. isArguments: isArguments,
  3689. isElement: isElement,
  3690. isDocument: isDocument,
  3691. isWindow: isWindow,
  3692. isFormData: isFormData,
  3693. isMap: isMap,
  3694. isWeakMap: isWeakMap,
  3695. isSet: isSet,
  3696. isWeakSet: isWeakSet,
  3697. isLeapYear: isLeapYear,
  3698. isMatch: isMatch,
  3699. isEqual: isEqual,
  3700. isEqualWith: isEqualWith,
  3701. getType: getType,
  3702. uniqueId: uniqueId,
  3703. getSize: getSize,
  3704. indexOf: indexOf,
  3705. lastIndexOf: lastIndexOf,
  3706. findIndexOf: findIndexOf,
  3707. findLastIndexOf: findLastIndexOf,
  3708. toStringJSON: toStringJSON,
  3709. toJSONString: toJSONString,
  3710. keys: keys,
  3711. values: values,
  3712. entries: entries,
  3713. pick: pick,
  3714. omit: omit,
  3715. first: first,
  3716. last: last,
  3717. each: each,
  3718. forOf: forOf,
  3719. lastForOf: lastForOf,
  3720. lastEach: lastEach,
  3721. has: has,
  3722. get: get,
  3723. set: set,
  3724. groupBy: groupBy,
  3725. countBy: countBy,
  3726. clone: clone,
  3727. clear: clear,
  3728. remove: remove,
  3729. range: range,
  3730. destructuring: destructuring,
  3731. // number
  3732. random: random,
  3733. min: min,
  3734. max: max,
  3735. commafy: commafy,
  3736. round: round,
  3737. ceil: ceil,
  3738. floor: floor,
  3739. toFixed: toFixed,
  3740. toNumber: toNumber,
  3741. toNumberString: toNumberString,
  3742. toInteger: toInteger,
  3743. add: add,
  3744. subtract: subtract,
  3745. multiply: multiply,
  3746. divide: divide,
  3747. sum: sum,
  3748. mean: mean,
  3749. // date
  3750. now: now,
  3751. timestamp: timestamp,
  3752. isValidDate: isValidDate,
  3753. isDateSame: isDateSame,
  3754. toStringDate: toStringDate,
  3755. toDateString: toDateString,
  3756. getWhatYear: getWhatYear,
  3757. getWhatMonth: getWhatMonth,
  3758. getWhatWeek: getWhatWeek,
  3759. getWhatDay: getWhatDay,
  3760. getYearDay: getYearDay,
  3761. getYearWeek: getYearWeek,
  3762. getMonthWeek: getMonthWeek,
  3763. getDayOfYear: getDayOfYear,
  3764. getDayOfMonth: getDayOfMonth,
  3765. getDateDiff: getDateDiff,
  3766. // string
  3767. trim: trim,
  3768. trimLeft: trimLeft,
  3769. trimRight: trimRight,
  3770. escape: escape,
  3771. unescape: unescape,
  3772. camelCase: camelCase,
  3773. kebabCase: kebabCase,
  3774. repeat: repeat,
  3775. padStart: padStart,
  3776. padEnd: padEnd,
  3777. startsWith: startsWith,
  3778. endsWith: endsWith,
  3779. template: template,
  3780. toFormatString: toFormatString,
  3781. toString: toValueString,
  3782. toValueString: toValueString,
  3783. // function
  3784. noop: noop,
  3785. property: property,
  3786. bind: bind,
  3787. once: once,
  3788. after: after,
  3789. before: before,
  3790. throttle: throttle,
  3791. debounce: debounce,
  3792. delay: delay,
  3793. // url
  3794. unserialize: unserialize,
  3795. serialize: serialize,
  3796. parseUrl: parseUrl,
  3797. // web
  3798. getBaseURL: getBaseURL,
  3799. locat: locat,
  3800. browse: browse,
  3801. cookie: cookie
  3802. })
  3803. return XEUtils
  3804. }))