// ==UserScript== // @name 别样红辅助(大道店) // @namespace https://bbs.tampermonkey.net.cn/ // @version 0.5.7.2 // @description 别样红辅助 // @author EmpyrealTear // @icon https://c1.beyondh.com/maxpms/5.2.1.2/favicon.ico // @match .*://mn.beyondh.com:8101/* // @require https://cdn.bootcdn.net/ajax/libs/jquery/3.7.1/jquery.min.js // @require https://scriptcat.org/lib/513/2.0.0/ElementGetter.js // @grant unsafeWindow // @grant GM_setValue // @grant GM_getValue // @grant GM_registerMenuCommand // @grant GM_unregisterMenuCommand // @grant GM_setClipboard // ==/UserScript== (function () { 'use strict' const options = { menus: { isShowCleaner: { toStr: (x) => '显示保洁人员:' + (x ? '✓' : '✕') }, isShowCleanerCount: { toStr: (x) => '显示保洁统计:' + (x ? '✓' : '✕') }, isShowDetailContractName: { toStr: (x) => '显示中介明细:' + (x ? '✓' : '✕') }, isHighlightBalancenePrice: { toStr: (x) => '高亮余额≠房价:' + (x ? '✓' : '✕') }, isShowGuestRecords: { toStr: (x) => '显示客历入住:' + (x ? '✓' : '✕') }, isAllowUploadFaceImg: { toStr: (x) => '显示上传照片:' + (x ? '✓' : '✕') }, isShowFaceImg: { toStr: (x) => '显示住客照片:' + (x ? '✓' : '✕') }, isShowDetail: { toStr: (x) => '显示住客信息:' + (x ? '✓' : '✕') }, isFilterEx: { toStr: (x) => '细分其他过滤:' + (x ? '✓' : '✕') }, isShowUnwashedAC: { toStr: (x) => '显示待洗空调:' + (x ? '✓' : '✕') }, }, loads: function () { Object.keys(this.menus).forEach(v => { let val = GM_getValue(v) this.menus[v]['_menu'] = GM_registerMenuCommand(this.menus[v].toStr(val), () => { GM_setValue(v, !val) Object.keys(this.menus).forEach(v => GM_unregisterMenuCommand(this.menus[v]['_menu'])) this.loads() }) }) } } options.loads() const getReactStateNode = (dom) => { let key = Object.keys(dom).find(key => key.startsWith("__reactInternalInstance$")) let internalInstance = dom[key] if (internalInstance == null) return null return internalInstance._debugOwner ?? internalInstance.return ?? internalInstance._currentElement._owner } const sum = (arr) => { if (arr == undefined) return undefined let res = 0 for (let i of arr) res += i return res } const getCleaner = (id) => { if ((id >= 301 && id <= 311) || (id >= 601 && id <= 611)) return '谢' else if ((id >= 401 && id <= 411) || id == 706 || (id >= 708 && id <= 711) || (id >= 808 && id <= 812)) return '钟' // return '易' else if ((id >= 401 && id <= 411)) return '段' // else if (id == 706 || (id >= 708 && id <= 711) || (id >= 807 && id <= 812)) // return '谢' // else if (id >= 807 && id <= 812) // return '谢' // else if (id >= 701 && id <= 711) // return '钟' else return '段' } const getRoomPriceAndBalance = (arr) => { let price = 0, balance = 0, groupNumber = -1, rooms = [] if (Object.prototype.toString.call(arr) == '[object Array]') { for (let i = 0; i < arr.length; i++) { let res = getRoomPriceAndBalance(arr[i]) price += res.price balance += res.balance groupNumber = res.groupNumber if (res.rooms.length > 0) rooms = [...rooms, ...res.rooms] } } else { // 每日5:49后完成夜审 let prices = arr.DailyPrices?.filter(v => new Date(v.Date).AddDays(1).AddMinutes(60 * 6 - 11) > Date.now()) if (prices != undefined) price += sum(prices.map(v => v.ActualPrice ?? 0)) balance += arr.Balance rooms.push(arr) groupNumber = arr.GroupOrRelationNumber } return { price: Math.round(price * 100) / 100, balance: Math.round(balance * 100) / 100, rooms: rooms, groupNumber: groupNumber } } const asyncPool = async (arr, process = (item, arr) => item, begin = (v) => v, end = (v) => v, poolLimit = 10, percentComplete = (pct) => pct) => { let ret = [] let executing = new Set() let arr_res = new Array(arr.length) let completeCount = 0 // begin percentComplete(completeCount / arr.length * 100) arr = begin(arr) // process for (let [index, item] of arr.entries()) { var p = Promise.resolve().then(async () => { try { let res = await process(item, arr) arr_res[index] = res } catch (err) { console.warn(err) arr_res[index] = err } return }).finally(() => { percentComplete((++completeCount) / arr.length * 100) }) ret.push(p) executing.add(p) let clean = () => executing.delete(p) p.then(clean).catch(clean) if (executing.size >= poolLimit) { await Promise.race(executing) } } // end return Promise.all(ret).then(() => { percentComplete(completeCount / arr.length * 100) return end(arr_res) }) } const ajax = { GET: (url, isAsync = false) => { let res = null $.ajax({ url: url, type: 'GET', xhrFields: { withCredentials: true }, dataType: 'json', async: isAsync, success: (e) => { res = e } }) return res }, POST: (url, data, isAsync = false) => { let res = null $.ajax({ url: url, type: 'POST', xhrFields: { withCredentials: true }, data: JSON.stringify(data), contentType: 'application/json', dataType: 'json', async: isAsync, success: (e) => { res = e } }) return res } } const apis = { GetMixAllRoomDetails: () => { // 获取房态信息 let res = ajax.GET('https://mn.beyondh.com:8111/api/Room/GetMixAllRoomDetails') if (res == null) return res for (let i = 0; i < res.Content.length; i++) { let id = parseInt(res.Content[i].RoomNumber) res.Content[i].Cleaner = getCleaner(id) } return res }, GetMixSingleRoomOccupationInfo: (id) => { // 获取房间摘要信息 return ajax.GET(`https://mn.beyondh.com:8111/api/Room/GetMixSingleRoomOccupationInfo?roomNumber=${id}`) }, SearchGuestFolioCheckIn: () => { // 获取住客在住单 return ajax.GET(`https://mn.beyondh.com:8111/API/GuestFolio/SearchGuestFolio?PageSize=100&PageIndex=1&IsDesc=false&Status%5B%5D=I&IsMeiTuanOrder=false&DateRangeType=RecentWithOneYear`) }, SearchGuestFolioCheckOut: (keywords, pageIndex = 1, pageSize = 10) => { // 获取住客退房单(不含挂账单) return ajax.GET(`https://mn.beyondh.com:8111/API/GuestFolio/SearchGuestFolio?PageSize=${pageSize}&PageIndex=${pageIndex}&IsDesc=false&Status%5B%5D=O&Keywords=${keywords}&IsMeiTuanOrder=false&DateRangeType=RecentWithOneYear`) }, GetCustomerForEdit: (id) => { // 获取住客信息 return ajax.GET(`https://mn.beyondh.com:8111/API/Customer/GetCustomerForEdit?CustomerId=${id}`) }, GetBillStatistics: (billIds) => { // 获取订单收付总额 return ajax.POST( `https://mn.beyondh.com:8111/API/Bill/GetBillStatistics`, Object.prototype.toString.call(billIds) === '[object Array]' ? billIds : [billIds] ) }, GetCustomerPhoto: (id) => { // 获取住客照片 return ajax.GET(`https://mn.beyondh.com:8111/API/Customer/GetCustomerPhoto?customerId=${id}`) }, UploadCustomerPhoto: (id, photoBase64) => { // 上传住客照片 return ajax.POST( `https://mn.beyondh.com:8111/API/Customer/UploadCustomerPhoto`, { CustomerId: id, CustomerPhoto: photoBase64 }) }, GetCheckinMenuModelMix: (checkinId) => { return ajax.GET(`https://mn.beyondh.com:8111/API/Checkin/GetCheckinMenuModelMix?checkinId=${checkinId}`) }, GetCheckInList: () => { let date = new Date() let checkDate = new Date(new Date().Format('yyyy-MM-dd') + ' 05:49') let cur = (date < checkDate ? date.AddDays(-1) : date).Format('yyyy-MM-dd') let params = { 'SearchType[]': 'Checkin', 'PageIndex': 1, 'PageSize': 500, 'IsDesc': true, 'BeginCheckinTime': `${cur} 08:00`, 'EndCheckinTime': `${cur} 23:59` } let res = [] let data do { data = ajax.GET('https://mn.beyondh.com:8111/API/GuestFolio/AdvanceSearchGuestFolio?' + Object.entries(params).map(v => `${encodeURIComponent(v[0])}=${encodeURIComponent(v[1])}`).join('&')) res.push(...data.Content.Content) } while (res.length < data.Content.RecordCount) return res } } unsafeWindow.jQuery = $ unsafeWindow.getReactStateNode = getReactStateNode unsafeWindow.ajax = ajax unsafeWindow.apis = apis elmGetter.selector('jquery') const xhrSend = XMLHttpRequest.prototype.send; const CheckinType = { Normal: '正常', Hour2: '2H', Hour4: '4H', Hour8: '8H', Internal: '自用', LongTerm: '长包', Free: '免费' } XMLHttpRequest.prototype.send = function (data) { const xhr = this if (GM_getValue('isShowGuestRecords') && /API\/Customer\/SearchCustomerCheckinsByPost/.test(xhr.url)) { // 修复客历无法显示历史入住记录 let xhrSendBody = JSON.parse(data) // let idCardNumber = apis.GetCustomerForEdit(xhrSendBody.CustomerId).Content.IdCardNumber let props = getReactStateNode(jQuery('div.ant-col-11 > div:nth-child(2)')[0]).return.stateNode.props let idCardNumber = props.customerDetail.IdCardNumber let checkouts = apis.SearchGuestFolioCheckOut(idCardNumber, xhrSendBody.PageIndex, xhrSendBody.PageSize) const getter = Object.getOwnPropertyDescriptor(XMLHttpRequest.prototype, "responseText").get Object.defineProperty(xhr, "responseText", { get: () => { let result = JSON.parse(getter.call(xhr)) let rows = [] for (let i of checkouts.Content.Content) { let detail = i.GuestFolioDetailModels[0] let bill = undefined // apis.GetBillStatistics([i.BillId]) rows.push({ RoomNumber: detail.RoomNumber[0], // 房号 ArriveTime: i.ArriveTime, // 抵店时间 DepatureTime: i.DepatureTime, // 离店时间 Days: i.CheckinTypeId.startsWith('Hour') ? CheckinType[i.CheckinTypeId] : detail.DailyPrices.length, // 天数 PriceRoomTypeName: `${detail.ControlRoomTypeName}`, // 房型 FirstDayPrice: detail.FirstDayPrice, // 房价 DailyPrices: detail.DailyPrices, // 房价 RoomChargeAmount: bill == undefined ? (i.Contract.Name ?? '散客') : bill.Content?.CreditAmount, // 总房费 OtherChargeAmount: i.CheckinMemo, // 其他消费 }) } result.Content = { ...checkouts.Content, Content: rows } return JSON.stringify(result) }, }) } else if (GM_getValue('isAllowUploadFaceImg') && /API\/Configure\/UploadFaceImage/.test(xhr.url)) { // 上传住客图片 let props = getReactStateNode(jQuery('div.ant-col-11 > div:nth-child(2)')[0]).return.stateNode.props let reader = new FileReader() reader.readAsDataURL(data) reader.onload = (e) => apis.UploadCustomerPhoto(props.customerDetail.CustomerId, e.target.result.replace('data:image/jpeg;base64,', '')) const getter = Object.getOwnPropertyDescriptor(XMLHttpRequest.prototype, "response").get Object.defineProperty(xhr, "response", { get: () => { let result = JSON.parse(getter.call(xhr)) if (result.Content == false) result = { Code: 0, Message: "上传完毕", Content: true } return JSON.stringify(result) } }) } return xhrSend.apply(xhr, arguments) } const xhrOpen = XMLHttpRequest.prototype.open; XMLHttpRequest.prototype.open = function () { const xhr = this if (/api\/Room\/GetMixAllRoomDetails/.test(arguments[1])) { const getter = Object.getOwnPropertyDescriptor(XMLHttpRequest.prototype, "responseText").get Object.defineProperty(xhr, "responseText", { get: () => { let result = getter.call(xhr) let roomInfo = JSON.parse(result) roomInfo.Content = roomInfo.Content.map(v => ({ ...v, ApplyRoundsRoom: false })) let roomSet = {} let groupSet = {} let roomDetails = apis.SearchGuestFolioCheckIn().Content.Content for (let i = 0; i < roomInfo.Content.length; i++) { let id = parseInt(roomInfo.Content[i].RoomNumber) roomInfo.Content[i].Cleaner = getCleaner(id) let guests = roomDetails .filter(v => v.GuestFolioDetailModels[0].RoomNumber.some(num => num == roomInfo.Content[i].RoomNumber)) roomInfo.Content[i].Remark = guests.map(v => v.CheckinMemo).join(';') if (guests.length > 0) { roomInfo.Content[i].Balance = sum(guests.map(v => v.Balance)) roomInfo.Content[i].TodayPrice = guests[0].GuestFolioDetailModels[0].FirstDayPrice.ActualPrice roomInfo.Content[i].DailyPrices = guests[0].GuestFolioDetailModels[0].DailyPrices } roomSet[roomInfo.Content[i].RoomNumber] = roomInfo.Content[i] if (roomInfo.Content[i].GroupOrRelationNumber != -1) { let groupNumber = roomInfo.Content[i].GroupOrRelationNumber if (groupNumber in groupSet) { groupSet[groupNumber].push(roomInfo.Content[i]) } else { groupSet[groupNumber] = [roomInfo.Content[i]] } } } for (let k of Object.keys(groupSet)) { if (groupSet[k].length == 1) { let checkinId = groupSet[k][0].CheckinId let menulist = apis.GetCheckinMenuModelMix(checkinId) if (menulist?.Content?.MenuItemList?.length > 1) { // console.log(menulist?.Content?.MenuItemList.filter(v => v.CheckinId != checkinId)) groupSet[k].push(...(menulist?.Content?.MenuItemList.filter(v => v.CheckinId != checkinId).map(v => ({ ...v, GroupOrRelationNumber: parseInt(k) })) ?? [])) } } } // console.log(roomSet) console.log(groupSet) let cleans = {} let rooms = roomInfo.Content for (let i = 0; i < rooms.length; i++) { let k = rooms[i].Cleaner if (!(k in cleans)) cleans[k] = 0 // VD=空脏; VC=空净; V_C=空夜; OOO=维修; OK=已检; OD=住脏; OC=住净; ED=预退; if (rooms[i].CustomerStatus == 'IsEstArrival' || /VD|OD|OC/.test(rooms[i].RoomStatus)) // if (rooms[i].CustomerStatus == 'IsEstArrival' || /OD|OC/.test(rooms[i].RoomStatus)) cleans[k] += 1 } // 统计需保洁数量 if (GM_getValue('isShowCleanerCount')) { jQuery('#cleaner').remove() jQuery('div>div.small:contains("出租率")').parent().prepend(`
(${Object.keys(cleans).map((v) => `${v}:${cleans[v]}`).join(';')})
`) } elmGetter.each('.room_number_default', (e) => { let roomId = e.text() let id = roomId.replace(/\s\(.+\)/, '') // 添加保洁显示 if (GM_getValue('isShowCleaner')) { if (!/[\(\)]/.test(roomId)) e.text(`${roomId} (${roomSet[roomId].Cleaner})`) } else { if (/[\(\)]/.test(roomId)) e.text(roomId.replace(/\s\(.+\)/, '')) } // 修改中介小图标 e.nextAll('.icon_group').each((i, v) => { elmGetter.get('.iconfont', v).then(elm => { let name = roomSet[id]?.ContractName if (name != undefined && /小程序|直客通|赫程/.test(name)) { if (GM_getValue('isShowDetailContractName')) { if (/[小直赫介银金钻]/.test(elm.text())) { let txt = name[0], icon = 'icon_circle' if (/小程序|直客通/.test(name)) { let remark = roomSet[id]?.Remark if (/[银金钻][卡石]/.test(remark)) { txt = remark.replace(/\n/g, '').replace(/.*([银金钻])[卡石].*/, '$1') icon = 'icon_circle icon_circle_' + (/直客通/.test(name) ? 'lv' : 'vacant_status') } } else if (/赫程/.test(name)) { txt = '赫' icon = 'icon_circle icon_circle_xie' } elm.parent().attr('class', icon) elm.text(txt) } } else { if (elm.text() != '介' && !elm.text().includes('抵')) { elm.parent().attr('class', 'icon_circle') elm.text('介') } } } }) }) let roomInfo = roomSet[id] // 高亮余额≠房价 if (GM_getValue('isHighlightBalancenePrice') && roomInfo.Balance != undefined) { // 每日5:49后完成夜审 let cur = getRoomPriceAndBalance(roomInfo) let grp = roomInfo.GroupOrRelationNumber == -1 ? cur : getRoomPriceAndBalance(groupSet[roomInfo.GroupOrRelationNumber]) e.parent().css('border', (cur.price > 0 && cur.balance == cur.price) || (grp.price > 0 && grp.balance == grp.price) || (/钻[卡石].*免费房/.test(roomInfo.Remark) && grp.price - grp.balance == 149) || (!/钻[卡石].*免费房/.test(roomInfo.Remark) && (cur.balance == cur.price || grp.price == grp.balance)) ? '' : '2px dashed #f24e4c') } else e.parent().css('border', '') if (GM_getValue('isShowUnwashedAC')) { $.ajax({ url: `https://mn.beyondh.com:8111/api/Room/GetMixSingleRoomOccupationInfo?roomNumber=${id}`, type: 'GET', xhrFields: { withCredentials: true }, dataType: 'json', async: true, success: (data) => { let roomRemark = data.Content.RoomRemark e.parent().find('.room_type').css('font-style', !/待洗空调/.test(roomRemark) ? '' : 'italic') e.parent().find('.room_type').css('text-decoration', !/待洗空调/.test(roomRemark) ? '' : 'underline') } }) } else { e.parent().find('.room_type').css('font-style', '') e.parent().find('.room_type').css('text-decoration', '') } }) let iconFilter = $('[owl-ignore="true"][class="row_dl real_time"] > dd[class="filter clear_float"]') let filters = { xie: { name: '赫程', icon: '', regex: /^赫/, isToggle: false }, vip: { name: '会员', icon: '', regex: /^[会银金钻]/, isToggle: false }, offline: { name: '会员', icon: '', regex: /^[小直]/, isToggle: false }, other: { name: '中介', icon: '', regex: /^介/, isToggle: false }, } if (iconFilter.length > 0 && GM_getValue('isFilterEx')) { let other = iconFilter.find('span:contains("其它")').closest('div') other.click(e => { if (other.attr('class') != 'select_tag selected') { Object.keys(filters).forEach(k => { let flt = filters[k] if ($(`#scriptcat_filter_${k}`).length == 0) { let id = `scriptcat_filter_${k}` let ele = $(`
${flt.icon}${flt.name}
`) ele.click((e) => { flt.isToggle = !flt.isToggle $('.room_number_default').each((i, v) => { let roomElement = $(v).closest('[owl-ignore="true"]') roomElement.css('display', !flt.isToggle || flt.regex.test(roomElement.find('.icon_group').text()) ? '' : 'none') }) ele.attr('class', flt.isToggle ? 'select_tag selected' : 'select_tag') ele.find('[class="iconfont g_icon"]').css('display', flt.isToggle ? '' : 'none') },) iconFilter.append(ele) } }) $('[owl-ignore][class$=FloorBreak]').before($('
')) elmGetter.get('span[style^=color]:contains("清空")').then(div => { console.log(div) $(div).click(e => { $('[id^=scriptcat_filter_][class="select_tag selected"]').click() $('[id^=scriptcat_filter_]').remove() $('[class^=script_floorbreak]').remove() }) }) } else { $('[id^=scriptcat_filter_][class="select_tag selected"]').click() $('[id^=scriptcat_filter_]').remove() $('[class^=script_floorbreak]').remove() } }) } // return JSON.stringify(roomInfo) return result }, }) } else if (GM_getValue('isAllowUploadFaceImg') && /API\/Checkin\/GetShowUploadFaceImageButton/.test(arguments[1])) { // 显示上传图片按钮 const getter = Object.getOwnPropertyDescriptor(XMLHttpRequest.prototype, "responseText").get Object.defineProperty(xhr, "responseText", { get: () => { let result = JSON.parse(getter.call(xhr)) result.Content = true if (unsafeWindow.pms) { unsafeWindow.pms['noEyes'] = true } return JSON.stringify(result) }, }) // 显示住客照片 const showFaceimg = () => { if (!GM_getValue('isShowFaceImg')) return let elm = document.querySelector('div.ant-col-11 > div:nth-child(2)') let props = getReactStateNode(elm).return.stateNode.props let src = '' if (props.customerDetail.PhotoStatus) src = apis.GetCustomerPhoto(props.customerDetail.CustomerId).Content if (jQuery('#zjz').length == 0) jQuery('form > div[class=ant-row]').append(jQuery(`
`)) else jQuery('#zjz img').attr('src', src) } jQuery('div.united_r div[owl-ignore] > div.cell_info').click(() => setTimeout(showFaceimg, 200)) jQuery(document.querySelector('div.ant-col-11 > div:nth-child(2)')).click(e => { let inputLabel = jQuery(e.target).closest('.ant-form-item').find('.ant-form-item-label').text()?.trim() let inputContent = jQuery(e.target).closest('.ant-form-item-control-input-content').text()?.trim() if (`${inputContent}`.length > 0 && `${inputLabel}` == '客人姓名') GM_setClipboard(inputContent) }) setTimeout(showFaceimg, 300) } else if (/GetCheckinCustomerDetail/.test(arguments[1])) { if (unsafeWindow.pms) { // 修改全局规则为展示隐藏信息 unsafeWindow.pms['noEyes'] = true && !!GM_getValue('isShowDetail') } } else if (/SearchGuestFolio|QuickSearchCustomer|GetMixRoomFuturePreorders|getordermix/.test(arguments[1])) { if (unsafeWindow.pms) { // 在触发查询客单时关闭展示隐藏信息开关,防止未知崩溃 unsafeWindow.pms['noEyes'] = false } } return xhrOpen.apply(xhr, arguments) } })();