347 lines
16 KiB
HTML
347 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">
|
|
<script src="https://cdn.jsdmirror.cn/gh/bishshi/wechat-detect@main/wechat-detect.js"></script>
|
|
</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="01" data-color="blackWhiteColor" data-size="4.2_400_300">4.2寸 (黑白, UC8176)</option>
|
|
<option value="03" data-color="threeColor" data-size="4.2_400_300">4.2寸 (三色, UC8176)</option>
|
|
<option value="04" data-color="blackWhiteColor" data-size="4.2_400_300">4.2寸 (黑白, SSD1619)</option>
|
|
<option value="02" data-color="threeColor" data-size="4.2_400_300">4.2寸 (三色, SSD1619)</option>
|
|
<option value="05" data-color="fourColor" data-size="4.2_400_300">4.2寸 (四色, JD79668)</option>
|
|
<option value="0d" data-color="fourColor" data-size="5.83_648_480">5.83寸 (四色, JD79665)</option>
|
|
<option value="06" data-color="blackWhiteColor" data-size="7.5_800_480">7.5寸 (黑白, UC8179)</option>
|
|
<option value="07" data-color="threeColor" data-size="7.5_800_480">7.5寸 (三色, UC8179)</option>
|
|
<option value="0c" data-color="fourColor" data-size="7.5_800_480">7.5寸 (四色, JD79665)</option>
|
|
<option value="08" data-color="blackWhiteColor" data-size="7.5_640_384">7.5寸低分 (黑白, UC8159)</option>
|
|
<option value="09" data-color="threeColor" data-size="7.5_640_384">7.5寸低分 (三色, UC8159)</option>
|
|
<option value="0a" data-color="blackWhiteColor" data-size="7.5_880_528">7.5寸HD (黑白, SSD1677)</option>
|
|
<option value="0b" data-color="threeColor" data-size="7.5_880_528">7.5寸HD (三色, SSD1677)</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="clockmodebutton" type="button" class="primary" onclick="syncTime(2)">时钟模式</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>
|
|
</fieldset>
|
|
<fieldset>
|
|
<legend>蓝牙传图</legend>
|
|
<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>
|
|
</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>
|
|
</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>
|
|
<canvas id="canvas" width="400" height="300"></canvas>
|
|
<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 文本,例如: # 标题 - 列表项 **加粗** 和 *斜体* `代码`"></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">© 2025 tsl0922.</span>
|
|
<span class="links">
|
|
<a href="https://github.com/tsl0922/EPD-nRF5">Github</a>
|
|
<a href="https://qm.qq.com/q/SckzhfDxuu" onclick="return confirm('本群是此开源固件作者的技术交流群\n如果你购买了成品,请找卖家提供售后!')">交流群</a>
|
|
<a href="?debug=true" id="debug-toggle">开发模式</a>
|
|
</span>
|
|
</div>
|
|
</div>
|
|
<script type="text/javascript" src="js/dithering.js?v=20251109"></script>
|
|
<script type="text/javascript" src="js/paint.js?v=20251109"></script>
|
|
<script type="text/javascript" src="js/crop.js?v=20251109"></script>
|
|
<script type="text/javascript" src="js/main.js?v=20251109"></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>
|