Commonutils.js 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506
  1. // import { whoami, getConfigApi } from '../api/auth'
  2. import { USER_ID, T_STORAGE_KEY, USER_INFO, USER_TOKEN } from '../constant'
  3. // const wx = require('weixin-js-sdk')
  4. // import { isNull } from 'lodash-es'
  5. /**
  6. * @description 解决小数计算精度问题(en,你应该使用big.js)
  7. * @param {Number, String} data 数字
  8. * @param {Number} accuracy 保留几位小数
  9. * @returns
  10. */
  11. export const fomartNumber = (data, accuracy = 2) => {
  12. const temp = data + ''
  13. if (temp.includes('.')) {
  14. return (data * 1).toFixed(accuracy)
  15. }
  16. return data
  17. }
  18. /**
  19. * @description 批量清除缓存
  20. * @param {String[]} cacheArr 要清除的缓存string数组
  21. */
  22. export const removeCache = (cacheArr) => {
  23. if (!Array.isArray(cacheArr)) {
  24. return
  25. }
  26. for (const item of cacheArr) {
  27. uni.removeStorageSync(item)
  28. }
  29. }
  30. /**
  31. * 检测登录是否有效
  32. */
  33. export const checkWhoami = () => {
  34. new Promise(async (resolve, reject) => {
  35. const userId = getUserId()
  36. const res = await whoami(userId)
  37. if (res.errno !== 0) {
  38. uni.navigateTo({
  39. url: '/pages/login/login'
  40. })
  41. }
  42. })
  43. }
  44. /**
  45. * 获取用户userid
  46. * @returns
  47. */
  48. export const getUserId = () => {
  49. const userId = uni.getStorageSync(USER_ID)
  50. if (!userId) {
  51. // uni.showToast({
  52. // title: "登录已失效,请重新登录",
  53. // duration: 2000,
  54. // icon: "none",
  55. // });
  56. // uni.navigateTo({
  57. // url: "/pages/login/login",
  58. // });
  59. uni.showModal({
  60. title: '提示',
  61. content: '您还未登录,是否去登录?',
  62. success(res) {
  63. if (res.confirm) {
  64. uni.navigateTo({
  65. url: '/pages/login/login'
  66. })
  67. } else if (res.cancel) {
  68. // uni.navigateBack();
  69. }
  70. }
  71. })
  72. return
  73. }
  74. return userId
  75. }
  76. /**
  77. * 获取新团蜂用户id
  78. * @returns
  79. */
  80. export const getStorageUserId = () => {
  81. const userInfo = uni.getStorageSync(T_STORAGE_KEY) || {}
  82. if (!userInfo.buyerUserId) {
  83. uni.showModal({
  84. title: '提示',
  85. content: '您还未登录,是否去登录?',
  86. success(res) {
  87. if (res.confirm) {
  88. uni.navigateTo({
  89. url: '/pages/login/login'
  90. })
  91. } else if (res.cancel) {
  92. // uni.navigateBack();
  93. }
  94. }
  95. })
  96. return
  97. }
  98. return userInfo.buyerUserId
  99. }
  100. /**
  101. * 获取新团蜂token
  102. * @returns
  103. */
  104. export const getStorageKeyToken = () => {
  105. const userInfo = uni.getStorageSync(USER_INFO)
  106. if (!userInfo || !userInfo.userId) {
  107. uni.showModal({
  108. title: '提示',
  109. content: '您还未登录,是否去登录?',
  110. success(res) {
  111. if (res.confirm) {
  112. uni.navigateTo({
  113. url: '/pages/login/login'
  114. })
  115. } else if (res.cancel) {
  116. // uni.navigateBack();
  117. }
  118. }
  119. })
  120. return
  121. }
  122. if (!userInfo || !userInfo.phone) {
  123. uni.showModal({
  124. title: '提示',
  125. content: '未绑定手机号码,是否去绑定?',
  126. success(res) {
  127. if (res.confirm) {
  128. uni.switchTab({
  129. url: '/pages/index/index'
  130. })
  131. } else if (res.cancel) {
  132. // uni.navigateBack();
  133. }
  134. }
  135. })
  136. return
  137. }
  138. const storageKey = uni.getStorageSync(T_STORAGE_KEY)
  139. if (!storageKey || !storageKey.token) {
  140. uni.showModal({
  141. title: '提示',
  142. content: '系统出错,请重新登陆',
  143. success(res) {
  144. if (res.confirm) {
  145. uni.navigateTo({
  146. url: '/pages/login/login'
  147. })
  148. } else if (res.cancel) {
  149. // uni.navigateBack();
  150. }
  151. }
  152. })
  153. return
  154. }
  155. return storageKey.token
  156. }
  157. /**
  158. * 判断当前H5是否在webview中打开
  159. */
  160. export const isH5InWebview = () => {
  161. const ua = navigator.userAgent.toLowerCase()
  162. return typeof ua === 'string' && (ua.includes('webview') || ua.includes('miniprogramhtmlwebview'))
  163. }
  164. // 判断当前是否处于微信环境
  165. export const isInWx = () => {
  166. // #ifdef H5
  167. var ua = navigator.userAgent.toLowerCase()
  168. return ua.match(/MicroMessenger/i) == 'micromessenger'
  169. // #endif
  170. // #ifdef APP
  171. return false
  172. // #endif
  173. }
  174. /**
  175. * 跳转到新团蜂入驻端项目
  176. * @returns
  177. */
  178. export const jumpToOtherProject = ({ isInMiniProgram, id, appId, url, toType, query, montageTerminal }, cb = () => { }) => {
  179. if (toType === 'H5') {
  180. // #ifdef H5
  181. window.location.href = url
  182. // #endif
  183. // #ifdef APP
  184. plus.runtime.openURL(url, cb)
  185. // #endif
  186. // #ifdef MP
  187. uni.redirectTo({
  188. url: `/user/view?target=${url}`
  189. })
  190. // #endif
  191. } else if (toType === 'MP') {
  192. if (isInWx()) {
  193. if (isInMiniProgram || isH5InWebview()) {
  194. wx.miniProgram.navigateTo({ // 先跳去本小程序其它页面,再跳去其它小程序页面
  195. url: query && montageTerminal && montageTerminal.includes(6) ? `/${url}${query}` : `/${url}`,
  196. fail: () => {
  197. setTimeout(() => { uni.switchTab({ url: '/pages/index/index' }) }, 2000)
  198. }
  199. })
  200. } else {
  201. if (!getStorageKeyToken()) return
  202. const currentUrl = window.location.href.replace('#', 'ericToken')
  203. getConfigApi({
  204. url: currentUrl,
  205. token: uni.getStorageSync(USER_TOKEN)
  206. }).then(({ data }) => {
  207. wx.config({
  208. debug: false, // 开启调试模式
  209. appId: data.appId, // 必填,公众号的唯一标识
  210. timestamp: data.timestamp, // 必填,生成签名的时间戳
  211. nonceStr: data.nonceStr, // 必填,生成签名的随机串
  212. signature: data.signature, // 必填,签名,见附录1
  213. jsApiList: [
  214. 'updateAppMessageShareData',
  215. 'updateTimelineShareData',
  216. 'onMenuShareAppMessage',
  217. 'onMenuShareTimeline'
  218. ], // 必填,需要使用的JS接口列表,所有JS接口列表见附录2
  219. openTagList: [ 'wx-open-launch-weapp' ]
  220. })
  221. wx.ready(function () {
  222. // config信息验证成功
  223. // console.log(res);
  224. })
  225. wx.error(function (res) {
  226. // config信息验证失败会执行error函数,如签名过期导致验证失败,具体错误信息可以打开config的debug模式查看,也可以在返回的res参数中查看,对于SPA可以在这里更新签名。
  227. // alert('error:'+JSON.stringify(res));
  228. })
  229. })
  230. }
  231. } else {
  232. // #ifdef H5
  233. location.href = 'weixin://dl/business/?appid=wxb446588ba0dbb9d7&path=pages/index/index'
  234. // #endif
  235. // #ifdef APP
  236. plus.share.getServices(function (result) {
  237. let sweixin = null
  238. for (const i in result) {
  239. if (result[i].id == 'weixin') {
  240. sweixin = result[i]
  241. }
  242. }
  243. if (sweixin) {
  244. sweixin.launchMiniProgram({
  245. id, // 微信小程序的原始ID("g_"开头的字符串)
  246. type: 0,
  247. path: query && montageTerminal && montageTerminal.includes(1) ? url + query : url
  248. })
  249. }
  250. }, function (e) {
  251. console.log('获取分享服务列表失败:' + e.message)
  252. })
  253. // #endif
  254. // #ifdef MP
  255. uni.navigateToMiniProgram({
  256. appId,
  257. path: query && montageTerminal && (montageTerminal.includes(2) || montageTerminal.includes(4)) ? url + query : url,
  258. extraData: {},
  259. success: (res) => {
  260. console.log('打开成功')
  261. },
  262. fail: (err) => {
  263. console.log('打开失败', err)
  264. },
  265. complete: (result) => {
  266. console.log(result)
  267. }
  268. })
  269. // #endif
  270. }
  271. }
  272. }
  273. /**
  274. * 点击复制
  275. * @param {*} text
  276. */
  277. export const useCopy = (text) => {
  278. const input = document.createElement('input')
  279. input.value = text
  280. document.body.appendChild(input)
  281. input.select()
  282. document.execCommand('Copy')
  283. document.body.removeChild(input)
  284. uni.showToast({
  285. title: '单号复制成功'
  286. })
  287. }
  288. /**
  289. * @description 防抖函数
  290. * @param {*} func
  291. * @param {*} wait
  292. * @param {*} immediate
  293. * @returns
  294. */
  295. export function handleDebounce(func, wait, immediate) {
  296. let timeout
  297. return function () {
  298. const context = this
  299. const args = arguments
  300. if (timeout) clearTimeout(timeout)
  301. if (immediate) {
  302. var callNow = !timeout
  303. timeout = setTimeout(() => {
  304. timeout = null
  305. }, wait)
  306. if (callNow) func.apply(context, args)
  307. } else {
  308. timeout = setTimeout(function () {
  309. func.apply(context, args)
  310. }, wait)
  311. }
  312. }
  313. }
  314. export function getRandom(min, max) {
  315. return Math.floor(Math.random() * (max - min) + min)
  316. }
  317. export const randomRGB = () => {
  318. const r = Math.floor(Math.random() * 255)
  319. const g = Math.floor(Math.random() * 255)
  320. const b = Math.floor(Math.random() * 255)
  321. return `rgb(${r}, ${g}, ${b})`
  322. }
  323. export const timestampToTime = (timestamp) => {
  324. // 时间戳为10位需*1000,时间戳为13位不需乘1000
  325. // var date = new Date(timestamp * 1000);
  326. var date = new Date(timestamp)
  327. var Y = date.getFullYear() + '-'
  328. var M = (date.getMonth() + 1 < 10 ? '0' + (date.getMonth() + 1) : date.getMonth() + 1) + '-'
  329. var D = (date.getDate() < 10 ? '0' + date.getDate() : date.getDate()) + ' '
  330. var h = date.getHours() + ':'
  331. var m = date.getMinutes() + ':'
  332. var s = date.getSeconds()
  333. return Y + M + D + h + m + s
  334. // console.log(timestampToTime(1670145353)); //2022-12-04 17:15:53
  335. }
  336. /**
  337. * 时间格式化
  338. */
  339. export const timeFormatting = (timeDifference) => {
  340. // 天数
  341. const day = Math.floor(timeDifference / 3600 / 24)
  342. // 小时
  343. const hr = Math.floor(timeDifference / 3600 % 24)
  344. // 分钟
  345. const min = Math.floor(timeDifference / 60 % 60)
  346. // 秒
  347. const sec = Math.floor(timeDifference % 60)
  348. return {
  349. day: day < 10 ? '0' + day : day,
  350. hour: hr < 10 ? '0' + hr : hr,
  351. min: min < 10 ? '0' + min : min,
  352. sec: sec < 10 ? '0' + sec : sec
  353. }
  354. }
  355. export const throttle = (fn, interval) => {
  356. let lastTime = 0
  357. const _throttle = function (...args) {
  358. const nowTime = new Date().getTime()
  359. const remainTime = interval - (nowTime - lastTime)
  360. if (remainTime <= 0) {
  361. fn.apply(this, args)
  362. lastTime = nowTime
  363. }
  364. }
  365. return _throttle
  366. }
  367. // 获取 微信 code
  368. // #ifdef H5
  369. export const getUrlCode = () => {
  370. var url = location.search
  371. var theRequest = new Object()
  372. if (url.indexOf('?') != -1) {
  373. var str = url.substr(1)
  374. var strs = str.split('&')
  375. for (var i = 0; i < strs.length; i++) {
  376. theRequest[strs[i].split('=')[0]] = strs[i].split('=')[1]
  377. }
  378. }
  379. console.log('code结果', theRequest)
  380. return theRequest
  381. }
  382. // #endif
  383. /**
  384. * 大数转小数 12345.123 = 1.23万
  385. */
  386. export const convertToDecimal = (number) => {
  387. if (!number || isNull(number)) return 0
  388. if (number < 10000) {
  389. return number.toString()
  390. } else if (number < 100000000) {
  391. const decimalNumber = (number / 10000).toFixed(2)
  392. return decimalNumber + '万'
  393. }
  394. const decimalNumber = (number / 100000000).toFixed(2)
  395. return decimalNumber + '亿'
  396. }
  397. export const isSubarray = (arr, subarr) => {
  398. const mainSet = new Set(arr)
  399. for (const element of subarr) {
  400. if (!mainSet.has(element)) {
  401. return false
  402. }
  403. }
  404. return true
  405. }
  406. export const tradeOrderNo = function () {
  407. const now = new Date()
  408. const year = now.getFullYear()
  409. let month = now.getMonth() + 1
  410. let day = now.getDate()
  411. let hour = now.getHours()
  412. let minutes = now.getMinutes()
  413. let seconds = now.getSeconds()
  414. String(month).length < 2 ? month = Number('0' + month) : month
  415. String(day).length < 2 ? day = Number('0' + day) : day
  416. String(hour).length < 2 ? hour = Number('0' + hour) : hour
  417. String(minutes).length < 2 ? minutes = Number('0' + minutes) : minutes
  418. String(seconds).length < 2 ? seconds = Number('0' + seconds) : seconds
  419. const yyyyMMddHHmmss = `${year}${month}${day}${hour}${minutes}${seconds}`
  420. return yyyyMMddHHmmss + Math.random().toString(36)
  421. .substr(2, 9)
  422. }
  423. /**
  424. * 判断当前资源是否是视频格式
  425. * @param {string} url
  426. * @returns {boolean}
  427. */
  428. // export function isVideo(url) {
  429. // // ['png', 'jpg', 'jpeg', 'bmp', 'gif','webp'] ['mp4', 'm2v', 'mkv', 'webm', 'ogg', 'flv']
  430. // const videoExtensions = ['.avi', '.wmv', '.mpg', '.mpeg', '.mov', '.rm', '.ram', '.swf', '.flv', '.mp4']
  431. // const lowercasedUrl = url.toLowerCase()
  432. // return videoExtensions.some(type => lowercasedUrl.includes(type))
  433. // }
  434. export function isVideoSource(src) {
  435. // return ['.avi', '.wmv', '.mpg', '.mpeg', '.mov', '.rm', '.ram', '.swf', '.flv', '.mp4'].some((item) => src.indexOf(item) !== -1)
  436. return ['.avi', '.wmv', '.mpg', '.mpeg', '.mov', '.rm', '.ram', '.swf', '.flv', '.mp4'].includes(src.substring(src.lastIndexOf('.')))
  437. }
  438. export const saveImg = (url, cb) => {
  439. // #ifdef H5
  440. if (isInWx()) {
  441. uni.showToast({
  442. title: '请长按图片保存',
  443. duration: 2000,
  444. icon: 'none'
  445. })
  446. } else {
  447. const uniappA = document.createElement('a')
  448. uniappA.download = ''
  449. uniappA.href = url
  450. document.body.appendChild(uniappA)
  451. uniappA.click()
  452. uniappA.remove()
  453. cb && typeof cb === 'function' && cb()
  454. }
  455. // #endif
  456. // #ifdef APP || MP
  457. uni.saveImageToPhotosAlbum({
  458. filePath: url,
  459. success() {
  460. cb && typeof cb === 'function' && cb()
  461. }
  462. })
  463. // #endif
  464. }