You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							476 lines
						
					
					
						
							12 KiB
						
					
					
				
			
		
		
		
			
			
			
				
					
				
				
					
				
			
		
		
	
	
							476 lines
						
					
					
						
							12 KiB
						
					
					
				| // 获取路径参数 | |
| import Vue from "vue"; | |
| // 全局变量 切换测试/正式 | |
| Vue.prototype.envPropObj = 1 ? { | |
| 	// 测试环境 | |
| 	// 我的-为您推荐 标签id | |
| 	use_recommend_tag_id: 684, | |
| 	// 积分商城-每日精选 | |
| 	score_daily_recommend: 679, | |
| 	// 积分商城-分类-全部 | |
| 	score_type_all: 680, | |
| 	// 积分商城-活动规则 | |
| 	score_rule_id: 2, | |
| } : { | |
| 	// 正式环境 | |
| 	use_recommend_tag_id: 674, | |
| 	score_daily_recommend: 673, | |
| 	score_type_all: 672, | |
| 	score_rule_id: 7, | |
| } | |
| 
 | |
| // 格式化富文本 | |
| Vue.prototype.formateRichText = str => { | |
| 	if (!str) return ""; | |
| 	var reg = new RegExp("<img", "g"); | |
| 	// #ifdef MP-WEIXIN | |
| 	str = str.replace(reg, "<img class='sz-xcx-fwb-img' width='100%'") | |
| 	reg = new RegExp("<IMG", "g"); | |
| 	str = str.replace(reg, "<img class='sz-xcx-fwb-img' width='100%'") | |
| 	reg = new RegExp(" ", "g"); | |
| 	// #endif | |
| 	// #ifdef H5 | |
| 	str = str.replace(reg, "<img class='sz-xcx-fwb-img' style='font-size:0;line-height:0' width='100%'") | |
| 	reg = new RegExp("<IMG", "g"); | |
| 	str = str.replace(reg, "<img class='sz-xcx-fwb-img' style='font-size:0;line-height:0' width='100%'") | |
| 	reg = new RegExp(" ", "g"); | |
| 	// #endif | |
| 	 | |
| 	str = str.replace(reg, '<span style="width: 8rpx;display: inline-block;"></span>') | |
| 	reg = new RegExp("section", "g"); | |
| 	str = str.replace(reg, 'div'); | |
| 	reg = new RegExp("↵", "g"); | |
| 	str = str.replace(reg, '<br />'); | |
| 	 | |
| 	let startStr = `<div style="font-size:0">` | |
| 	 | |
| 	 | |
| 	str = `<div style="font-size:0">` + str + `</div>` | |
| 	 | |
| 	return str; | |
| } | |
| 
 | |
| // 格式化价格 | |
| Vue.prototype.formatePrice = str => { | |
| 	try { | |
| 		str = str.toFixed(2) | |
| 		let priceObj = {allPrice: str, iNum: str.split('.')[0], fNum: str.split('.')[1]} | |
| 		  | |
| 		return `<div class="fixPrice">${priceObj.iNum}<span class="fixNum">.${priceObj.fNum}</span></div>`  | |
| 	} catch(e) { | |
| 		console.log(e) | |
| 		return str | |
| 	} | |
| } | |
| 
 | |
| // 获取路径参数 | |
| Vue.prototype.getUrlPara = url => { | |
| 	let arrUrl = url.split("?"); | |
| 	let para = arrUrl[1]; | |
| 	return para ? para.split('&') : false; | |
| } | |
| 
 | |
| // 中文姓名规则 | |
| Vue.prototype.idChinaName = (val) => { | |
| 	var pattern = /^[\u4E00-\u9FA5]{2,4}$/ | |
| 	return pattern.test(val); | |
| } | |
| 
 | |
| // 身份证验证规则 | |
| Vue.prototype.idCardNumber = (val) => { | |
| 	var pattern = /^\d{17}(\d{1}|[X|x])$/ | |
| 	return pattern.test(val); | |
| } | |
| 
 | |
| // 护照验证正则 | |
| Vue.prototype.passportValid = (val) => { | |
| 	return /^([a-zA-z]|[0-9]){5,17}$/.test(val); | |
| } | |
| 
 | |
| // 台胞证正则 | |
| Vue.prototype.taiwanValid = (val) => { | |
| 	return /^\d{8}|^[a-zA-Z0-9]{10}|^\d{18}$/.test(val); | |
| } | |
| 
 | |
| // 港澳通行证正则 | |
| Vue.prototype.gangaoValid = (val) => { | |
| 	return /^([A-Z]\d{6,10}(\(\w{1}\))?)$/.test(val); | |
| } | |
| 
 | |
| // 外国人永久居留证正则 | |
| Vue.prototype.foreignerValid = (val) => { | |
| 	return /(^[A-Za-z]{3})([0-9]{12}$)/.test(val); | |
| } | |
| 
 | |
| // 军官证正则 | |
| Vue.prototype.officerValid = (val) => { | |
| 	return /^[\u4E00-\u9FA5](字第)([0-9a-zA-Z]{4,8})(号?)$/.test(val); | |
| } | |
| 
 | |
| // 邮箱验证正则 | |
| Vue.prototype.emailValid = (val) => { | |
| 	return /^[A-Za-z0-9\u4e00-\u9fa5]+@[a-zA-Z0-9_-]+(\.[a-zA-Z0-9_-]+)+$/.test(val) | |
| } | |
| 
 | |
| //判断电话号码格式 | |
| Vue.prototype.IsTel = tel => { | |
| 	var pattern = /^(13[0-9]|14[01456879]|15[0-35-9]|16[2567]|17[0-8]|18[0-9]|19[0-35-9])\d{8}$/; | |
| 	return pattern.test(tel); | |
| } | |
| 
 | |
| //判断澳门电话号码格式 | |
| Vue.prototype.IsTelMacau = tel => { | |
| 	var pattern = /^\d{8}$/; | |
| 	return pattern.test(tel); | |
| } | |
| 
 | |
| //验证码格式 | |
| Vue.prototype.IsCode = code => { | |
| 	var pattern = /^\d{6}$/; | |
| 	return pattern.test(code); | |
| } | |
| 
 | |
| // 图片显示判断 | |
| Vue.prototype.showImg = img => { | |
| 	if (!img) return; | |
| 	if (img.indexOf('https://') != -1 || img.indexOf('http://') != -1) { | |
| 		return img; | |
| 	} else { | |
| 		// return Vue.prototype.NEWAPIURL + img; | |
| 		return 'https://static.ticket.sz-trip.com'+img; | |
| 	} | |
| } | |
| 
 | |
| // 获取经纬度 | |
| Vue.prototype.getLocation = () => { | |
| 	uni.startLocationUpdate({ | |
| 		success: res => { | |
| 			uni.onLocationChange(data => { | |
| 				uni.setStorageSync('location', { | |
| 					lat: data.latitude, | |
| 					lon: data.longitude | |
| 				}); | |
| 			}) | |
| 		} | |
| 	}) | |
| } | |
| 
 | |
| // 路由页面跳转 | |
| Vue.prototype.gotoPath = path => { | |
| 	uni.navigateTo({ | |
| 		url: path | |
| 	}) | |
| } | |
| 
 | |
| // 返回上一页 | |
| Vue.prototype.goBack = () => { | |
| 	getCurrentPages().length > 1 ? uni.navigateBack({}) : uni.switchTab({ | |
| 		url: '/pages/index/index' | |
| 	}) | |
| } | |
| 
 | |
| // 打开地图 | |
| Vue.prototype.openLocation = (lat, lon) => { | |
| 	uni.openLocation({ | |
| 		latitude: Number(lat), | |
| 		longitude: Number(lon), | |
| 		success: function() { | |
| 			console.log('success'); | |
| 		} | |
| 	}); | |
| } | |
| 
 | |
| // 拨打电话 | |
| Vue.prototype.clickPhone = (phone) => { | |
| 	uni.makePhoneCall({ | |
| 		phoneNumber: phone | |
| 	}) | |
| } | |
| 
 | |
| //周几 | |
| Vue.prototype.ShowDateDay = day => { | |
| 	let stateTxt = ""; | |
| 	switch (day) { | |
| 		case 0: | |
| 			stateTxt = '周日' | |
| 			break; | |
| 		case 1: | |
| 			stateTxt = '周一' | |
| 			break; | |
| 		case 2: | |
| 			stateTxt = '周二' | |
| 			break; | |
| 		case 3: | |
| 			stateTxt = '周三' | |
| 			break; | |
| 		case 4: | |
| 			stateTxt = '周四' | |
| 			break; | |
| 		case 5: | |
| 			stateTxt = '周五' | |
| 			break; | |
| 		case 6: | |
| 			stateTxt = '周六' | |
| 			break; | |
| 	} | |
| 	return stateTxt | |
| } | |
| 
 | |
| Date.prototype.Format = function(fmt) { //author: meizz | |
| 	var o = { | |
| 		"Y+": this.getFullYear(), //月份 | |
| 		"M+": this.getMonth() + 1, //月份 | |
| 		"d+": this.getDate(), //日 | |
| 		"h+": this.getHours(), //小时 | |
| 		"m+": this.getMinutes(), //分 | |
| 		"s+": this.getSeconds(), //秒 | |
| 		"q+": Math.floor((this.getMonth() + 3) / 3), //季度 | |
| 		"S": this.getMilliseconds() //毫秒 | |
| 	}; | |
| 	if (/(y+)/.test(fmt)) | |
| 		fmt = fmt.replace(RegExp.$1, (this.getFullYear() + "").substr(4 - RegExp.$1.length)); | |
| 	for (var k in o) | |
| 		if (new RegExp("(" + k + ")").test(fmt)) | |
| 			fmt = fmt.replace(RegExp.$1, (RegExp.$1.length == 1) ? (o[k]) : (("00" + o[k]).substr(("" + o[k]) | |
| 				.length))); | |
| 	return fmt; | |
| } | |
| 
 | |
| // 是否接收订阅消息 | |
| Vue.prototype.getSubscribeMessage = () => { | |
| 	const templateIds = [ | |
| 		// 退款成功通知 | |
| 		'hRZoiEES2BWtKb6Xgsnn8khLQH9un5j_11qu0bwlhfE', | |
| 		// 订单核销通知 | |
| 		// '7D-JP7o0nQ_NiQk2w8mBs8jdT1_7ofvyBN-G9NLY2Zk', | |
| 		// 订单支付成功通知 | |
| 		// '6cHez9KDlCDp1_nWUlUSV7qEaahIQWmYVlOCE-J6ODQ', | |
| 		// 出票失败提醒 | |
| 		// 'G-N85zK2gPwgTRZWQrtHZo_-5TFcdAqBxSk4qsqcvVc', | |
| 		// 出票结果通知 | |
| 		'YyTCUIYBnrj9CyKks8cOjNX_Rk8a4yVdswMP-zXVbhc' | |
| 	] | |
| 	uni.requestSubscribeMessage({ | |
| 		tmplIds: templateIds, | |
| 		complete(res) { | |
| 			uni.navigateTo({ | |
| 				url: '/subPackages/order/trades' | |
| 			}) | |
| 		} | |
| 	}) | |
| } | |
| 
 | |
| // 金刚区头图 | |
| Vue.prototype.getHeadImg = id => { | |
| 	return Vue.prototype.Post({ | |
| 			id, | |
| 		}, | |
| 		'/api/multimedia/detail' | |
| 	).then(res => { | |
| 		return res.data.head_img | |
| 	});	 | |
| } | |
| 
 | |
| // 获取最大优惠券 params:{money:100,sku_ids:1,2} money单位分 | |
| Vue.prototype.getMaxCoupon = async function (param) { | |
| 	param.money = (param.money||0) | |
| 	let res = await this.Post(param, "/api/coupon/use_max_coupon_list") | |
| 	if (res.code == 1 && res.data.id) { | |
| 		this.$store.commit("choseCoupon",res.data); | |
| 	} | |
| 	return res.data | |
| } | |
| 
 | |
| // H5小程序跳转外部链接 | |
| Vue.prototype.gotoWebUrl = url => { | |
| 	uni.navigateTo({ | |
| 		url: '/subPackages/webPage/webPage?url=' + encodeURIComponent(url) | |
| 	}) | |
| 	 | |
| 	// #ifdef MP-WEIXIN | |
| 	uni.navigateTo({ | |
| 		url: '/subPackages/webPage/webPage?url=' + encodeURIComponent(url) | |
| 	}) | |
| 	// #endif | |
| 	// #ifdef H5 | |
| 	window.location.href = url | |
| 	// #endif | |
| } | |
| 
 | |
| // 根据类型跳转详情 | |
| Vue.prototype.gotoDetailByType = item => { | |
| 	let url = '' | |
| 	if(item.is_package) { | |
| 		url = '/subPackages/food/detail?id=' + (item.product_id || item.id) | |
| 		 | |
| 		Vue.prototype.gotoPath(url) | |
| 		return; | |
| 	} | |
| 	switch (item.type){ | |
| 		case 'line': | |
| 			url = '/subPackages/line/lineDetail?id=' + (item.id || item.product_id) | |
| 			break; | |
| 		case 'hotel': | |
| 			url = '/subPackages/homestay/detail?id=' + (item.scene_id || item.id) | |
| 			break; | |
| 		case 'ticket': | |
| 			url = '/subPackages/ticket/detail?id=' + (item.scene_id || item.id) | |
| 			break; | |
| 		case 'post': | |
| 			url = '/subPackages/techan/detail?id=' + (item.product_id || item.id) | |
| 			break; | |
| 		case 'scenic': | |
| 			url = '/subPackages/ticket/detail?id=' + (item.scene_id || item.id) | |
| 			break; | |
| 		default: | |
| 			break; | |
| 	} | |
| 	console.log(url) | |
| 	Vue.prototype.gotoPath(url) | |
| } | |
| 
 | |
| // 跳转购物车 | |
| Vue.prototype.goCartPage = ()=>{ | |
| 	uni.switchTab({ | |
| 		url: "/pages/cart/cart" | |
| 	}) | |
| }, | |
| 
 | |
| // 购物车下单去下个页面 | |
| Vue.prototype.goCartNextPage= function(currentPageIndex){ | |
| 	// currentPage "" 购物车  'techan邮寄' 'techan配送' 'techan自提' 'ticket' 'food' 'hotel' | |
| 	// 0 1 2 3 4 | |
| 	let techanOrderList = this.$store.state.user.techanOrderList; | |
| 	let techanList1 = techanOrderList.list1 | |
| 	let techanList2 = techanOrderList.list2 | |
| 	let techanList3 = techanOrderList.list3 | |
| 	let ticketOrderList = this.$store.state.user.ticketOrderList; | |
| 	let foodOrderList = this.$store.state.user.foodOrderList; | |
| 	let hotelOrderList = this.$store.state.user.hotelOrderList; | |
| 	console.log(hotelOrderList) | |
| 	 | |
| 	let orderPage = [ | |
| 		{path: '/subPackages/techan/cartOrder', length: techanList1.length},  | |
| 		{path: '/subPackages/techan/cartOrder2', length: techanList2.length},  | |
| 		{path: '/subPackages/techan/cartOrder1', length: techanList3.length},  | |
| 		{path: '/subPackages/ticket/order', length: ticketOrderList.length}, | |
| 		{path: '/subPackages/food/order', length: foodOrderList.length}, | |
| 		{path: '/subPackages/homestay/order',length: hotelOrderList.length} | |
| 	] | |
| 	let nextPage = orderPage.find((v,index)=>v.length>0&&index>=currentPageIndex) | |
| 	if (nextPage) { | |
| 		uni.navigateTo({ | |
| 			url: nextPage.path+'?isShoppingCart=1' | |
| 		}) | |
| 	} else { | |
| 		console.log(orderPage, nextPage, '结束') | |
| 		// todo 最后下单页面 | |
| 		uni.navigateTo({ | |
| 			url: '/subPackages/order/cartOrder' | |
| 		}) | |
| 		 | |
| 	} | |
| } | |
| 
 | |
| 
 | |
| Vue.prototype.sharePage = function(param) { | |
| 	const pages = getCurrentPages(); // 获取加载的页面 | |
| 	const view = pages[pages.length - 1]; // 获取当前页面的对象 | |
| 	 | |
| 	let shareParam = { | |
| 		title: '时味苏州', // 分享的名称 | |
| 		path: `${view.$page.fullPath}`, | |
| 		imageUrl: "https://static.ticket.sz-trip.com/uploads/20250818/5ea2c18a15db8a438f2ce642194b6051.jpg", | |
| 		mpId: 'wx699ed131345cf8dd', // 此处配置微信小程序的 AppId | |
| 		...param, | |
| 	} | |
| 	 | |
| 	return shareParam; | |
| } | |
| 
 | |
| 
 | |
| // 配送时间 | |
| /** | |
|  * 解析"小时:分钟"格式的时间为总分钟数 | |
|  * @param {string} timeStr - 时间字符串,格式为"HH:MM" | |
|  * @returns {number} 从0点开始的总分钟数 | |
|  */ | |
| Vue.prototype.parseTimeToMinutes = (timeStr)=> { | |
|     const [hours, minutes] = timeStr.split(':').map(Number); | |
|     if (isNaN(hours) || isNaN(minutes) || hours < 0 || hours > 23 || minutes < 0 || minutes > 59) { | |
|         throw new Error('时间格式不正确,请使用"HH:MM"格式(例如"08:01")'); | |
|     } | |
|     return hours * 60 + minutes; | |
| } | |
| 
 | |
| /** | |
|  * 判断当前时间是否在指定的时间范围内 | |
|  * @param {string} startTime - 起始时间,格式为"HH:MM"(例如"08:01") | |
|  * @param {string} endTime - 结束时间,格式为"HH:MM"(例如"20:02") | |
|  * @param {Date} [date] - 可选参数,指定要检查的日期,默认使用当前时间 | |
|  * @returns {boolean} 如果在时间范围内返回true,否则返回false | |
|  */ | |
| Vue.prototype.isInTimeRange=(startTime, endTime, date = new Date())=> { | |
|     try { | |
|         // 解析起始和结束时间为总分钟数 | |
|         const startTotalMinutes = Vue.prototype.parseTimeToMinutes(startTime); | |
|         const endTotalMinutes = Vue.prototype.parseTimeToMinutes(endTime); | |
|          | |
|         // 获取指定时间的总分钟数 | |
|         const hours = date.getHours(); | |
|         const minutes = date.getMinutes(); | |
|         const currentTotalMinutes = hours * 60 + minutes; | |
|          | |
|         // 判断是否在时间范围内 | |
|         return currentTotalMinutes >= startTotalMinutes && currentTotalMinutes <= endTotalMinutes; | |
|     } catch (error) { | |
|         console.error('时间范围检查出错:', error.message); | |
|         return false; | |
|     } | |
| } | |
| 
 | |
| // 邮寄自提配送文字 | |
| Vue.prototype.getDeliveryMethodStr=(str, obj={"1":"邮寄","2":"自提", "3":"配送"})=>{ | |
| 	if (str) { | |
| 		let code = str.split(",") | |
| 		console.log(code) | |
| 		let arr = code.map(v=>obj[v]).filter(v=>v) | |
| 		return arr | |
| 	} | |
| 	return [] | |
| } | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| 
 | |
| // H5 | |
| // #ifdef H5 | |
| let jWeixin = require('jweixin-module') | |
| 
 | |
| Vue.prototype.jWeixinInit = function () { | |
| 	if (window.navigator.userAgent.toLowerCase().match(/MicroMessenger/i) == "micromessenger") { | |
| 		this.Post({web_url: encodeURIComponent(window.location.href)}, "/api/wx/jsSdk").then(res=>{ | |
| 			if (res && res.data && res.data.jssdk) { | |
| 				let jssdk = res.data.jssdk | |
| 				jWeixin.config({ | |
| 					debug: false, | |
| 					appId: jssdk.appId, | |
| 					timestamp: jssdk.timestamp, | |
| 					nonceStr: jssdk.nonceStr, | |
| 					signature: jssdk.signature, | |
| 					jsApiList: ['getLocation', | |
| 						'openLocation', | |
| 						'updateTimelineShareData', | |
| 						'updateAppMessageShareData', | |
| 						'hideMenuItems', | |
| 						'showMenuItems', | |
| 						'hideAllNonBaseMenuItem' | |
| 					], | |
| 					openTagList: ['wx-open-launch-weapp'] | |
| 				}); | |
| 			} | |
| 		}) | |
| 	} | |
| } | |
| // #endif
 |