forked from boranton/testcafe-workshop
221 lines
5.5 KiB
JavaScript
221 lines
5.5 KiB
JavaScript
/* eslint no-invalid-this: 0 */
|
|
|
|
var helpers = require('./helpers')
|
|
var qs = helpers.qs
|
|
var qsa = helpers.qsa
|
|
var $on = helpers.$on
|
|
var $parent = helpers.$parent
|
|
var $delegate = helpers.$delegate
|
|
|
|
module.exports = View
|
|
|
|
/**
|
|
* View that abstracts away the browser's DOM completely.
|
|
* It has two simple entry points:
|
|
*
|
|
* - bind(eventName, handler)
|
|
* Takes a todo application event and registers the handler
|
|
* - render(command, parameterObject)
|
|
* Renders the given command with the options
|
|
*/
|
|
function View(template) {
|
|
this.template = template
|
|
|
|
this.ENTER_KEY = 13
|
|
this.ESCAPE_KEY = 27
|
|
|
|
this.$todoList = qs('.todo-list')
|
|
this.$todoItemCounter = qs('.todo-count')
|
|
this.$clearCompleted = qs('.clear-completed')
|
|
this.$main = qs('.main')
|
|
this.$footer = qs('.footer')
|
|
this.$toggleAll = qs('.toggle-all')
|
|
this.$newTodo = qs('.new-todo')
|
|
}
|
|
|
|
View.prototype._removeItem = function(id) {
|
|
var elem = qs('[data-id="' + id + '"]')
|
|
|
|
if (elem) {
|
|
this.$todoList.removeChild(elem)
|
|
}
|
|
}
|
|
|
|
View.prototype._clearCompletedButton = function(completedCount, visible) {
|
|
this.$clearCompleted.innerHTML = this.template.clearCompletedButton(completedCount)
|
|
this.$clearCompleted.style.display = visible ? 'block' : 'none'
|
|
}
|
|
|
|
View.prototype._setFilter = function(currentPage) {
|
|
qs('.filters .selected').className = ''
|
|
qs('.filters [href="#/' + currentPage + '"]').className = 'selected'
|
|
}
|
|
|
|
View.prototype._elementComplete = function(id, completed) {
|
|
var listItem = qs('[data-id="' + id + '"]')
|
|
|
|
if (!listItem) {
|
|
return
|
|
}
|
|
|
|
listItem.className = completed ? 'completed' : ''
|
|
|
|
// In case it was toggled from an event and not by clicking the checkbox
|
|
qs('input', listItem).checked = completed
|
|
}
|
|
|
|
View.prototype._editItem = function(id, title) {
|
|
var listItem = qs('[data-id="' + id + '"]')
|
|
|
|
if (!listItem) {
|
|
return
|
|
}
|
|
|
|
listItem.className = listItem.className + ' editing'
|
|
|
|
var input = document.createElement('input')
|
|
input.className = 'edit'
|
|
|
|
listItem.appendChild(input)
|
|
input.focus()
|
|
input.value = title
|
|
}
|
|
|
|
View.prototype._editItemDone = function(id, title) {
|
|
var listItem = qs('[data-id="' + id + '"]')
|
|
|
|
if (!listItem) {
|
|
return
|
|
}
|
|
|
|
var input = qs('input.edit', listItem)
|
|
listItem.removeChild(input)
|
|
|
|
listItem.className = listItem.className.replace('editing', '')
|
|
|
|
qsa('label', listItem).forEach(function(label) {
|
|
label.textContent = title
|
|
})
|
|
}
|
|
|
|
View.prototype.render = function(viewCmd, parameter) {
|
|
var that = this
|
|
var viewCommands = {
|
|
showEntries: function() {
|
|
that.$todoList.innerHTML = that.template.show(parameter)
|
|
},
|
|
removeItem: function() {
|
|
that._removeItem(parameter)
|
|
},
|
|
updateElementCount: function() {
|
|
that.$todoItemCounter.innerHTML = that.template.itemCounter(parameter)
|
|
},
|
|
clearCompletedButton: function() {
|
|
that._clearCompletedButton(parameter.completed, parameter.visible)
|
|
},
|
|
contentBlockVisibility: function() {
|
|
that.$main.style.display = that.$footer.style.display = parameter.visible ? 'block' : 'none'
|
|
},
|
|
toggleAll: function() {
|
|
that.$toggleAll.checked = parameter.checked
|
|
},
|
|
setFilter: function() {
|
|
that._setFilter(parameter)
|
|
},
|
|
clearNewTodo: function() {
|
|
that.$newTodo.value = ''
|
|
},
|
|
elementComplete: function() {
|
|
that._elementComplete(parameter.id, parameter.completed)
|
|
},
|
|
editItem: function() {
|
|
that._editItem(parameter.id, parameter.title)
|
|
},
|
|
editItemDone: function() {
|
|
that._editItemDone(parameter.id, parameter.title)
|
|
}
|
|
}
|
|
|
|
viewCommands[viewCmd]()
|
|
}
|
|
|
|
View.prototype._itemId = function(element) {
|
|
var li = $parent(element, 'li')
|
|
return parseInt(li.dataset.id, 10)
|
|
}
|
|
|
|
View.prototype._bindItemEditDone = function(handler) {
|
|
var that = this
|
|
$delegate(that.$todoList, 'li .edit', 'blur', function() {
|
|
if (!this.dataset.iscanceled) {
|
|
handler({
|
|
id: that._itemId(this),
|
|
title: this.value
|
|
})
|
|
}
|
|
})
|
|
|
|
$delegate(that.$todoList, 'li .edit', 'keypress', function(event) {
|
|
if (event.keyCode === that.ENTER_KEY) {
|
|
// Remove the cursor from the input when you hit enter just like if it
|
|
// were a real form
|
|
this.blur()
|
|
}
|
|
})
|
|
}
|
|
|
|
View.prototype._bindItemEditCancel = function(handler) {
|
|
var that = this
|
|
$delegate(that.$todoList, 'li .edit', 'keyup', function(event) {
|
|
if (event.keyCode === that.ESCAPE_KEY) {
|
|
this.dataset.iscanceled = true
|
|
this.blur()
|
|
|
|
handler({id: that._itemId(this)})
|
|
}
|
|
})
|
|
}
|
|
|
|
View.prototype.bind = function(event, handler) { // eslint-disable-line
|
|
var that = this
|
|
if (event === 'newTodo') {
|
|
$on(that.$newTodo, 'change', function() {
|
|
handler(that.$newTodo.value)
|
|
})
|
|
|
|
} else if (event === 'removeCompleted') {
|
|
$on(that.$clearCompleted, 'click', function() {
|
|
handler()
|
|
})
|
|
|
|
} else if (event === 'toggleAll') {
|
|
$on(that.$toggleAll, 'click', function() {
|
|
handler({completed: this.checked})
|
|
})
|
|
|
|
} else if (event === 'itemEdit') {
|
|
$delegate(that.$todoList, 'li label', 'dblclick', function() {
|
|
handler({id: that._itemId(this)})
|
|
})
|
|
|
|
} else if (event === 'itemRemove') {
|
|
$delegate(that.$todoList, '.destroy', 'click', function() {
|
|
handler({id: that._itemId(this)})
|
|
})
|
|
|
|
} else if (event === 'itemToggle') {
|
|
$delegate(that.$todoList, '.toggle', 'click', function() {
|
|
handler({
|
|
id: that._itemId(this),
|
|
completed: this.checked
|
|
})
|
|
})
|
|
|
|
} else if (event === 'itemEditDone') {
|
|
that._bindItemEditDone(handler)
|
|
|
|
} else if (event === 'itemEditCancel') {
|
|
that._bindItemEditCancel(handler)
|
|
}
|
|
}
|