Close read Vue official documentation series 🎉
In my opinion, the core of a Todo application lies in the definition of data structure:
{
id:Number
title:String.completed:Boolean
}
Copy the code
Then add, delete, change and save around this data structure. Looking at storage, the official example encapsulates a simple todoStorage object and provides fetch and save methods.
var todoStorage = {
key:'TODOS_STORAGE_KEY'.fetch(){
var todos = JSON.parse(localStorage.getItem(this.key) || '[]');
todos.forEach((todo, index) = >{
todo.id = index;
});
todoStorage.uid = todos.length;
return todos;
},
save(todos){
return localStorage.setItem(this.key, JSON.stringify(todos)); }}Copy the code
Data storage can be handled automatically by directly listening for changes in the data source itself, thus eliminating the need for developers to consider more interactive scenarios.
{
data(){
return {
todos:todoStorage.fetch(),
newTodo:' '.editTodo:null.status:'all'}},watch: {todos: {handler(newValue){
todoStorage.save(newValue);
},
deep:true}}}Copy the code
Look at the ** in addition, delete, change **
In the text input box, use V-Model to bidirectional update the value of this.newTodo, and then click Add, wrap the value of this.newTodo, and then add it to this.todos, and then trigger deep listening, automatic storage of data. Finally, empty this.newTodo = ‘ ‘to clear the input record in the text box after adding.
{
methods: {addTodo(){
if (this.newTodo && this.newTodo.trim()) {
var todo = {
id: todoStorage.uid++,
title: this.newTodo,
completed: false
};
this.todos.push(todo);
this.newTodo = ' '; }}}}Copy the code
As for deleting, it is easy to pass the current TOdo object into the delete method directly from the toDOS loop list.
this.todos.splice(this.todos.indexOf(todo), 1)
Copy the code
If you provide modification for each toDO option in the toDOS loop list, you can directly bind the title of the current toDO option to the input box, which is more concise and elegant.
<input v-if="editTodo === todo" v-todo-focus="editTodo === todo" v-model="todo.title" @keydown.enter="doneEditTodo(todo)" @keyup.esc="cancelEditTodo(todo)" />
<span v-if=! "" editTodo" @dbclick="editTodo(todo)">{{todo.title}}</span>
Copy the code
In case the user changes the title but does not save it and then cancels it, the title of the current todo option must be saved in advance when the change is triggered. Since the variable that stores the value does not need to be a reactive object, we will add a property directly on the component instance.
{
methods: {editTodo(todo){
this.cacheTodoTitle = todo.title;
this.editTodo = todo; }}}Copy the code
When the user triggers a change but hits Cancel while typing, the previous value is restored from cacheTodoTitle.
{
methods: {cancelEditTodo(todo){
todo.title = this.cacheTodoTitle;
this.editTodo = null; }}}Copy the code
If the user is done, click Save, and trigger the doneEditTodo method, because the saving is deep listening, it’s all automatic, but we can make some improvements in this method.
{
methods: {doneEditTodo(todo){
todo.title = todo.title.trim();
if(! todo.title){this.removeTodo(todo)
}
this.editTodo = null; }}}Copy the code
Finally, there are two more points worth discussing in detail.
Todos list loop rendering and state filtering
As in the previous example, we prefer to filter conditions or states in the calculated properties, and the data for the loop rendering list is not the data source itself, but the calculated properties are used to make the loop rendering list.
var filters = {
all(todos){
return todos;
},
active(todos){
return todos.filter(todo= >! todo.completed); },completed(todos){
return todos.filter(todo= >todo.completed); }}; {computed: {filtered({todos}){
return filters[this.visibility](todos); }}}Copy the code
Visibility is a variable whose value is the key of the filters object.
Because filters are mutually exclusive and cannot be aggregated or juxtaposed, the official instance wraps them into a filters object that is then triggered indirectly in the calculation property by the reactive variable this.visibility. If the filter condition changes, the calculation property is triggered to filter the data. Changes in the data source will also trigger the calculation of attributes and re-filtering.
<li v-for="todo in filtered" :key="todo.id">
<input v-if="editTodo === todo" v-model="todo.title" @keydown.enter="doneEditTodo(todo)" @keyup.esc="cancelEditTodo(todo)" />
<span v-if=! "" editTodo" @dbclick="editTodo(todo)">{{todo.title}}</span>
</li>
Copy the code
Complete all functions
The complete functionality can be implemented in a number of ways:
- Listen for events, then execute methods, and put all
todo
çš„completed
Set totrue
. v-model
Update the state of a reactive variable, then listen for the reactive data, and perform the corresponding processing.
Here, the official example shows a more concise solution, complete with a single calculated property that does all the work.
{
computed: {remaining(){
return filters.active(this.todos).length;
},
allDone: {set(value){
this.todos.forEach(todo= >{
todo.completed = value;
});
},
get(){
return this.remaining === 0; }}}}Copy the code
<input type="checkbox" v-model="allDone" />
Copy the code
Finally, there is a custom directive:
{
directives: {
"todo-focus": function(el, binding) {
if(binding.value) { el.focus(); }}}}Copy the code