(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o b[prop]) { return 1; } if (a[prop] < b[prop]) { return -1; } return 0; }); render(); } function add() { var n = originalData[Math.floor(Math.random() * 10)]; data = [{rank: nextKey++, title: n.title, desc: n.desc, elmHeight: 0}].concat(data); render(); render(); } function remove(movie) { data = data.filter(function(m) { return m !== movie; }); render(); } function movieView(movie) { return h('div.row', { key: movie.rank, style: {opacity: '0', transform: 'translate(-200px)', 'a-transform': 'translateY(' + movie.offset + 'px)', 'a-opacity': '1'}, oninsert: function(vnode) { movie.elmHeight = vnode.elm.offsetHeight; }, onremove: function(vnode, rm) { vnode.elm.style.transform = vnode.elm.style.transform + ' translateX(200px)'; vnode.elm.style.opacity = '0'; setTimeout(rm, 500); }, }, [ h('div', {style: {fontWeight: 'bold'}}, movie.rank), h('div', movie.title), h('div', movie.desc), h('div.btn.rm-btn', {on: {click: [remove, movie]}}, 'x'), ]); } function render() { data = data.reduce(function(acc, m) { var last = acc[acc.length - 1]; m.offset = last ? last.offset + last.elmHeight + margin : margin; return acc.concat(m); }, []); totalHeight = data[data.length - 1].offset + data[data.length - 1].elmHeight; vnode = snabbdom.patch(vnode, view(data)); } function view(data) { return h('div', [ h('h1', 'Top 10 movies'), h('div', [ h('a.btn.add', {on: {click: add}}, 'Add'), 'Sort by: ', h('span.btn-group', [ h('a.btn.rank', {class: {active: sortBy === 'rank'}, on: {click: [changeSort, 'rank']}}, 'Rank'), h('a.btn.title', {class: {active: sortBy === 'title'}, on: {click: [changeSort, 'title']}}, 'Title'), h('a.btn.desc', {class: {active: sortBy === 'desc'}, on: {click: [changeSort, 'desc']}}, 'Description'), ]), ]), h('div.list', {style: {height: totalHeight+'px'}}, data.map(movieView)), ]); } window.addEventListener('DOMContentLoaded', function() { var container = document.getElementById('container'); vnode = snabbdom.patch(snabbdom.emptyNodeAt(container), view(data)); render(); }); },{"../../h.js":2,"../../modules/class":4,"../../modules/eventlisteners":5,"../../modules/props":6,"../../modules/style":7,"../../snabbdom.js":8}],2:[function(require,module,exports){ var VNode = require('./vnode'); var is = require('./is'); module.exports = function h(sel, b, c) { var data = {}, children, text, i; if (arguments.length === 3) { data = b; if (is.array(c)) { children = c; } else if (is.primitive(c)) { text = c; } } else if (arguments.length === 2) { if (is.array(b)) { children = b; } else if (is.primitive(b)) { text = b; } else { data = b; } } if (is.array(children)) { for (i = 0; i < children.length; ++i) { if (is.primitive(children[i])) children[i] = VNode(undefined, undefined, undefined, children[i]); } } return VNode(sel, data, children, text, undefined); }; },{"./is":3,"./vnode":9}],3:[function(require,module,exports){ module.exports = { array: Array.isArray, primitive: function(s) { return typeof s === 'string' || typeof s === 'number'; }, }; },{}],4:[function(require,module,exports){ function updateClass(oldVnode, vnode) { var cur, name, elm = vnode.elm, oldClass = oldVnode.data.class || {}, klass = vnode.data.class || {}; for (name in klass) { cur = klass[name]; if (cur !== oldClass[name]) { elm.classList[cur ? 'add' : 'remove'](name); } } } module.exports = {create: updateClass, update: updateClass}; },{}],5:[function(require,module,exports){ var is = require('../is'); function arrInvoker(arr) { return function() { arr[0](arr[1]); }; } function updateEventListeners(oldVnode, vnode) { var name, cur, old, elm = vnode.elm, oldOn = oldVnode.data.on || {}, on = vnode.data.on; if (!on) return; for (name in on) { cur = on[name]; old = oldOn[name]; if (old === undefined) { elm.addEventListener(name, is.array(cur) ? arrInvoker(cur) : cur); } else if (is.array(old)) { old[0] = cur[0]; // Deliberately modify old array since it's old[1] = cur[1]; // captured in closure created with `arrInvoker` } } } module.exports = {create: updateEventListeners, update: updateEventListeners}; },{"../is":3}],6:[function(require,module,exports){ function updateProps(oldVnode, vnode) { var key, cur, old, elm = vnode.elm, oldProps = oldVnode.data.props || {}, props = vnode.data.props || {}; for (key in props) { cur = props[key]; old = oldProps[key]; if (old !== cur) { elm[key] = cur; } } } module.exports = {create: updateProps, update: updateProps}; },{}],7:[function(require,module,exports){ var raf = requestAnimationFrame || setTimeout; var nextFrame = function(fn) { raf(function() { raf(fn); }); }; function setNextFrame(obj, prop, val) { nextFrame(function() { obj[prop] = val; }); } function updateStyle(oldVnode, vnode) { var cur, name, elm = vnode.elm, oldStyle = oldVnode.data.style || {}, style = vnode.data.style || {}; for (name in style) { cur = style[name]; if (cur !== oldStyle[name]) { if (name[0] === 'a' && name[1] === '-') { setNextFrame(elm.style, name.slice(2), cur); } else { elm.style[name] = cur; } } } } module.exports = {create: updateStyle, update: updateStyle}; },{}],8:[function(require,module,exports){ // jshint newcap: false 'use strict'; var VNode = require('./vnode'); var is = require('./is'); function isUndef(s) { return s === undefined; } function emptyNodeAt(elm) { return VNode(elm.tagName, {}, [], undefined, elm); } var emptyNode = VNode('', {}, [], undefined, undefined); var frag = document.createDocumentFragment(); var insertedVnodeQueue; function createElm(vnode) { var i, elm, children = vnode.children, sel = vnode.sel; if (!isUndef(sel)) { // Parse selector var hashIdx = sel.indexOf('#'); var dotIdx = sel.indexOf('.', hashIdx); var hash = hashIdx > 0 ? hashIdx : sel.length; var dot = dotIdx > 0 ? dotIdx : sel.length; var tag = hashIdx !== -1 || dotIdx !== -1 ? sel.slice(0, Math.min(hash, dot)) : sel; elm = vnode.elm = document.createElement(tag); if (hash < dot) elm.id = sel.slice(hash + 1, dot); if (dotIdx > 0) elm.className = sel.slice(dot+1).replace(/\./g, ' '); if (is.array(children)) { for (i = 0; i < children.length; ++i) { elm.appendChild(createElm(children[i])); } } else if (is.primitive(vnode.text)) { //elm.textContent = vnode.text; elm.appendChild(document.createTextNode(vnode.text)); } for (i = 0; i < createCbs.length; ++i) createCbs[i](emptyNode, vnode); if (vnode.data.oncreate) vnode.data.oncreate(vnode); if (vnode.data.oninsert) insertedVnodeQueue.push(vnode); } else { elm = vnode.elm = document.createTextNode(vnode.text); } return elm; } function sameVnode(vnode1, vnode2) { return vnode1.key === vnode2.key && vnode1.sel === vnode2.sel; } function createKeyToOldIdx(children, beginIdx, endIdx) { var i, map = {}, key; for (i = beginIdx; i <= endIdx; ++i) { key = children[i].key; if (!isUndef(key)) map[key] = i; } return map; } function addVnodes(parentElm, before, vnodes, startIdx, endIdx) { if (isUndef(before)) { for (; startIdx <= endIdx; ++startIdx) { parentElm.appendChild(createElm(vnodes[startIdx])); } } else { var elm = before.elm; for (; startIdx <= endIdx; ++startIdx) { parentElm.insertBefore(createElm(vnodes[startIdx]), elm); } } } function removeVnodes(parentElm, vnodes, startIdx, endIdx) { for (; startIdx <= endIdx; ++startIdx) { var ch = vnodes[startIdx]; if (!isUndef(ch)) { if (ch.data.onremove) { ch.data.onremove(ch, parentElm.removeChild.bind(parentElm, ch.elm)); } else { parentElm.removeChild(ch.elm); } ch.elm = undefined; } } } function updateChildren(parentElm, oldCh, newCh) { var oldStartIdx = 0, newStartIdx = 0; var oldEndIdx = oldCh.length - 1; var oldStartVnode = oldCh[0]; var oldEndVnode = oldCh[oldEndIdx]; var newEndIdx = newCh.length - 1; var newStartVnode = newCh[0]; var newEndVnode = newCh[newEndIdx]; var oldKeyToIdx, idxInOld, elmToMove; while (oldStartIdx <= oldEndIdx && newStartIdx <= newEndIdx) { if (isUndef(oldStartVnode)) { oldStartVnode = oldCh[++oldStartIdx]; // Vnode has been moved left } else if (isUndef(oldEndVnode)) { oldEndVnode = oldCh[--oldEndIdx]; } else if (sameVnode(oldStartVnode, newStartVnode)) { patchVnode(oldStartVnode, newStartVnode); oldStartVnode = oldCh[++oldStartIdx]; newStartVnode = newCh[++newStartIdx]; } else if (sameVnode(oldEndVnode, newEndVnode)) { patchVnode(oldEndVnode, newEndVnode); oldEndVnode = oldCh[--oldEndIdx]; newEndVnode = newCh[--newEndIdx]; } else if (sameVnode(oldStartVnode, newEndVnode)) { // Vnode moved right patchVnode(oldStartVnode, newEndVnode); parentElm.insertBefore(oldStartVnode.elm, oldEndVnode.elm.nextSibling); oldStartVnode = oldCh[++oldStartIdx]; newEndVnode = newCh[--newEndIdx]; } else if (sameVnode(oldEndVnode, newStartVnode)) { // Vnode moved left patchVnode(oldEndVnode, newStartVnode); parentElm.insertBefore(oldEndVnode.elm, oldStartVnode.elm); oldEndVnode = oldCh[--oldEndIdx]; newStartVnode = newCh[++newStartIdx]; } else { if (isUndef(oldKeyToIdx)) oldKeyToIdx = createKeyToOldIdx(oldCh, oldStartIdx, oldEndIdx); idxInOld = oldKeyToIdx[newStartVnode.key]; if (isUndef(idxInOld)) { // New element parentElm.insertBefore(createElm(newStartVnode), oldStartVnode.elm); newStartVnode = newCh[++newStartIdx]; } else { elmToMove = oldCh[idxInOld]; patchVnode(elmToMove, newStartVnode); oldCh[idxInOld] = undefined; parentElm.insertBefore(elmToMove.elm, oldStartVnode.elm); newStartVnode = newCh[++newStartIdx]; } } } if (oldStartIdx > oldEndIdx) addVnodes(parentElm, oldStartVnode, newCh, newStartIdx, newEndIdx); else if (newStartIdx > newEndIdx) removeVnodes(parentElm, oldCh, oldStartIdx, oldEndIdx); } function patchVnode(oldVnode, vnode) { var i, elm = vnode.elm = oldVnode.elm, oldCh = oldVnode.children, ch = vnode.children; if (!isUndef(vnode.data)) { for (i = 0; i < updateCbs.length; ++i) updateCbs[i](oldVnode, vnode); } if (isUndef(vnode.text)) { if (!isUndef(oldCh) && !isUndef(ch)) { updateChildren(elm, oldCh, ch); } else if (!isUndef(ch)) { addVnodes(elm, undefined, ch, 0, ch.length - 1); } else if (!isUndef(oldCh)) { removeVnodes(elm, oldCh, 0, oldCh.length - 1); } } else if (oldVnode.text !== vnode.text) { elm.childNodes[0].nodeValue = vnode.text; } return vnode; } function patch(oldVnode, vnode) { insertedVnodeQueue = []; patchVnode(oldVnode, vnode); for (var i = 0; i < insertedVnodeQueue.length; ++i) { insertedVnodeQueue[i].data.oninsert(insertedVnodeQueue[i]); } insertedVnodeQueue = undefined; return vnode; } var createCbs = []; var updateCbs = []; function init(modules) { modules.forEach(function(module) { if (module.create) createCbs.push(module.create); if (module.update) updateCbs.push(module.create); }); } module.exports = {createElm: createElm, init: init, patch: patch, emptyNodeAt: emptyNodeAt}; },{"./is":3,"./vnode":9}],9:[function(require,module,exports){ module.exports = function(sel, data, children, text, elm) { var key = data === undefined ? undefined : data.key; return {sel: sel, data: data, children: children, text: text, elm: elm, key: key}; }; },{}]},{},[1]);