mirror of
				https://github.com/LiteyukiStudio/LiteyukiBot.git
				synced 2025-10-26 05:16:32 +00:00 
			
		
		
		
	✨ 新版状态页面
This commit is contained in:
		| @@ -94,6 +94,9 @@ def get_local_data(lang_code) -> dict: | ||||
|             "minutes"         : lang.get("status.minutes"), | ||||
|             "seconds"         : lang.get("status.seconds"), | ||||
|             "runtime"         : lang.get("status.runtime"), | ||||
|             "threads"         : lang.get("status.threads"), | ||||
|             "cores"           : lang.get("status.cores"), | ||||
|             "process"         : lang.get("status.process"), | ||||
|  | ||||
|     } | ||||
|  | ||||
| @@ -149,7 +152,7 @@ async def get_hardware_data() -> dict: | ||||
|     all_processes = psutil.Process().children(recursive=True) | ||||
|     all_processes.append(psutil.Process()) | ||||
|  | ||||
|     mem_used_bot = 0 | ||||
|     mem_used_process = 0 | ||||
|     process_mem = {} | ||||
|     for process in all_processes: | ||||
|         try: | ||||
| @@ -157,7 +160,7 @@ async def get_hardware_data() -> dict: | ||||
|             if ps_name not in process_mem: | ||||
|                 process_mem[ps_name] = 0 | ||||
|             process_mem[ps_name] += process.memory_info().rss | ||||
|             mem_used_bot += process.memory_info().rss | ||||
|             mem_used_process += process.memory_info().rss | ||||
|         except Exception: | ||||
|             pass | ||||
|     swap = psutil.swap_memory() | ||||
| @@ -181,6 +184,7 @@ async def get_hardware_data() -> dict: | ||||
|                     "total"      : mem.total, | ||||
|                     "used"       : mem.used, | ||||
|                     "free"       : mem.free, | ||||
|                     "usedProcess": mem_used_process, | ||||
|             }, | ||||
|             "swap"  : { | ||||
|                     "percent": swap.percent, | ||||
| @@ -194,6 +198,8 @@ async def get_hardware_data() -> dict: | ||||
|     for disk in psutil.disk_partitions(all=True): | ||||
|         try: | ||||
|             disk_usage = psutil.disk_usage(disk.mountpoint) | ||||
|             if disk_usage.total == 0: | ||||
|                 continue    # 虚拟磁盘 | ||||
|             result["disk"].append({ | ||||
|                     "name"   : disk.mountpoint, | ||||
|                     "percent": disk_usage.percent, | ||||
|   | ||||
| @@ -10,9 +10,9 @@ require("nonebot_plugin_alconna") | ||||
| from nonebot_plugin_alconna import on_alconna, Alconna, Args, Subcommand, Arparma, UniMessage | ||||
|  | ||||
| status_alc = on_alconna( | ||||
|     aliases={"#状态"}, | ||||
|     aliases={"状态"}, | ||||
|     command=Alconna( | ||||
|         "#status", | ||||
|         "status", | ||||
|         Subcommand( | ||||
|             "memory", | ||||
|             alias={"mem", "m", "内存"}, | ||||
|   | ||||
| @@ -153,3 +153,6 @@ status.days=天 | ||||
| status.hours=时 | ||||
| status.minutes=分 | ||||
| status.seconds=秒 | ||||
| status.cores=核心 | ||||
| status.threads=线程 | ||||
| status.process=进程 | ||||
| @@ -2,6 +2,8 @@ | ||||
|     --main-text-color: #fff; | ||||
|     --sub-text-color: #bbb; | ||||
|     --tip-text-color: #888; | ||||
|  | ||||
|     --device-info-width: 240px; | ||||
| } | ||||
|  | ||||
| .bot-info { | ||||
| @@ -58,3 +60,71 @@ | ||||
|     margin: 10px 0; | ||||
|     width: 100%; | ||||
| } | ||||
|  | ||||
| #hardware-info { | ||||
|     display: flex; | ||||
|     justify-content: space-evenly; | ||||
| } | ||||
|  | ||||
| .device-info { | ||||
|     max-width: 30%; | ||||
| } | ||||
|  | ||||
| .device-chart { | ||||
|     display: flex; | ||||
|     height: var(--device-info-width); | ||||
|     width: var(--device-info-width); | ||||
|     margin-bottom: 20px; | ||||
|     justify-content: center; | ||||
| } | ||||
|  | ||||
| .device-tags { | ||||
|     text-align: center; | ||||
|     color: var(--sub-text-color); | ||||
|     font-size: 24px; | ||||
|     max-width: var(--device-info-width); | ||||
|     word-wrap: break-word; | ||||
| } | ||||
|  | ||||
| .disk-info { | ||||
|     display: flex; | ||||
|     position: relative; | ||||
|     width: 100%; | ||||
|     height: 60px; | ||||
|     background-color: #ffffff44; | ||||
|     border-radius: 30px; | ||||
|     align-items: center; | ||||
| } | ||||
|  | ||||
| .disk-usage { | ||||
|     background-color: #a2d8f4; | ||||
|     height: 100%; | ||||
|     border-radius: 30px; | ||||
|     position: absolute; | ||||
|     z-index: 1; | ||||
| } | ||||
|  | ||||
| .disk-title { | ||||
|     position: absolute; | ||||
|     color: var(--main-text-color); | ||||
|     font-size: 24px; | ||||
|     margin-bottom: 10px; | ||||
|     margin-left: 20px; | ||||
|     text-align: left; | ||||
|     z-index: 2; | ||||
| } | ||||
|  | ||||
| #motto-text { | ||||
|     font-size: 36px; | ||||
|     word-wrap: break-word; | ||||
|     color: var(--main-text-color); | ||||
|     text-align: center; | ||||
|     margin: 30px 0 10px 0; | ||||
| } | ||||
|  | ||||
| #motto-from { | ||||
|     font-size: 24px; | ||||
|     font-style: italic; | ||||
|     color: var(--sub-text-color); | ||||
|     text-align: right; | ||||
| } | ||||
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 2.6 MiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 6.1 MiB | 
										
											Binary file not shown.
										
									
								
							| Before Width: | Height: | Size: 2.6 MiB | 
| @@ -1,14 +0,0 @@ | ||||
| const bgs = [ | ||||
|     "bg1.png", | ||||
|     "bg2.png", | ||||
|     "bg3.png", | ||||
|     "bg4.png", | ||||
|     "bg5.png", | ||||
|     "bg6.png", | ||||
|     "bg7.png", | ||||
|     "bg8.png", | ||||
|     "bg9.png", | ||||
|     "bg10.png", | ||||
| ] | ||||
| // 随机选择背景图片 | ||||
| document.body.style.backgroundImage = `url(./img/${bgs[Math.floor(Math.random() * bgs.length)]})`; | ||||
| @@ -1,14 +1,11 @@ | ||||
| const bgs = [ | ||||
|     "bg1.png", | ||||
|     "bg2.png", | ||||
|     "bg3.png", | ||||
|     "bg4.png", | ||||
|     "bg5.png", | ||||
|     "bg6.png", | ||||
|     "bg7.png", | ||||
|     "bg8.png", | ||||
|     "bg9.png", | ||||
|     "bg10.png", | ||||
| ] | ||||
| // 随机选择背景图片 | ||||
| document.body.style.backgroundImage = `url(./img/${bgs[Math.floor(Math.random() * bgs.length)]})`; | ||||
|   | ||||
| @@ -6,7 +6,7 @@ const mottos = [ | ||||
|         "source": "《琵琶行》" | ||||
|     }, | ||||
|     { | ||||
|         "text": "海Memory知己,天涯若比邻。", | ||||
|         "text": "海内存知己,天涯若比邻。", | ||||
|         "author": "王勃", | ||||
|         "source": "《送杜少府之任蜀州》" | ||||
|     }, | ||||
| @@ -20,6 +20,16 @@ const mottos = [ | ||||
|         "author": "苏轼", | ||||
|         "source": "《水调歌头》" | ||||
|     }, | ||||
|     { | ||||
|         "text": "人生自古谁无死,留取丹心照汗青。", | ||||
|         "author": "文天祥", | ||||
|         "source": "《过零丁洋》" | ||||
|     }, | ||||
|     { | ||||
|         "text": "山重水复疑无路,柳暗花明又一村。", | ||||
|         "author": "陆游", | ||||
|         "source": "《游山西村》" | ||||
|     }, | ||||
|     { | ||||
|         "text": "逸一时,误一世,逸久逸久罢已龄", | ||||
|         "author": "田所浩二", | ||||
| @@ -46,8 +56,58 @@ const mottos = [ | ||||
|         "source": "轻雪文档" | ||||
|     }, | ||||
|     { | ||||
|         "text": "你知道吗,轻雪的主题是基于Vue.js开发的", | ||||
|         "text": "你知道吗,轻雪的主题是基于HTML5开发的", | ||||
|         "author": "SnowyKami", | ||||
|         "source": "轻雪文档" | ||||
|     }, | ||||
|     { | ||||
|         "text": "路漫漫其修远兮,吾将上下而求索。", | ||||
|         "author": "屈原", | ||||
|         "source": "《离骚》" | ||||
|     }, | ||||
|     { | ||||
|         "text": "千里之行,始于足下。", | ||||
|         "author": "老子", | ||||
|         "source": "《道德经》" | ||||
|     }, | ||||
|     { | ||||
|         "text": "读书破万卷,下笔如有神。", | ||||
|         "author": "杜甫", | ||||
|         "source": "《奉赠韦左丞丈二十韵》" | ||||
|     }, | ||||
|     { | ||||
|         "text": "不登高山,不知天之高也;不临深溪,不知地之厚也。", | ||||
|         "author": "荀子", | ||||
|         "source": "《劝学》" | ||||
|     }, | ||||
|     { | ||||
|         "text": "知之者不如好之者,好之者不如乐之者。", | ||||
|         "author": "孔子", | ||||
|         "source": "《论语》" | ||||
|     }, | ||||
|     { | ||||
|         "text": "天行健,君子以自强不息;地势坤,君子以厚德载物。", | ||||
|         "source": "《易经》", | ||||
|         "author": "" | ||||
|     }, | ||||
|     { | ||||
|         "text": "书山有路勤为径,学海无涯苦作舟。", | ||||
|         "author": "韩愈", | ||||
|         "source": "《读书有感》" | ||||
|     }, | ||||
|     { | ||||
|         "text": "前事不忘,后事之师。", | ||||
|         "author": "孔子", | ||||
|         "source": "《论语》" | ||||
|     }, | ||||
|     { | ||||
|         "text": "志当存高远,若樽俎断绝。", | ||||
|         "author": "陶渊明", | ||||
|         "source": "《饮酒》" | ||||
|     }, | ||||
|     { | ||||
|         "text": "不以物喜,不以己悲。", | ||||
|         "author": "傅子", | ||||
|         "source": "《傅子》" | ||||
|     } | ||||
| ] | ||||
| @@ -19,7 +19,8 @@ function createPieChartOption(title, data) { | ||||
|             top: 'center', | ||||
|             textStyle: { | ||||
|                 color: '#fff', | ||||
|                 fontSize: 30 | ||||
|                 fontSize: 30, | ||||
|                 lineHeight: 36 | ||||
|             } | ||||
|         }, | ||||
|         tooltip: { | ||||
| @@ -90,8 +91,17 @@ function convertSize(size, precision = 2, addUnit = true, suffix = " XiB") { | ||||
|  * @param title | ||||
|  * @param percent 数据 | ||||
|  */ | ||||
| function createBarChartOption(title, percent) { | ||||
| function createBarChart(title, percent) { | ||||
|     // percent为百分比,最大值为100 | ||||
|     let diskDiv = document.createElement('div') | ||||
|     diskDiv.setAttribute('class', 'disk-info') | ||||
|     diskDiv.style.marginBottom = '20px' | ||||
|     diskDiv.innerHTML = ` | ||||
|         <div class="disk-title">${title}</div> | ||||
|         <div class="disk-usage" style="width: ${percent}%"></div> | ||||
|     ` | ||||
|  | ||||
|     return diskDiv | ||||
| } | ||||
|  | ||||
| function secondsToTextTime(seconds) { | ||||
| @@ -112,7 +122,7 @@ function main() { | ||||
|  | ||||
|             // 设置机器人信息 | ||||
|             botInfoDiv.className = 'info-box bot-info' | ||||
|             console.log(botInfoDiv.querySelector('.bot-icon-img')) | ||||
|  | ||||
|             botInfoDiv.querySelector('.bot-icon-img').setAttribute('src', bot['icon']) | ||||
|             botInfoDiv.querySelector('.bot-name').innerText = bot['name'] | ||||
|             let tagArray = [ | ||||
| @@ -144,7 +154,7 @@ function main() { | ||||
|     liteyukiInfoDiv.className = 'info-box bot-info' | ||||
|     liteyukiInfoDiv.querySelector('.bot-icon-img').setAttribute('src', './img/liteyuki.png') | ||||
|     liteyukiInfoDiv.querySelector('.bot-name').innerText = liteyukiData['name'] | ||||
|     console.log(liteyukiData) | ||||
|  | ||||
|     let tagArray = [ | ||||
|         `Liteyuki ${liteyukiData['version']}`, | ||||
|         `Nonebot ${liteyukiData['nonebot']}`, | ||||
| @@ -173,12 +183,12 @@ function main() { | ||||
|  | ||||
|     const cpuTagArray = [ | ||||
|         cpuData['name'], | ||||
|         `${cpuData['cores']}C ${cpuData['threads']}T`, | ||||
|         `${cpuData['cores']}${localData['cores']} ${cpuData['threads']}${localData['threads']}`, | ||||
|         `${(cpuData['freq'] / 1000).toFixed(2)}GHz` | ||||
|     ] | ||||
|  | ||||
|     const memTagArray = [ | ||||
|         `Bot ${convertSize(memData['bot'])}`, | ||||
|         `${localData['process']} ${convertSize(memData['usedProcess'])}`, | ||||
|         `${localData['used']} ${convertSize(memData['used'])}`, | ||||
|         `${localData['free']} ${convertSize(memData['free'])}`, | ||||
|         `${localData['total']} ${convertSize(memData['total'])}` | ||||
| @@ -189,7 +199,96 @@ function main() { | ||||
|         `${localData['free']} ${convertSize(swapData['free'])}`, | ||||
|         `${localData['total']} ${convertSize(swapData['total'])}` | ||||
|     ] | ||||
|     console.log(cpuTagArray, memTagArray, swapTagArray) | ||||
|     let cpuDeviceInfoDiv = document.importNode(document.getElementById('device-info').content, true) | ||||
|     let memDeviceInfoDiv = document.importNode(document.getElementById('device-info').content, true) | ||||
|     let swapDeviceInfoDiv = document.importNode(document.getElementById('device-info').content, true) | ||||
|  | ||||
|     cpuDeviceInfoDiv.querySelector('.device-info').setAttribute('id', 'cpu-info') | ||||
|     memDeviceInfoDiv.querySelector('.device-info').setAttribute('id', 'mem-info') | ||||
|     swapDeviceInfoDiv.querySelector('.device-info').setAttribute('id', 'swap-info') | ||||
|     cpuDeviceInfoDiv.querySelector('.device-chart').setAttribute('id', 'cpu-chart') | ||||
|     memDeviceInfoDiv.querySelector('.device-chart').setAttribute('id', 'mem-chart') | ||||
|     swapDeviceInfoDiv.querySelector('.device-chart').setAttribute('id', 'swap-chart') | ||||
|  | ||||
|     let devices = { | ||||
|         'cpu': cpuDeviceInfoDiv, | ||||
|         'mem': memDeviceInfoDiv, | ||||
|         'swap': swapDeviceInfoDiv | ||||
|     } | ||||
|     // 遍历添加标签 | ||||
|     for (let device in devices) { | ||||
|         let tagArray = [] | ||||
|         switch (device) { | ||||
|             case 'cpu': | ||||
|                 tagArray = cpuTagArray | ||||
|                 break | ||||
|             case 'mem': | ||||
|                 tagArray = memTagArray | ||||
|                 break | ||||
|             case 'swap': | ||||
|                 tagArray = swapTagArray | ||||
|                 break | ||||
|         } | ||||
|         tagArray.forEach( | ||||
|             (tag, index) => { | ||||
|                 let tagDiv = document.createElement('div') | ||||
|                 tagDiv.className = 'device-tag' | ||||
|                 tagDiv.innerText = tag | ||||
|                 // 给最后一个标签不添加后缀 | ||||
|                 tagDiv.setAttribute('suffix', index === tagArray.length - 1 ? '0' : '1') | ||||
|                 devices[device].querySelector('.device-tags').appendChild(tagDiv) | ||||
|             } | ||||
|         ) | ||||
|     } | ||||
|  | ||||
|  | ||||
|     // 插入 | ||||
|     document.getElementById('hardware-info').appendChild(cpuDeviceInfoDiv) | ||||
|     document.getElementById('hardware-info').appendChild(memDeviceInfoDiv) | ||||
|     document.getElementById('hardware-info').appendChild(swapDeviceInfoDiv) | ||||
|  | ||||
|     let cpuChart = echarts.init(document.getElementById('cpu-chart')) | ||||
|     let memChart = echarts.init(document.getElementById('mem-chart')) | ||||
|     let swapChart = echarts.init(document.getElementById('swap-chart')) | ||||
|  | ||||
|  | ||||
|     cpuChart.setOption(createPieChartOption(`${localData['cpu']}\n${cpuData['percent'].toFixed(1)}%`, [ | ||||
|         {name: 'used', value: cpuData['percent']}, | ||||
|         {name: 'free', value: 100 - cpuData['percent']} | ||||
|     ])) | ||||
|  | ||||
|     memChart.setOption(createPieChartOption(`${localData['memory']}\n${memData['percent'].toFixed(1)}%`, [ | ||||
|         {name: 'process', value: memData['usedProcess']}, | ||||
|         {name: 'used', value: memData['used'] - memData['usedProcess']}, | ||||
|         {name: 'free', value: memData['free']} | ||||
|     ])) | ||||
|  | ||||
|  | ||||
|     swapChart.setOption(createPieChartOption(`${localData['swap']}\n${swapData['percent'].toFixed(1)}%`, [ | ||||
|         {name: 'used', value: swapData['used']}, | ||||
|         {name: 'free', value: swapData['free']} | ||||
|     ])) | ||||
|  | ||||
|  | ||||
|     // 磁盘信息 | ||||
|     const diskData = hardwareData['disk'] | ||||
|     diskData.forEach( | ||||
|         (disk) => { | ||||
|             let diskTitle = `${disk['name']}  ${localData['free']} ${convertSize(disk['free'])}  ${localData['total']} ${convertSize(disk['total'])}` | ||||
|             // 最后一个把margin-bottom去掉 | ||||
|             let diskDiv = createBarChart(diskTitle, disk['percent']) | ||||
|             if (disk === diskData[diskData.length - 1]) { | ||||
|                 diskDiv.style.marginBottom = '0' | ||||
|             } | ||||
|             document.getElementById('disk-info').appendChild(createBarChart(diskTitle, disk['percent'])) | ||||
|         }) | ||||
|     // 随机一言 | ||||
|     let motto = mottos[Math.floor(Math.random() * mottos.length)] | ||||
|     let mottoText = motto['text'] | ||||
|     let mottoFrom = `${motto['author']} ${motto['source']}` | ||||
|     document.getElementById('motto-text').innerText = mottoText | ||||
|     document.getElementById('motto-from').innerText = mottoFrom | ||||
|  | ||||
|  | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -6,6 +6,7 @@ | ||||
|     <link rel="stylesheet" href="./css/card.css"> | ||||
|     <link rel="stylesheet" href="./css/status.css"> | ||||
|     <link rel="stylesheet" href="./css/fonts.css"> | ||||
|  | ||||
| </head> | ||||
| <body> | ||||
| <template id="bot-template"> | ||||
| @@ -25,11 +26,24 @@ | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <template id="device-info"> | ||||
|     <div class="device-info"> | ||||
|         <div class="device-chart"> | ||||
|         </div> | ||||
|         <div class="device-tags"> | ||||
|         </div> | ||||
|     </div> | ||||
| </template> | ||||
|  | ||||
| <div class="data-storage" id="data">{{ data | tojson }}</div> | ||||
| <div class="info-box" id="hardware-info"></div> | ||||
| <div class="info-box" id="disk-info"></div> | ||||
| <div class="info-box" id="motto-info"></div> | ||||
|  | ||||
| <div class="info-box" id="motto-info"> | ||||
|     <div id="motto-text"></div> | ||||
|     <div id="motto-from"></div> | ||||
| </div> | ||||
| <script src="https://cdnjs.cloudflare.com/ajax/libs/echarts/4.3.0/echarts.min.js"></script> | ||||
| <script src="./js/motto.js"></script> | ||||
| <script src="./js/card.js"></script> | ||||
| <script src="./js/status.js"></script> | ||||
| </body> | ||||
|   | ||||
		Reference in New Issue
	
	Block a user