
<p>一个完整且实用的番茄工作法计时器(Pomodoro Timer)Web应用源码及开发详解。</p>
<p><strong>在线演示:</strong> [此处替换为你的在线演示链接] <strong>完整源码:</strong> [此处替换为你的GitHub仓库链接或Gist链接]</p>
<section>
<h2>项目概述与核心功能</h2>
<p>本次我们构建一个经典的番茄工作法计时器应用,番茄工作法是一种高效的时间管理技巧,核心是将工作时间划分为25分钟的专注时段(称为一个“番茄钟”)和5分钟的短休息时段,每完成4个番茄钟后进行一次15-20分钟的长休息,我们的Web应用将实现以下核心功能:</p>
<ul>
<li><strong>可配置计时:</strong> 允许用户自定义工作、短休息和长休息的时间长度(默认25分钟/5分钟/15分钟)。</li>
<li><strong>阶段循环:</strong> 自动在工作 -> 短休息 -> 工作 -> ... -> 长休息的循环中切换。</li>
<li><strong>直观显示:</strong> 清晰展示当前阶段(工作/休息)、倒计时时间、进度环动画。</li>
<li><strong>控制操作:</strong> 开始、暂停、重置计时器。</li>
<li><strong>任务追踪(可选):</strong> 允许用户输入当前专注的任务名称。</li>
<li><strong>状态持久化:</strong> 使用浏览器本地存储(LocalStorage)保存用户配置和当前状态(即使刷新页面也能恢复)。</li>
<li><strong>响应式设计:</strong> 适配不同屏幕尺寸的设备。</li>
</ul>
</section>
<section>
<h2>技术栈选择</h2>
<p>为了简洁高效并专注于核心逻辑,我们选择纯前端技术栈:</p>
<ul>
<li><strong>HTML5:</strong> 构建页面结构和语义。</li>
<li><strong>CSS3 (含动画):</strong> 实现界面样式、布局和进度环动画。</li>
<li><strong>JavaScript (ES6+):</strong> 处理所有计时逻辑、状态管理、用户交互和数据持久化。</li>
<li><strong>Web APIs:</strong> 主要利用 `setInterval` / `clearInterval` 进行计时,`localStorage` 进行数据存储。</li>
</ul>
<p><strong>为什么选择纯前端?</strong> 对于这类轻量级、以交互和状态管理为核心的工具型应用,纯前端方案部署简单、访问快速、无需后端服务器,非常适合初学者理解和实践。</p>
</section>
<section>
<h2>开发步骤详解</h2>
<h3>1. 构建HTML结构 (index.html)</h3>
<p>创建清晰、语义化的结构,为CSS和JS操作提供锚点。</p>
<pre><code><!DOCTYPE html>
<html lang="zh-CN">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>番茄工作法计时器</title>
<link rel="stylesheet" href="style.css">
</head>
<body>
<div class="container">
<h1>番茄时钟</h1>
<div class="progress-ring">
<svg width="200" height="200" viewBox="0 0 200 200">
<circle class="progress-ring__circle" stroke="#e0e0e0" stroke-width="10" fill="transparent" r="90" cx="100" cy="100"/>
<circle class="progress-ring__circle progress-ring__circle--progress" stroke="#4CAF50" stroke-width="10" fill="transparent" r="90" cx="100" cy="100" stroke-dasharray="565.48" stroke-dashoffset="0"/>
</svg>
<div class="timer-display">25:00</div>
<div class="current-status">准备开始工作</div>
</div>
<div class="task-input">
<input type="text" id="taskName" placeholder="输入当前任务...">
</div>
<div class="controls">
<button id="startPauseBtn">开始</button>
<button id="resetBtn">重置</button>
</div>
<div class="settings">
<h3>设置</h3>
<label>工作时间 (分钟): <input type="number" id="workDuration" min="1" value="25"></label>
<label>短休息 (分钟): <input type="number" id="shortBreakDuration" min="1" value="5"></label>
<label>长休息 (分钟): <input type="number" id="longBreakDuration" min="1" value="15"></label>
<label>番茄钟数: <input type="number" id="pomodorosBeforeLongBreak" min="1" value="4"></label>
<button id="saveSettingsBtn">保存设置</button>
</div>
</div>
<script src="script.js"></script>
</body>
</html></code></pre>
<p><strong>关键点:</strong> 使用SVG绘制两个圆形构成进度环;`timer-display`和`current-status`用于动态更新时间和状态;控制按钮;设置区域输入框及保存按钮。</p>
<h3>2. 设计样式与动画 (style.css)</h3>
<p>实现美观、响应式的界面和流畅的进度环动画。</p>
<pre><code> { box-sizing: border-box; margin: 0; padding: 0; }
body { font-family: 'Segoe UI', sans-serif; background: #f5f5f5; color: #333; min-height: 100vh; display: flex; justify-content: center; align-items: center; }
.container { background: white; border-radius: 20px; box-shadow: 0 10px 30px rgba(0, 0, 0, 0.1); padding: 2rem; max-width: 500px; width: 90%; text-align: center; }
h1 { margin-bottom: 1.5rem; color: #2c3e50; }
.progress-ring { position: relative; margin: 0 auto 1.5rem; width: 200px; height: 200px; }
.timer-display { position: absolute; top: 50%; left: 50%; transform: translate(-50%, -50%); font-size: 3rem; font-weight: bold; color: #2c3e50; }
.current-status { margin-bottom: 1.5rem; font-size: 1.2rem; color: #7f8c8d; }
.task-input { margin-bottom: 1.5rem; }
.task-input input { width: 80%; padding: 0.8rem; border: 2px solid #ddd; border-radius: 50px; font-size: 1rem; text-align: center; outline: none; transition: border-color 0.3s; }
.task-input input:focus { border-color: #4CAF50; }
.controls button { background: #4CAF50; color: white; border: none; padding: 0.8rem 1.5rem; margin: 0 0.5rem; border-radius: 50px; font-size: 1rem; cursor: pointer; transition: background 0.3s, transform 0.1s; }
.controls button:hover { background: #43A047; }
.controls button:active { transform: scale(0.98); }
#resetBtn { background: #e74c3c; }
#resetBtn:hover { background: #c0392b; }
.settings { margin-top: 1.5rem; padding-top: 1.5rem; border-top: 1px solid #eee; text-align: left; }
.settings h3 { margin-bottom: 1rem; color: #2c3e50; text-align: center; }
.settings label { display: block; margin-bottom: 0.8rem; }
.settings input[type="number"] { width: 60px; padding: 0.3rem; border: 1px solid #ddd; border-radius: 4px; text-align: center; float: right; }
#saveSettingsBtn { display: block; width: 100%; background: #3498db; color: white; border: none; padding: 0.8rem; border-radius: 50px; font-size: 1rem; cursor: pointer; margin-top: 1rem; transition: background 0.3s; }
#saveSettingsBtn:hover { background: #2980b9; }
/ 状态颜色 /
.working .progress-ring__circle--progress { stroke: #4CAF50; } / 工作状态-绿色 /
.short-break .progress-ring__circle--progress { stroke: #3498db; } / 短休息-蓝色 /
.long-break .progress-ring__circle--progress { stroke: #9b59b6; } / 长休息-紫色 /
</code></pre>
<p><strong>关键点:</strong> 使用Flexbox居中布局;进度环使用绝对定位叠加时间显示;按钮设计有悬停和点击反馈;设置项标签清晰;通过CSS类(`.working`, `.short-break`, `.long-break`)控制不同状态下的进度环颜色。</p>
<h3>3. 实现核心逻辑 (script.js)</h3>
<p>这是应用的大脑,处理计时、状态转换、用户交互和数据存储。</p>
<pre><code>// DOM元素引用
const timerDisplay = document.querySelector('.timer-display');
const currentStatus = document.querySelector('.current-status');
const startPauseBtn = document.getElementById('startPauseBtn');
const resetBtn = document.getElementById('resetBtn');
const taskInput = document.getElementById('taskName');
const workDurationInput = document.getElementById('workDuration');
const shortBreakDurationInput = document.getElementById('shortBreakDuration');
const longBreakDurationInput = document.getElementById('longBreakDuration');
const pomodorosBeforeLongBreakInput = document.getElementById('pomodorosBeforeLongBreak');
const saveSettingsBtn = document.getElementById('saveSettingsBtn');
const progressCircle = document.querySelector('.progress-ring__circle--progress');
const container = document.querySelector('.container');
// 应用状态变量
let timerInterval = null;
let timeLeft = 0; // 当前阶段剩余时间(秒)
let isRunning = false;
let currentPhase = 'work'; // 'work', 'shortBreak', 'longBreak'
let completedPomodoros = 0; // 已完成的工作番茄钟计数
const circumference = 565.48; // 2 π r (90), 用于SVG进度计算
// 默认配置(会被localStorage覆盖)
let config = {
workDuration: 25 60, // 秒
shortBreakDuration: 5 60,
longBreakDuration: 15 60,
pomodorosBeforeLongBreak: 4
};
// 初始化:加载保存的设置和状态
function init() {
// 尝试从localStorage加载配置
const savedConfig = localStorage.getItem('pomodoroConfig');
if (savedConfig) {
try {
config = JSON.parse(savedConfig);
// 更新设置输入框的值
workDurationInput.value = config.workDuration / 60;
shortBreakDurationInput.value = config.shortBreakDuration / 60;
longBreakDurationInput.value = config.longBreakDuration / 60;
pomodorosBeforeLongBreakInput.value = config.pomodorosBeforeLongBreak;
} catch (e) {
console.error('Error parsing saved config', e);
}
}
// 尝试加载任务和状态(可选,更复杂的状态恢复)
const savedState = localStorage.getItem('pomodoroState');
if (savedState) {
try {
const state = JSON.parse(savedState);
timeLeft = state.timeLeft;
currentPhase = state.currentPhase;
completedPomodoros = state.completedPomodoros;
isRunning = state.isRunning; // 注意:恢复时通常不自动开始,需要用户点击
taskInput.value = state.taskName || '';
updateTimerDisplay(timeLeft);
updateStatusText();
setPhaseClass(); // 设置正确的进度环颜色类
updateProgressRing();
startPauseBtn.textContent = isRunning ? '暂停' : '开始'; // 更新按钮文本
} catch (e) {
console.error('Error parsing saved state', e);
}
} else {
// 没有保存状态,初始化为工作阶段
timeLeft = config.workDuration;
updateTimerDisplay(timeLeft);
updateStatusText();
}
}
// 格式化时间显示 (MM:SS)
function formatTime(seconds) {
const mins = Math.floor(seconds / 60).toString().padStart(2, '0');
const secs = (seconds % 60).toString().padStart(2, '0');
return `${mins}:${secs}`;
}
// 更新计时器显示
function updateTimerDisplay(seconds) {
timerDisplay.textContent = formatTime(seconds);
}
// 更新状态文本
function updateStatusText() {
let statusText;
switch (currentPhase) {
case 'work':
statusText = '专注工作时间';
break;
case 'shortBreak':
statusText = '短休息时间';
break;
case 'longBreak':
statusText = '长休息时间';
break;
default:
statusText = '准备开始';
}
currentStatus.textContent = statusText;
}
// 设置当前阶段的CSS类 (控制进度环颜色)
function setPhaseClass() {
container.classList.remove('working', 'short-break', 'long-break');
container.classList.add(currentPhase === 'work' ? 'working' :
(currentPhase === 'shortBreak' ? 'short-break' : 'long-break'));
}
// 更新SVG进度环
function updateProgressRing() {
let totalTime;
switch (currentPhase) {
case 'work': totalTime = config.workDuration; break;
case 'shortBreak': totalTime = config.shortBreakDuration; break;
case 'longBreak': totalTime = config.longBreakDuration; break;
}
const progressOffset = circumference - (timeLeft / totalTime) circumference;
progressCircle.style.strokeDashoffset = progressOffset;
}
// 开始倒计时
function startTimer() {
if (timerInterval) return; // 防止重复启动
isRunning = true;
startPauseBtn.textContent = '暂停';
timerInterval = setInterval(() => {
timeLeft--;
updateTimerDisplay(timeLeft);
updateProgressRing();
// 检查时间是否用完
if (timeLeft <= 0) {
clearInterval(timerInterval);
timerInterval = null;
isRunning = false;
startPauseBtn.textContent = '开始';
// 播放提示音 (这里用系统beep模拟,实际应用建议用<audio>播放文件)
console.log('x07'); // 终端可能会响,浏览器通常需要用户交互后才允许播放声音
// 更佳实践: new Audio('beep.mp3').play().catch(e => console.log('Audio play failed:', e));
// 阶段结束,进入下一阶段
switchToNextPhase();
}
saveState(); // 定期保存当前状态(可选,频率可调整)
}, 1000); // 每秒更新一次
}
// 暂停倒计时
function pauseTimer() {
if (!timerInterval) return;
clearInterval(timerInterval);
timerInterval = null;
isRunning = false;
startPauseBtn.textContent = '开始';
saveState();
}
// 重置计时器到当前阶段的初始状态
function resetTimer() {
pauseTimer(); // 先停止计时
switch (currentPhase) {
case 'work':
timeLeft = config.workDuration;
break;
case 'shortBreak':
timeLeft = config.shortBreakDuration;
break;
case 'longBreak':
timeLeft = config.longBreakDuration;
break;
}
updateTimerDisplay(timeLeft);
updateProgressRing();
saveState();
}
// 阶段结束,切换到下一阶段
function switchToNextPhase() {
if (currentPhase === 'work') {
completedPomodoros++;
// 检查是否达到长休息条件
if (completedPomodoros % config.pomodorosBeforeLongBreak === 0) {
currentPhase = 'longBreak';
timeLeft = config.longBreakDuration;
} else {
currentPhase = 'shortBreak';
timeLeft = config.shortBreakDuration;
}
} else { // 当前是shortBreak或longBreak
currentPhase = 'work';
timeLeft = config.workDuration;
}
updateTimerDisplay(timeLeft);
updateStatusText();
setPhaseClass();
updateProgressRing();
saveState();
}
// 保存当前应用状态到localStorage
function saveState() {
const state = {
timeLeft,
currentPhase,
completedPomodoros,
isRunning,
taskName: taskInput.value
};
localStorage.setItem('pomodoroState', JSON.stringify(state));
}
// 保存配置到localStorage和内存
function saveSettings() {
// 从输入框获取新值并转换为秒
config.workDuration = parseInt(workDurationInput.value) 60;
config.shortBreakDuration = parseInt(shortBreakDurationInput.value) 60;
config.longBreakDuration = parseInt(longBreakDurationInput.value) 60;
config.pomodorosBeforeLongBreak = parseInt(pomodorosBeforeLongBreakInput.value);
// 保存配置
localStorage.setItem('pomodoroConfig', JSON.stringify(config));
// 如果当前阶段时间需要根据新配置调整(可选,这里简单重置)
resetTimer(); // 重置会使用新配置的时长
alert('设置已保存并生效!');
}
// 事件监听
startPauseBtn.addEventListener('click', () => {
if (isRunning) {
pauseTimer();
} else {
startTimer();
}
});
resetBtn.addEventListener('click', resetTimer);
saveSettingsBtn.addEventListener('click', saveSettings);
// 输入框改变时实时保存任务?(可选,根据需求选择频率)
taskInput.addEventListener('input', () => {
saveState(); // 频繁输入可能太频繁,可以加防抖
});
// 初始化应用
init();</code></pre>
<p><strong>关键逻辑与专业解决方案:</strong></p>
<ul>
<li><strong>状态管理:</strong> 使用变量 (`timeLeft`, `currentPhase`, `completedPomodoros`, `isRunning`, `config`) 清晰管理应用核心状态。</li>
<li><strong>初始化与持久化:</strong> 使用 `localStorage` 保存用户配置 (`pomodoroConfig`) 和当前会话状态 (`pomodoroState`),实现刷新页面恢复,这是提升用户体验的关键。</li>
<li><strong>计时精度:</strong> 使用 `setInterval` 每秒更新一次,虽然存在微小误差,但对于分钟级计时足够,更精确方案可考虑 `requestAnimationFrame` + 时间戳差值计算。</li>
<li><strong>进度环计算:</strong> 利用SVG `stroke-dasharray` 和 `stroke-dashoffset` 属性,根据剩余时间比例动态计算偏移量,实现平滑的环形进度条。</li>
<li><strong>阶段自动切换:</strong> `switchToNextPhase` 函数根据当前阶段和完成的番茄钟数智能决定下一阶段(工作->短休息/长休息->工作)。</li>
<li><strong>状态反馈:</strong> 通过改变父容器类名(`.working`, `.short-break`, `.long-break`)触发CSS改变进度环颜色,提供直观视觉反馈。</li>
<li><strong>错误处理:</strong> 在解析 `localStorage` 数据时使用 `try...catch`,避免无效数据导致应用崩溃。</li>
<li><strong>音频提示:</strong> 注释中说明了 `console.log('x07')` 的局限性,并给出了使用 `<audio>` 标签播放声音文件的更佳实践方向。</li>
<li><strong>防抖优化(提示):</strong> 在 `taskInput` 的 `input` 事件处理中提示了频繁保存可能需要防抖(如 `setTimeout` 或 Lodash `_.debounce`)。</li>
</ul>
</section>
<section>
<h2>部署与扩展思路</h2>
<p><strong>部署:</strong> 将 `index.html`, `style.css`, `script.js` 三个文件放在同一个目录下,直接在浏览器中打开 `index.html` 即可运行,也可轻松部署到任何静态网站托管服务(如GitHub Pages, Netlify, Vercel)。</p>
<p><strong>功能扩展方向:</strong></p>
<ul>
<li><strong>更完善的音频系统:</strong> 使用 `<audio>` 标签预加载不同提示音(开始、结束、休息结束),并确保在用户交互后允许自动播放。</li>
<li><strong>任务列表与统计:</strong> 添加功能记录每个番茄钟对应的任务,并统计每天/每周完成的番茄钟数量和专注任务分布。</li>
<li><strong>通知提醒:</strong> 集成浏览器的 Notification API,在阶段切换时发送桌面通知(需用户授权)。</li>
<li><strong>主题切换:</strong> 实现深色/浅色模式切换。</li>
<li><strong>多语言支持:</strong> 国际化 (i18n) 支持。</li>
<li><strong>后端集成:</strong> 如果需要跨设备同步数据,可以添加后端(如Node.js + Express + MongoDB 或 Firebase)和用户认证。</li>
<li><strong>PWA应用:</strong> 添加Manifest和Service Worker,使其可安装到桌面并离线使用。</li>
</ul>
</section>
<section>
<h2>lt;/h2>
<p>这个番茄钟项目麻雀虽小,五脏俱全,涵盖了前端开发的多个核心概念:DOM操作、事件处理、状态管理、CSS动画、SVG应用、本地存储(LocalStorage)以及响应式设计,通过构建它,你实践了如何将用户需求转化为功能模块,并组合成一个完整的交互式应用。</p>
<p>源码结构清晰,注释完善,非常适合学习、修改和扩展,你可以直接使用它作为你的时间管理工具,也可以将其作为学习JavaScript和Web开发的跳板,添加更多你感兴趣的功能。</p>
</section>
<p><strong>你已经尝试过番茄工作法了吗?你觉得这个计时器最实用的功能是什么?或者你希望它还能加入什么酷炫的新功能?欢迎在评论区分享你的想法和使用体验!</strong></p>
原创文章,作者:世雄 - 原生数据库架构专家,如若转载,请注明出处:https://idctop.com/article/27164.html