Todo List

Addy Osmani在新窗口打开 是一位高产的大师, 在 2011 年, 他创建了 TodoMVC在新窗口打开

Todo MVC 不仅 适用于其本身想解决的问题, 也经常用于前端技术教学的 案例。

提示

以下 是 Todo MVC 希望解决的问题的英文原文 和 简单的翻译

Helping you select an MV* framework

帮助你选择 MV* 框架

Developers these days are spoiled with choice when it comes to selecting an MV* framework for structuring and organizing JavaScript web apps.

程序员在为 Javascript 程序选择 MV* 框架 这件事上实在是被惯死了。

Backbone, Ember, AngularJS... the list of new and stable solutions goes on and on, but just how do you decide on which to use in a sea of so many options?

各种稳定可靠的解决方案层出不穷,但是如何在海量方案中做出选择?

To help solve this problem, we created TodoMVC - a project which offers the same Todo application implemented using MV* concepts in most of the popular JavaScript MV* frameworks of today.

为了解决这个问题,我们创建了 TodoMVC,用不同主流前端 MV* 解决方案 来实现 同一个 待办列表应用。 这样孰优孰劣,你可以根据自己的判断来选择。

范例

相较于 Todo MVC Vue 版本在新窗口打开,我做了一些简化。 此外,这个例子是用 Vue 3在新窗口打开 编写的。

它的功能清单如下:

  • 新增一个 待办事项

    • 输入为空 或者 输入重复的事项 会无效
  • 删除一个 待办事项

    • 点击每个事项后面的 X 文字可以删除当前的待办事项

以下的 范例 是 可以操作的, 你可以试试。

我的待办(范例)
  • 快学前端 (X)
Thu Mar 28 2024 15:27:29 GMT+0800 (中国标准时间)

注意

  • 范例源代码的知识点 没有任何解释说明
  • 因为目前你无需理解这个范例的源代码
范例源代码(Vue3) (点击展开)

MyTodoList.vue

<template>
  <div>
    <em>我的待办(范例)</em>
    <hr />
    <!-- add todo -->
    <input
      ref="newTodo"
      @blur="addTodo"
      @keyup.enter="addTodo"
      placeholder="输入待办标题 然后敲回车"
    />
    <!-- todo list -->
    <ul>
      <li v-for="todo in todos" :key="todo.title">
        {{ todo.title }} (<span @click="removeTodo(todo.title)">X</span>)
      </li>
    </ul>
    <cite>
      {{ now }}
    </cite>
  </div>
</template>

<script setup>
import { ref } from "vue";
import { store } from "./MyTodoList.store";

const newTodo = ref(null);

const todos = store.state.todos;

const now = new Date();

function addTodo(evt) {
  const newTodoTitle = evt.target.value.trim();
  // 非空 并且 不重复
  if (newTodoTitle && !store.has(newTodoTitle)) {
    store.add(evt.target.value);
    newTodo.value.value = "";
    newTodo.value.focus();
  }
}

function removeTodo(title) {
  store.remove(title);
}
</script>

<style scoped>
* {
  background-color: rgb(240, 241, 242);
  font-size: 1rem;
}
div {
  width: 400px;
  padding: 1.5rem 1rem;
  display: flex;
  flex-direction: column;
}
input {
  margin-top: 1rem;
  border: none;
  border-bottom: 1px solid black;
  font-size: 0.9rem;
}
input:focus {
  outline: none;
}
@keyframes rotate {
  0% {
    transform: rotate(0deg);
  }
  50% {
    transform: rotate(90deg);
  }
  100% {
    transform: rotate(180deg);
    color: red;
  }
}
span {
  display: inline-block;
  background-color: transparent;
}
span:hover {
  animation: rotate 0.4s ease;
  animation-fill-mode: both;
  cursor: pointer;
}
em {
  width: 50%;
  font-size: 1rem;
}
cite {
  margin-top: 3rem;
  font-size: 0.5rem;
  align-self: flex-end;
}
</style>

MyTodoList.store.js

import { reactive } from "vue";

class Store {
  constructor() {
    const todos = [
      {
        title: "快学前端",
      },
    ];
    this.state = reactive({
      todos,
    });
  }
  add(title) {
    this.state.todos.push({
      title,
    });
  }
  has(title) {
    return this.state.todos.some((todo) => todo.title === title);
  }
  remove(title) {
    const index = this.state.todos.findIndex((todo) => todo.title === title);
    this.state.todos.splice(index, 1);
  }
}

export const store = new Store();

静态还原

眼见为实

接下来我们用 ES6在新窗口打开 来说重新实现这个 范例。 你需要开始掌握一些基础的 HTML在新窗口打开CSS在新窗口打开 知识, 不过你可以先继续看完本节。

提示

作为前端程序员, 你真的是集 万千宠爱于一身 , 前端 三剑客在新窗口打开 html css javascript书籍教程 多如 夜空中的繁星。 我深感眼拙, 列出几个最吸引我的如下:

这些教程有的有中文翻译,有的没有,很抱歉我不推荐我没有看过的资料。我相信你也能很容易找到适合入门的 基础教程

以下是 源代码,这里 不涉及到 Javascript, 它 100% 还原了 上一节的 范例 静态 渲染的 部分。 而 动态 的部分 留在下一节处理。

提示

动态 一般是指

  • 用户浏览器 进行交互,例如 输入用户名,点击一个按钮,在页面上下滚动浏览,导致 界面 发生了变化。
  • state 也就是 状态 本身发生了变化。
<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <meta http-equiv="X-UA-Compatible" content="ie=edge" />
    <title>Static Template</title>
    <style>
      * {
        background-color: rgb(240, 241, 242);
        font-size: 1rem;
      }
      div {
        width: 400px;
        padding: 1.5rem 1rem;
        display: flex;
        flex-direction: column;
      }
      input {
        margin-top: 1rem;
        border: none;
        border-bottom: 1px solid black;
        font-size: 0.9rem;
      }
      input:focus {
        outline: none;
      }
      @keyframes rotate {
        0% {
          transform: rotate(0deg);
        }
        50% {
          transform: rotate(90deg);
        }
        100% {
          transform: rotate(180deg);
          color: red;
        }
      }
      span {
        display: inline-block;
        background-color: transparent;
      }
      span:hover {
        animation: rotate 0.4s ease;
        animation-fill-mode: both;
        cursor: pointer;
      }
      em {
        width: 50%;
        font-size: 1rem;
      }
      cite {
        margin-top: 3rem;
        font-size: 0.5rem;
        align-self: flex-end;
      }
    </style>
  </head>
  <body>
    <div>
      <em>我的待办(范例)</em>
      <hr />
      <!-- add todo -->
      <input placeholder="输入待办标题 然后敲回车" />
      <!-- todo list -->
      <ul>
        <li>待办事项A (<span>X</span>)</li>
        <li>待办事项B (<span>X</span>)</li>
        <li>待办事项C (<span>X</span>)</li>
      </ul>
      <cite> Fri Jun 17 2022 11:23:33 GMT+0800 (中国标准时间) </cite>
    </div>
  </body>
</html>







 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

















运行效果:

很明显:

  • 从第 8 行 到 第 58 行全部都是 CSS 代码 (高亮部分)
  • 其他全部都是 HTML 代码

接下来我们来展开一些你需要掌握并且不容易被掌握的部分:

语义化

HTML三剑客 里面 能被 活学 知识最少的一位。也就是 三剑客 的软肋和短板。 这是开玩笑。

现阶段 我觉得 HTML 所需要掌握的知识点如下:

  • 常用的的 元素 或者叫 标签样式 角度来看 几乎可以认为都是完全一样的。

    这意味着 我们可以通过 样式 把 常用标签 装扮地 不分你我

    <!-- 理论上来讲, 通过 css 属性,可以让以下两个不同类型的标签 外观样式完全一样 -->
    <div></div>
    <span></span>
    
  • 因此标签的区别 在我看来主要在于 语义化在新窗口打开 的区别 它 是 HTML 编写的一个重要 原则。

    简单来说,不同的 标签 名 代表了不同的 业务 含义。 自然会有 去消费这些 语义。 例如 最典型的例子就是 Google 爬虫在新窗口打开。 它会根据 页面 里面 特定 标签 的内容做分析和排名。

伪元素

HTMLCSS 分别 负责 定义 网页上的 内容样式 尽管 这并非绝对的。

CSS 有能力 定义 内容, 尽管这不是 CSS 的职责。 不仅如此,CSS 也有能力 实现 动态效果。 后文你会看到。

如果样式这样写

输入

<div class="i-am-a-pseudo-element" />
div.i-am-a-pseudo-element::before {
  content: "hello world!";
}

输出

上面的 hello world 是用 css 输出的,这是 伪元素在新窗口打开

CSS 通过 选择器 Selector在新窗口打开 选中 HTML 中的 元素 element/elements 然后给 它/它们 添加 样式

伪元素 是 用来 选中 元素的一部分 例如 ::before 表示选中 div 元素的 前面 这个 部分.

注意

A CSS pseudo-element is used to style specified parts of an element.

还好的是 HTML 没有本事 去 定义 样式

伪类

注意

A pseudo-class is used to define a special state of an element.

input:focus {
  outline: none;
}

来看上面这段代码片段, :focus 是另外一种 CSS 选择器 伪类选择器在新窗口打开

如果说 伪元素 就很难理解了, 那么 伪类 更奇怪,实际上它选择的不是 元素 而是 元素 在某一时刻的 状态。 例如这个例子里面 这个 状态 是指 input 文本框 获取到了 焦点

我们再来看一个例子:

输入

<input class="i-am-a-pseudo-class" /><span></span>
input.i-am-a-pseudo-class + span {
  margin-left: 0.5rem;
}
input.i-am-a-pseudo-class + span::before {
  content: "我还没有获取焦点,把鼠标移动到文本框中试试";
}
input.i-am-a-pseudo-class:focus + span::before {
  content: "我获取了焦点";
  color: red;
}

输出

在这个例子里面 当 input 元素 处于 获取 焦点 这个状态,选择器 命中 元素,样式设置生效。 结合 上一节的 伪类。实现了 动态 改变 内容 的效果。

如果你能理解 这两节内容。 恭喜你!你的速度 堪比 🚀 CSS语法在新窗口打开 你完全掌握了!

不过 CSS 远不止 语法 这一个重要的知识点。 但是,你掌握了 CSS 最基础 最核心的 知识点 你解锁了一项 里程碑 式的技能。🏁

动画

TBD

最佳实践

Best Practices

你应该注意到 在案例里面 我们用到的是 Internal CSS在新窗口打开 以及 大部分都是 元素选择器在新窗口打开

这两点在教学层面是没有问题的,但是在 真实项目里面 我们并不会这么做。

如下所示, 对于初学者,我觉得有必要让你 及早知道 为什么 这不是 好的实践方式。

  • Internal CSS

    显然如果把大量的 CSSHTML 写在一起 这违法了 编程的 单一职责 原则 SRP in 100 seconds在新窗口打开

    不过在 浏览器性能优化 领域 把 CSS 通过 <style></style> 内联 到 HTML 确也是一种 必要的手段。

    编程世界里面 没有 绝对, 适用场景很关键。 但是一般而言 从 代码编写和组织角度来讲, 把文件分开 一直都是 正确 的选择。 因为, 有很多种技术手段可以自动达到 性能优化 的目标。

  • 元素选择器

    元素选择器 拥有 较低的 CSS 选择器权重在新窗口打开。因此 它会选中 所有相同的 元素 不管它在哪里。

    显然这容易带来很多问题, 这是 CSS 层叠样式表 特性在新窗口打开 决定的。

    很多著名的 CSS 架构方案, 例如 ITCSS在新窗口打开 都尝试解决 选择器 滥用 导致的 CSS 架构层面的 问题。

    CSS 最复杂的技术问题莫过于此。 我们会在后面介绍一些 优秀的解决方案。

注意

一般而言 CSS 选择器权重 是一个必讲的知识点, 它看上去那么的 技术。但是我不打算讲它, 也不建议你精通, 只需要理解就可以了。

因为在后面会讲到的 CSS 各种 最佳工程实践中,你几乎不需要知道有 CSS 选择器权重 这件事情存在。

<!-- 这是 Internal CSS -->
<style>
  /* 这是 元素选择器 Element Selector */
  div {
    ...;
  }
</style>

调试技巧

TBD

本节最后我想补充一些基本的 开发调试技能...

state

看到的不一定就是真相

fn 是外衣, state 才是灵魂。 上一节, 我们静态还原了 范例。 但是,很显然这个 界面是不会 的。你在文本输入框 输入任何内容都不会导致 待办列表的变化。

这一节,我们就来考虑 的部分。

响应式

谈实现,先聊一下设计目标,假设如果有一个 容器 能存放 新待办事项的标题,另外有一个 容器 能存放 所有 待办事项。并且对 新待办事项的标题 或者 待办事项 的操作 能自动 反映到 界面 上。

这个容器就是 变量在新窗口打开 。类型分别是 String 类型在新窗口打开 以及 数组在新窗口打开。 一样一个就够了。

刚才说的是 状态 改变 可以 导致 ui 变化。 但是单这一个方向是不够的的,比如, 用户在文本输入框里面输入内容,它改变了 ui,如果 state 能随着 ui 变化而变化那就更好了。

这样 变化就是双向的。 它 也被称为 双向绑定

如何实现呢?现在是 2022 年 我就不赘述 10 年前的 JQuery 时代的实现方式了。

这里我们引入 响应式在新窗口打开 的概念,你可以继续读完本章后再回过头来看它。

新待办事项 的实现

我们先来实现 新待办事项 的双向绑定。

我们设计了一个数据状态 data.newTodo(5-7 行) 当我们往这个状态里面设置内容 58 行,你可以看到 这段内容出现在了 文本输入框内。

继续往下看 如果我们往输入框输入内容,那么 state.newTodo 就会变化。 而这是由 37-52 行代码决定的, 在这里需要掌握 前端编程里面最重要的 概念 事件在新窗口打开

事件在前端领域非常重要,现在你只需要记住 事件 等于 用户输入,不过如果你能理解 异步回调在新窗口打开,那恭喜你离前端入门已经不远了。

注意

前端 动态 离不开 JavaScript在新窗口打开,虽然 你很可能学过其他编程语言,比如 Java,甚至掌握了 循环函数对象 等等基本编程概念。

但是 JavaScript 有很多不同与其他编程语言的地方,必须引起你的注意,你无需记忆他们。 而是理解他们的设计意图。

例如 异步回调 只不过是 前端领域 处理用户操作的一种编程范式。此外 作用域上下文闭包 都是非常基础的概念,他们在 Javascript 语言 运行时 中的特性才是你需要关注的地方。

换句话说他们并不是一些通用的 编程语言概念 而是和 运行时 相关的 概念。

the good parts
// 存储副作用函数的桶
const bucket = new Set();

// 原始数据
const data = {
  newTodo: null,
};

// 对原始数据的代理
const state = new Proxy(data, {
  // 拦截读取操作
  get(target, key) {
    // 将副作用函数 effect 添加到存储副作用函数的桶中
    bucket.add(effect);
    // 返回属性值
    return target[key];
  },
  // 拦截设置操作
  set(target, key, newVal) {
    // 设置属性值
    target[key] = newVal;
    // 把副作用函数从桶里取出并执行
    bucket.forEach((fn) => fn());
  },
});

// 副作用
function effect() {
  document.getElementById("newTodo").value = state.newTodo;
}

// 更新 newTodo 状态
function updateNewTodo(content) {
  state.newTodo = content || "";
}

// 响应用户在页面上的回车事件
document.addEventListener("keydown", function (e) {
  if (e.code === "Enter") {
    console.log("回车被触发");
    const newVal = e.target.value && e.target.value.trim();
    if (newVal) {
      console.log("回车被有效触发");
      updateNewTodo(newVal);
    }
  }
});

// 响应用户在文本框输入
document.getElementById("newTodo").onchange = function (e) {
  updateNewTodo(e.target.value);
};

// 先关联副作用函数
console.log(state.newTodo);

// 第一次设置
state.newTodo = "我是初始值";




 
 
 





























 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 




 
 
新待办事项 (点击展开)

待办列表 的实现

如果你感觉到这里学起来都很费劲,恭喜你上路了。

// 存储副作用函数的桶
const bucket = new Set();

// todo 模板
const getTodoListItemTemplate = (todo) => {
  return `<li>${todo} (<span>X</span>)</li>`;
};

// todos 模板
const getTodoListTemplate = (todos) =>
  todos.map((todo) => getTodoListItemTemplate(todo)).join("");

// 原始数据
const data = {
  newTodo: null,
  todos: [],
};

// 对原始数据的代理
const state = new Proxy(data, {
  // 拦截读取操作
  get(target, key) {
    // 将副作用函数 effect 添加到存储副作用函数的桶中
    bucket.add(effect);
    // 返回属性值
    return target[key];
  },
  // 拦截设置操作
  set(target, key, newVal) {
    // 设置属性值
    target[key] = newVal;
    // 把副作用函数从桶里取出并执行
    bucket.forEach((fn) => fn());
  },
});

// 副作用
function effect() {
  document.getElementById("newTodo").value = state.newTodo;
  document.getElementById("todoList").innerHTML = getTodoListTemplate(
    state.todos
  );
}

// 更新 todo, newTodo 状态
function updateState(content) {
  state.todos = [...state.todos, content];
  state.newTodo = "";
}

// 响应用户在页面上的回车事件
document.addEventListener("keydown", function (e) {
  if (e.code === "Enter") {
    console.log("回车被触发");
    const newVal = e.target.value && e.target.value.trim();
    if (newVal) {
      console.log("回车被有效触发");
      updateState(newVal);
    }
  }
});

// 响应用户在文本框输入
document.getElementById("newTodo").onchange = function (e) {
  updateState(e.target.value);
};

// 先关联副作用函数
console.log(state.newTodo);

// 第一次设置
state.newTodo = "我是初始值";
待办列表 (点击展开)

知识点

TBD