Commit 76d18d5a authored by Sindre Sorhus's avatar Sindre Sorhus

PlastronJS: Convert to tabs

parent 3d71ff07
<!doctype html> <!doctype html>
<html lang="en"> <html lang="en">
<head> <head>
<meta charset="utf-8"> <meta charset="utf-8">
<title>PlastronJS • TodoMVC</title> <title>PlastronJS • TodoMVC</title>
<link rel="stylesheet" href="../../../assets/base.css"> <link rel="stylesheet" href="../../../assets/base.css">
<!--[if IE]> <!--[if IE]>
<script src="../../../assets/ie.js"></script> <script src="../../../assets/ie.js"></script>
<![endif]--> <![endif]-->
</head> </head>
<body> <body>
<section id="todoapp"> <section id="todoapp">
<header id="header"> <header id="header">
<h1>todos</h1> <h1>todos</h1>
<input id="new-todo" class="todo-entry" placeholder="What needs to be done?" autofocus> <input id="new-todo" class="todo-entry" placeholder="What needs to be done?" autofocus>
</header> </header>
<section id="main"> <section id="main">
<input id="toggle-all" class="toggle-all" type="checkbox"> <input id="toggle-all" class="toggle-all" type="checkbox">
<label for="toggle-all">Mark all as complete</label> <label for="toggle-all">Mark all as complete</label>
<ul id="todo-list"></ul> <ul id="todo-list"></ul>
</section> </section>
<footer id="footer"> <footer id="footer">
<span id="todo-count"><strong>0</strong> item left</span> <span id="todo-count"><strong>0</strong> item left</span>
<ul id="filters"> <ul id="filters">
<li> <li>
<a class="selected" href="#/">All</a> <a class="selected" href="#/">All</a>
</li> </li>
<li> <li>
<a href="#/active">Active</a> <a href="#/active">Active</a>
</li> </li>
<li> <li>
<a href="#/completed">Completed</a> <a href="#/completed">Completed</a>
</li> </li>
</ul> </ul>
<button id="clear-completed" class="clear-completed">Clear completed</button> <button id="clear-completed" class="clear-completed">Clear completed</button>
</footer> </footer>
</section> </section>
<footer id="info"> <footer id="info">
<p>Double-click to edit a todo</p> <p>Double-click to edit a todo</p>
<p>Created by <a href="http://rhysbrettbowen.com">Rhys Brett-Bowen</a> (<a href="https://twitter.com/#!/RhysBB">RhysBB</a>).</p> <p>Template by <a href="https://github.com/sindresorhus">Sindre Sorhus</a></p>
<p>Using <a href="https://github.com/rhysbrettbowen/PlastronJS">PlastronJS</a> and <a href="https://developers.google.com/closure/">Closure Tools</a></p> <p>Created by <a href="http://rhysbrettbowen.com">Rhys Brett-Bowen</a> (<a href="https://twitter.com/#!/RhysBB">RhysBB</a>)</p>
<p>Part of <a href="http://todomvc.com">TodoMVC</a>.</p> <p>Using <a href="https://github.com/rhysbrettbowen/PlastronJS">PlastronJS</a> and <a href="https://developers.google.com/closure/">Closure Tools</a></p>
</footer> <p>Part of <a href="http://todomvc.com">TodoMVC</a></p>
<script src="../../../assets/base.js"></script> </footer>
<!-- <script src="http://localhost:9810/compile?id=todomvc&mode=raw"></script> --> <script src="../../../assets/base.js"></script>
<script src="js/compiled.js"></script> <!-- <script src="http://localhost:9810/compile?id=todomvc&mode=raw"></script> -->
<script src="js/compiled.js"></script>
</body> </body>
</html> </html>
\ No newline at end of file
...@@ -17,8 +17,8 @@ goog.require('todomvc.todocontrol'); ...@@ -17,8 +17,8 @@ goog.require('todomvc.todocontrol');
* @extends {mvc.Control} * @extends {mvc.Control}
*/ */
todomvc.listcontrol = function(list) { todomvc.listcontrol = function(list) {
goog.base(this, list); goog.base(this, list);
this.filter_ = todomvc.listcontrol.Filter.ALL; this.filter_ = todomvc.listcontrol.Filter.ALL;
}; };
goog.inherits(todomvc.listcontrol, mvc.Control); goog.inherits(todomvc.listcontrol, mvc.Control);
...@@ -27,9 +27,9 @@ goog.inherits(todomvc.listcontrol, mvc.Control); ...@@ -27,9 +27,9 @@ goog.inherits(todomvc.listcontrol, mvc.Control);
* @enum {Function} * @enum {Function}
*/ */
todomvc.listcontrol.Filter = { todomvc.listcontrol.Filter = {
ALL: function() {return true}, ALL: function() {return true},
ACTIVE: function(model) {return !model.get('completed')}, ACTIVE: function(model) {return !model.get('completed')},
COMPLETED: function(model) {return model.get('completed')} COMPLETED: function(model) {return model.get('completed')}
}; };
...@@ -39,77 +39,77 @@ todomvc.listcontrol.Filter = { ...@@ -39,77 +39,77 @@ todomvc.listcontrol.Filter = {
* @inheritDoc * @inheritDoc
*/ */
todomvc.listcontrol.prototype.enterDocument = function() { todomvc.listcontrol.prototype.enterDocument = function() {
goog.base(this, 'enterDocument'); goog.base(this, 'enterDocument');
var list = /** @type {Object} */(this.getModel()); var list = /** @type {Object} */(this.getModel());
// create new model from text box // create new model from text box
var input = this.getEls('input')[0]; var input = this.getEls('input')[0];
this.on(goog.events.EventType.KEYUP, function(e) { this.on(goog.events.EventType.KEYUP, function(e) {
// on return get trimmed text // on return get trimmed text
if (e.keyCode !== goog.events.KeyCodes.ENTER) return; if (e.keyCode !== goog.events.KeyCodes.ENTER) return;
var text = goog.string.trim(input.value); var text = goog.string.trim(input.value);
if (text === '') return; if (text === '') return;
//create new model //create new model
list.newModel({'title': text}); list.newModel({'title': text});
input.value = ''; input.value = '';
}, 'todo-entry'); }, 'todo-entry');
// clear completed // clear completed
this.click(function(e) { this.click(function(e) {
goog.array.forEach(list.get('completed'), function(model) { goog.array.forEach(list.get('completed'), function(model) {
model.dispose(); model.dispose();
}); });
}, 'clear-completed'); }, 'clear-completed');
// toggle completed // toggle completed
this.click(function(e) { this.click(function(e) {
var checked = e.target.checked; var checked = e.target.checked;
goog.array.forEach(list.getModels(), function(model) { goog.array.forEach(list.getModels(), function(model) {
model.set('completed', checked); model.set('completed', checked);
}); });
}, 'toggle-all'); }, 'toggle-all');
// refresh the view on changes that effect the models order // refresh the view on changes that effect the models order
this.anyModelChange(function() { this.anyModelChange(function() {
this.refresh(); this.refresh();
list.save(); list.save();
}, this); }, this);
// hide/show footer and main body // hide/show footer and main body
this.modelChange(function() { this.modelChange(function() {
this.showMainFooter(!!list.getLength()); this.showMainFooter(!!list.getLength());
}, this); }, this);
this.showMainFooter(!!list.getLength()); this.showMainFooter(!!list.getLength());
// update counts // update counts
this.bind('completed', function(completedModels) { this.bind('completed', function(completedModels) {
// update "left" count // update "left" count
soy.renderElement(goog.dom.getElement('todo-count'), soy.renderElement(goog.dom.getElement('todo-count'),
todomvc.templates.itemsLeft, { todomvc.templates.itemsLeft, {
left: list.getLength() - completedModels.length left: list.getLength() - completedModels.length
}); });
// update clear button // update clear button
var clearButton = goog.dom.getElement('clear-completed'); var clearButton = goog.dom.getElement('clear-completed');
goog.dom.setTextContent(clearButton, goog.dom.setTextContent(clearButton,
'Clear completed (' + completedModels.length + ')'); 'Clear completed (' + completedModels.length + ')');
goog.style.showElement(clearButton, completedModels.length); goog.style.showElement(clearButton, completedModels.length);
// update checkbox // update checkbox
var checkBox = this.getEls('.toggle-all')[0]; var checkBox = this.getEls('.toggle-all')[0];
checkBox.checked = completedModels.length === list.getLength(); checkBox.checked = completedModels.length === list.getLength();
}); });
// get the saved todos // get the saved todos
list.fetch(); list.fetch();
}; };
...@@ -119,11 +119,11 @@ todomvc.listcontrol.prototype.enterDocument = function() { ...@@ -119,11 +119,11 @@ todomvc.listcontrol.prototype.enterDocument = function() {
* @param {boolean=} opt_hide whether to hide the footer. * @param {boolean=} opt_hide whether to hide the footer.
*/ */
todomvc.listcontrol.prototype.showMainFooter = function(opt_hide) { todomvc.listcontrol.prototype.showMainFooter = function(opt_hide) {
var main = goog.dom.getElement('main'); var main = goog.dom.getElement('main');
var footer = goog.dom.getElementsByTagNameAndClass('footer')[0]; var footer = goog.dom.getElementsByTagNameAndClass('footer')[0];
goog.style.showElement(main, opt_hide); goog.style.showElement(main, opt_hide);
goog.style.showElement(footer, opt_hide); goog.style.showElement(footer, opt_hide);
}; };
...@@ -133,8 +133,8 @@ todomvc.listcontrol.prototype.showMainFooter = function(opt_hide) { ...@@ -133,8 +133,8 @@ todomvc.listcontrol.prototype.showMainFooter = function(opt_hide) {
* @param {Function} filter to decide models returned. * @param {Function} filter to decide models returned.
*/ */
todomvc.listcontrol.prototype.setFilter = function(filter) { todomvc.listcontrol.prototype.setFilter = function(filter) {
this.filter_ = filter; this.filter_ = filter;
this.refresh(); this.refresh();
}; };
...@@ -143,17 +143,17 @@ todomvc.listcontrol.prototype.setFilter = function(filter) { ...@@ -143,17 +143,17 @@ todomvc.listcontrol.prototype.setFilter = function(filter) {
*/ */
todomvc.listcontrol.prototype.refresh = function() { todomvc.listcontrol.prototype.refresh = function() {
// dispose and remove all the children. // dispose and remove all the children.
this.forEachChild(function(child) {child.dispose();}); this.forEachChild(function(child) {child.dispose();});
this.removeChildren(true); this.removeChildren(true);
// create new controls for the models // create new controls for the models
goog.array.forEach(this.getModel().getModels(this.filter_), goog.array.forEach(this.getModel().getModels(this.filter_),
function(model) { function(model) {
var newModelControl = new todomvc.todocontrol(model); var newModelControl = new todomvc.todocontrol(model);
this.addChild(newModelControl); this.addChild(newModelControl);
newModelControl.render(goog.dom.getElement('todo-list')); newModelControl.render(goog.dom.getElement('todo-list'));
}, this); }, this);
}; };
...@@ -14,7 +14,7 @@ goog.require('todomvc.templates'); ...@@ -14,7 +14,7 @@ goog.require('todomvc.templates');
* @extends {mvc.Control} * @extends {mvc.Control}
*/ */
todomvc.todocontrol = function(model) { todomvc.todocontrol = function(model) {
goog.base(this, model); goog.base(this, model);
}; };
goog.inherits(todomvc.todocontrol, mvc.Control); goog.inherits(todomvc.todocontrol, mvc.Control);
...@@ -27,9 +27,9 @@ goog.inherits(todomvc.todocontrol, mvc.Control); ...@@ -27,9 +27,9 @@ goog.inherits(todomvc.todocontrol, mvc.Control);
* @inheritDoc * @inheritDoc
*/ */
todomvc.todocontrol.prototype.createDom = function() { todomvc.todocontrol.prototype.createDom = function() {
var el = soy.renderAsFragment(todomvc.templates.todoItem, { var el = soy.renderAsFragment(todomvc.templates.todoItem, {
model: this.getModel().toJson() model: this.getModel().toJson()
}, null); }, null);
this.setElementInternal(/** @type {Element} */(el)); this.setElementInternal(/** @type {Element} */(el));
}; };
...@@ -41,34 +41,34 @@ todomvc.todocontrol.prototype.createDom = function() { ...@@ -41,34 +41,34 @@ todomvc.todocontrol.prototype.createDom = function() {
*/ */
todomvc.todocontrol.prototype.enterDocument = function() { todomvc.todocontrol.prototype.enterDocument = function() {
var model = this.getModel(); var model = this.getModel();
// toggle complete // toggle complete
this.click(function(e) { this.click(function(e) {
model.set('completed', e.target.checked); model.set('completed', e.target.checked);
}, 'toggle'); }, 'toggle');
// delete the model // delete the model
this.click(function(e) { this.click(function(e) {
model.dispose(); model.dispose();
}, 'destroy'); }, 'destroy');
// dblclick to edit // dblclick to edit
this.on(goog.events.EventType.DBLCLICK, function(e) { this.on(goog.events.EventType.DBLCLICK, function(e) {
this.makeEditable(true); this.makeEditable(true);
}, 'view'); }, 'view');
// save on edit // save on edit
var inputEl = this.getEls('.edit')[0]; var inputEl = this.getEls('.edit')[0];
this.on(goog.events.EventType.KEYUP, function(e) { this.on(goog.events.EventType.KEYUP, function(e) {
if (e.keyCode === goog.events.KeyCodes.ENTER) { if (e.keyCode === goog.events.KeyCodes.ENTER) {
model.set('title', inputEl.value); model.set('title', inputEl.value);
} }
}, 'edit'); }, 'edit');
this.on(goog.events.EventType.BLUR, function(e) { this.on(goog.events.EventType.BLUR, function(e) {
model.set('title', inputEl.value); model.set('title', inputEl.value);
}, 'edit'); }, 'edit');
}; };
...@@ -79,12 +79,12 @@ todomvc.todocontrol.prototype.enterDocument = function() { ...@@ -79,12 +79,12 @@ todomvc.todocontrol.prototype.enterDocument = function() {
*/ */
todomvc.todocontrol.prototype.makeEditable = function(editable) { todomvc.todocontrol.prototype.makeEditable = function(editable) {
var inputEl = this.getEls('.edit')[0]; var inputEl = this.getEls('.edit')[0];
goog.dom.classes.enable(this.getElement(), 'editing', editable); goog.dom.classes.enable(this.getElement(), 'editing', editable);
if (editable) { if (editable) {
inputEl.value = this.getModel().get('title'); inputEl.value = this.getModel().get('title');
inputEl.select(); inputEl.select();
} }
}; };
...@@ -11,23 +11,23 @@ goog.require('todomvc.todomodel'); ...@@ -11,23 +11,23 @@ goog.require('todomvc.todomodel');
*/ */
todomvc.listmodel = function() { todomvc.listmodel = function() {
var todosSchema = { var todosSchema = {
'completed': { 'completed': {
get: function() { get: function() {
return this.getModels(function(mod) { return this.getModels(function(mod) {
return mod.get('completed'); return mod.get('completed');
}); });
}, },
models: true models: true
} }
}; };
goog.base(this, { goog.base(this, {
'id': 'todos-plastronjs', 'id': 'todos-plastronjs',
'sync': new todomvc.listsync(), 'sync': new todomvc.listsync(),
'schema': todosSchema, 'schema': todosSchema,
'modelType': todomvc.todomodel 'modelType': todomvc.todomodel
}); });
}; };
goog.inherits(todomvc.listmodel, mvc.Collection); goog.inherits(todomvc.listmodel, mvc.Collection);
...@@ -36,8 +36,8 @@ goog.inherits(todomvc.listmodel, mvc.Collection); ...@@ -36,8 +36,8 @@ goog.inherits(todomvc.listmodel, mvc.Collection);
* @return {Object} todos as json. * @return {Object} todos as json.
*/ */
todomvc.listmodel.prototype.toJson = function() { todomvc.listmodel.prototype.toJson = function() {
return goog.array.map(this.getModels(), function(mod) { return goog.array.map(this.getModels(), function(mod) {
return mod.toJson(); return mod.toJson();
}); });
}; };
...@@ -12,17 +12,17 @@ goog.require('mvc.Model.ValidateError'); ...@@ -12,17 +12,17 @@ goog.require('mvc.Model.ValidateError');
* @extends {mvc.Model} * @extends {mvc.Model}
*/ */
todomvc.todomodel = function(opt_options) { todomvc.todomodel = function(opt_options) {
goog.base(this, opt_options); goog.base(this, opt_options);
this.setter('title', function(title) { this.setter('title', function(title) {
var updated = goog.string.trim(title); var updated = goog.string.trim(title);
if (!updated.length) if (!updated.length)
throw new mvc.Model.ValidateError('null string'); throw new mvc.Model.ValidateError('null string');
return updated; return updated;
}); });
this.errorHandler(function() { this.errorHandler(function() {
this.dispose(); this.dispose();
}); });
}; };
goog.inherits(todomvc.todomodel, mvc.Model); goog.inherits(todomvc.todomodel, mvc.Model);
...@@ -9,7 +9,7 @@ goog.require('mvc.LocalSync'); ...@@ -9,7 +9,7 @@ goog.require('mvc.LocalSync');
* @extends {mvc.LocalSync} * @extends {mvc.LocalSync}
*/ */
todomvc.listsync = function() { todomvc.listsync = function() {
goog.base(this); goog.base(this);
}; };
goog.inherits(todomvc.listsync, mvc.LocalSync); goog.inherits(todomvc.listsync, mvc.LocalSync);
...@@ -18,13 +18,13 @@ goog.inherits(todomvc.listsync, mvc.LocalSync); ...@@ -18,13 +18,13 @@ goog.inherits(todomvc.listsync, mvc.LocalSync);
* @inheritDoc * @inheritDoc
*/ */
todomvc.listsync.prototype.read = function(model, opt_callback) { todomvc.listsync.prototype.read = function(model, opt_callback) {
var id = /** @type {string} */(model.get('id')); var id = /** @type {string} */(model.get('id'));
var todos = this.store_.get(id) || []; var todos = this.store_.get(id) || [];
goog.array.forEach(/** @type {Array} */(todos), goog.array.forEach(/** @type {Array} */(todos),
function(todo) { function(todo) {
model.newModel(todo, true); model.newModel(todo, true);
}); });
model.change(); model.change();
}; };
{ {
"id" : "todomvc", "id": "todomvc",
"inputs" : "js/app.js", "inputs": "js/app.js",
"paths" : ["js/","template/"], "paths": ["js/", "template/"],
"output-wrapper" : "(function(){%output%})();", "output-wrapper": "(function(){%output%})();",
"mode" : "ADVANCED", "mode": "ADVANCED",
"level" : "VERBOSE", "level": "VERBOSE",
"output-file" : "js/compiled.js", "output-file": "js/compiled.js",
"define" : { "define": {
"goog.LOCALE": "en_GB" "goog.LOCALE": "en_GB"
}, },
"checks": { "checks": {
// Unfortunately, the Closure Library violates these in many places. "checkRegExp": "WARNING",
// "accessControls": "ERROR", "checkTypes": "WARNING",
// "visibility": "ERROR" "checkVars": "WARNING",
"deprecated": "WARNING",
"checkRegExp": "WARNING", "fileoverviewTags": "WARNING",
"checkTypes": "WARNING", "invalidCasts": "WARNING",
"checkVars": "WARNING", "missingProperties": "WARNING",
"deprecated": "WARNING", "nonStandardJsDocs": "WARNING",
"fileoverviewTags": "WARNING", "undefinedVars": "WARNING"
"invalidCasts": "WARNING",
"missingProperties": "WARNING",
"nonStandardJsDocs": "WARNING",
"undefinedVars": "WARNING"
} }
} }
\ No newline at end of file
...@@ -4,19 +4,19 @@ ...@@ -4,19 +4,19 @@
* @param model * @param model
*/ */
{template .todoItem} {template .todoItem}
<li {if $model['completed']}class="completed"{/if}> <li {if $model['completed']}class="completed"{/if}>
<div class="view"> <div class="view">
<input class="toggle" type="checkbox" {if $model['completed']}checked{/if}> <input class="toggle" type="checkbox" {if $model['completed']}checked{/if}>
<label>{$model['title']}</label> <label>{$model['title']}</label>
<button class="destroy"></button> <button class="destroy"></button>
</div> </div>
<input class="edit" type="text" value=""> <input class="edit" type="text" value="">
</li> </li>
{/template} {/template}
/** /**
* @param left * @param left
*/ */
{template .itemsLeft} {template .itemsLeft}
<strong>{$left}</strong> item{if $left != 1}s{/if} left <strong>{$left}</strong> item{if $left != 1}s{/if} left
{/template} {/template}
\ No newline at end of file
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment