| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680 | <template>    <!-- 设备详情 -->    <view class="content">        <view class="name_box">            <view class="name_content">{{ devInfo.devName }}</view>        </view>        <view class="contentBg"> </view>        <view            class="radar-box"            :style="{                left: `${offsetLeft}rpx`,                top: `${offsetTop}rpx`,            }"        >            <view                :style="{                    width: `${rotatedRect.width}rpx`,                    height: `${rotatedRect.height}rpx`,                    position: 'absolute',                    top: `${rotatedRect.top - 8}rpx`,                    left: `${rotatedRect.left - 8}rpx`,                    transform: `scale(${zoomTimes})`,                    border: `${9 / zoomTimes}rpx solid #333333`,                }"                class="tranStyle"            >            </view>            <template v-if="modules.length > 0">                <view                    v-for="(item, index) in modules"                    :key="index"                    class="moduleContent"                >                    <view                        :class="item.type"                        :style="{                            width: `${item.width}rpx`,                            height: `${item.height}rpx`,                            top: `${item.top}rpx`,                            left: `${item.left}rpx`,                            transform: `rotate(${item.rotate}deg)`,                            'transform-origin': 'center center',                            zIndex: 44,                        }"                    >                        <image                            style="width: 100%; height: 100%; display: block"                            :src="`../../static/furnitures/${item.type}.png`"                            mode=""                        />                    </view>                </view>            </template>            <template>                <image                    v-for="item in targetPoints"                    :key="item.id"                    class="action-icon-M"                    :src="`../../static/${lnbAction}.png`"                    :style="{                        position: 'absolute',                        transform: `translate(-50%, -50%) translate3d(${item.displayX}rpx, ${item.displayY}rpx, 0)`,                        transition: 'transform 1s linear',                        willChange: 'transform',                        zIndex: 9999,                        top: '0',                        left: '0',                        width: '70rpx',                        height: '70rpx',                    }"                />            </template>            <image class="redar-pic" src="../../static/rander.png" mode="" />        </view>        <view class="switchBox">            <text class="name">呼吸灯</text>            <switch                :checked="statusLight == 1"                @change="handleLightChange"                :active-value="1"                :inactive-value="0"                size="24px"                active-color="#07c160"                inactive-color="#eeeff1"                style="transform: scale(0.8)"            />        </view>        <view class="notice-info">            <view class="notice_title" @click="discrepancy">                <text style="color: #95a4b3; font-size: 28rpx">{{                    todayDate                }}</text>                <image src="../../static/rightArrow.png"></image>            </view>            <view                @click="getFrenEcharts()"                class="title"                v-if="                    devInfo.installPosition == 'Bedroom' ||                    (breathRate !== 0 && breathRate !== '')                "            >                <view                    class="title-text"                    style="color: #111111; font-size: 30rpx"                    >当前呼吸率</view                >                <view                    style="                        display: flex;                        justify-content: center;                        align-items: center;                    "                >                    <view                        class="title-btn"                        style="color: #111111; font-size: 30rpx"                        v-if="breathRate !== ''"                    >                        {{ breathRate }}次/分钟</view                    >                    <view                        v-else                        class="title-btn"                        style="color: #111111; font-size: 30rpx"                    >                        暂无                    </view>                    <image                        src="../../static/rightArrow.png"                        style="margin-left: auto; width: 30rpx; height: 30rpx"                    ></image>                </view>            </view>            <view class="title">                <view class="title-text" style="color: #22dea7"                    >今日进出频次</view                >                <view class="title-btn" style="color: #22dea7">                    {{ freQuenceList.length || 0 }}次                </view>            </view>            <view class="stayDetail" v-if="freQuenceList.length > 0">                <!-- 只有一条时直接展示 -->                <view v-if="freQuenceList.length === 1">                    <view                        v-for="item in freQuenceList"                        :key="item.id"                        class="stayDetail-item"                    >                        <view class="stayDetail-text">                            {{ formatTime(item.enterTime) }}目标进入                        </view>                        <view class="stayDetail-btn">                            {{ formatTime(item.leaveTime) }}目标离开                        </view>                    </view>                </view>                <!-- 多条时用 swiper 展示 -->                <swiper                    v-else                    class="auto-scroll-swiper"                    :indicator-dots="false"                    :autoplay="false"                    :interval="3000"                    :circular="true"                    :display-multiple-items="1"                    :vertical="false"                    :current="currentIndex"                    @change="onSwiperChange"                >                    <swiper-item                        v-for="item in freQuenceList"                        :key="item.id"                        class="stayDetail-item"                    >                        <view class="stayDetail-text">                            {{ formatTime(item.enterTime) }}目标进入                        </view>                        <view class="stayDetail-btn">                            {{ formatTime(item.leaveTime) }}目标离开                        </view>                    </swiper-item>                </swiper>            </view>        </view>        <view class="box" v-if="!breathShow">            <view class="handle-btn">                <view class="btn1" @click="shareDevice" v-if="shareJudge"                    >分享</view                >                <view class="noShareFlag" @click="noShareFlag" v-else                    >分享</view                >                <view                    class="btn1"                    @click="gotoSetting"                    v-if="devInfo.online == 1"                    >设置</view                >                <view class="noShareFlag" @click="gotoSetting" v-else                    >设置</view                >                <view class="btn2" @click="healthAlarm">守护计划</view>            </view>        </view>        <view class="box" v-else>            <view class="handle-btn">                <view class="closeBreath" @click="breathShow = false"                    >关闭呼吸率曲线图</view                >                <view class="btn2" @click="saveBreath"> 保存呼吸率曲线图 </view>            </view>        </view>        <!-- 遮罩层 -->        <view class="mask" v-if="shareModel" @click="shareModel = false"></view>        <!-- 弹窗内容 -->        <view class="share-modal" v-if="shareModel">            <view class="modal-header">                <text class="cancel-btn" @click="shareModel = false">取消</text>                <text class="modal-title">分享</text>                <text class="confirm-btn" @click="onShareConfirm()">确认</text>            </view>            <view class="modal-body">                <view class="info-row">                    <view class="label">设备序列号:</view>                    <view class="value">{{ devInfo.clientId }}</view>                </view>                <view class="info-row">                    <view class="label">设备名称:</view>                    <view class="value">{{ devInfo.devName }}</view>                </view>                <view class="input-row phoneInfo">                    <view>被分享人</view>                    <input                        class="input"                        placeholder="请输入手机号"                        type="number"                        v-model="sharedPhone"                    />                </view>                <view class="funChoice">                    <view class="funItem" @click="smChange()">                        <label class="simple-radio">                            <radio                                :value="messageFlag"                                color="#7c5345"                                :checked="messageFlag == true"                                class="hide-original"                            />                            <text>短信权限</text>                        </label>                    </view>                    <view class="funItem">                        <label class="simple-radio" @click="snChange()">                            <radio                                :value="serviceNumberFlag"                                color="#7c5345"                                :checked="serviceNumberFlag == true"                                class="hide-original"                            />                            <text>服务号通知</text>                        </label>                    </view>                    <view class="funItem">                        <label class="simple-radio" @click="vfChange()">                            <radio                                :value="voipFlag"                                color="#7c5345"                                :checked="voipFlag == true"                                class="hide-original"                            />                            <text>语音通话</text>                        </label>                    </view>                </view>            </view>        </view>        <view class="modal-mask" v-if="choiceVisible" @click="closeChoice">            <view class="modal-container">                <view class="modal-header">                    <text class="title">请选择分享方式</text>                </view>                <view class="modal-buttons">                    <!-- 手机号分析按钮 -->                    <button class="btn phone-btn" @click="handlePhoneAnalysis">                        <text>手机号分享</text>                    </button>                    <!-- 链接分享按钮 -->                    <button class="btn link-btn" @click="handleLinkShare">                        <text>链接分享</text>                    </button>                </view>            </view>        </view>        <!-- ECharts图标模块 -->        <view v-if="breathShow" class="echartsClass">            <image                src="../../static/closePng.png"                class="closePng"                @click="breathShow = false"            ></image>            <l-echart                id="chart-canvas"                canvas-id="chart-canvas"                ref="chartRef"                @finished="initChart"                v-if="breathShow"                @click="breathShow = false"            ></l-echart>        </view>        <alarModel v-if="showModle" ref="alarmModel" />    </view></template><script>import * as echarts from "../../uni_modules/lime-echart/static/echarts.min";import MqttService from "../../utils/globalMqtt.js";import {    convert_region_r2c,    rotateRect_cw,    convert_furniture_r2c,} from "../../utils/changezb.js";export default {    data() {        return {            clientId: "",            width: 0, //检测区域宽度            length: 0, //检测区域长度            devInfo: "",            actionName: "",            startDate: "",            endDate: "",            softWare: "",            statusLight: 0,            lnbAction: "",            stayDetail: "",            todayDate: "",            dev_id: "",            nowTime: "",            devName: "",            devType: "",            localPhone: uni.getStorageSync("phone"),            freQuenceList: [],            // mqtt相关            currentIndex: 0,            modules: [],            autoPlayinterval: "",            choiceVisible: "",            // 手机号分享授权模块            shareModel: false,            alarmModel: false,            sharedPhone: "",            messageFlag: false,            serviceNumberFlag: false,            voipFlag: false,            // mqtt模块            targetPoints: {},            inactivityTimer: null,            left: 0,            top: 0,            breathRate: "",            breathShow: false,            breathRpmList: [],            option: {                title: [                    {                        text: "呼吸率曲线",                        left: "center",                        top: 20,                        textStyle: { fontSize: 14 },                    },                    {                        text: "BPM: 0 次/分钟",                        left: "center",                        top: 50,                        textStyle: { fontSize: 12, color: "#805246" },                    },                    {                        text: "时间 (秒)",                        left: "right",                        top: "bottom",                        textStyle: { fontSize: 12, color: "#888888" },                    },                ],                grid: { left: 60, right: 20, top: 80, bottom: 50 },                xAxis: {                    type: "category",                    data: Array.from({ length: 60 }, (_, i) => i + 1),                    boundaryGap: false,                    splitLine: { show: false },                },                yAxis: {                    type: "value",                    min: 0,                    max: 40,                    name: "呼吸率(次)",                    nameLocation: "end",                    nameGap: 20,                    nameTextStyle: { fontSize: 12, color: "#888888" },                    splitLine: { show: true },                },                series: [                    {                        type: "line",                        smooth: true,                        symbol: "none",                        lineStyle: { color: "#805246", width: 2 },                        data: Array(60).fill(0),                    },                ],                animation: true,                animationDuration: 100,            },            showModle: false,            zoomTimes: 2,            // 设备分享权限判断            shareJudge: false,            unsubscribeFn: null,            fallingEventChange: null,            lnbActionJudean: false,            setIntervalVal: null,            // 重新计算雷达坐标位置            x_radar: 250,            y_radar: 250,            offsetLeft: 0,            offsetTop: 0,            //检测区域计算            rotatedRect: {},        };    },    computed: {},    methods: {        // 获取设备信息        getdevInfo(devId) {            this.$http                .get(`wap/device/queryDeviceInfoById`, {                    devId: devId,                })                .then((res) => {                    if (res.data.data) {                        this.devInfo = res.data.data;                        this.devType = this.devInfo.devType;                        this.width = Math.abs(                            this.devInfo.yyEnd - this.devInfo.yyStart                        );                        this.length = Math.abs(                            this.devInfo.xxEnd - this.devInfo.xxStart                        );                        this.statusLight = this.devInfo.statusLight;                        this.calculate(this.width, this.length);                        this.calculateRegion();                        //  this.offsetLeft = -(x_cm_start + x_cm_stop) / 2;                        //             this.offsetTop = (y_cm_start + y_cm_stop) / 2 + 80;                        this.calculateOffset();                        // 设备分享权限判断                        this.shareJudge =                            this.devInfo.userId == uni.getStorageSync("userId")                                ? true                                : false;                    }                })                .catch((err) => {});        },        getdevRoomInfo(devId) {            this.$http                .get(`wap/room/readRoom`, {                    devId: devId,                })                .then((res) => {                    const pRadar = {                        x: this.x_radar,                        y: this.y_radar,                        angle: uni.getStorageSync("northAngle"),                        mountPlain: this.devInfo.mountPlain,                    };                    if (res.data.data) {                        this.modules = res.data.data.furnitures;                    }                    console.log(this.modules, 88888);                    this.modules.forEach((item) => {                        // 1️⃣ 计算家具在参考系中的矩形                        const rect = convert_furniture_r2c(item, pRadar);                        console.log(rect, "rect222222");                        const rotatedRect = rotateRect_cw(                            rect,                            pRadar,                            pRadar.angle                        );                        console.log(rotatedRect, "rotatedRect222222");                        item.left = rotatedRect.left;                        item.top = rotatedRect.top;                        item.width = rotatedRect.width;                        item.height = rotatedRect.height;                        item.name = item.name;                        item.type = item.type;                    });                    console.log(this.modules, 99999);                });        },        // 处理设备状态灯        handleLightChange(e) {            let newValue = e.detail.value == true ? 1 : 0;            this.$http                .post(`wap/device/statusLight`, {                    statusLight: newValue,                    devId: this.devInfo.devId,                })                .then((res) => {                    if (res.data.code == 200) {                        uni.showToast({                            title: "操作成功",                            icon: "success",                        });                        this.statusLight = newValue;                    } else {                        wx.showToast({                            title: res.data.message,                            icon: "none",                        });                    }                });        },        // 分享功能模块        shareDevice() {            this.choiceVisible = true;        },        noShareFlag() {            uni.showToast({                title: "您没有分享权限",                icon: "none",            });        },        onShareConfirm() {            if (!this.sharedPhone) {                uni.showModal({                    content: "请填写手机号!",                    showCancel: false,                });                return;            }            let reg_tel =                /^(13[0-9]|14[01456879]|15[0-3,5-9]|16[2567]|17[0-8]|18[0-9]|19[0-3,5-9])\d{8}$/;            if (!reg_tel.test(this.sharedPhone)) {                uni.showModal({                    content: "请填写正确手机号!",                    showCancel: false,                });                return;            }            let shareParam = {                sharerUserId: uni.getStorageSync("userId"),                devId: this.devInfo.devId,                sharerPhone: uni.getStorageSync("phone"),                sharedUserId: "",                sharedPhone: this.sharedPhone,                messageFlag: this.messageFlag == true ? 0 : 1,                serviceNumberFlag: this.serviceNumberFlag == true ? 0 : 1,                voipFlag: this.voipFlag == true ? 0 : 1,            };            this.$http                .post("wap/share/deviceShare", shareParam, {                    header: {                        "Content-Type": "application/json;charset=UTF-8",                    },                })                .then((res) => {                    if (res.data.code == 200) {                        uni.showToast({                            title: "分享成功",                            icon: "success",                            duration: 1500,                        });                    } else {                        uni.showToast({                            title: res.data.message,                            icon: "none",                            duration: 1500,                        });                    }                });            this.shareModel = false;        },        closeChoice() {            this.choiceVisible = false;        },        handlePhoneAnalysis() {            this.choiceVisible = false;            this.shareModel = true;        },        handleLinkShare() {            this.choiceVisible = false;            uni.navigateTo({                url:                    "/pagesA/linkShare/linkShare?devInfo=" +                    JSON.stringify(this.devInfo),            });        },        smChange() {            this.messageFlag = !this.messageFlag;        },        snChange() {            this.serviceNumberFlag = !this.serviceNumberFlag;        },        vfChange() {            this.voipFlag = !this.voipFlag;        },        gotoSetting() {            if (this.devInfo.online == 0) {                uni.showToast({                    title: "离线设备不支持设置",                    icon: "none",                });                return;            }            uni.navigateTo({                url:                    "/pagesA/deviceSetting/deviceSetting?devInfo=" +                    JSON.stringify(this.devInfo),            });        },        getFrequency(devId) {            let params = {                devId: devId,                createTimeStart: this.$time(new Date()),                createTimeEnd: this.$time(new Date()),                pageNo: 1,                pageSize: 999,            };            this.$http                .post(`wap/stats/inOutEventQuery`, params, {                    header: {                        "Content-Type": "application/json;charset=UTF-8",                    },                })                .then((res) => {                    if (res.data.code == 200) {                        if (res.data.data.rows.length > 0) {                            this.freQuenceList = res.data.data.rows;                        } else {                            this.freQuenceList = [];                        }                    } else {                        uni.showToast({                            title: res.data.message,                            icon: "none",                        });                    }                });        },        // 健康闹钟方法        healthAlarm() {            uni.navigateTo({                url:                    "/pagesA/healthAlarm/healthAlarm?devInfo=" +                    JSON.stringify(this.devInfo),            });        },        onSwiperChange(event) {            const current = event.detail.current;            const totalItems = this.freQuenceList.length;            if (current === totalItems - 1) {                this.autoplay = false;            }            this.currentIndex = current;        },        autoSwipe() {            if (this.freQuenceList && this.freQuenceList.length > 0) {                this.setIntervalVal = setInterval(() => {                    let nextIndex = this.currentIndex + 1;                    if (nextIndex >= this.freQuenceList.length) {                        nextIndex = 0; // 循环到第一个项目                    }                    this.currentIndex = nextIndex;                }, 3000); // 每3秒自动滚动一次            }        },        discrepancy() {            uni.navigateTo({                url:                    "/pagesA/discrepancy/discrepancy?freQuenceList=" +                    JSON.stringify(this.freQuenceList),            });        },        receptionChange(val) {            this.targetPoints = val;        },        receptHealth(val) {            this.breathRate = val;            this.setEcharts(val);        },        // echarts图表模块        getOption(list) {            // 固定 X 轴 [0 ~ 60]            const xData = Array.from({ length: 61 }, (_, i) => i);            const recent = list.slice(-61);            const data = new Array(61).fill(null);            for (let i = 0; i < recent.length; i++) {                data[i] = recent[i];            }            return {                title: [                    { text: "呼吸率曲线", left: "center", top: 20 },                    {                        text: `BPM: ${recent[recent.length - 1] || 0} 次/分钟`,                        left: "center",                        top: 50,                        textStyle: { fontSize: 12, color: "#805246" },                    },                    {                        text: "时间 (秒)",                        left: "right",                        top: "bottom",                        textStyle: {                            fontSize: 10,                            color: "#888888",                            fontWeight: "normal",                        },                    },                ],                grid: { left: 60, right: 20, top: 80, bottom: 50 },                xAxis: {                    type: "category",                    data: xData,                    boundaryGap: false,                },                yAxis: {                    type: "value",                    min: 0,                    max: 40,                    name: "呼吸率(次)",                    nameLocation: "end",                    nameGap: 20,                    nameTextStyle: {                        fontSize: 10,                        color: "#888888",                        fontWeight: "normal",                    },                    splitLine: { show: true },                },                series: [                    {                        type: "line",                        smooth: true,                        symbol: "none",                        data: data,                        showSymbol: false,                        lineStyle: { color: "#7a4e42", width: 2 },                    },                ],            };        },        getFrenEcharts() {            if (this.breathRate === "" || this.breathRate === null) {                uni.showToast({                    title: "暂无呼吸率",                    icon: "none",                    duration: 1500,                });                return;            }            this.breathShow = true;            this.$nextTick(() => {                this.initChart();            });        },        initChart() {            if (!this.$refs.chartRef) return;            if (this.chartInstance) {                this.chartInstance.dispose();                this.chartInstance = null;            }            this.$refs.chartRef.init(echarts, (chart) => {                this.chartInstance = chart;                chart.setOption(this.option, true);            });        },        setEcharts(val) {            if (!Array.isArray(this.breathRpmList)) {                this.breathRpmList = [];            }            this.breathRpmList.push(val);            if (!this.chartInstance) return;            const option = this.getOption(this.breathRpmList);            this.chartInstance.setOption(option, { notMerge: true });        },        saveBreath() {            const chart = this.$refs.chartRef;            if (!chart) {                uni.showToast({ title: "图表未渲染", icon: "none" });                return;            }            chart.canvasToTempFilePath({                success: (res) => {                    uni.saveImageToPhotosAlbum({                        filePath: res.tempFilePath,                        success: () => {                            uni.showToast({                                title: "保存成功",                                icon: "success",                            });                        },                        fail: (err) => {                            uni.showToast({                                title: "保存失败",                                icon: "none",                            });                        },                    });                },                fail: (err) => {                    uni.showToast({                        title: "导出失败",                        icon: "none",                    });                },            });        },        // 处理点位消息        handleMessage(topic, message, clientId) {            // 清除不活动定时器            clearTimeout(this.inactivityTimer);            this.inactivityTimer = setTimeout(() => {                this.targetPoints = {};                // console.log("没有点位,消除点位数据");            }, 1500);            // 验证topic格式            const match = topic.match(/^\/dev\/(.+)\/tracker_targets$/);            if (!match || match[1] !== clientId) return;            try {                const data = JSON.parse(message.toString());                // console.log(data.tracker_targets, "点位消息");                if (data.health) {                    if (                        data.health.breath_rpm ||                        data.health.breath_rpm === 0                    ) {                        this.receptHealth(Math.floor(data.health.breath_rpm));                    } else {                    }                }                this.processTrackerData(data.tracker_targets);                // console.log(data.tracker_targets, "MQTT消息解析成功22222");            } catch (e) {                console.error("MQTT消息解析失败", e);            }        },        processTrackerData(arr) {            if (Array.isArray(arr) && arr.length > 0 && Array.isArray(arr[0])) {                this.targetPoints = {};                const currentIds = new Set();                const newTargetPoints = {};                // 处理每个追踪目标                arr.forEach((item) => {                    if (!Array.isArray(item) || item.length < 4) return;                    const [x, y, z, id] = item;                    currentIds.add(id.toString());                    // 处理新点或更新现有点                    if (!this.targetPoints[id]) {                        newTargetPoints[id] = this.createNewTargetPoint(                            x,                            y,                            z,                            id                        );                    } else {                        newTargetPoints[id] = this.updateExistingTargetPoint(                            this.targetPoints[id],                            x,                            y,                            z,                            2                        );                    }                });                // 移除不存在的点                Object.keys(this.targetPoints).forEach((id) => {                    if (!currentIds.has(id)) {                        delete this.targetPoints[id];                    }                });                // 更新目标点                this.targetPoints = {                    ...this.targetPoints,                    ...newTargetPoints,                };                const northAngle = uni.getStorageSync("northAngle");                const radar = [this.x_radar, this.y_radar];                let points = [];                if (Array.isArray(this.targetPoints)) {                    points = this.targetPoints;                } else if (                    this.targetPoints &&                    typeof this.targetPoints === "object"                ) {                    points = Object.values(this.targetPoints);                }                this.targetPoints = points.map((p) => {                    return this.rotatePoint(p, radar, northAngle);                });                // console.log(this.targetPoints, "更新后的点位数据333");            }        },        createNewTargetPoint(x, y, z, id) {            return {                x,                y,                z,                id,                displayX: x,                displayY: y,                lastX: x,                lastY: y,            };        },        updateExistingTargetPoint(existingPoint, x, y, z, THRESHOLD) {            const dx = x - existingPoint.lastX;            const dy = y - existingPoint.lastY;            const distance = Math.sqrt(dx * dx + dy * dy);            if (distance > THRESHOLD) {                return {                    ...existingPoint,                    x,                    y,                    z,                    lastX: x,                    lastY: y,                    displayX: x,                    displayY: y,                };            }            return existingPoint;        },        // 计算点位问题        rotatePoint(point, radar, angle) {            const cx = this.x_radar;            const cy = this.y_radar;            const rad = (angle * Math.PI) / 180;            const cosA = Math.cos(rad);            const sinA = Math.sin(rad);            const xRot = point.x * cosA + point.y * sinA;            const yRot = -point.x * sinA + point.y * cosA;            point.displayX = cx + xRot;            point.displayY = cy - yRot;            return point;        },        // 计算缩放比例        calculate(width, length) {            let zoomTime = width > length ? width : length;            let practical = zoomTime;            this.zoomTimes = 600 / practical;            this.zoomTimes = this.zoomTimes - 0.2;            this.zoomTimes = Number(this.zoomTimes.toFixed(2));            console.log(this.xOffset, this.yOffset, this.zoomTimes, "偏移量");            this.lnbActionJudean = true;        },        calculateRegion() {            let x_cm_start = this.devInfo.xxStart;            let y_cm_start = this.devInfo.yyStart;            let x_cm_stop = this.devInfo.xxEnd;            let y_cm_stop = this.devInfo.yyEnd;            const trackingRegion = {                x_cm_start,                y_cm_start,                x_cm_stop,                y_cm_stop,            };            const pRadar = {                x: this.x_radar,                y: this.y_radar,                angle: uni.getStorageSync("northAngle"),                mountPlain: this.devInfo.mountPlain,            };            const rect = convert_region_r2c(trackingRegion, pRadar);            this.rotatedRect = rotateRect_cw(rect, pRadar, pRadar.angle);            console.log(this.rotatedRect, "rotatedRect");        },        calculateOffset() {            let x_cm_start = "";            let x_cm_stop = "";            let y_cm_start = "";            let y_cm_stop = "";            if (this.devInfo.northAngle == 0) {                x_cm_start = this.devInfo.xxStart;                x_cm_stop = this.devInfo.xxEnd;                y_cm_start = this.devInfo.yyStart;                y_cm_stop = this.devInfo.yyEnd;                this.offsetLeft = -(x_cm_start + x_cm_stop) / 2;                this.offsetTop = (y_cm_start + y_cm_stop) / 2 + 80;            }            if (this.devInfo.northAngle == 90) {                x_cm_start = this.devInfo.xxStart;                x_cm_stop = this.devInfo.xxEnd;                y_cm_start = -this.devInfo.yyEnd;                y_cm_stop = -this.devInfo.yyStart;                this.offsetLeft = -(x_cm_start + x_cm_stop) / 2;                this.offsetTop = (y_cm_start + y_cm_stop) / 2 + 80;            }            if (this.devInfo.northAngle == 180) {                x_cm_start = -this.devInfo.xxEnd;                x_cm_stop = -this.devInfo.xxStart;                y_cm_start = -this.devInfo.yyEnd;                y_cm_stop = -this.devInfo.yyStart;                this.offsetLeft = -(x_cm_start + x_cm_stop) / 2;                this.offsetTop = (y_cm_start + y_cm_stop) / 2 + 80;            }            if (this.devInfo.northAngle == 270) {                x_cm_start = -this.devInfo.xxEnd;                x_cm_stop = -this.devInfo.xxStart;                y_cm_start = this.devInfo.yyStart;                y_cm_stop = this.devInfo.yyEnd;                this.offsetLeft = -(x_cm_start + x_cm_stop) / 2;                this.offsetTop = (y_cm_start + y_cm_stop) / 2;            }        },        formatTime(time) {            return time ? time.slice(11, 19) : "暂无";        },        // 订阅消息,处理mqtt消息        initSubscriptions() {            const topicList = [                {                    topic: `/dev/${this.clientId}/tracker_targets`,                    key: "unsubscribeFn",                    callback: (message, msgTopic) => {                        const dataMatch = msgTopic.match(                            /^\/dev\/(.+)\/tracker_targets$/                        );                        const cmdMatch = msgTopic.match(                            /^\/mps\/wx_(.+)\/notice$/                        );                        if (dataMatch && dataMatch[1] === this.clientId) {                            this.handleMessage(                                msgTopic,                                message,                                this.clientId                            );                        } else if (cmdMatch) {                            this.$refs.alarmModel.hanOtherMessage(                                msgTopic,                                message                            );                        }                    },                },                {                    topic: `/dev/${this.clientId}/falling_event_change`,                    key: "fallingEventChange",                    callback: (message, msgTopic) => {                        const dataMatch = msgTopic.match(                            /^\/dev\/(.+)\/falling_event_change$/                        );                        if (dataMatch && dataMatch[1] === this.clientId) {                            const dataMessage = JSON.parse(message.toString());                            console.log(dataMessage, 888888);                            if (dataMessage.falling == 1) {                                this.falling = dataMessage.falling;                                this.lnbAction = "actionWarn";                            } else if (                                dataMessage.falling == 2 ||                                dataMessage.falling == 3                            ) {                                this.falling = dataMessage.falling;                                this.lnbAction = "actionSerious";                            } else {                                this.lnbAction = "action8";                            }                        }                    },                },            ];            topicList.forEach((item) => {                // 避免重复订阅                if (this[item.key]) return;                const subscribeFunc = () => {                    const unsubscribe = MqttService.subscribe(                        "DATA",                        item.topic,                        item.callback                    );                    if (unsubscribe) {                        this[item.key] = unsubscribe;                        console.log(`✅ 已成功订阅主题: ${item.topic}`);                    }                };                if (MqttService.dataConnected) {                    subscribeFunc();                } else {                    // MQTT 未连接,等待重连成功再订阅                    const handler = () => {                        subscribeFunc();                        uni.$off("mqttData-ready", handler);                    };                    uni.$on("mqttData-ready", handler);                }            });        },    },    onLoad(options) {        this.lnbAction = "action8";        const devItem = JSON.parse(options.devItem);        const { devId, clientId } = devItem;        // 缓存到页面实例        this.clientId = clientId;        uni.setStorageSync("currentDevItem", devItem);        this.getFrequency(devId);        this.dev_id = devId;        this.todayDate = this.$time(new Date(), 2);        // 清理旧定时器        clearTimeout(this.autoPlayinterval);        this.autoPlayinterval = setTimeout(() => this.autoSwipe(), 3000);        if (MqttService.dataConnected && MqttService.dataClient) {            // 已连接,直接订阅            this.initSubscriptions();        }    },    onShow() {        this.showModle = true;        this.lnbAction = "action8";        // MQTT 初始化        const userId = uni.getStorageSync("userId");        if (!userId) return console.error("MQTT 初始化失败:缺少 userId");        if (!MqttService.dataConnected) {            // 未连接,先 connect            MqttService.connectData(userId)                .then((client) => {                    // 页面订阅逻辑,等 connect 成功再执行                    const handler = () => {                        MqttService.resubscribeAll("DATA");                        uni.$off("mqttData-ready", handler);                    };                    uni.$on("mqttData-ready", handler);                })                .catch((err) => console.error("DATA MQTT 初始化失败", err));        }        this.getdevInfo(this.dev_id);        this.getdevRoomInfo(this.dev_id);    },    onUnload() {        // 清理定时器        clearTimeout(this.autoPlayinterval);        // 清理自动滑动定时器        clearInterval(this.setIntervalVal);        this.setIntervalVal = null;        this.autoPlayinterval = null;        // 取消订阅        ["unsubscribeFn", "fallingEventChange"].forEach((key) => {            if (this[key]) {                this[key]();                this[key] = null;            }        });    },    onHide() {        // MqttService.dataConnected = false;        this.showModle = false;    },};</script><style lang="less" scoped>.content {    height: 100vh;    background: linear-gradient(180deg, #faede2 0%, #f4f4f4 100%);    box-sizing: border-box;    padding-top: 20rpx;    position: relative;    .name_box {        display: flex;        justify-content: center;        align-items: center;        margin: 0 auto 60rpx 0;        .name_content {            display: inline-flex;            background: rgba(255, 255, 255, 0.65);            border-radius: 37rpx;            padding: 6rpx 18rpx;        }    }    .contentBg {        position: absolute;        border-radius: 37.5rpx;        top: 31%;        left: 50%;        width: 710rpx;        height: 680rpx;        background-color: #ffffff;        -webkit-transform: translate(-50%, -50%);        transform: translate(-50%, -50%);        z-index: 0;    }    .radar-box {        margin: 0 auto;        position: relative;        // display: flex;        // align-items: center;        // justify-content: center;        width: 500rpx;        height: 500rpx;        // background: #ffffff;        border-radius: 37.5rpx;        box-sizing: border-box;        .tranStyle {            position: absolute;            background-repeat: no-repeat;            background-position: center;            z-index: 1; // 父元素层级要高于伪元素容器            &::after {                content: "";                position: absolute;                inset: 0;                background: url("https://hflnxx.oss-cn-shanghai.aliyuncs.com/IMAGE/20250919/toilet_bg.png")                    no-repeat center;                background-size: 150% 150%;                z-index: -1;            }        }        .moduleContent {            view {                position: absolute;            }            .module-img {                width: 100%;                height: 100%;                display: block;            }        }        .action-icon-G {            position: absolute;            width: 100rpx;            height: 100rpx;        }        .action-icon-M {            // position: absolute;            // width: 50rpx;            // height: 50rpx;            // transform: translate(-50%, -50%);            // transition: transform 1s linear;            // will-change: transform;            // z-index: 9999;        }        .redar-pic {            position: absolute;            top: 50%;            left: 50%;            width: 40rpx;            height: 40rpx;            transform: translate(-50%, -50%);            z-index: 20;        }    }    .switchBox {        width: 710rpx;        height: 94rpx;        margin: 190rpx auto 20rpx auto;        padding: 0 37rpx;        display: flex;        align-items: center;        background: #ffffff;        border-radius: 37rpx;        box-sizing: border-box;        .name {            color: #111111;            font-size: 32rpx;        }    }    .notice-info {        width: 710rpx;        max-height: 300rpx;        // overflow: hidden;        padding: 30rpx 37rpx 10rpx 37rpx;        background: #ffffff;        border-radius: 37rpx;        box-sizing: border-box;        margin: 0 auto;        .notice_title {            display: flex;            align-items: space-between;            image {                margin-left: auto;                width: 30rpx;                height: 30rpx;            }        }        .title {            margin-top: 10rpx;            margin-bottom: 10rpx;            font-family: PingFang SC, PingFang SC;            font-weight: 500;            font-size: 34rpx;            color: #1757dd;            display: flex;            align-items: center;            justify-content: space-between;            .title-text {                font-size: 28rpx;                font-weight: 500;                color: #1757dd;            }            .title-btn {                font-size: 28rpx;                font-weight: 500;                color: #5a5a5a;            }        }        .stayDetail {            height: 80rpx;            // overflow: hidden;            margin-top: 10rpx;            margin-bottom: 10rpx;            padding-left: 20rpx;            font-family: PingFang SC, PingFang SC;            font-weight: 500;            font-size: 34rpx;            .stayDetail-item {                display: flex;                padding-bottom: 5rpx !important;                padding-top: 10rpx !important;                box-sizing: border-box;                line-height: 60rpx;                .stayDetail-text {                    width: 40%;                    font-size: 28rpx;                    height: 30rpx !important;                }                .stayDetail-btn {                    padding-left: 30rpx;                    font-size: 28rpx;                    height: 30rpx !important;                }            }        }    }    .box {        position: fixed;        bottom: 0;        left: 0;        width: 100vw;        height: 168rpx;        padding: 0 37rpx;        background: #ffffff;        box-sizing: border-box;        .handle-btn {            margin-top: 40rpx;            display: flex;            align-items: center;            justify-content: space-between;            .btn1 {                display: flex;                align-items: center;                justify-content: center;                width: 155rpx;                height: 94rpx;                background: #ffebe4;                border-radius: 28rpx;                font-weight: 500;                color: #111111;                font-size: 32rpx;                text-align: center;            }            .noShareFlag {                display: flex;                align-items: center;                justify-content: center;                width: 155rpx;                height: 94rpx;                border-radius: 28rpx;                font-weight: 500;                font-size: 32rpx;                text-align: center;                background: #f0f0f0;                color: #999999;                opacity: 0.6;            }            .btn2 {                display: flex;                align-items: center;                justify-content: center;                width: 328rpx;                height: 94rpx;                background: linear-gradient(                    105.95deg,                    #a27867 0%,                    #74483d 100%                );                border-radius: 28rpx;                box-shadow: 0rpx 4.69rpx 18.75rpx rgba(72, 41, 29, 0.15),                    0rpx 9.38rpx 9.38rpx rgba(154, 132, 89, 0.2),                    0rpx -4.69rpx 28.13rpx 4.69rpx #a16647 inset;                font-family: MiSans;                font-weight: 500;                color: #ffffff;                font-size: 32rpx;            }            .closeBreath {                display: flex;                align-items: center;                justify-content: center;                width: 328rpx;                height: 94rpx;                background: #ffebe4;                border-radius: 28rpx;                font-weight: 500;                color: #111111;                font-size: 32rpx;                text-align: center;            }        }    }    /* 弹窗部分样式 */    .mask {        position: fixed;        left: 0;        top: 0;        right: 0;        bottom: 0;        background: rgba(0, 0, 0, 0.3);        z-index: 100;    }    .share-modal {        position: fixed;        left: 0;        right: 0;        bottom: 0;        background: #fff;        border-radius: 24rpx 24rpx 0 0;        z-index: 101;        height: 750rpx;        .modal-header {            display: flex;            justify-content: space-between;            align-items: center;            padding: 32rpx 32rpx 0 32rpx;            font-size: 30rpx;            .cancel-btn {                color: #888;                font-size: 32rpx;            }            .modal-title {                color: #111111;                font-size: 36rpx;            }            .confirm-btn {                color: #996e5f;                font-size: 32rpx;            }        }        .modal-body {            padding: 50rpx;            .info-row {                display: flex;                margin-bottom: 24rpx;                font-size: 28rpx;                padding: 0 20rpx;                .label {                    color: #111111;                    width: 200rpx;                    font-size: 32rpx;                    text-align-last: justify;                }                .value {                    text-align: right;                    width: 65%;                    color: #111111;                    font-size: 32rpx;                }            }            .input-row {                display: flex;                align-items: center;                margin-top: 32rpx;                border-radius: 40rpx;                background: #f7f7f7;                padding: 0 24rpx;                height: 80rpx;                .input {                    text-align: right;                    flex: 1;                    border: none;                    background: transparent;                    font-size: 28rpx;                    color: #333;                }            }            .funChoice {                .funItem {                    .simple-radio {                        display: flex;                        align-items: center;                        padding: 20rpx 0;                    }                }            }        }    }    // 分享弹窗样式    .modal-mask {        position: fixed;        top: 0;        left: 0;        right: 0;        bottom: 0;        background: rgba(0, 0, 0, 0.5);        display: flex;        justify-content: center;        align-items: center;        z-index: 555;        .modal-container {            width: 80%;            background: #fff;            border-radius: 16rpx;            // overflow: hidden;            animation: fadeIn 0.3s;            .modal-header {                padding: 30rpx;                text-align: center;                border-bottom: 1rpx solid #f5f5f5;                .title {                    font-size: 36rpx;                    display: block;                    margin-bottom: 10rpx;                }            }            .modal-buttons {                display: flex;                flex-direction: column;                padding: 20rpx;                .btn {                    flex: 1;                    height: 90rpx;                    margin: 15rpx 0;                    border-radius: 45rpx;                    display: flex;                    align-items: center;                    justify-content: center;                    font-size: 32rpx;                    border: none;                    background: none;                    position: relative;                }                .btn-icon {                    width: 40rpx;                    height: 40rpx;                    margin-right: 15rpx;                }                .phone-btn {                    background: linear-gradient(                        105.95deg,                        #ba978a 0%,                        #a27867 100%                    );                    color: white;                }                .link-btn {                    background: linear-gradient(                        105.95deg,                        #a27867 0%,                        #74483d 100%                    );                    color: white;                }            }        }    }    .echartsClass {        position: absolute;        width: 650rpx;        height: 600rpx;        top: -8%;        left: -37%;        transform: translate(50%, 50%);        background: #ffffff;        .closePng {            width: 40rpx;            height: 40rpx;            position: fixed;            top: 10rpx;            right: 10rpx;            z-index: 555;        }    }}</style>
 |