标为完成

2019-01-077173 阅读6 评论

用户点击标为完成按钮,则将对应的 todo 标为完成,为了标记 todo 是已完成还是未完成的状态,我们需要稍微改造一下我们的 todo 模型,一开始我们的 todo 有 id、title 两个属性,现在加一个 completed 属性,默认值是 false,表示未完成。

todo = {id:1, title:'todo', completed:false}

当然我们对应的 addTodo 方法也要修改一下:

addTodo: function () {
    this.todos.push(
        // 修改后的 todo 模型
        {id: id++, title: this.newTodoTitle, completed: false}
    );
    this.newTodoTitle = '';
}

标为已完成的思路和之前是一样的,监听用户点击标为完成按钮的 click 事件,然后调用相应的处理方法,把 todo 的 completed 的属性由 false 改为 true 即可。这里的一个关键是,我们如何知道用户点击的是哪一个 todo,或者说我们可以通过坐在用户身边的方式用肉眼观察用户点击了哪个 todo,但是 Vue 如何知道?

好在 Vue 允许我们向绑定的方法传递参数,我们可以把当前循环中 todo 的对象传给绑定的方法,绑定的方法代码如下:

<div id="todo-app">
  ...
  <!--todo list-->
  <ul>
    <li v-for='todo in todos' :key='todo.id'>
      <span>{{ todo.title }}</span>
      <input type="button" 
             value="标为完成"
             @click="markAsCompleted(todo)"/>
      ...
    </li>
  </ul>
  <!-- end todo list -->
  ...
</div>

注意到我们给标为完成的按钮的点击事件绑定了一个 markAsCompleted 方法,这个方法将用来处理将 todo 标为完成的动作。同时和之前绑定的的方法不同的是,我们还给他传递了一个参数,其值为 todo,这个 todo 是我们在 v-for='todo in todos' 获取,当前循环的是哪个 todo,这个 todo 就指向哪个 todo 对象。从而 Vue 就知道了原来我们我们想要操作 todos 列表中这个 todo 对象。

markAsCompleted的逻辑非常简单:

var app = new Vue({
    ...
    methods: {
        addTodo: function () {
            ...
        },
        markAsCompleted: function (todo) {
            todo.completed = true
        }
    }
})

视觉上我们应该让已经标记了已完成的 todo 显示为已删除的状态,一条横线穿过。这需要为我们的内容增加一点样式,这又要用到我们动态绑定样式的方法了:

<li v-for='todo in todos' :key='todo.id'>
    <span :class="{completed: todo.completed}">{{ todo.title }}</span>
    <input type="button" value="标为完成"
           @click="markAsCompleted (todo)"/>
    ...
</li>

这个completed 的样式我们定义在html文档最开头的head标签的style标签里,为元素提供贯穿线的样式。这里根据 todo.completed 来决定是否应该给元素添加 class=‘completed’ 的样式。

练习

练习一:目前我们点击标为完成按钮后,这个按钮是不会消失的,为了更好的用户体验,已完成的 todo 后面不该再显示一个标为完成的按钮了,想办法把它去掉。

练习二:进一步地,如果用户突然发现其实事情还没有做完,他想把这个 todo 标记回未完成状态,添加一个标为未完成的按钮实现这个需求,要注意不同按钮出现的场合应该符合逻辑。

-- EOF --

6 评论
登录后回复
明天去接小河童
2020-06-13 21:09:43

很感谢你的分享,打算今天把你的教程撸完

回复
宋宇航192
2020-01-12 16:38:31

你好,我点击了标记完成按钮,要在下次输入或更新之后,完成才会划掉。。感觉执行有延时,这要怎么处理?还有一个问题,请问有什么好的软件或插件,可以实现在浏览器上边看边修改vue的功能?

回复
追梦人物 宋宇航192
2020-02-25 11:33:19

点击了标记完成按钮,要在下次输入或更新之后,完成才会划掉。

猜测是你的代码逻辑中存在问题。

请问有什么好的软件或插件,可以实现在浏览器上边看边修改vue的功能

F12 打开调试窗口,在 console 里调试就行。

回复
johnandcat
2019-05-05 09:29:05

您好 请问 todo模型 todo = {id:1, title:'todo', completed:false}  这一段代码是添加在哪个地方  之前的代码中并未见到 与之相关的代码片段

回复
Bethink johnandcat
2019-08-09 11:31:55

就是原来的todos里面的项目,增加一个属性completed:false

回复
nicangtianws01 johnandcat
2019-11-06 16:58:12

这是作者给出的对象模型,你真正需要做的是在push的时候给对象添加一个completed:false属性。

回复

目录