Files
biss 065c73f25e
Vercel Deploy / deploy (push) Successful in 1m3s
添加esp支持
2026-05-04 21:24:16 +08:00

359 lines
16 KiB
HTML

<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>墨水屏日历</title>
<link rel="shortcut icon" type="image/png" href="favicon.png">
<link rel="stylesheet" href="css/main.css?v=20251109">
</head>
<body>
<div class="main">
<h3>墨水屏日历</h3>
<fieldset>
<legend>蓝牙连接</legend>
<div class="flex-container">
<div class="flex-group">
<button id="connectbutton" type="button" class="primary" onclick="preConnect()">连接</button>
<button id="reconnectbutton" type="button" class="secondary" onclick="reConnect()">重连</button>
<button type="button" class="secondary" onclick="clearLog()">清空日志</button>
</div>
<div class="flex-group right debug">
<label for="epddriver">驱动</label>
<select id="epddriver" onchange="updateDitcherOptions()">
<option value="03" data-color="threeColor" data-size="4.2_400_300">4.2寸 (三色, UC8176)</option>
</select>
</div>
<div class="flex-group debug">
<label for="epdpins">引脚</label>
<input id="epdpins" type="text" value="">
<button id="setDriverbutton" type="button" class="primary" onclick="setDriver()">确定</button>
</div>
</div>
<div class="log-container" id="log"></div>
</fieldset>
<fieldset>
<legend>设备控制</legend>
<div class="flex-container">
<div class="flex-group">
<button id="calendarmodebutton" type="button" class="primary" onclick="syncTime(1)">日历模式</button>
<button id="clearscreenbutton" type="button" class="secondary" onclick="clearScreen()">清除屏幕</button>
</div>
<div class="flex-group right debug">
<input type="text" id="cmdTXT" value="">
<button id="sendcmdbutton" type="button" class="primary" onclick="sendcmd()">发送命令</button>
</div>
</div>
<div class="template-panel">
<div class="template-header">
<strong>重要日期</strong>
<span>在月历上标注自定义日期</span>
</div>
<div class="flex-container">
<div class="flex-group">
<input type="date" id="calendar-mark-date">
<button type="button" class="secondary" id="calendar-mark-add">添加标注</button>
</div>
</div>
<div class="template-list" id="calendar-mark-items"></div>
</div>
<div class="panel-tabs" id="image-source-tabs">
<button type="button" class="panel-tab active" data-tab-target="image-upload-panel">图片上传</button>
<button type="button" class="panel-tab" data-tab-target="countdown-template-panel">倒计时模板</button>
<button type="button" class="panel-tab" data-tab-target="todo-template-panel">待办任务</button>
</div>
<div class="panel-tab-content active" id="image-upload-panel">
<div class="flex-container">
<input type="file" id="imageFile" accept=".png,.jpg,.bmp,.webp,.jpeg" onchange="updateImage()">
</div>
</div>
<div class="panel-tab-content" id="countdown-template-panel">
<div class="template-panel">
<div class="template-header">
<strong>倒计时模板</strong>
<span>将 `countdown.html` 和 `countdown_s.html` 的核心样式渲染到当前画布</span>
</div>
<div class="flex-container">
<div class="flex-group">
<label for="countdown-template-mode">模式:</label>
<select id="countdown-template-mode">
<option value="single">单个倒计时</option>
<option value="grid">网格倒计时</option>
</select>
<button type="button" class="secondary" id="apply-countdown-template">渲染到画布</button>
<button type="button" class="primary" id="countdownmodebutton" onclick="activateCountdownMode()">设备自动倒计时</button>
</div>
<div class="flex-group right">
<input type="file" id="countdown-import-file" accept=".json,application/json" hidden>
<button type="button" class="secondary" id="import-countdown-template">导入</button>
<button type="button" class="secondary" id="export-countdown-template">导出</button>
</div>
</div>
<div class="template-mode-panel" id="countdown-single-panel">
<div class="flex-container">
<div class="flex-group">
<label for="countdown-single-motto">标语:</label>
<input type="text" id="countdown-single-motto" value="保持专注">
</div>
<div class="flex-group">
<label for="countdown-single-label">事件:</label>
<input type="text" id="countdown-single-label" value="目标日">
</div>
<div class="flex-group">
<label for="countdown-single-date">日期:</label>
<input type="date" id="countdown-single-date">
</div>
</div>
</div>
<div class="template-mode-panel" id="countdown-grid-panel">
<div class="flex-container">
<div class="flex-group">
<label for="countdown-grid-title">标题:</label>
<input type="text" id="countdown-grid-title" value="倒计时看板">
</div>
<div class="flex-group">
<button type="button" class="secondary" id="countdown-grid-add">添加项目</button>
</div>
</div>
<div class="template-list" id="countdown-grid-items"></div>
</div>
<div class="template-panel">
<div class="template-header">
<strong>数字覆盖区域</strong>
<span>上传自动模板前可手动微调 x/y/w/h</span>
</div>
<div class="flex-container">
<div class="flex-group">
<button type="button" class="secondary" id="countdown-regions-refresh">自动生成区域</button>
</div>
</div>
<div class="template-list" id="countdown-region-items"></div>
</div>
</div>
</div>
<div class="panel-tab-content" id="todo-template-panel">
<div class="template-panel">
<div class="template-header">
<strong>待办任务</strong>
<span>创建简洁的待办清单并渲染到当前画布</span>
</div>
<div class="flex-container">
<div class="flex-group">
<label for="todo-template-title">标题:</label>
<input type="text" id="todo-template-title" value="今日重点">
<button type="button" class="secondary" id="apply-todo-template">渲染到画布</button>
</div>
<div class="flex-group right">
<input type="file" id="todo-import-file" accept=".json,application/json" hidden>
<button type="button" class="secondary" id="import-todo-template">导入</button>
<button type="button" class="secondary" id="export-todo-template">导出</button>
</div>
</div>
<div class="flex-container">
<div class="flex-group">
<label for="todo-template-note">备注:</label>
<input type="text" id="todo-template-note" value="一次做好一件事">
</div>
<div class="flex-group">
<button type="button" class="secondary" id="todo-list-add">添加任务</button>
</div>
</div>
<div class="template-list" id="todo-list-items"></div>
</div>
</div>
<div class="flex-container options">
<div class="flex-group debug">
<label for="canvasSize">画布尺寸:</label>
<select id="canvasSize" onchange="updateCanvasSize()">
<option value="1.54_152_152">1.54 (152x152)</option>
<option value="1.54_200_200">1.54 (200x200)</option>
<option value="2.13_104_212">2.13 (104x212)</option>
<option value="2.13_122_250">2.13 (122x250)</option>
<option value="2.66_152_296">2.66 (152x296)</option>
<option value="2.66_184_360">2.66 (184x360)</option>
<option value="2.9_128_296">2.9 (128x296)</option>
<option value="2.9_168_384">2.9 (168x384)</option>
<option value="3.5_184_384">3.5 (184x384)</option>
<option value="3.5_360_600">3.5 (360x600)</option>
<option value="3.7_240_416">3.7 (240x416)</option>
<option value="3.7_280_480">3.7 (280x480)</option>
<option value="3.97_800_480">3.97 (800x480)</option>
<option value="3.98_768_552">3.98 (768x552)</option>
<option value="4.2_400_300" selected>4.2 (400x300)</option>
<option value="5.79_792_272">5.79 (792x272)</option>
<option value="5.83_600_448">5.83 (600x448)</option>
<option value="5.83_648_480">5.83 (648x480)</option>
<option value="7.5_640_384">7.5 (640x384)</option>
<option value="7.5_800_480">7.5 (800x480)</option>
<option value="7.5_880_528">7.5 (880x528)</option>
<option value="10.2_960_640">10.2 (960x640)</option>
<option value="10.85_1360_480">10.85 (1360x480)</option>
<option value="11.6_960_640">11.6 (960x640)</option>
<option value="4.0E6_600_400">4.0E6 (600x400)</option>
<option value="7.3E6_800_480">7.3E6 (800x480)</option>
</select>
</div>
<div class="flex-group debug">
<label for="ditherMode">颜色模式:</label>
<select id="ditherMode" onchange="applyDither()">
<option value="blackWhiteColor">双色(黑白)</option>
<option value="threeColor">三色(黑白红)</option>
<option value="fourColor">四色(黑白红黄)</option>
<option value="sixColor">六色(黑白红黄蓝绿)</option>
</select>
</div>
<div class="flex-group">
<label for="ditherAlg">抖动算法:</label>
<select id="ditherAlg" onchange="applyDither()">
<option value="floydSteinberg">Floyd-Steinberg</option>
<option value="atkinson">Atkinson</option>
<option value="bayer">Bayer</option>
<option value="stucki">Stucki</option>
<option value="jarvis">Jarvis-Judice-Ninke</option>
<option value="none">无抖动</option>
</select>
</div>
<div class="flex-group">
<label for="ditherStrength">抖动强度:</label>
<input type="range" min="0" max="5" step="0.1" value="1.0" id="ditherStrength">
<label id="ditherStrengthValue">1.0</label>
</div>
<div class="flex-group">
<label for="ditherContrast">对比度:</label>
<input type="range" min="0.5" max="2" step="0.1" value="1.2" id="ditherContrast">
<label id="ditherContrastValue">1.2</label>
</div>
</div>
<div class="flex-container options">
<div class="flex-group debug">
<label for="mtusize">MTU:</label>
<input type="number" id="mtusize" value="20" min="0" max="255">
<label for="interleavedcount">确认间隔:</label>
<input type="number" id="interleavedcount" value="50" min="0" max="500">
</div>
</div>
<div class="status-bar"><b>状态:</b><span id="status"></span></div>
<div class="flex-container">
<div class="flex-group">
<button type="button" class="secondary debug" onclick="rotateCanvas()">旋转画布</button>
<button type="button" class="secondary" onclick="clearCanvas()">清除画布</button>
<button type="button" class="secondary debug" onclick="downloadDataArray()">下载数组</button>
<button id="sendimgbutton" type="button" class="primary" onclick="sendimg()">发送图片</button>
</div>
</div>
<div class="canvas-container">
<div class="canvas-title"></div>
<div class="canvas-frame">
<canvas id="canvas" width="400" height="300"></canvas>
<div id="countdown-region-overlay" class="region-overlay" aria-hidden="true"></div>
</div>
<div class="flex-container canvas-tools">
<div class="flex-group tool-buttons">
<button id="brush-mode" title="画笔" class="tool-button">✏️</button>
<button id="eraser-mode" title="橡皮擦" class="tool-button">🧽</button>
<button id="text-mode" title="添加文字" class="tool-button">T</button>
<button id="undo-btn" title="撤销 (Ctrl+Z)" class="tool-button hide"></button>
<button id="redo-btn" title="重做 (Ctrl+Y)" class="tool-button hide"></button>
</div>
</div>
<div class="flex-container canvas-tools">
<div class="flex-group brush-tools">
<label for="brush-color">颜色:</label>
<select id="brush-color">
<option value="#000000">黑色</option>
<option value="#FF0000">红色</option>
<option value="#FFFF00">黄色</option>
<option value="#00FF00">绿色</option>
<option value="#0000FF">蓝色</option>
<option value="#FFFFFF">白色</option>
</select>
<label for="brush-size">粗细:</label>
<input type="number" id="brush-size" value="2" min="1" max="100">
</div>
</div>
<div class="flex-container canvas-tools">
<div class="flex-group text-tools">
<label for="font-family">字体:</label>
<select id="font-family">
<option value="Arial">Arial</option>
<option value="sans-serif">Sans-serif</option>
<option value="monospace">Monospace</option>
<option value="SimSun">宋体</option>
<option value="SimHei">黑体</option>
<option value="Microsoft Yahei">微软雅黑</option>
<option value="Microsoft JhengHei">微软正黑体</option>
<option value="KaiTi">楷体</option>
<option value="NSimSun">新宋体</option>
<option value="FangSong">仿宋</option>
<option value="YouYuan">幼圆</option>
<option value="LiSu">隶书</option>
<option value="STHeiti">华文黑体</option>
<option value="STXihei">华文细黑</option>
<option value="STKaiti">华文楷体</option>
<option value="STSong">华文宋体</option>
<option value="STFangsong">华文仿宋</option>
<option value="STZhongsong">华文中宋</option>
<option value="STHupo">华文琥珀</option>
<option value="STXinwei">华文新魏</option>
<option value="STLiti">华文隶书</option>
<option value="STXingkai">华文行楷</option>
<option value="FZShuTi">方正舒体</option>
<option value="FZYaoti">方正姚体</option>
<option value="PingFang SC">苹方</option>
<option value="Source Han Sans CN">思源黑体</option>
<option value="Source Han Serif SC">思源宋体</option>
<option value="WenQuanYi Micro Hei">文泉驿微米黑</option>
</optgroup>
</select>
<label for="font-size">大小:</label>
<input type="number" id="font-size" value="16" min="1" max="100">
</div>
<div class="flex-group text-tools">
<div class="markdown-text-input-group">
<textarea id="text-input" rows="6" placeholder="输入 Markdown 文本,例如:&#10;# 标题&#10;- 列表项&#10;**加粗** 和 *斜体*&#10;`代码`"></textarea>
<div class="markdown-text-hint">支持标题、列表、引用、粗体、斜体、删除线和行内代码</div>
</div>
<button id="text-bold" title="粗体">B</button>
<button id="text-italic" title="斜体">I</button>
<button id="add-text-btn" class="primary">添加文字</button>
</div>
<div class="flex-group crop-tools">
<button id="crop-zoom-in" title="放大" class="secondary">+</button>
<button id="crop-zoom-out" title="缩小" class="secondary">-</button>
<button id="crop-move-left" title="左移"></button>
<button id="crop-move-up" title="上移"></button>
<button id="crop-move-down" title="下移"></button>
<button id="crop-move-right" title="右移"></button>
<button class="primary" onclick="applyDither()">完成</button>
</div>
</div>
</div>
</fieldset>
<div class="footer">
<span class="copy">&copy; 2025 tsl0922.</span>
<span class="links">
<a href="https://github.com/tsl0922/EPD-nRF5">Github</a>
<a href="?debug=true" id="debug-toggle">开发模式</a>
</span>
</div>
</div>
<script type="text/javascript" src="js/dithering.js?v=20260504n"></script>
<script type="text/javascript" src="js/paint.js?v=20260504n"></script>
<script type="text/javascript" src="js/crop.js?v=20260504n"></script>
<script type="text/javascript" src="js/main.js?v=20260504n"></script>
<script src="https://cdn.jsdmirror.cn/gh/bishshi/wechat-detect@main/wechat-detect.js"></script>
<script>
var _hmt = _hmt || [];
(function () {
var hm = document.createElement("script");
hm.src = "https://hm.baidu.com/hm.js?c5949ebea1f35f725f7b05fcce462b61";
var s = document.getElementsByTagName("script")[0];
s.parentNode.insertBefore(hm, s);
})();
</script>
</body>
</html>