1394 lines
79 KiB
PHP
1394 lines
79 KiB
PHP
<?php include ("./link/session.php") ?>
|
||
<!doctype html>
|
||
<html lang="uft-8">
|
||
<head>
|
||
<?php include ("./public/head.inc") ?>
|
||
<link href="assets/plugins/fileinput/css/fileinput.min.css" rel="stylesheet" >
|
||
</head>
|
||
<body>
|
||
<?php include ("./public/menu.inc") ?>
|
||
<div data-simplebar class="mb-4">
|
||
<main class="page-content sys" id="app" v-cloak>
|
||
<div class="row">
|
||
<div class="col-lg-6">
|
||
<ul class="nav nav-tabs nav-primary" role="tablist">
|
||
<li class="nav-item" role="presentation" v-if="Object.keys(netAdapter).length > 0 && Object.keys(netManagerConf).length > 0" v-for="(item,index) in Object.values(netAdapter)" :key="index">
|
||
<a v-if="netManagerConf.interface.hasOwnProperty(item.dev)" :class="['nav-link',{'active':index===0}]" data-bs-toggle="tab" :href="'#tab'+(index+1)" role="tab" aria-selected="true">
|
||
<div v-if="item.type === 'lan' || item.type === 'other'" class="d-flex align-items-center">
|
||
<div class="tab-icon"><i :class="['fa-solid me-1',{'fa-code-merge':index%2===1},{'fa-code-fork':index%2===0}]"></i></i></div>
|
||
<div class="tab-title">
|
||
<cn>网口</cn>
|
||
<en>LAN</en>
|
||
<span v-if="index > 0">{{index+1}}</span>
|
||
</div>
|
||
</div>
|
||
<div v-if="item.type === 'wifi'" class="d-flex align-items-center">
|
||
<wifi-flag :icon="'wifi-'+(item.rssi > 3 ? 4 : (item.rssi < 3 ? (item.rssi === 0 ? 1 : 2) : 3))" :width="20" :height="20" :stroke="'#cccccc'" :color="'#777777'" :stroke-width="2.3"></wifi-flag>
|
||
<div class="tab-title">
|
||
<cn>无线网</cn>
|
||
<en>WIFI</en>
|
||
</div>
|
||
</div>
|
||
<div v-if="item.type === 'dongle'" class="d-flex align-items-center">
|
||
<antenan-flag :icon="'antenan-'+(item.rssi > 3 ? 4 : (item.rssi < 3 ? (item.rssi === 0 ? 0 : 2) : 3))" :width="20" :height="20" :stroke="'#cccccc'" :color="'#777777'" :stroke-width="2.3"></antenan-flag>
|
||
<div class="tab-title">
|
||
<cn>移动网络</cn>
|
||
<en>Cellular network</en>
|
||
</div>
|
||
</div>
|
||
</a>
|
||
</li>
|
||
|
||
<li v-if="Object.keys(hardwareConf).length > 0 && hardwareConf.function.wifi && !Object.values(netAdapter).some(item => item.type === 'wifi')" class="nav-item lp-cursor-pointer" ref="wifiHandler">
|
||
<a class="nav-link">
|
||
<div class="d-flex align-items-center">
|
||
<wifi-flag icon="wifi-off" :width="20" :height="20" :stroke="'#999999'" :stroke-width="2.3"></wifi-flag>
|
||
<div class="tab-title">
|
||
<cn>无线网</cn>
|
||
<en>WIFI</en>
|
||
</div>
|
||
</div>
|
||
</a>
|
||
</li>
|
||
|
||
<li v-if="!Object.values(netAdapter).some(item => item.type === 'dongle')" class="nav-item lp-cursor-pointer" ref="antenanHandler">
|
||
<a class="nav-link">
|
||
<div class="d-flex align-items-center">
|
||
<antenan-flag icon="antenan-off" :width="20" :height="20" :stroke="'#999999'" :stroke-width="2.3"></antenan-flag>
|
||
<div class="tab-title">
|
||
<cn>移动网络</cn>
|
||
<en>Cellular network</en>
|
||
</div>
|
||
</div>
|
||
</a>
|
||
</li>
|
||
</ul>
|
||
|
||
<div class="tab-content py-3 pe-2 ps-2">
|
||
<div v-if="Object.keys(netAdapter).length > 0 && Object.keys(netManagerConf).length > 0" v-for="(item,index) in Object.values(netAdapter)" :class="['tab-pane fade',{'show active':index===0}]" :key="index" :id="'tab'+(index+1)" role="tabpanel">
|
||
<div v-if="netManagerConf.interface.hasOwnProperty(item.dev) && (item.type === 'lan' || item.type === 'other')">
|
||
<div v-if="Object.keys(hardwareConf).length > 0 && hardwareConf.function.dhcp" class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
DHCP
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-5">
|
||
<bs-switch v-model="netManagerConf.interface[item.dev].dhcp" :size="'normal'"></bs-switch>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
IP
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<input class="form-control" v-model.trim.lazy="netManagerConf.interface[item.dev].ip">
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>掩码</cn>
|
||
<en>Mask</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<input class="form-control" v-model.trim.lazy="netManagerConf.interface[item.dev].mask">
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>网关</cn>
|
||
<en>Gateway</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<input class="form-control" v-model.trim.lazy="netManagerConf.interface[item.dev].gw">
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
DNS
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<input class="form-control" v-model.trim.lazy="netManagerConf.interface[item.dev].dns">
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
MAC
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<div class="input-group">
|
||
<input class="form-control" type="text" :disabled="macLock" v-model.trim.lazy="netManagerConf.interface[item.dev].mac">
|
||
<span class="input-group-text input-group-addon lp-cursor-pointer" @click="macLock=!macLock"><i :class="['fa-solid',{'fa-lock':macLock},{'fa-unlock':!macLock}]"></i></span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-4">
|
||
<div class="col-lg-12 text-center">
|
||
<button type="button" class="btn border-3 btn-primary px-4" @click="updateDefNetwork(item.dev)"><cn>保存</cn><en>Save</en></button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div v-if="Object.keys(hardwareConf).length > 0 && hardwareConf.function.wifi && netManagerConf.interface.hasOwnProperty(item.dev) && item.type === 'wifi'">
|
||
<div class="row mt-4">
|
||
<div class="col-lg-6 border-right">
|
||
<div class="row mt-4">
|
||
<div class="col-lg-3 lp-align-center">
|
||
<label>
|
||
<cn>启用</cn>
|
||
<en>Enable</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-8">
|
||
<bs-switch v-model="netManagerConf.interface[item.dev].enable" :size="'normal'" @switch-change="enableWifi"></bs-switch>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-3 lp-align-center">
|
||
<label>
|
||
WIFI
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-8">
|
||
<div class="input-group">
|
||
<select class="form-select" v-model="wifiConnectId">
|
||
<option v-for="(it,idx) in wifiList" :key="idx+10" :value="it.ssid">{{it.ssid}}</option>
|
||
</select>
|
||
<span class="input-group-text input-group-addon lp-cursor-pointer" @click="refreshWifi"><i :class="['fa-solid fa-arrows-rotate',{'spin':wifiRefresh}]"></i></span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-3 lp-align-center">
|
||
<label>
|
||
<cn>密码</cn>
|
||
<en>Password</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-8">
|
||
<div class="input-group">
|
||
<input class="form-control" :type="!showPasswd.wifipwd ? 'password' : 'text'" v-model.trim.lazy="wifiPassword">
|
||
<span class="input-group-text input-group-addon lp-cursor-pointer" @click="showPasswd.wifipwd = !showPasswd.wifipwd"><i :class="['fa-regular',{'fa-eye-slash':!showPasswd.wifipwd},{'fa-eye':showPasswd.wifipwd}]"></i></span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-4">
|
||
<div class="col-lg-3 lp-align-center">
|
||
<label>
|
||
<cn>状态</cn>
|
||
<en>Status</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-8">
|
||
<label v-if="!netManagerConf.interface[item.dev].enable">
|
||
<cn>未启用</cn><en>Disabled</en>
|
||
</label>
|
||
<label v-else-if="(!item.ssid && wifiPassword) || (!item.linkup && item.ssid)">
|
||
<cn class="pointLoading">连接中</cn><en class="pointLoading">connecting</en>
|
||
</label>
|
||
<label v-else-if="item.linkup && item.ssid">
|
||
<cn>已连接</cn><en>connected </en>
|
||
{{item.ssid}}
|
||
</label>
|
||
<label v-else>
|
||
<cn>未连接</cn><en>not connected</en>
|
||
</label>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-4">
|
||
<div class="col-lg-12 text-center">
|
||
<button type="button" class="btn border-3 btn-primary px-4 me-2" @click="connectWifi"><cn>连接</cn><en>Connect</en></button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<div class="row mt-4">
|
||
<div class="col-lg-3 lp-align-center">
|
||
<label>
|
||
DHCP
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-8">
|
||
<bs-switch v-model="netManagerConf.interface[item.dev].dhcp" :size="'normal'"></bs-switch>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-3 lp-align-center">
|
||
<label>
|
||
IP
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-8">
|
||
<input class="form-control" v-model.trim.lazy="netManagerConf.interface[item.dev].ip">
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-3 lp-align-center">
|
||
<label>
|
||
<cn>掩码</cn>
|
||
<en>Mask</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-8">
|
||
<input class="form-control" v-model.trim.lazy="netManagerConf.interface[item.dev].mask">
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-3 lp-align-center">
|
||
<label>
|
||
<cn>网关</cn>
|
||
<en>Gateway</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-8">
|
||
<input class="form-control" v-model.trim.lazy="netManagerConf.interface[item.dev].gw">
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-3 lp-align-center">
|
||
<label>
|
||
DNS
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-8">
|
||
<input class="form-control" v-model.trim.lazy="netManagerConf.interface[item.dev].dns">
|
||
</div>
|
||
</div>
|
||
<div class="row mt-4 mb-4">
|
||
<div class="col-lg-12 text-center">
|
||
<button type="button" class="btn border-3 btn-primary px-4" @click="updateNetManagerConf"><cn>保存</cn><en>Save</en></button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div v-if="item.type === 'dongle'">
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>运营商</cn>
|
||
<en>Operator</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<div v-if="!netAdapter[item.dev].oper" v-html="`<cn class='pointLoading'>检测中</cn><en class='pointLoading'>Detecting...</en>`"></div>
|
||
<input v-else class="form-control" v-model.trim.lazy="netAdapter[item.dev].oper" readonly>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>服务</cn>
|
||
<en>Service</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<input class="form-control" v-model.trim.lazy="netAdapter[item.dev].service" readonly>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>上行</cn>
|
||
<en>Service</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<input class="form-control" :value="formatNetSpeed(netAdapter[item.dev].tx)" readonly>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>下行</cn>
|
||
<en>Service</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<input class="form-control" :value="formatNetSpeed(netAdapter[item.dev].rx)" readonly>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
IP
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<input class="form-control" v-model.trim.lazy="netAdapter[item.dev].ip" readonly>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>掩码</cn>
|
||
<en>Mask</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<input class="form-control" v-model.trim.lazy="netAdapter[item.dev].mask" readonly>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3 mb-2">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>网关</cn>
|
||
<en>Gateway</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<input class="form-control" v-model.trim.lazy="netAdapter[item.dev].gw" readonly>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<div class="row">
|
||
<div class="col-lg-12">
|
||
<div class="card">
|
||
<div class="card-header bg-transparent">
|
||
<div class="p-2 mb-0 d-flex align-items-end">
|
||
<cn>密码设置</cn>
|
||
<en>Password</en>
|
||
</div>
|
||
</div>
|
||
<div class="card-body">
|
||
<div class="row">
|
||
<div class="row">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>旧密码</cn>
|
||
<en>Current</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<div class="input-group">
|
||
<form>
|
||
<input class="form-control" :type="!showPasswd.oldpwd ? 'password' : 'text'" v-model.trim.lazy="userPasswd.oldpwd" autocomplete="off">
|
||
</form>
|
||
<span class="input-group-text input-group-addon lp-cursor-pointer" @click="showPasswd.oldpwd = !showPasswd.oldpwd"><i :class="['fa-regular',{'fa-eye-slash':showPasswd.oldpwd},{'fa-eye':!showPasswd.oldpwd}]"></i></span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<label>
|
||
<cn>新密码</cn>
|
||
<en>New</en>
|
||
</label>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<div class="input-group">
|
||
<form>
|
||
<input class="form-control" :type="!showPasswd.newpwd ? 'password' : 'text'" v-model.trim.lazy="userPasswd.newpwd" autocomplete="off">
|
||
</form>
|
||
<span class="input-group-text input-group-addon lp-cursor-pointer" @click="showPasswd.newpwd = !showPasswd.newpwd"><i :class="['fa-regular',{'fa-eye-slash':showPasswd.newpwd},{'fa-eye':!showPasswd.newpwd}]"></i></span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>确认密码</cn>
|
||
<en>Confirm</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<div class="input-group">
|
||
<form>
|
||
<input class="form-control" :type="!showPasswd.confirm ? 'password' : 'text'" v-model.trim.lazy="userPasswd.confirm" autocomplete="off">
|
||
</form>
|
||
<span class="input-group-text input-group-addon lp-cursor-pointer" @click="showPasswd.confirm = !showPasswd.confirm"><i :class="['fa-regular',{'fa-eye-slash':showPasswd.confirm},{'fa-eye':!showPasswd.confirm}]"></i></span>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-12 text-center">
|
||
<button type="button" class="btn border-3 btn-primary px-4" @click="updateUserPasswd(userPasswd)"><cn>保存</cn><en>Save</en></button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="col-lg-12">
|
||
<div class="card">
|
||
<div class="card-header bg-transparent">
|
||
<div class="p-2 mb-0 d-flex align-items-end">
|
||
<cn>应用场景</cn>
|
||
<en>Application scenario</en>
|
||
</div>
|
||
</div>
|
||
<div class="card-body py-4">
|
||
<div class="row py-2">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>场景</cn>
|
||
<en>Scene</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<select class="form-select" v-if="Object.keys(videoBufferConf).length > 0" v-model="videoBufferConf.used">
|
||
<option v-for="item in handleSysScene" :value="item">{{item}}</option>
|
||
</select>
|
||
</div>
|
||
<div class="col-lg-2">
|
||
<button type="button" class="btn border-3 btn-primary change" @click="updateVideoBufferConf"><cn>切换</cn><en>Change</en></button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="row">
|
||
<div class="col-lg-6">
|
||
<div class="card">
|
||
<div class="card-header bg-transparent">
|
||
<div class="p-2 mb-0 d-flex align-items-end">
|
||
<cn>定时维护</cn>
|
||
<en>Auto reboot</en>
|
||
</div>
|
||
</div>
|
||
<div class="card-body py-4">
|
||
<div class="row">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>系统时间</cn>
|
||
<en>system time</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<input class="form-control" v-model.trim.lazy="sysTime">
|
||
</div>
|
||
<div class="col-lg-2">
|
||
<button type="button" class="btn border-3 btn-primary px-2" @click="syncTimeFromPc"><cn>本地同步</cn><en>Sync from PC</en></button>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3" v-if="Object.keys(ntpConf).length > 0">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>NTP同步</cn>
|
||
<en>NTP sync</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-3">
|
||
<input class="form-control" v-model.trim.lazy="ntpConf.server">
|
||
</div>
|
||
<div class="col-lg-3">
|
||
<div class="input-group">
|
||
<span class="input-group-text input-group-addon">
|
||
<cn>间隔</cn>
|
||
<en>inr</en>
|
||
</span>
|
||
<input class="form-control" v-model.trim.lazy="ntpConf.interval">
|
||
<span class="input-group-text input-group-addon">
|
||
<cn>分钟</cn>
|
||
<en>min</en>
|
||
</span>
|
||
</div>
|
||
</div>
|
||
<div class="col-lg-2">
|
||
<bs-switch v-model="ntpConf.enable"></bs-switch>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>时区设置</cn>
|
||
<en>time zone</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-3">
|
||
<select class="form-select" v-model="timezoneConf.timeArea" @change="onTimeAreaChange">
|
||
<option value="Africa">Africa</option>
|
||
<option value="America">Americas</option>
|
||
<option value="Antarctica">Antarctica</option>
|
||
<option value="Asia">Asia</option>
|
||
<option value="Atlantic">Atlantic Ocean</option>
|
||
<option value="Australia">Australia</option>
|
||
<option value="Europe">Europe</option>
|
||
<option value="Indian">Indian Ocean</option>
|
||
<option value="Pacific">Pacific Ocean</option>
|
||
</select>
|
||
</div>
|
||
<div class="col-lg-3">
|
||
<select class="form-select" v-model="timezoneConf.timeCity">
|
||
<option v-for="item in timezoneCitys" :value="item.name">{{item.name}}</option>
|
||
</select>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>维护时间</cn>
|
||
<en>reboot time</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-3">
|
||
<select class="form-select" v-model="cronDay">
|
||
<option cn="从不" en="never" value="x" v-language-option></option>
|
||
<option cn="每天" en="everyday" value="*" v-language-option></option>
|
||
<option cn="每周一" en="monday" value="1" v-language-option></option>
|
||
<option cn="每周二" en="tuesday" value="2" v-language-option></option>
|
||
<option cn="每周三" en="wednesday" value="3" v-language-option></option>
|
||
<option cn="每周四" en="thursday" value="4" v-language-option></option>
|
||
<option cn="每周五" en="friday" value="5" v-language-option></option>
|
||
<option cn="每周六" en="saturday" value="6" v-language-option></option>
|
||
<option cn="每周日" en="sunday" value="0" v-language-option></option>
|
||
</select>
|
||
</div>
|
||
<div class="col-lg-3">
|
||
<select class="form-select" v-model="cronTime">
|
||
<option value="0">0:00</option>
|
||
<option value="1">1:00</option>
|
||
<option value="2">2:00</option>
|
||
<option value="3">3:00</option>
|
||
<option value="4">4:00</option>
|
||
<option value="5">5:00</option>
|
||
<option value="6">6:00</option>
|
||
<option value="7">7:00</option>
|
||
<option value="8">8:00</option>
|
||
<option value="9">9:00</option>
|
||
<option value="10">10:00</option>
|
||
<option value="11">11:00</option>
|
||
<option value="12">12:00</option>
|
||
<option value="13">13:00</option>
|
||
<option value="14">14:00</option>
|
||
<option value="15">15:00</option>
|
||
<option value="16">16:00</option>
|
||
<option value="17">17:00</option>
|
||
<option value="18">18:00</option>
|
||
<option value="19">19:00</option>
|
||
<option value="20">20:00</option>
|
||
<option value="21">21:00</option>
|
||
<option value="22">22:00</option>
|
||
<option value="23">23:00</option>
|
||
</select>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-4">
|
||
<div class="col-lg-12 text-center">
|
||
<button type="button" class="btn border-3 btn-primary px-4 me-4" @click="saveSysConf"><cn>保存</cn><en>Save</en></button>
|
||
<button type="button" class="btn border-3 btn-primary px-4 me-4" @click="rebootConfirm('是否立即重启系统')"><cn>立即重启</cn><en>Reboot</en></button>
|
||
<button type="button" class="btn border-3 btn-primary px-4" @click="resetConfirm"><cn>恢复出厂设置</cn><en>Reset default</en></button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<div class="col-lg-6">
|
||
<div class="card">
|
||
<div class="card-header bg-transparent">
|
||
<div class="p-2 mb-0 d-flex align-items-end">
|
||
<cn>配置文件</cn>
|
||
<en>Config file</en>
|
||
<small>
|
||
<cn>按需导出配置文件</cn>
|
||
<en>Export configuration files on demand</en>
|
||
</small>
|
||
</div>
|
||
</div>
|
||
<div class="card-body py-4">
|
||
<div class="row row-cols-4 row-cols-lg-4 px-5">
|
||
<div class="text-center">
|
||
<cn>编解码配置</cn>
|
||
<en>Default Config</en>
|
||
</div>
|
||
<div class="text-center">
|
||
<cn>布局配置</cn>
|
||
<en>Layout Config</en>
|
||
</div>
|
||
<div class="text-center">
|
||
<cn>推流配置</cn>
|
||
<en>Push Config</en>
|
||
</div>
|
||
<div class="text-center">
|
||
<cn>密码配置</cn>
|
||
<en>Password Config</en>
|
||
</div>
|
||
</div>
|
||
<hr>
|
||
<div>
|
||
<div class="row row-cols-4 row-cols-lg-4 px-5">
|
||
<div class="lp-align-center">
|
||
<bs-switch v-model="exportConfigs.config"></bs-switch>
|
||
</div>
|
||
<div class="lp-align-center">
|
||
<bs-switch v-model="exportConfigs.defLays"></bs-switch>
|
||
</div>
|
||
<div class="lp-align-center">
|
||
<bs-switch v-model="exportConfigs.push"></bs-switch>
|
||
</div>
|
||
<div class="lp-align-center">
|
||
<bs-switch v-model="exportConfigs.passwd"></bs-switch>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="row row-cols-4 row-cols-lg-4 mt-4 px-5">
|
||
<div class="text-center">
|
||
<cn>录制配置</cn>
|
||
<en>Record Config</en>
|
||
</div>
|
||
<div class="text-center">
|
||
<cn>端口配置</cn>
|
||
<en>Port Config</en>
|
||
</div>
|
||
<div class="text-center">
|
||
<cn>维护配置</cn>
|
||
<en>NTP Config</en>
|
||
</div>
|
||
<div class="text-center">
|
||
<cn>场景配置</cn>
|
||
<en>Scene Config</en>
|
||
</div>
|
||
</div>
|
||
<hr>
|
||
<div>
|
||
<div class="row row-cols-4 row-cols-lg-4 px-5">
|
||
<div class="lp-align-center">
|
||
<bs-switch v-model="exportConfigs.record"></bs-switch>
|
||
</div>
|
||
<div class="lp-align-center">
|
||
<bs-switch v-model="exportConfigs.port"></bs-switch>
|
||
</div>
|
||
<div class="lp-align-center">
|
||
<bs-switch v-model="exportConfigs.cron"></bs-switch>
|
||
</div>
|
||
<div class="lp-align-center">
|
||
<bs-switch v-model="exportConfigs.videoBuffer"></bs-switch>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-4">
|
||
<div class="col-lg-12 text-center mt-3">
|
||
<button type="button" class="btn border-3 btn-primary px-4 me-3" @click="exportConf"><cn>导出</cn><en>Export</en></button>
|
||
<button type="button" class="btn border-3 btn-primary px-4 " @click="importConf"><cn>导入</cn><en>Import</en></button>
|
||
<input type="file" accept=".zip" style="display:none;" ref="importHandle" />
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="row" v-if="Object.keys(hardwareConf).length > 0 && hardwareConf.function.portCtrl">
|
||
<div class="col-lg-12">
|
||
<div class="card">
|
||
<div class="card-header bg-transparent">
|
||
<div class="p-2 mb-0 d-flex align-items-end">
|
||
<cn>端口配置</cn>
|
||
<en>Port config</en>
|
||
</div>
|
||
</div>
|
||
<div class="card-body" v-if="Object.keys(portConf).length > 0">
|
||
<div class="row mb-2">
|
||
<div class="col-10">
|
||
<div class="row">
|
||
<div class="col-3"></div>
|
||
<div class="col text-center">HTTP</div>
|
||
<div class="col text-center">RTSP</div>
|
||
<div class="col text-center">RTMP</div>
|
||
<div class="col text-center">HTTPTS</div>
|
||
<div class="col text-center">Telnet</div>
|
||
<div class="col text-center">SSH</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<hr>
|
||
<div class="row mt-2">
|
||
<div class="col-10">
|
||
<div class="row">
|
||
<div class="col-3 d-flex align-items-center justify-content-end">
|
||
<cn>固定端口</cn>
|
||
<en>Static port</en>
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.http[0]" readonly disabled>
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.rtsp[0]" readonly disabled>
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.rtmp[0]" readonly disabled>
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.httpts[0]" readonly disabled>
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.telnet[0]" readonly disabled>
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.ssh[0]" readonly disabled>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-3 d-flex align-items-center justify-content-end">
|
||
<cn>备用端口</cn>
|
||
<en>Reserve port</en>
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.http[1]">
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.rtsp[1]">
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.rtmp[1]">
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.httpts[1]">
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.telnet[1]">
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.ssh[1]">
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-3 d-flex align-items-center justify-content-end">
|
||
<cn>映射(外网)端口</cn>
|
||
<en>NAT port</en>
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.http[2]">
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.rtsp[2]">
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.rtmp[2]">
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.httpts[2]">
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.telnet[2]">
|
||
</div>
|
||
<div class="col">
|
||
<input type="text" class="form-control text-center" v-model.trim.lazy="portConf.ssh[2]">
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="row my-2">
|
||
<div class="col-lg-12 text-center mt-3">
|
||
<button type="button" class="btn border-3 btn-primary px-5" @click="updatePortConf"><cn>保存</cn><en>Save</en></button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="row">
|
||
<div class="col-lg-6">
|
||
<div class="row">
|
||
<div class="col-lg-12">
|
||
<div class="card">
|
||
<div class="card-header bg-transparent">
|
||
<div class="p-2 mb-0 d-flex align-items-end">
|
||
<cn>远程协助</cn>
|
||
<en>Remote Assistance</en>
|
||
</div>
|
||
</div>
|
||
<div class="card-body">
|
||
<div class="row my-3">
|
||
<div class="col-lg-3 d-flex align-items-center justify-content-end">
|
||
<cn>授权码</cn>
|
||
<en>Auth code</en>
|
||
</div>
|
||
<div class="col-lg-3 lp-align-center">
|
||
<input type="text" class="form-control" v-model.trim.lazy="helpCode" readonly disabled>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<button type="button" class="btn border-3 btn-primary px-4 me-2" @click="startHelp"><cn>开始协助</cn><en>Start</en></button>
|
||
<button type="button" class="btn border-3 btn-primary px-4" @click="stopHelp"><cn>停止协助</cn><en>Stop</en></button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="col-lg-12">
|
||
<div class="card">
|
||
<div class="card-header bg-transparent">
|
||
<div class="p-2 mb-0 d-flex align-items-end">
|
||
<cn>网络测试</cn>
|
||
<en>Network Test</en>
|
||
</div>
|
||
</div>
|
||
<div class="card-body">
|
||
<div class="row my-3">
|
||
<div class="col-lg-3 d-flex align-items-center justify-content-end">
|
||
<cn>服务地址</cn>
|
||
<en>Server</en>
|
||
</div>
|
||
<div class="col-lg-5 lp-align-center" v-if="Object.keys(netManagerConf).length > 0">
|
||
<input type="text" class="form-control" v-model.trim.lazy="netManagerConf.onlineServer">
|
||
</div>
|
||
<div class="col-lg-4">
|
||
<button type="button" class="btn border-3 btn-primary px-3 me-2" @click="updateNetManagerConf"><cn>保存</cn><en>Save</en></button>
|
||
<button type="button" class="btn border-3 btn-primary px-4 net-test" @click="systemNetTest"><cn>网络测试</cn><en>Start Test</en></button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<div class="card">
|
||
<div class="card-header bg-transparent">
|
||
<div class="p-2 mb-0 d-flex align-items-end">
|
||
<cn>系统升级</cn><en>Upgrade</en>
|
||
</div>
|
||
</div>
|
||
<div class="card-body" v-if="Object.keys(versionConf).length > 0">
|
||
<div class="row mt-1">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>应用版本</cn>
|
||
<en>App version</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<label>{{versionConf.app}}</label>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>SDK版本</cn>
|
||
<en>SDK version</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<label>{{versionConf.sdk}}</label>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>系统版本</cn>
|
||
<en>Sys version</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<label>{{versionConf.sys}}</label>
|
||
</div>
|
||
</div>
|
||
<hr class="my-4">
|
||
<div class="row mt-3">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>上传升级</cn>
|
||
<en>upload packet</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<button type="button" class="btn border-3 btn-primary px-3 me-2" @click="showBootstrapModal('upload')"><cn>选择文件</cn><en>File</en></button>
|
||
<button type="button" class="btn border-3 btn-primary px-3" @click="showBootstrapModal('log')"><cn>版本日志</cn><en>Logs</en></button>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-1 mb-4">
|
||
<div class="col-lg-2 offset-lg-1 lp-align-center">
|
||
<label>
|
||
<cn>在线升级</cn>
|
||
<en>online update</en>
|
||
</label>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<loading-button custom-class="btn border-3 btn-primary px-3 me-2 checkUpdate" @button-click="checkUpdatePatch" :had-loading="checkLoading">
|
||
<cn>检测更新</cn>
|
||
<en>Search</en>
|
||
</loading-button>
|
||
<button type="button" class="btn border-3 btn-primary px-3 search-packet checkUpdate" @click="searchUpdatePatch"><i class="fa fa-search"></i></button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
<upload-modal :modal-title="'上传升级包&Upload'" :modal-show="showUploadModal" :modal-fade="true"
|
||
:upload-allow="['bin']" :upload-action="'/link/upd/uploadPatch.php'" :upload-count="1"
|
||
:upload-tip="'请把升级包拖动到此处...&Please drag the upgrade package here...'"
|
||
@upload-success="uploadSuccess" @upload-error="uploadError">
|
||
</upload-modal>
|
||
|
||
<logger-modal :modal-size="'modal-lg'" :had-header="false" :had-footer="false" :modal-show="showVerLogModal" :modal-fade="true" :content-class="'logCtx'" :body-class="'logBody'" >
|
||
<div data-simplebar class="mt-2">
|
||
<div v-for="(item,index) in verLogsConf" :key="index" class="mt-3">
|
||
<button v-if="index===0" type="button" class="btn-clear close" @click="showBootstrapModal('log')"><i class="fa-solid fa-x"></i></button>
|
||
<div class="row">
|
||
<div class="col-lg-10 offset-lg-1">
|
||
<h3>{{item.version}}</h3>
|
||
</div>
|
||
</div>
|
||
<div class="row mt-2">
|
||
<div class="col-lg-10 offset-lg-1">
|
||
<ul>
|
||
<li v-for="(it,idx) in item.logs" :key="idx" style="font-size: 14px;white-space:pre-wrap;">{{it}}</li>
|
||
</ul>
|
||
</div>
|
||
</div>
|
||
<hr>
|
||
</div>
|
||
</div>
|
||
</logger-modal>
|
||
|
||
<search-modal :modal-title="'固件搜索&Search'" :modal-show="showSearchModal"
|
||
:confirm-btn-name="'搜索&Search'" :cancel-btn-name="'取消&Cancel'"
|
||
@confirm-btn-click="searchPatchBySn">
|
||
<div class="row my-3">
|
||
<div class="col-lg-3 offset-lg-1">
|
||
<div class="snTitle">
|
||
<cn>固件编号:</cn>
|
||
<en>Patch Serial:</en>
|
||
</div>
|
||
</div>
|
||
<div class="col-lg-6">
|
||
<input class="snInput" v-model.trim.lazy="patchSN" autocomplete="off"/>
|
||
</div>
|
||
</div>
|
||
</search-modal>
|
||
|
||
<upgrade-modal v-model:check-upgrade="checkUpgrade" :patch-sn="patchSN" :modal-fade="true"></upgrade-modal>
|
||
</main>
|
||
</div>
|
||
<?php include ("./public/foot.inc") ?>
|
||
<script src="assets/plugins/fileinput/js/fileinput.min.js" type="module"></script>
|
||
<script src="assets/plugins/fileinput/js/locales/zh.js" type="module"></script>
|
||
<script src="assets/plugins/fileinput/themes/fa6/theme.min.js" type="module"></script>
|
||
<script type="module">
|
||
|
||
import vue from "./assets/js/vue.build.js";
|
||
import JsZip from "./assets/plugins/jszip/jszip.esm.js"
|
||
import * as fileSave from "./assets/plugins/jszip/filesaver.esm.js";
|
||
import { rpc2,alertMsg,func,queryData,popover,formatDate,rebootConfirm,resetConfirm,clearReactiveObject } from "./assets/js/lp.utils.js";
|
||
import { useHardwareConf,useNetManagerConf,usePasswordConf,useVideoBufferConf,useNtpConf,useTimezoneConf,usePortConf,useVersionConf,useVerLogsConf,useWpaConf } from "./assets/js/vue.hooks.js";
|
||
import { ignoreCustomElementPlugin,bootstrapSwitchComponent,languageOptionDirective,uploadModalComponent,upgradeModalComponent,customModalComponent,loadingButtonComponent } from "./assets/js/vue.helper.js"
|
||
import { wifiFlagComponent,antenanFlagComponent } from "./assets/js/vue.flags.js";
|
||
|
||
const { createApp,ref,reactive,watch,watchEffect,computed,onMounted } = vue;
|
||
const app = createApp({
|
||
directives:{
|
||
"language-option": languageOptionDirective
|
||
},
|
||
components:{
|
||
"bs-switch" : bootstrapSwitchComponent,
|
||
"wifi-flag": wifiFlagComponent,
|
||
"antenan-flag": antenanFlagComponent,
|
||
"upload-modal": uploadModalComponent,
|
||
"logger-modal": customModalComponent,
|
||
"upgrade-modal": upgradeModalComponent,
|
||
"search-modal": customModalComponent,
|
||
"loading-button": loadingButtonComponent
|
||
},
|
||
setup(props,context) {
|
||
|
||
const { hardwareConf } = useHardwareConf();
|
||
const { netManagerConf,updateNetManagerConf } = useNetManagerConf();
|
||
const { wpaConf } = useWpaConf();
|
||
const { updateUserPasswd } = usePasswordConf();
|
||
const { videoBufferConf,updateVideoBufferConf } = useVideoBufferConf();
|
||
const { ntpConf } = useNtpConf();
|
||
const { timezoneConf } = useTimezoneConf();
|
||
const { portConf,updatePortConf } = usePortConf();
|
||
const { versionConf } = useVersionConf();
|
||
const { verLogsConf } = useVerLogsConf();
|
||
const { saveAs } = fileSave;
|
||
|
||
const state = {
|
||
netAdapter: reactive({}),
|
||
wifiHandler: ref(null),
|
||
wifiPopover: {},
|
||
antenanHandler:ref(null),
|
||
antenanPopover: {},
|
||
wifiRefresh: ref(false),
|
||
wifiList:reactive([]),
|
||
wifiPassword:ref(""),
|
||
wifiConnectId:ref(""),
|
||
sysTime: ref("1970-01-01 08:00:00"),
|
||
timezoneCitys: reactive([]),
|
||
cronDay: ref("never"),
|
||
cronTime: ref("0"),
|
||
importHandle:ref(null),
|
||
helpCode:ref(""),
|
||
userPasswd:reactive({}),
|
||
showPasswd:reactive({}),
|
||
exportConfigs:reactive({}),
|
||
showUploadModal:ref(false),
|
||
showVerLogModal:ref(false),
|
||
showSearchModal:ref(false),
|
||
checkLoading:ref(false),
|
||
checkUpgrade:ref(false),
|
||
patchSN:ref(""),
|
||
macLock:ref(true)
|
||
}
|
||
|
||
watch(state.wifiHandler,() => {
|
||
if(state.wifiHandler.value) {
|
||
if(Object.keys(state.wifiPopover).length > 0) {
|
||
state.wifiPopover.dispose();
|
||
state.wifiPopover = {};
|
||
}
|
||
state.wifiPopover = popover(state.wifiHandler.value, {
|
||
placement:"bottom",
|
||
trigger:"hover",
|
||
content:`<cn>请先插入USB WIFI网卡</cn><en>Please insert the USB WIFI network card first</en>`,
|
||
});
|
||
if(document.querySelector('a[href="#tab1"]'))
|
||
document.querySelector('a[href="#tab1"]').click();
|
||
} else {
|
||
if(!state.wifiHandler.value && Object.keys(state.wifiPopover).length !== 0) {
|
||
state.wifiPopover.dispose();
|
||
state.wifiPopover = {};
|
||
}
|
||
}
|
||
})
|
||
|
||
watch(state.antenanHandler, () => {
|
||
if(state.antenanHandler.value) {
|
||
if(Object.keys(state.antenanPopover).length > 0) {
|
||
state.antenanPopover.dispose();
|
||
state.antenanPopover = {};
|
||
}
|
||
state.antenanPopover = popover(state.antenanHandler.value, {
|
||
placement:"bottom",
|
||
trigger:"hover",
|
||
content:`<cn>请先插入移动网卡</cn><en>Please insert the Cellular network card first</en>`,
|
||
})
|
||
if(document.querySelector('a[href="#tab1"]'))
|
||
document.querySelector('a[href="#tab1"]').click();
|
||
} else {
|
||
if(!state.antenanHandler.value && Object.keys(state.antenanPopover).length !== 0) {
|
||
state.antenanPopover.dispose();
|
||
state.antenanPopover = {};
|
||
}
|
||
}
|
||
})
|
||
|
||
watchEffect(()=>{
|
||
if(!state.checkUpgrade.value)
|
||
state.checkLoading.value = false;
|
||
|
||
if(wpaConf.length > 0 && state.wifiConnectId.value === "") {
|
||
state.wifiConnectId.value = wpaConf[0].ssid
|
||
state.wifiPassword.value = wpaConf[0].psk;
|
||
}
|
||
|
||
if(Object.keys(timezoneConf).length > 0 && state.timezoneCitys.length === 0)
|
||
onTimeAreaChange();
|
||
})
|
||
|
||
const getAdapterNetState = () => {
|
||
rpc2("net.getState").then(data => {
|
||
clearReactiveObject(state.netAdapter);
|
||
Object.assign(state.netAdapter,data.interface);
|
||
});
|
||
setTimeout(getAdapterNetState,2000);
|
||
}
|
||
|
||
const handleSysScene = computed(()=>{
|
||
return Object.keys(videoBufferConf).filter((item,index)=>{
|
||
return item !== "used";
|
||
})
|
||
});
|
||
|
||
const updateDefNetwork = (dev) => {
|
||
updateNetManagerConf().then(()=>{
|
||
if(dev === netManagerConf["gw"])
|
||
setTimeout(() => window.location.href="http://"+netManagerConf["interface"][dev]["ip"]+"/sys.php",1000)
|
||
})
|
||
}
|
||
|
||
const enableWifi = state => {
|
||
updateNetManagerConf("noTip").then(()=>{
|
||
if(state)
|
||
alertMsg('<cn>WIFI已启用</cn><en>WIFI enable successfully!</en>', 'success');
|
||
else
|
||
alertMsg('<cn>WIFI已关闭</cn><en>WIFI disable successfully!</en>', 'success');
|
||
})
|
||
}
|
||
|
||
const refreshWifi = (tip = 'loading',refresh = true) => {
|
||
const scanwifi = type => {
|
||
rpc2("net.scanWifi",[refresh]).then(data => {
|
||
if(data.length === 0)
|
||
data.push({ssid:state.wifiConnectId});
|
||
state.wifiList.splice(0, state.wifiList.length, ...data);
|
||
state.wifiRefresh.value = false;
|
||
})
|
||
}
|
||
if(tip === "noLoading") {
|
||
scanwifi(refresh);
|
||
return;
|
||
}
|
||
state.wifiRefresh.value = true;
|
||
setTimeout(scanwifi,2000);
|
||
}
|
||
|
||
const connectWifi = () => {
|
||
state.wifiPassword.value = state.wifiPassword.value.replace(/\s/g, '');
|
||
if(state.wifiPassword.value.length < 8) {
|
||
alertMsg('<cn>WIFI密码格式错误,密码长度不能少于8位</cn><en>The wifi password format is wrong. The password length cannot be less than 8 characters!</en>', 'error');
|
||
return;
|
||
}
|
||
rpc2("net.setSimpleWifi", [state.wifiConnectId.value,state.wifiPassword.value]).then(data => {
|
||
if(data)
|
||
alertMsg('<cn>设置成功,如果长时间处于连接状态,请检查wifi密码是否正确</cn><en>The setting is successful. If you are connected for a long time, please check whether the wifi password is correct</en>', 'success');
|
||
else
|
||
alertMsg('<cn>设置失败</cn><en>Save failed!</en>', 'error');
|
||
});
|
||
}
|
||
|
||
const formatNetSpeed = speed => {
|
||
if (speed < 1024) {
|
||
return speed + " kb/s";
|
||
} else {
|
||
const formattedSpeed = parseFloat(speed / 1024).toFixed(1);
|
||
return formattedSpeed.replace(/\.0$/, '') + " mb/s";
|
||
}
|
||
}
|
||
|
||
const syncTimeFromPc = () => {
|
||
let time1 = formatDate("yyyy/MM/dd/hh/mm/ss");
|
||
let time2 = formatDate("yyyy-MM-dd hh:mm:ss");
|
||
func("/system/setSystemTime",{time1:time1,time2:time2}).then((data)=>{
|
||
if(data.status === "success") {
|
||
alertMsg('<cn>时间同步成功</cn><en>system time synchronization successful!</en>', 'success');
|
||
state.sysTime.value = time2;
|
||
}
|
||
})
|
||
}
|
||
|
||
const onTimeAreaChange = (evt) => {
|
||
queryData("/timezone/zoneinfo/"+timezoneConf.timeArea+"/").then(data=>{
|
||
state.timezoneCitys.splice(0, state.timezoneCitys.length, ...data);
|
||
if(evt !== undefined)
|
||
timezoneConf.timeCity = state.timezoneCitys[0].name;
|
||
})
|
||
}
|
||
|
||
const saveSysConf = () => {
|
||
Promise.all([
|
||
func("/conf/updateNtpConf", ntpConf),
|
||
func("/conf/updateTimezoneConf", timezoneConf),
|
||
func("/system/setSystemCrontab", { day: state.cronDay.value, time: state.cronTime.value }),
|
||
]).then((results) => {
|
||
if(results.every(ret => typeof ret === "boolean" ? ret : (ret?.status === "success")))
|
||
alertMsg('<cn>保存设置成功</cn><en>Save config successfully!</en>', 'success');
|
||
else
|
||
alertMsg('<cn>保存设置失败</cn><en>Save config failed!</en>', 'error');
|
||
})
|
||
}
|
||
|
||
const exportConf = () => {
|
||
let confs = ["lang.json"];
|
||
for(let i=0;i<Object.keys(state.exportConfigs).length;i++) {
|
||
let path = Object.keys(state.exportConfigs)[i];
|
||
if(state.exportConfigs[path]) {
|
||
if(path === "cron") {
|
||
confs.push("ntp.json");
|
||
confs.push("auto/root.cron");
|
||
confs.push("misc/timezone/tzselect.json");
|
||
} else if(path === "videoBuffer") {
|
||
confs.push("board.json");
|
||
confs.push("videoBuffer.json");
|
||
} else {
|
||
confs.push(path+".json");
|
||
}
|
||
}
|
||
}
|
||
|
||
const promiseArray = confs.map((conf) => {
|
||
return queryData("config/" + conf, { responseType: 'blob' }).then(data => ({ name: conf, data }));
|
||
});
|
||
|
||
Promise.all(promiseArray)
|
||
.then((results) => {
|
||
const zip = new JsZip();
|
||
results.forEach(({ name, data }) => {
|
||
zip.file(name, data, { binary: true });
|
||
});
|
||
|
||
if (Object.keys(zip.files).length > 0) {
|
||
return zip.generateAsync({ type: 'blob' });
|
||
} else {
|
||
throw new Error('下载全部失败');
|
||
}
|
||
})
|
||
.then(blob => {
|
||
saveAs(blob, 'configs.zip');
|
||
})
|
||
.catch(error => {
|
||
console.error(error);
|
||
});
|
||
}
|
||
|
||
const importConf = () =>{
|
||
state.importHandle.value.click();
|
||
state.importHandle.value.addEventListener('change', event => {
|
||
let data = new FormData();
|
||
let file=event.target.files[0];
|
||
let name=file.name;
|
||
data.append("file",file);
|
||
data.append("name",name);
|
||
axios({
|
||
url: '/link/upd/uploadConf.php',
|
||
method: 'post',
|
||
data: data,
|
||
responseType: 'json',
|
||
headers: {'Content-Type': 'multipart/form-data'}
|
||
}).then(response => {
|
||
if (response.data.isSuccess) {
|
||
alertMsg('<cn>导入成功</cn><en>Import success</en>!','success');
|
||
} else {
|
||
alertMsg('<cn>导入失败</cn><en>Import faild</en>!','error');
|
||
}
|
||
})
|
||
.catch(error => {
|
||
console.error('请求发生错误:', error);
|
||
});
|
||
});
|
||
}
|
||
|
||
const startHelp = () => {
|
||
state.helpCode.value = Math.floor(Math.random()*1000);
|
||
func("/system/startHelp",{helpCode: state.helpCode.value}).then((data)=>{
|
||
if(data.status === "success")
|
||
alertMsg('<cn>连接成功,请向客服提供授权码以便控制您的编码器。</cn><en>Connect success, please provide auth code to customer service to control your encoder!</en>', 'success');
|
||
})
|
||
}
|
||
|
||
const stopHelp = () => {
|
||
func("/system/stopHelp").then((data)=>{
|
||
if(data.status === "success") {
|
||
state.helpCode.value = "";
|
||
alertMsg('<cn>已断开连接</cn><en>Disconnect success</en>', 'success');
|
||
}
|
||
})
|
||
}
|
||
|
||
const systemNetTest = () => {
|
||
func("/system/systemNetTest",netManagerConf.onlineServer).then(data => {
|
||
const str = data.data.join();
|
||
if(str === "")
|
||
alertMsg('<cn>域名解析超时</cn><en>DNS timeout</en>!', 'error');
|
||
else if(str.indexOf(" 0%")>0)
|
||
alertMsg('<cn>网络可用</cn><en>Network available</en>!', 'success');
|
||
else
|
||
alertMsg('<cn>网络不可用</cn><en>Network Unavailable</en>!', 'error');
|
||
})
|
||
}
|
||
|
||
const showBootstrapModal = type => {
|
||
if(type === "upload")
|
||
state.showUploadModal.value = !state.showUploadModal.value;
|
||
if(type === "log")
|
||
state.showVerLogModal.value = !state.showVerLogModal.value;
|
||
}
|
||
|
||
const uploadSuccess = data => {
|
||
const response = data.response;
|
||
if(response.upload === "0")
|
||
rebootConfirm("上传成功,是否立即重启系统完成更新?");
|
||
if(response.upload === "-1")
|
||
alertMsg("<cn>上传失败,升级包机型不匹配!</cn><en>Upload failed, upgrade package model does not match!</en>","error");
|
||
if(response.upload === "-2")
|
||
alertMsg("<cn>上传失败,升级包与系统版本不匹配!</cn><en>Upload failed, the upgrade package does not match the system versio!</en>","error");
|
||
}
|
||
|
||
const uploadError = errMsg => {
|
||
alertMsg(errMsg, 'error');
|
||
}
|
||
|
||
const checkUpdatePatch = () => {
|
||
state.patchSN.value = "";
|
||
state.checkLoading.value = true;
|
||
setTimeout(()=>{
|
||
state.checkUpgrade.value = true;
|
||
},1000)
|
||
}
|
||
|
||
const searchUpdatePatch = () => {
|
||
state.showSearchModal.value = !state.showSearchModal.value;
|
||
}
|
||
|
||
const searchPatchBySn = () => {
|
||
if(state.patchSN.value === "") {
|
||
alertMsg("<cn>请输入固件编号</cn><en>Please enter the patch sn.</en>", "error");
|
||
return;
|
||
}
|
||
state.checkUpgrade.value = true;
|
||
state.showSearchModal.value = false;
|
||
}
|
||
|
||
const updateRenderData = () => {
|
||
Object.assign(state.userPasswd,{oldpwd: '',newpwd: '',confirm: ''});
|
||
Object.assign(state.showPasswd,{wifipwd: false,oldpwd: false,newpwd: false,confirm: false});
|
||
Object.assign(state.exportConfigs,{config: true,defLays: false,push: false,
|
||
record: false,port: false,passwd: false,videoBuffer: true,cron: true
|
||
});
|
||
}
|
||
|
||
const getSysAbortTime = () => {
|
||
func("/system/getSystemTime").then(result => {
|
||
state.sysTime.value = result.data;
|
||
setInterval(()=>{
|
||
let currentTime = new Date(state.sysTime.value);
|
||
currentTime.setSeconds(currentTime.getSeconds() + 1);
|
||
const year = currentTime.getFullYear();
|
||
const month = String(currentTime.getMonth() + 1).padStart(2, '0');
|
||
const day = String(currentTime.getDate()).padStart(2, '0');
|
||
const hours = String(currentTime.getHours()).padStart(2, '0');
|
||
const minutes = String(currentTime.getMinutes()).padStart(2, '0');
|
||
const seconds = String(currentTime.getSeconds()).padStart(2, '0');
|
||
state.sysTime.value = `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;
|
||
},1000);
|
||
});
|
||
|
||
func("/system/getSystemCrontab").then(result => {
|
||
if ( result.data === null || result.data.split( " " ).length !== 6 ) {
|
||
state.cronDay.value = "x";
|
||
state.cronTime.value = "0";
|
||
} else {
|
||
state.cronTime.value = result.data.split( " " )[ 1 ];
|
||
state.cronDay.value = result.data.split( " " )[ 4 ];
|
||
}
|
||
});
|
||
}
|
||
|
||
onMounted(()=>{
|
||
updateRenderData();
|
||
getSysAbortTime();
|
||
getAdapterNetState();
|
||
refreshWifi("noLoading",false);
|
||
})
|
||
|
||
return {...state,hardwareConf,netManagerConf,videoBufferConf,ntpConf,timezoneConf,portConf,versionConf,verLogsConf,
|
||
enableWifi,refreshWifi,connectWifi,updateNetManagerConf,handleSysScene, updateUserPasswd,updateVideoBufferConf,updateDefNetwork,
|
||
updatePortConf,showBootstrapModal,formatNetSpeed, uploadSuccess,uploadError,rebootConfirm,resetConfirm, onTimeAreaChange,
|
||
syncTimeFromPc,saveSysConf,exportConf,importConf,startHelp,stopHelp, systemNetTest,checkUpdatePatch,searchUpdatePatch,searchPatchBySn}
|
||
}
|
||
});
|
||
app.use(ignoreCustomElementPlugin);
|
||
app.mount('#app');
|
||
</script>
|
||
</body>
|
||
</html>
|