| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202 | <template>  <div class="statistics-dashboard">    <!-- 数据概览 (Data Overview) -->    <el-card class="overview-section" shadow="never">      <el-row :gutter="20">        <el-col :xs="24" :sm="24" :md="16" :lg="16" :xl="16" class="companybox">          <img src="../assets/images/topbg.png" alt="" class="topimg">          <img src="../assets/images/topbg.png" alt="" class="bottomimg">          <div class="companyboxtitle">            企业数据          </div>          <div class="companyflex">            <div class="topbg companycard cardafter">              <div class="card-title1">                <img src="../assets/images/tab_company.png" alt="" class="icon-img">                分公司数量              </div>              <div class="card-value highlight1">                <count-to :start-val="0" :end-val="dealderCount" :duration="3600"                          class="card-panel-num companynumber" />              </div>            </div>            <div class="companycard cardafter">              <div class="card-title1">                <img src="../assets/images/salesperson.png" alt="" class="icon-img">                销售数量              </div>              <div class="card-value highlight1">                <count-to :start-val="0" :end-val="groupMgrCount" :duration="3600"                          class="card-panel-num companynumber" />              </div>            </div>            <div class="companycard cardafter">              <div class="card-title1">                <img src="../assets/images/member.png" alt="" class="icon-img">                会员数量              </div>              <div class="card-value highlight1">                <count-to :start-val="0" :end-val="memberCount" :duration="3600" class="card-panel-num companynumber" />                <span class="highlight-today-add companyadd">+{{todayIncreaseUserNum}}</span>              </div>            </div>            <div class="cardafter companycard">              <div class="card-title1">                <img src="../assets/images/tab_enterprise.png" alt="" class="icon-img">                企微数量              </div>              <div class="card-value highlight1">                <count-to :start-val="0" :end-val="qwMemberNum" :duration="3600" class="card-panel-num companynumber" />              </div>            </div>            <div class="botttombg companycard">              <div class="card-title1">                <svg-icon icon-class="phone" />                pad使用情况              </div>              <div class="card-value highlight1">                <count-to :start-val="0" :end-val="padUsedNum" :duration="3600" class="card-panel-num companynumber" />                /                <template v-if="typeof padTotalNum === 'number'">                  <count-to :start-val="0" :end-val="padTotalNum" :duration="1800" class="card-panel-num companynumber" />                </template>                <template v-else>                  <span class="card-panel-num companynumber">{{ padTotalNum }}</span>                </template>              </div>            </div>          </div>        </el-col>        <el-col :xs="24" :sm="24" :md="8" :lg="8" :xl="8" class="propertyboxtitle">          <div class="property_title">            资产概览          </div>          <div class="propertyboxflex">            <div class="property-card propertyline">              <div class="property-title">                <i class="el-icon-money"></i>                企业资产(元)              </div>              <div class="card-value highlight">                <count-to :start-val="0" :end-val="balance" :duration="3600" class="card-panel-num" />              </div>            </div>            <div class="property-card">              <div class="property-title">                <span>今日消耗 (元)</span>              </div>              <div class="card-value highlight" style="color: rgba(32, 33, 36, 1);margin-top: 10px;">                <count-to :start-val="0" :end-val="todayComsumption" :duration="3600" class="card-panel-num" />              </div>              <div class="card-compare">                较昨日 <span>+1</span>              </div>            </div>          </div>        </el-col>      </el-row>      <el-row :gutter="20">        <el-col :xs="24" :sm="24" :md="16" :lg="16" :xl="16">          <div class="operatetitle">            经营数据          </div>          <div class="operatetitle-col">            <div class="operatetitle-card">              <div class="card-title">                <i class="el-icon-shopping-cart-full"></i>                收款总数              </div>              <div class="operate-value highlight">                <count-to :start-val="0" :end-val="recvTotalNum" :duration="3600" class="card-panel-num" />                <div class="yesterdaybox">                  较昨日 <span class="highlight-today-add2">+{{recvTodayNum}}</span>                </div>              </div>              <div class="card-badge">              </div>            </div>            <div class="operatetitle-card">              <div class="card-title">                <i class="el-icon-shopping-cart-full"></i>                订单总数              </div>              <div class="operate-value highlight">                <count-to :start-val="0" :end-val="orderTotalNum" :duration="3600" class="card-panel-num" />                <div class="yesterdaybox">                  较昨日 <span class="highlight-today-add2">+{{todayOrderNum}}</span>                </div>              </div>              <div class="card-badge">              </div>            </div>            <div class="operatetitle-card">              <div class="card-title">                平台今日看课人数              </div>              <div class="operate-value highlight">                <count-to :start-val="0" :end-val="todayWatchUserCount" :duration="3600" class="card-panel-num" />              </div>              <div class="card-sub">                <span>配额上限</span>                <span class="sub-value">                  <count-to :start-val="0" :end-val="todayWatchUserCount" :duration="3600" class="card-panel-num"                            style="color: rgba(49, 185, 154, 1);" />                  /                  <count-to :start-val="0" :end-val="versionLimit" :duration="3600" class="card-panel-num" />                </span>              </div>              <el-progress :percentage="versionLimitPercent" :show-text="false"                           color="#409EFF"></el-progress>            </div>            <div class="operatetitle-card">              <div class="card-title">                <i class="el-icon-shopping-cart-full"></i>                商品总数              </div>              <div class="operate-value highlight">                <count-to :start-val="0" :end-val="goodsTotalNum" :duration="3600" class="card-panel-num" />                <div class="yesterdaybox">                  较昨日 <span class="highlight-today-add2">+{{todayGoodsNum}}</span>                </div>              </div>              <div class="card-badge">              </div>            </div>          </div>        </el-col>        <el-col :xs="24" :sm="24" :md="8" :lg="8" :xl="8" style="padding-left: 15px;">          <div class="internetbox">            <div class="internet-cardtop">              <div class="cardinnerbox">                <div class="cardtopimg">                  <img src="../assets/images/liuliang.png" alt=""><span>剩余流量</span>                </div>                <div class="cardtopnumber">                  <span>{{formatBytes(this.trafficCount)}}</span>                </div>              </div>              <div class="progress">                <el-progress :percentage="90" :show-text="false" define-back-color="#000">                </el-progress>              </div>              <div class="cardinnerbox2">                <div>                  今日消耗 <span>{{formatBytes(this.todayTraffic)}}</span>                </div>                <div>                  本月 <span>{{formatBytes(this.thisMonthTraffic)}}</span>                </div>              </div>            </div>            <div class="internetbox-messge">              <div class="internet-card">                <img src="../assets/images/message.png" alt="">                <span class="internet-title">                  短信剩余条数 (条)                </span>              </div>              <div class="internet-number">                0              </div>            </div>          </div>        </el-col>      </el-row>    </el-card>    <!-- 分析概览 (Analysis Overview) -->    <div class="analysis-section" shadow="never">      <div slot="header" class="header">        <div>分析概览</div>        <div class="tab-group">          <el-radio-group v-model="queryTime" size="medium" @change="handleAnalysis">            <el-radio-button label="今日"></el-radio-button>            <el-radio-button label="昨日"></el-radio-button>            <el-radio-button label="本周"></el-radio-button>            <el-radio-button label="本月"></el-radio-button>            <el-radio-button label="上月"></el-radio-button>          </el-radio-group>        </div>        <div class="action-group">          <div v-if="medicalMallConfig.statics">            <!-- 选择部门 -->            <el-select v-model="deptId" placeholder="请选择部门" size="small" @change="handleDeptChange" style="width: 150px">              <el-option                v-for="company in deptOptions"                :key="company.deptId"                :label="company.deptName"                :value="company.deptId"              />            </el-select>            <!-- 选择销售公司 -->            <el-select  v-model="companyId" placeholder="请选择销售公司" size="small" clearable @change="handleCompanyChange" style="width: 180px" >              <el-option                v-for="company in companyOptions"                :key="company.companyId"                :label="company.companyName"                :value="company.companyId"              />            </el-select>          </div>          <el-radio-group v-model="userTypeText" @change="handleUserType">            <el-radio-button label="会员"></el-radio-button>            <el-radio-button label="企微"></el-radio-button>          </el-radio-group>          <el-dropdown @command="handleAutoRefresh" trigger="click">            <el-button size="small" plain>              自动刷新              <i class="el-icon-arrow-down el-icon--right"></i>            </el-button>            <el-dropdown-menu slot="dropdown">              <el-dropdown-item :command="0" :class="{ 'is-active': !autoRefreshInterval }">关闭</el-dropdown-item>              <el-dropdown-item :command="5" :class="{ 'is-active': autoRefreshInterval === 5 }">5分钟</el-dropdown-item>              <el-dropdown-item :command="10"                                :class="{ 'is-active': autoRefreshInterval === 10 }">10分钟</el-dropdown-item>              <el-dropdown-item :command="15"                                :class="{ 'is-active': autoRefreshInterval === 15 }">15分钟</el-dropdown-item>            </el-dropdown-menu>          </el-dropdown>          <el-button size="small" plain icon="el-icon-refresh" type="primary" @click="manualRefresh">手动刷新</el-button>        </div>      </div>    </div>    <div>      <el-row :gutter="20">        <el-col :xs="24" :sm="24" :md="12" :lg="12" :xl="12" style="position: relative">          <div class="analysis-card-check" :class="selectedDiv===0?'analysis-card-check-selected color':''"               @click="handleToggleDiv(0)">            <div class="analysis-card">              <img class="card-icon" src="../assets/images/cishu_views.png"></img>              <div class="card-content">                <div class="card-row">                  <span>观看人数</span>                  <span class="highlight">                    <count-to :start-val="0" :end-val="watchUserCount" :duration="3600" class="card-panel-num" />                  </span>                </div>                <div class="card-row">                  <span>完播人数</span>                  <span class="highlight">                    <count-to :start-val="0" :end-val="completedUserCount" :duration="3600" class="card-panel-num" />                  </span>                </div>                <div class="card-row">                  <span>完播率</span>                  <span class="highlight">{{completedRate}}%</span>                </div>              </div>            </div>            <div class="analysis-card">              <img class="card-icon" src="../assets/images/number_views.png"></img>              <div class="card-content">                <div class="card-row">                  <span>观看次数</span>                  <span class="highlight-red">                    <count-to :start-val="0" :end-val="watchCount" :duration="3600" class="card-panel-num" /></span>                </div>                <div class="card-row">                  <span>完播次数</span>                  <span class="highlight-red">                    <count-to :start-val="0" :end-val="completedCount" :duration="3600" class="card-panel-num" />                  </span>                </div>                <div class="card-row">                  <span>视频完播率</span>                  <span class="highlight-red">{{watchRate}}%</span>                </div>              </div>            </div>          </div>        </el-col>        <el-col :xs="24" :sm="12" :md="6" :lg="6" :xl="6" style="position: relative">          <div class="analysis-card-check" :class="selectedDiv===1?'analysis-card-check-selected color':''"               @click="handleToggleDiv(1)">            <div class="analysis-card">              <img class="card-icon" src="../assets/images/renshu_views.png"></img>              <div class="card-content">                <div class="card-row">                  <span>答题人数</span>                  <span class="highlight-black">                    <count-to :start-val="0" :end-val="answerMemberCount" :duration="3600" class="card-panel-num" />                  </span>                </div>                <div class="card-row">                  <span>正确人数</span>                  <span class="highlight-black">                    <count-to :start-val="0" :end-val="correctUserCount" :duration="3600" class="card-panel-num" />                  </span>                </div>                <div class="card-row">                  <span>正确率</span>                  <span class="highlight-black">{{correctRate}}%</span>                </div>              </div>            </div>          </div>        </el-col>        <el-col :xs="24" :sm="12" :md="6" :lg="6" :xl="6"  style="position: relative">          <div class="analysis-card-check" :class="selectedDiv===2?'analysis-card-check-selected color':''"               @click="handleToggleDiv(2)">            <div class="analysis-card">              <img class="card-icon" src="../assets/images/hongbao_views.png"></img>              <div class="card-content">                <div class="card-row">                  <span>答题红包个数</span>                  <span class="highlight-black">                    <count-to :start-val="0" :end-val="rewardCount" :duration="3600" class="card-panel-num" />                  </span>                </div>                <div class="card-row">                  <span>答题红包金额(元)</span>                  <span class="highlight-black">                    <count-to :start-val="0" :end-val="rewardMoney" :duration="3600" class="card-panel-num" /></span>                </div>              </div>            </div>          </div>        </el-col>      </el-row>    </div>    <!-- 图表区域 (Charts Area) -->    <transition name="fade">      <el-row :gutter="20" class="charts-section" v-show="selectedDiv===0">        <el-col :span="12">          <el-card shadow="never">            <div slot="header" class="chart-header">              <span>会员观看、完播人数趋势图</span>              <div class="legend">                <div class="legend-item">                  <span class="dot viewer-dot"></span>                  <span>观看人数</span>                </div>                <div class="legend-item">                  <span class="dot complete-dot"></span>                  <span>完播人数</span>                </div>              </div>              <!--              <el-button size="small" plain class="view-more">平台每日统计 <i class="el-icon-arrow-right"></i></el-button>-->            </div>            <div ref="viewerChart" class="chart-container"></div>          </el-card>        </el-col>        <el-col :span="12">          <el-card shadow="never">            <div slot="header" class="chart-header">              <span>经销商会员观看TOP10</span>              <div class="legend">                <el-radio-group v-model="viewerType" size="small" @change="handleDealerChartData">                  <el-radio-button label="0">按观看人数</el-radio-button>                  <el-radio-button label="1">按完播人数</el-radio-button>                </el-radio-group>              </div>              <!--              <el-button size="small" plain class="view-more">经销商统计 <i class="el-icon-arrow-right"></i></el-button>-->            </div>            <div ref="dealerChart" class="chart-container"></div>          </el-card>        </el-col>      </el-row>    </transition>    <transition name="fade">      <el-row :gutter="20" class="charts-section" v-show="selectedDiv===1">        <el-card shadow="never">          <div slot="header" class="chart-header">            <span>课程观看TOP10</span>            <div class="legend">              <el-radio-group v-model="viewerType" size="small" @change="handleCourseWatchChart">                <el-radio-button label="0">按观看人数</el-radio-button>                <el-radio-button label="1">按完播人数</el-radio-button>                <el-radio-button label="2">按答题人数</el-radio-button>                <el-radio-button label="3">按正确人数</el-radio-button>              </el-radio-group>            </div>            <div class="legend">              <el-radio-group v-model="delerSort" @change="handleCourseWatchChart">                <el-radio label="DESC">前10名</el-radio>                <el-radio label="ASC">倒数10名</el-radio>              </el-radio-group>            </div>            <div class="legend">              <div class="legend-item">                <span class="dot viewer-dot"></span>                <span>观看人数</span>              </div>              <div class="legend-item">                <span class="dot complete-dot"></span>                <span>完播人数</span>              </div>              <div class="legend-item">                <span class="dot" style="background-color: #E6A23C"></span>                <span>答题人数</span>              </div>              <div class="legend-item">                <span class="dot" style="background-color: #F56C6C"></span>                <span>正确人数</span>              </div>            </div>            <!--            <el-button size="small" plain class="view-more">经销商统计 <i class="el-icon-arrow-right"></i></el-button>-->          </div>          <div ref="courseWatchChart" class="chart-container"></div>        </el-card>      </el-row>    </transition>    <transition name="fade">      <el-row :gutter="20" class="charts-section" v-show="selectedDiv===2">        <el-col :span="12">          <el-card shadow="never">            <div slot="header" class="chart-header">              <span>答题红包金额TOP10</span>              <div class="legend">                <el-radio-group v-model="dataType" size="small" @change="handleAnswerRedPackViewerChart">                  <el-radio-button label="0">按经销商排行</el-radio-button>                  <el-radio-button label="1">按课程排行</el-radio-button>                </el-radio-group>              </div>              <!--              <el-button size="small" plain class="view-more">红包记录 <i class="el-icon-arrow-right"></i></el-button>-->            </div>            <div ref="answerRedPackViewerChart" class="chart-container"></div>          </el-card>        </el-col>        <el-col :span="12">          <el-card shadow="never">            <div slot="header" class="chart-header">              <span>答题红包金额趋势图</span>              <div class="legend">                <div class="legend-item">                  <span class="dot viewer-dot"></span>                  <span>答题红包金额</span>                </div>              </div>              <!--              <el-button size="small" plain class="view-more">红包记录 <i class="el-icon-arrow-right"></i></el-button>-->            </div>            <div ref="answerRedPackMoneyViewerChart" class="chart-container"></div>          </el-card>        </el-col>      </el-row>    </transition>    <el-row :gutter="20" class="charts-section">      <el-col :span="12">        <el-card shadow="never">          <div slot="header" class="chart-header">            <span>本月订单数</span>            <div class="legend">              <div class="legend-item">                <span class="dot viewer-dot"></span>                <span>订单数</span>              </div>              <div class="legend-item">                <span class="dot complete-dot"></span>                <span>订单金额</span>              </div>            </div>          </div>          <div ref="viewerOrderChart" class="chart-container"></div>        </el-card>      </el-col>      <el-col :span="12">        <el-card shadow="never">          <div slot="header" class="chart-header">            <span>本月收款数</span>            <div class="legend">              <div class="legend-item">                <span class="dot viewer-dot"></span>                <span>收款数</span>              </div>              <div class="legend-item">                <span class="dot complete-dot"></span>                <span>收款金额</span>              </div>            </div>          </div>          <div ref="viewerReceiveChart" class="chart-container"></div>        </el-card>      </el-col>    </el-row>    <br/>  </div></template><script>import * as echarts from 'echarts'import CountTo from "vue-count-to";import {  analysisPreview,  authorizationInfo,  dealerAggregated, deaMemberTopTen, rechargeComsumption, rewardMoneyTopTen, rewardMoneyTrend,  smsBalance, thisMonthOrderCount, thisMonthRecvCount, trafficLog,  watchCourseTopTen, watchEndPlayTrend} from "@/api/statistics/statistics";import dayjs from 'dayjs';import { listDept } from '@/api/system/dept'import { listCompany } from '@/api/his/company'import { getConfigByKey } from '@/api/system/config'const viewCharOption = {  tooltip: {    trigger: 'axis',    axisPointer: {      type: 'shadow'    }  },  grid: {    left: '3%',    right: '4%',    bottom: '3%',    containLabel: true  },  xAxis: {    type: 'category',    data: ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '10', '11', '12', '13', '14', '15', '16', '17', '18', '19', '20', '21', '22', '23']  },  yAxis: {    type: 'value'  },  series: [    {      name: '观看人数',      type: 'bar',      data: [],      itemStyle: {        color: '#409EFF'      }    },    {      name: '完播人数',      type: 'bar',      data: [],      itemStyle: {        color: '#67C23A'      }    }  ]}const thisMonthOrderCountOption = {  tooltip: {    trigger: 'axis',    axisPointer: {      type: 'shadow'    }  },  grid: {    left: '3%',    right: '4%',    bottom: '3%',    containLabel: true  },  xAxis: {    type: 'category',    data: ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '10', '11', '12', '13', '14', '15', '16', '17', '18', '19', '20', '21', '22', '23']  },  yAxis: {    type: 'value'  },  series: [    {      name: '订单数',      type: 'line',      data: [],      itemStyle: {        color: '#409EFF'      }    },    {      name: '订单金额',      type: 'line',      data: [],      itemStyle: {        color: '#67C23A'      }    }  ]}const thisMonthRecvCountOption = {  tooltip: {    trigger: 'axis',    axisPointer: {      type: 'shadow'    }  },  grid: {    left: '3%',    right: '4%',    bottom: '3%',    containLabel: true  },  xAxis: {    type: 'category',    data: ['0', '1', '2', '3', '4', '5', '6', '7', '8', '9', '10', '11', '12', '13', '14', '15', '16', '17', '18', '19', '20', '21', '22', '23']  },  yAxis: {    type: 'value'  },  series: [    {      name: '收款数',      type: 'line',      data: [],      itemStyle: {        color: '#409EFF'      }    },    {      name: '收款金额',      type: 'line',      data: [],      itemStyle: {        color: '#67C23A'      }    }  ]}const dealerOption = {  tooltip: {    trigger: 'axis',    axisPointer: {      type: 'shadow'    }  },  grid: {    left: '3%',    right: '4%',    bottom: '3%',    containLabel: true  },  xAxis: {    type: 'value'  },  yAxis: {    type: 'category',    data: []  },  series: [    {      name: '观看人数',      type: 'bar',      data: [],      itemStyle: {        color: '#409EFF'      }    }  ]}const courseWatchOption = {  tooltip: {    trigger: 'axis',    axisPointer: {      type: 'shadow'    }  },  grid: {    left: '3%',    right: '4%',    bottom: '8%',    top: '3%',    containLabel: true  },  xAxis: {    type: 'category',    data: [],    axisLabel: {      interval: 0,      rotate: 30,      fontSize: 10,      width: 100,      overflow: 'truncate'    }  },  yAxis: {    type: 'value',    splitLine: {      lineStyle: {        type: 'dashed'      }    }  },  series: [    {      name: '观看人数',      type: 'bar',      data: [],      itemStyle: {        color: '#409EFF'      }    },    {      name: '完播人数',      type: 'bar',      data: [],      itemStyle: {        color: '#67C23A'      }    },    {      name: '答题人数',      type: 'bar',      data: [],      itemStyle: {        color: '#E6A23C'      }    },    {      name: '正确人数',      type: 'bar',      data: [],      itemStyle: {        color: '#F56C6C'      }    }  ]}const lineChartOption = {  tooltip: {    trigger: 'axis',    axisPointer: {      type: 'cross' // 改为 'cross' 更适合折线图    }  },  grid: {    left: '3%',    right: '4%',    bottom: '8%', // 如果x轴标签旋转,可能需要更大的 bottom    top: '5%',    // 增加一点顶部空间给可能的 Y 轴名称    containLabel: true  },  xAxis: {    type: 'time', // X轴类型改为 'time'    // data: [], // 时间轴不需要单独设置 data,数据在 series 中提供    axisLabel: {      // interval: 0, // 时间轴通常自动处理间隔,可以先移除或注释掉      rotate: 30,   // 保留旋转,如果标签可能重叠      fontSize: 10,      // width: 100, // width 和 overflow 对于时间轴可能行为不同,按需调整      // overflow: 'truncate',      formatter: null // ECharts 会自动格式化时间,如需特定格式可用 function 或字符串模板    }  },  yAxis: {    type: 'value',    name: '金额 (元)', // 添加 Y 轴名称    nameLocation: 'end', // 名称位置    nameTextStyle: {      align: 'right',      padding: [0, 10, 0, 0] // 调整名称与轴线的距离    },    splitLine: {      lineStyle: {        type: 'dashed'      }    },    axisLabel: {      formatter: '{value} 元' // 可选:给 Y 轴刻度添加单位    }  },  series: [    {      name: '答题红包金额',      type: 'line', // 系列类型改为 'line'      data: [      ],      itemStyle: { // 控制数据点(标记)的样式        color: '#409EFF'      },      lineStyle: { // 控制线的样式        color: '#409EFF'      },      smooth: false, // 是否平滑曲线,可设为 true      symbol: 'circle', // 数据点标记形状,'emptyCircle', 'rect', 'roundRect', 'triangle', 'diamond', 'pin', 'arrow', 'none'      symbolSize: 4   // 数据点标记大小    }  ]};const redPackageOption = {  tooltip: {    trigger: 'axis',    axisPointer: {      type: 'shadow'    }  },  grid: {    left: '3%',    right: '4%',    bottom: '8%',    top: '3%',    containLabel: true  },  xAxis: {    type: 'category',    data: [],    axisLabel: {      interval: 0,      rotate: 30,      fontSize: 10,      width: 100,      overflow: 'truncate'    }  },  yAxis: {    type: 'value',    splitLine: {      lineStyle: {        type: 'dashed'      }    }  },  series: [    {      name: '答题红包金额',      type: 'bar',      data: [],      itemStyle: {        color: '#409EFF'      }    }  ]}export default {  name: 'StatisticsDashboard',  components: {CountTo},  data() {    return {      medicalMallConfig:{},      deptInitOptions:[],      deptOptions:[],      intiDeptId:this.$store.state.user.user.deptId,      deptId:this.$store.state.user.user.deptId,      staticParam : {companyId:null,deptId:this.$store.state.user.user.deptId},      companyIntiOptions:[],      companyOptions:[],      companyId:null,      percentage: 0,      // 预测message      remainMessage: '',      // 当天使用流量      todayTraffic: 0,      trafficCount: 0,      // 当月使用流量      thisMonthTraffic: 0,      dataType: '0',      delerSort: 'DESC',      smsRemainCount: 0,      viewerType: '0',      viewerChart: null,      userTypeText: process.env.VUE_APP_COURSE_DEFAULT==1?"会员":"企微",      userType: process.env.VUE_APP_COURSE_DEFAULT,      dealerChart: null,      // 分公司数量      dealderCount: 0,      // 销售数量      groupMgrCount: 0,      // 会员总数量      memberCount: 0,      // 企微数量      qwMemberNum: 0,      // pad使用情况      padTotalNum: 0,      // pad使用情况      padUsedNum: 0,      // 正常会员数量      normalNum: 0,      // 黑名单会员数量      blackNum: 0,      // 观看人数      watchUserCount: 0,      // 完播人数      completedUserCount: 0,      // 完播率      completedRate: 0,      // 观看次数      watchCount:0,      // 完播次数      completedCount: 0,      // 视频完播率      watchRate: 0,      // 答题人数      answerMemberCount: 0,      // 正确人数      correctUserCount: 0,      correctRate: 0.0,      // 答题红包个数      rewardCount: 0,      // 答题红包金额      rewardMoney: 0.0,      queryTime: '今日',      todayWatchUserCount: 0,      versionLimit: 0,      versionLimitPercent : 0.0,      /// 选中的分析概览      selectedDiv: 0,      filterType: 0,      answerRedPackViewerChart: null,      answerRedPackMoneyViewerChart: null,      todayComsumption: 0,      yesterdayComsumption: 0,      balance: 0,      autoRefreshInterval: null,      // 今日新增用户数      todayIncreaseUserNum: 0,      // 订单总数      orderTotalNum: 0,      // 今日新增订单数      todayOrderNum: 0,      // 收款总数      recvTotalNum: 0,      // 今日收款总数      recvTodayNum: 0,      // 商品总数      goodsTotalNum: 0,      // 今日商品总数      todayGoodsNum: 0    }  },  mounted() {    this.$nextTick(() => {      this.initViewerChart()      this.initDealerChart()      this.initCourseWatchChart();      this.initAnswerRedPackViewerChart();      this.initAnswerRedPackMoneyViewerChart();      this.initThisMonthOrderChart();      this.initThisMonthRecvChart()      // 监听窗口大小变化,重新渲染图表      window.addEventListener('resize', () => {        this.viewerChart && this.viewerChart.resize()        this.dealerChart && this.dealerChart.resize()      })    })  },  created() {    this.refresh();    listDept().then(res => {      this.deptInitOptions = res.data;      listCompany().then(res => {        this.companyIntiOptions = res.rows;        this.getDeptOptions(this.intiDeptId);        this.getCompanyOptions(this.intiDeptId);      });    });    getConfigByKey("medicalMall.func.switch").then(response => {      if(response.data && response.data.configValue) {        this.medicalMallConfig = JSON.parse(response.data.configValue);        console.log(this.medicalMallConfig)      }    });  },  methods: {    getDeptOptions(deptId) {      const deptInitOptions = this.deptInitOptions;      // 部门本身节点      let deptNode = deptInitOptions.filter(item => item.deptId === deptId);      // 递归查找所有子节点      function findChildren(parentId) {        //部门的子部门        let deptChildren = deptInitOptions.filter(item => item.parentId === parentId);        //添加子部门        deptChildren.forEach(child => {          deptNode.push(child);          findChildren(child.deptId); // 递归查找子节点的子节点        });      }      // 从目标节点开始查找子节点      findChildren(deptId);      this.deptOptions = deptNode;    },    getCompanyOptions(deptId) {      this.companyId = null;      //修改选择后清空查询参数      this.staticParam.companyId = null;      this.staticParam.deptId = deptId;      const deptInitOptions = this.deptInitOptions;      const companyInitOptions = this.companyIntiOptions;      // 部门下的公司      let companyNode = companyInitOptions.filter(item => item.deptId === deptId);      // 递归查找所有子节点      function findChildren(parentId) {        //部门的子部门        let deptChildren = deptInitOptions.filter(item => item.parentId === parentId);        //添加子部门        deptChildren.forEach(child => {          //子部门下的销售公司          let companyChildren = companyInitOptions.filter(item => item.deptId === child.deptId);          companyChildren.forEach(companyChild => {            companyNode.push(companyChild);          })          findChildren(child.deptId); // 递归查找子节点的子节点        });      }      // 从目标节点开始查找子节点      findChildren(deptId);      this.companyOptions = companyNode;    },    //首页统计选择部门、销售公司    handleDeptChange() {      this.getCompanyOptions(this.deptId);      this.refresh();    },    handleCompanyChange() {      this.staticParam.companyId = this.companyId;      this.refresh();    },    handleUserType() {      if (this.userTypeText === '会员') {        this.userType = 1      } else {        this.userType = 2      }      this.refresh()    },    /**     * 计算余额预计可持续的天数     * @param {number} balance - 当前账户余额     * @param {number} todayConsumption - 今日消耗金额     * @param {number} yesterdayConsumption - 昨日消耗金额     * @return {Object} 包含天数和进度百分比的对象     */    calculateRemainingDays(balance, todayConsumption, yesterdayConsumption) {      // 如果今日和昨日消耗都为0,则无法预测(避免除以0)      if (todayConsumption === 0 && yesterdayConsumption === 0) {        return {          days: Infinity,          percentage: 0,          message: '暂无消耗数据'        };      }      // 计算每日平均消耗量      const avgDailyConsumption = (todayConsumption + yesterdayConsumption) / 2;      // 如果平均消耗为0,则无法预测      if (avgDailyConsumption === 0) {        return {          days: Infinity,          percentage: 0,          message: '暂无消耗数据'        };      }      // 计算剩余天数(向下取整)      const remainingDays = Math.floor(balance / avgDailyConsumption);      // 计算进度条百分比,最大为100      // 这里假设100天是满值,可以根据需要调整      const maxDays = 100;      const percentage = Math.min(100, Math.max(0, Math.round((remainingDays / maxDays) * 100)));      let message = '';      if (remainingDays > 365) {        message = '预测余额充足';      } else {        message = `预测不足${remainingDays}天`;      }      return {        days: remainingDays,        percentage: 100 - percentage,        message: message      };    },    /**     * 将字节数转换为合适的单位表示(Byte、KB、MB、GB、TB)     * @param {number} bytes - 字节数     * @param {number} [decimals=2] - 小数点后保留的位数     * @returns {string} 格式化后的字符串,包含数值和单位     */    formatBytes(bytes, decimals = 2) {      const isNegative = bytes < 0;  // 判断是否为负数      bytes = Math.abs(bytes);  // 获取绝对值      if (bytes === 0) return '0 Byte';      const k = 1024;      const sizes = ['Byte', 'KB', 'MB', 'GB', 'TB'];      // 计算合适的单位级别      let i = Math.floor(Math.log(bytes) / Math.log(k));      // 转换为对应单位的值      const value = bytes / Math.pow(k, i);      if(this.deptId !== 1 ||  this.companyId !== null){        i += 1;      }      // 格式化为指定小数位的字符串      const result = parseFloat(value.toFixed(decimals)) + ' ' + sizes[Math.min(i, sizes.length - 1)];      // 如果是负数,返回带负号的值      return isNegative ? `-${result}` : result;    },    // 手动刷新    manualRefresh() {      this.refresh();    },    // 处理自动刷新选项    handleAutoRefresh(command) {      // 清除之前的定时器      if (this.timer) {        clearInterval(this.timer);        this.timer = null;      }      // 设置新的刷新间隔      this.autoRefreshInterval = parseInt(command);      // 如果间隔大于0,设置新的定时器      if (this.autoRefreshInterval > 0) {        this.timer = setInterval(() => {          this.refresh();        }, this.autoRefreshInterval * 60 * 1000); // 转换为毫秒        this.$message.success(`已设置${this.autoRefreshInterval}分钟自动刷新`);      } else {        this.$message.info('已关闭自动刷新');      }    },    refresh() {      rechargeComsumption(this.staticParam).then(res => {        if (res.code === 200) {          this.balance = res.data.balance;          this.todayComsumption = res.data.todayComsumption;          this.yesterdayComsumption = res.data.yesterdayComsumption;          let calculateRemainingDays1 = this.calculateRemainingDays(this.balance, this.todayComsumption, this.yesterdayComsumption);          this.percentage = calculateRemainingDays1.percentage;          this.remainMessage = calculateRemainingDays1.message;        }      });      trafficLog(this.staticParam).then(res => {        if (res.code === 200) {          this.todayTraffic = res.data.today;          this.thisMonthTraffic = res.data.thisMonth;          this.trafficCount = res.data.traffic;        }      })      dealerAggregated(this.staticParam).then(res => {        if (res.code === 200) {          this.dealderCount = res.data.dealderCount ?? 0;          this.groupMgrCount = res.data.groupMgrCount ?? 0;          this.memberCount = res.data.memberCount ?? 0;          this.qwMemberNum = res.data.qwMemberNum ?? 0;          const totalNum = res.data.padTotalNum;          //-1 不限 null 未设置          if(totalNum != null && totalNum !== -1 && totalNum > 0){            this.padTotalNum = totalNum;          }else{            this.padTotalNum = '不限';          }          this.padUsedNum = res.data.padUsedNum ?? 0;          this.padInfo = res.data.padInfo;          this.normalNum = res.data.normalNum ?? 0;          this.blackNum = res.data.blackNum ?? 0;          this.todayIncreaseUserNum = res.data.todayIncreaseUserNum ?? 0;          this.orderTotalNum = res.data.orderTotalNum ?? 0;          this.todayOrderNum = res.data.todayOrderNum ?? 0;          this.recvTotalNum = res.data.recvTotalNum ?? 0;          this.recvTodayNum = res.data.recvTodayNum ?? 0;          this.goodsTotalNum = res.data.goodsTotalNum ?? 0;          this.todayGoodsNum = res.data.todayGoodsNum ?? 0;        }      })      let param = this.getParam();      // 获取当前日期时间      const today = dayjs();      param.startTime = this.formatDate(today);      param.endTime = this.formatDate(today);      analysisPreview(param).then(res => {        if (res.code === 200) {          this.watchUserCount = res.data.watchUserCount;          this.completedUserCount = res.data.completedUserCount;          this.completedRate = res.data.completedRate;          this.watchCount = res.data.watchCount;          this.completedCount = res.data.completedCount;          this.answerMemberCount = res.data.answerMemberCount;          this.correctUserCount = res.data.correctUserCount;          this.correctRate = res.data.correctRate;          this.rewardCount = res.data.rewardCount;          this.rewardMoney = res.data.rewardMoney;          this.watchRate = res.data.watchRate;        }      })      smsBalance(this.staticParam).then(res => {        if (res.code === 200) {          if (res.data == null) {            this.smsRemainCount = 0;          } else {            this.smsRemainCount = res.data;          }        }      })      authorizationInfo(this.staticParam).then(res => {        if (res.code === 200 && res.data != null) {          this.todayWatchUserCount = res.data.todayWatchUserCount;          this.versionLimit = res.data.versionLimit;          if(this.versionLimit){            this.versionLimitPercent = this.todayWatchUserCount/this.versionLimit;          }        }      })      this.handleCourseWatchChart()      this.handleViewChartData()      // 经销商会员观看TOP10      this.handleDealerChartData()      this.handleAnswerRedPackViewerChart()      this.handleAnswerRedPackMoneyViewerChart()      this.handleThisMonthRecvCount();      this.handleThisMonthOrderCount();    },    /**     * 将数字添加千位分隔符     * @param {number|string} num - 需要格式化的数字     * @return {string} 添加千位分隔符后的字符串     */    formatNumberWithCommas(num) {      if (num === null || num === undefined || isNaN(Number(num))) {        return '0';      }      const numStr = String(num);      // 处理负数      const isNegative = numStr.startsWith('-');      const absNumStr = isNegative ? numStr.slice(1) : numStr;      // 分离整数部分和小数部分      const parts = absNumStr.split('.');      const integerPart = parts[0];      const decimalPart = parts.length > 1 ? '.' + parts[1] : '';      const formattedInteger = integerPart.replace(/\B(?=(\d{3})+(?!\d))/g, ',');      return (isNegative ? '-' : '') + formattedInteger + decimalPart;    },    handleToggleDiv(selected) {      this.selectedDiv = selected;      if (selected === 1) {        this.$nextTick(() => {          if (this.courseWatchChart) {            this.courseWatchChart.resize();          } else {          }        });      } else if (selected === 0) {        this.$nextTick(() => {          if (this.viewerChart) this.viewerChart.resize();          if (this.dealerChart) this.dealerChart.resize();        });      } else if (selected === 2) {        this.$nextTick(() => {          if (this.answerRedPackViewerChart) this.answerRedPackViewerChart.resize();          if (this.answerRedPackMoneyViewerChart) this.answerRedPackMoneyViewerChart.resize();        });      }      if (this.selectedDiv === 0) {        this.handleViewChartData()        this.handleDealerChartData()      } else if (this.selectedDiv === 1) {        this.handleCourseWatchChart()      } else if (this.selectedDiv === 2) {        this.handleAnswerRedPackViewerChart()        this.handleAnswerRedPackMoneyViewerChart()      }    },    formatDate(date) {      return dayjs(date).format('YYYY-MM-DD');    },    getParam() {      let param = {        startTime: '',        endTime: '',        userType: this.userType,        companyId: this.companyId,        deptId: this.deptId      };      // 获取当前日期时间      const today = dayjs();      let type = 0;      if (this.queryTime === '今日') {        param.startTime = this.formatDate(today);        param.endTime = this.formatDate(today);        type = 0;      } else if (this.queryTime === '昨日') {        const yesterday = today.subtract(1, 'day');        param.startTime = this.formatDate(yesterday);        param.endTime = this.formatDate(yesterday);        type = 1;      } else if (this.queryTime === '本周') {        param.startTime = this.formatDate(today.startOf('week'));        param.endTime = this.formatDate(today.endOf('week'));        type = 2;      } else if (this.queryTime === '本月') {        param.startTime = this.formatDate(today.startOf('month'));        param.endTime = this.formatDate(today.endOf('month'));        type = 3;      } else if (this.queryTime === '上月') {        const lastMonth = today.subtract(1, 'month');        param.startTime = this.formatDate(lastMonth.startOf('month'));        param.endTime = this.formatDate(lastMonth.endOf('month'));        type = 4;      } else {        console.warn(`未知的 queryTime: ${this.queryTime}, 默认使用今日`);        param.startTime = this.formatDate(today);        param.endTime = this.formatDate(today);      }      param.type = type;      param.sort = this.delerSort;      return param;    },    // 分析概览    handleAnalysis(e) {      let param = this.getParam();      analysisPreview(param).then(res => {        if (res.code === 200) {          this.watchUserCount = res.data.watchUserCount;          this.completedUserCount = res.data.completedUserCount;          this.completedRate = res.data.completedRate;          this.watchCount = res.data.watchCount;          this.completedCount = res.data.completedCount;          this.answerMemberCount = res.data.answerMemberCount;          this.correctUserCount = res.data.correctUserCount;          this.correctRate = res.data.correctRate;          this.rewardCount = res.data.rewardCount;          this.rewardMoney = res.data.rewardMoney;          this.watchRate = res.data.watchRate;        }      })      if (this.selectedDiv === 0) {        this.handleViewChartData()        this.handleDealerChartData()      } else if (this.selectedDiv === 1) {        this.handleCourseWatchChart()      } else if (this.selectedDiv === 2) {        this.handleAnswerRedPackViewerChart()        this.handleAnswerRedPackMoneyViewerChart()      }    },    handleAnswerRedPackViewerChart() {      let param = this.getParam();      param = { ...param, statisticalType: this.viewerType, dataType: this.dataType };      rewardMoneyTopTen(param).then(res => {        if (res.code === 200) {          let data = res.data;          let companyNameList = data.map(e => e.companyName)          let courseNameList = data.map(e => e.courseName)          let rewardMoneyList = data.map(e => e.rewardMoney)          if (this.dataType === '0') {            redPackageOption.xAxis.data = companyNameList;          } else {            redPackageOption.xAxis.data = courseNameList;          }          redPackageOption.series[0].data = rewardMoneyList;          this.answerRedPackViewerChart.setOption(redPackageOption)        }      })    },    handleAnswerRedPackMoneyViewerChart() {      let param = this.getParam();      param = { ...param, statisticalType: this.viewerType, dataType: this.dataType };      rewardMoneyTrend(param).then(res => {        if (res.code === 200) {          let data = res.data;          let option = data.map(e => [e.x, e.rewardMoney])          lineChartOption.series[0].data = option;          this.answerRedPackMoneyViewerChart.setOption(lineChartOption)        }      })    },    handleCourseWatchChart() {      let param = this.getParam();      param = { ...param, statisticalType: this.viewerType };      watchCourseTopTen(param).then(res => {        if (res.code === 200) {          let data = res.data;          let watchUserCountList = data.map(e => e.watchUserCount);          let completedUserCountList = data.map(e => e.completedUserCount);          let answerUserCountList = data.map(e => e.answerUserCount);          let correctUserCountList = data.map(e => e.correctUserCount);          let courseNameList = data.map(e => e.courseName);          courseWatchOption.xAxis.data = courseNameList;          courseWatchOption.series[0].data = watchUserCountList;          courseWatchOption.series[1].data = completedUserCountList;          courseWatchOption.series[2].data = answerUserCountList;          courseWatchOption.series[3].data = correctUserCountList;          this.courseWatchChart.setOption(courseWatchOption)        }      })    },    handleDealerChartData() {      let param = this.getParam();      // 经销商会员观看TOP10      deaMemberTopTen({ ...param, statisticalType: this.viewerType }).then(res => {        if (res.code === 200) {          let data = res.data;          let companyNameList = data.map(e => e.companyName);          let watchUserList = data.map(e => e.watchUserCount);          dealerOption.yAxis.data = companyNameList;          dealerOption.series[0].data = watchUserList;          this.dealerChart.setOption(dealerOption)        }      })    },    handleThisMonthOrderCount() {      thisMonthOrderCount(this.staticParam).then(res => {        if (res.code === 200) {          let dates = res.dates;          let orderCount = res.orderCount;          let payPrice = res.payPrice;          thisMonthOrderCountOption.series[0].data = orderCount;          thisMonthOrderCountOption.series[1].data = payPrice;          thisMonthOrderCountOption.xAxis.data = dates;          this.thisMonthOrderChart.setOption(thisMonthOrderCountOption)        }      })    },    handleThisMonthRecvCount() {      thisMonthRecvCount(this.staticParam).then(res => {        if (res.code === 200) {          let dates = res.dates;          let orderCount = res.orderCount;          let payMoney = res.payMoney;          thisMonthRecvCountOption.series[0].data = orderCount;          thisMonthRecvCountOption.series[1].data = payMoney;          thisMonthRecvCountOption.xAxis.data = dates;          this.thisMonthRecvChart.setOption(thisMonthRecvCountOption)        }      })    },    handleViewChartData() {      let param = this.getParam();      watchEndPlayTrend({ ...param }).then(res => {        if (res.code === 200) {          let data = res.data;          let watchUserCountList = data.map(e => e.watchUserCount);          let completedUserCountList = data.map(e => e.completedUserCount);          let xAxis = data.map(e => e.x);          viewCharOption.series[0].data = watchUserCountList;          viewCharOption.series[1].data = completedUserCountList;          viewCharOption.xAxis.data = xAxis;          this.viewerChart.setOption(viewCharOption);        }      })    },    initThisMonthOrderChart() {      this.thisMonthOrderChart = echarts.init(this.$refs.viewerOrderChart)      this.thisMonthOrderChart.setOption(thisMonthOrderCountOption)    },    initThisMonthRecvChart() {      this.thisMonthRecvChart = echarts.init(this.$refs.viewerReceiveChart)      this.thisMonthRecvChart.setOption(thisMonthOrderCountOption)    },    initViewerChart() {      this.viewerChart = echarts.init(this.$refs.viewerChart)      this.viewerChart.setOption(viewCharOption)    },    initDealerChart() {      this.dealerChart = echarts.init(this.$refs.dealerChart)      this.dealerChart.setOption(dealerOption)    },    initCourseWatchChart() {      this.courseWatchChart = echarts.init(this.$refs.courseWatchChart)      this.courseWatchChart.setOption(courseWatchOption)    },    initAnswerRedPackViewerChart() {      this.answerRedPackViewerChart = echarts.init(this.$refs.answerRedPackViewerChart)      this.answerRedPackViewerChart.setOption(redPackageOption)    },    initAnswerRedPackMoneyViewerChart() {      this.answerRedPackMoneyViewerChart = echarts.init(this.$refs.answerRedPackMoneyViewerChart)      this.answerRedPackMoneyViewerChart.setOption(lineChartOption)    }  },  beforeDestroy() {    // 组件销毁时清除定时器    if (this.timer) {      clearInterval(this.timer);      this.timer = null;    }    // window.removeEventListener('resize', this.resizeHandler)    this.viewerChart && this.viewerChart.dispose()    this.dealerChart && this.dealerChart.dispose()  }}</script><style scoped>.highlight-today-add {  color: green;  font-size: 17px;  font-weight: normal;}.highlight-today-add2 {  color: rgba(49, 185, 154, 1);  font-size: 14px;  font-weight: normal;  transform: scale(.9);}.action-group .el-button + .el-button,.action-group .el-dropdown {  margin-left: 10px;}.is-active {  color: #409EFF;  font-weight: bold;}::v-deep .el-radio-button__inner:hover {  color: #409EFF;  /* 鼠标悬浮时的文字颜色,可以根据需要调整 */}::v-deep .el-radio-button.is-active .el-radio-button__inner {  background-color: #409EFF;  /* 选中时的背景色 */  border-color: #409EFF;  /* 选中时的边框色 */  color: #FFFFFF;  /* 选中时的文字颜色 (通常是白色) */  box-shadow: -1px 0 0 0 #409EFF;  /* 处理按钮间的连接缝隙 */}/* 如果需要,也可以修改非选中状态下的聚焦(focus)或悬浮(hover)样式 *//* 例如,让非选中按钮悬浮时边框和文字也变蓝 */::v-deep .el-radio-button:not(.is-active) .el-radio-button__inner:hover {  color: #409EFF;  /* border-color: #b3d8ff;  Element UI 默认悬浮边框色,可以按需修改 */}/* 聚焦时的外框,如果需要的话 */::v-deep .el-radio-button:focus:not(.is-checked) .el-radio-button__inner {  /* border-color: #409EFF; */  /* Element UI 默认的 focus 颜色通常关联主题色 */  /* box-shadow: 0 0 2px 2px rgba(64, 158, 255, 0.2); */  /* 示例 focus 光晕 */}.statistics-dashboard {  padding: 20px;  background-color: #f5f7fa;}.overview-section,.analysis-section {  margin-bottom: 20px;  border-radius: 4px;}.header {  display: flex;  justify-content: space-between;  align-items: center;  font-size: 16px;  font-weight: 500;}.data-card {  background-color: #fff;  border-radius: 4px;  padding: 15px;  height: 120px;  display: flex;  flex-direction: column;  position: relative;  transition: background-color 0.3s ease-in-out;}.data-card:hover {  border: 1px solid #4592ff;  background-color: #e7f1ff;}.card-title {  color: #606266;  font-size: 14px;  margin-bottom: 10px;}.card-title1 {  color: white;  font-size: 14px;  margin-bottom: 10px;}.card-value {  font-size: 24px;  font-weight: bold;  margin-top: 20px;}.highlight {  color: #409EFF;}.card-sub {  display: flex;  justify-content: space-between;  font-size: 12px;  color: #909399;  margin-top: 5px;}.card-desc {  font-size: 12px;  color: #909399;  margin-top: 5px;}.card-badge {  position: absolute;  top: 15px;  right: 15px;  background: #f0f9eb;  color: #67c23a;  padding: 2px 5px;  border-radius: 4px;}.cdn-label {  background-color: #409EFF;  color: white;  padding: 2px 5px;  border-radius: 4px;  margin-right: 5px;  font-size: 12px;}.tab-group {  display: flex;  gap: 10px;}.action-group {  display: flex;  gap: 10px;}.analysis-card {  border-radius: 4px;  padding: 20px;  display: flex;  align-items: center;}.card-icon {  width: 50px;  height: 50px;  background-color: rgba(64, 158, 255, 0.1);  border-radius: 8px;  display: flex;  justify-content: center;  align-items: center;  font-size: 24px;  color: #409EFF;  margin-right: 20px;}.card-content {  display: flex;}.card-row {  display: flex;  justify-content: center;  justify-items: center;  flex-direction: column;  padding: 10px;  .highlight {    text-align: center;    margin-top: 1em;    font-family: BebasNeue;    color: #1677ff;    font-size: 21px;    line-height: 42px;    font-weight: 600;    margin-top: 8px;  }  font-size: 15px;  color: #000;}.charts-section {  margin-top: 20px;}.chart-header {  display: flex;  justify-content: space-between;  align-items: center;}.view-more {  font-size: 12px;}.legend {  display: flex;  gap: 15px;}.legend-item {  display: flex;  align-items: center;  font-size: 12px;}.dot {  width: 10px;  height: 10px;  border-radius: 50%;  margin-right: 5px;}.viewer-dot {  background-color: #409EFF;}.complete-dot {  background-color: #67C23A;}.chart-container {  height: 350px;  width: 100%;}.analysis-card-check {  display: flex;  flex-direction: row;  border: 1px solid transparent;  background-color: #fff;  border-radius: 4px;}.analysis-card-check:hover {  cursor: pointer;}.analysis-card-check-selected:after {  content: "";  display: block;  border-width: 15px;  position: absolute;  bottom: -30px;  left: 50%;  margin-left: -32px;  border-style: solid dashed dashed solid;  border-color: #4592FF transparent transparent transparent;  font-size: 0;  line-height: 0;  z-index: 1;}.analysis-card-check-selected:before {  content: "";  display: block;  border-width: 15px;  position: absolute;  bottom: -30px;  left: 50%;  margin-left: -32px;  border-style: solid dashed dashed solid;  border-color: #4592FF transparent transparent transparent;  font-size: 0;  line-height: 0;  z-index: 1;}.analysis-card-check-selected {  border: 1px solid #4592FF;  background-color: #e7f1ff;}.color {  position: relative;  border: 1px solid #4592FF;  background-color: #e7f1ff;}.color:after {  bottom: -27px;  border-color: #E7F1FF transparent transparent transparent;}.companybox {  color: white;  background-color: #006CFF;  padding: 10px 10px 40px 10px;  box-sizing: border-box;  position: relative;  border-radius: 6px;  .topimg {    width: 100px;    height: 80px;    position: absolute;    top: 0px;    left: 0;  }  .bottomimg {    width: 100px;    height: 80px;    position: absolute;    bottom: -10px;    right: 0;    transform: rotate(180deg);  }  .companyboxtitle {    height: 30px;    margin-bottom: 20px;    font-weight: 600;  }  .companyflex {    display: flex;    justify-content: space-around;  }}.companynumber {  color: white;}.companycard {  width: 25%;}.companyadd {  color: white;}.cardafter {  position: relative;}.cardafter::after {  content: "";  height: 60px;  border-right: 1px solid rgba(255, 255, 255, 0.20);  position: absolute;  right: 30px;  top: 0px;}.highlight1 {  margin-top: 10px;  margin-left: 20px;}.propertyboxtitle {  background-image: url(../assets/images/zcgl_bg.png);  height: 100px;}.propertyboxflex {  display: flex;  background-color: white;  justify-content: space-between;}.property-card {  width: 48%;  border-radius: 4px;  padding: 15px;  height: 100px;  display: flex;  flex-direction: column;  position: relative;  transition: background-color 0.3s ease-in-out;}.property_title {  height: 40px;  line-height: 40px;  color: rgba(32, 33, 36, 1);  box-sizing: border-box;  padding-left: 10px;  font-weight: 600;}.card-compare {  margin-top: 5px;  font-size: 14px;  color: rgba(92, 95, 102, 1);  span {    font-size: 13px;    transform: scale(.8);    color: rgba(49, 185, 154, 1);  }}.propertyline {  position: relative;}.propertyline::after {  position: absolute;  content: "";  height: 80px;  border-right: 1px solid rgba(237, 239, 242, 1);  right: 0;}.operatetitle {  font-weight: 600;  height: 50px;  line-height: 50px;}.operatetitle-col {  display: flex;  justify-content: space-between;  padding-bottom: 20px;  padding-right: 5px;  .operatetitle-card {    width: 24%;    border: 1px solid transparent;    background-color: rgba(245, 247, 250, 1);    border-radius: 4px;    padding: 15px;    display: flex;    flex-direction: column;    position: relative;    transition: background-color 0.3s ease-in-out;    .operate-value {      font-size: 24px;      font-weight: bold;    }  }}.yesterdaybox {  font-size: 14px;  color: rgba(92, 95, 102, 1);  font-weight: 200;  margin-top: 10px;}.internetbox {  background: linear-gradient(to right, rgba(255, 99, 0, 1), rgba(255, 159, 1, 1));  border-radius: 6px;  padding: 2px 5px 15px 5px;  width: 100%;  .internet-cardtop {    background-color: white;    border-radius: 3px;    margin-top: 5px;    justify-content: space-between;    padding: 15px;    box-sizing: border-box;    color: rgba(32, 33, 36, 1);    .cardinnerbox {      display: flex;      justify-content: space-between;      .cardtopimg {        display: flex;        align-items: center;        font-size: 15px;        img {          height: 18px;          width: 18px;        }      }      .cardtopnumber {        font-size: 25px;        color: rgba(32, 33, 36, 1);        font-weight: bold;      }    }    .cardinnerbox2 {      display: flex;      justify-content: space-between;      font-size: 14px;      color: rgba(92, 95, 102, 1);    }  }  .internetbox-messge {    color: white;    display: flex;    justify-content: space-between;    padding: 15px 10px 10px 10px;    .internet-card {      display: flex;      font-size: 14px;      align-items: center;      img {        height: 18px;        width: 18px;      }      span {        padding-left: 5px;      }    }    .internet-number {      font-size: 25px;    }  }  .internetbox {    display: flex;    color: white;    .internet-card {      width: 100%;      display: flex;      color: white;    }  }  .internet-title {    font-size: 14px;  }}.company {  background-color: #006CFF;}.highlight-red {  text-align: center;  margin-top: 1em;  font-family: BebasNeue;  color: rgba(255, 82, 82, 1);  font-size: 21px;  line-height: 42px;  font-weight: 600;  margin-top: 8px;}.highlight-black {  text-align: center;  margin-top: 1em;  font-family: BebasNeue;  color: rgba(32, 33, 36, 1);  font-size: 21px;  line-height: 42px;  font-weight: 600;  margin-top: 8px;}.operatetitle-card:hover {  border: 1px solid #4592ff;  background-color: #e7f1ff;}.icon-img {  height: 14px;  width: 14px;}.progress {  transform: rotate(180deg);  margin-top: 20px;}.progress ::v-deep .el-progress-bar__outer {  background-color: rgba(247, 152, 11, 0.2);}::v-deep .el-progress {  .el-progress-bar {    .el-progress-bar__inner {      background: linear-gradient(to right, #FF6300, #FF9F01)    }  }}</style>
 |