Upgrade to Pro — share decks privately, control downloads, hide ads and more …

物極必反 - riot.js 與 angular.js 雜談

物極必反 - riot.js 與 angular.js 雜談

第七次聚會

1b56cc5159a07e4eee8f819c1a2557e9?s=128

Johnson Liang

November 29, 2013
Tweet

More Decks by Johnson Liang

Other Decks in Programming

Transcript

  1. 物極必反 riot.js 與 angular.js 雜談 13年11⽉月30⽇日星期六

  2. ≡ Johnson (MrOrz) ≡ Mobile HCI lab ≡ Dev @

    VUSE ≡ ࢿ㘤ܥ౷܇࿅൝ )5.-ߨࢣ 13年11⽉月30⽇日星期六
  3. Front-end Engineering 13年11⽉月30⽇日星期六

  4. Ever Changing 13年11⽉月30⽇日星期六

  5. 13年11⽉月30⽇日星期六

  6. 13年11⽉月30⽇日星期六

  7. https://moot.it/blog/technology/riotjs-the-1kb-mvp-framework.html 13年11⽉月30⽇日星期六

  8. https://moot.it/blog/technology/riotjs-the-1kb-mvp-framework.html 13年11⽉月30⽇日星期六

  9. 戰文 13年11⽉月30⽇日星期六

  10. 13年11⽉月30⽇日星期六

  11. 回來來 回來來 回來來 13年11⽉月30⽇日星期六

  12. Adopt a library 13年11⽉月30⽇日星期六

  13. JS Library 13年11⽉月30⽇日星期六

  14. JS Library ✓ API ॱ؟ 13年11⽉月30⽇日星期六

  15. JS Library ✓ API ॱ؟ ✓ࢥ૝ሣ௧ 13年11⽉月30⽇日星期六

  16. riot.js 13年11⽉月30⽇日星期六

  17. MVC is more complex. The many arrows form a circle.

    The role of the controller is not clear, and the pattern can be interpreted in many different ways. In fact, this is the root cause for the explosion of client-side frameworks. ❌ https://moot.it/blog/technology/riotjs-the-1kb-mvp-framework.html 13年11⽉月30⽇日星期六
  18. Riot uses Model-View-Presenter (MVP) design pattern to organize your code

    so that it's modular, testable and easy to understand. https://moot.it/blog/technology/riotjs-the-1kb-mvp-framework.html 13年11⽉月30⽇日星期六
  19. 13年11⽉月30⽇日星期六

  20. $(function() { / 1. Initialize / var  todo  =  new

     Todo(); / 2. Listen to user events /      $("#new-­‐todo").keyup(function(e)  {            var  val  =  $.trim(this.value);            if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }      });      $("#clear-­‐completed").click(function()  {          todo.remove("completed");      }) // ... / 3. Listen to model events / // an entry was edited      todo.on("add",  ...)              .on("remove",  function(items)  {                      $.each(items,  function()  {                            $("#"  +  this.id).remove()                      });              }); // ... }) 13年11⽉月30⽇日星期六
  21. $(function() { / 1. Initialize / var  todo  =  new

     Todo(); / 2. Listen to user events /      $("#new-­‐todo").keyup(function(e)  {            var  val  =  $.trim(this.value);            if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }      });      $("#clear-­‐completed").click(function()  {          todo.remove("completed");      }) // ... / 3. Listen to model events / // an entry was edited      todo.on("add",  ...)              .on("remove",  function(items)  {                      $.each(items,  function()  {                            $("#"  +  this.id).remove()                      });              }); // ... }) 13年11⽉月30⽇日星期六
  22. $(function() { / 1. Initialize / var  todo  =  new

     Todo(); / 2. Listen to user events /      $("#new-­‐todo").keyup(function(e)  {            var  val  =  $.trim(this.value);            if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }      });      $("#clear-­‐completed").click(function()  {          todo.remove("completed");      }) // ... / 3. Listen to model events / // an entry was edited      todo.on("add",  ...)              .on("remove",  function(items)  {                      $.each(items,  function()  {                            $("#"  +  this.id).remove()                      });              }); // ... }) 13年11⽉月30⽇日星期六
  23. $(function() { / 1. Initialize / var  todo  =  new

     Todo(); / 2. Listen to user events /      $("#new-­‐todo").keyup(function(e)  {            var  val  =  $.trim(this.value);            if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }      });      $("#clear-­‐completed").click(function()  {          todo.remove("completed");      }) // ... / 3. Listen to model events / // an entry was edited      todo.on("add",  ...)              .on("remove",  function(items)  {                      $.each(items,  function()  {                            $("#"  +  this.id).remove()                      });              }); // ... }) 13年11⽉月30⽇日星期六
  24. $(function() { / 1. Initialize / var  todo  =  new

     Todo(); / 2. Listen to user events /      $("#new-­‐todo").keyup(function(e)  {            var  val  =  $.trim(this.value);            if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }      });      $("#clear-­‐completed").click(function()  {          todo.remove("completed");      }) // ... / 3. Listen to model events / // an entry was edited      todo.on("add",  add).on("remove",  function(items)  {            $.each(items,  function()  {                  $("#"  +  this.id).remove()            });      }); // ... }) 13年11⽉月30⽇日星期六
  25. function  Todo(db)  {    var  items  =  [];    this.add

     =  function(name)  {        var  item  =  {  ...  }        items[item.id]  =  item;        this.trigger("add",  item);    }    this.remove  =  function(filter)  {        var  els  =  this.items(filter);        $.each(els,  function()  {            delete  items[this.id]        })        this.trigger("remove",  els);    }    $.observable(this); } $(function() { / 1. Initialize / var  todo  =  new  Todo(); / 2. Listen to user events /      $("#new-­‐todo").keyup(function(e)  {            var  val  =  $.trim(this.value);            if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }      });      $("#clear-­‐completed").click(function()  {          todo.remove("completed");      }) // ... / 3. Listen to model events / // an entry was edited      todo.on("add",  add).on("remove",  function(items)  {            $.each(items,  function()  {                  $("#"  +  this.id).remove()            });      }); // ... }) 13年11⽉月30⽇日星期六
  26. function  Todo(db)  {    var  items  =  [];    this.add

     =  function(name)  {        var  item  =  {  ...  }        items[item.id]  =  item;        this.trigger("add",  item);    }    this.remove  =  function(filter)  {        var  els  =  this.items(filter);        $.each(els,  function()  {            delete  items[this.id]        })        this.trigger("remove",  els);    }    $.observable(this); } $(function() { / 1. Initialize / var  todo  =  new  Todo(); / 2. Listen to user events /      $("#new-­‐todo").keyup(function(e)  {            var  val  =  $.trim(this.value);            if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }      });      $("#clear-­‐completed").click(function()  {          todo.remove("completed");      }) // ... / 3. Listen to model events / // an entry was edited      todo.on("add",  add).on("remove",  function(items)  {            $.each(items,  function()  {                  $("#"  +  this.id).remove()            });      }); // ... }) 13年11⽉月30⽇日星期六
  27. function  Todo(db)  {    var  items  =  [];    this.add

     =  function(name)  {        var  item  =  {  ...  }        items[item.id]  =  item;        this.trigger("add",  item);    }    this.remove  =  function(filter)  {        var  els  =  this.items(filter);        $.each(els,  function()  {            delete  items[this.id]        })        this.trigger("remove",  els);    }    $.observable(this); } $(function() { / 1. Initialize / var  todo  =  new  Todo(); / 2. Listen to user events /      $("#new-­‐todo").keyup(function(e)  {            var  val  =  $.trim(this.value);            if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }      });      $("#clear-­‐completed").click(function()  {          todo.remove("completed");      }) // ... / 3. Listen to model events / // an entry was edited      todo.on("add",  add).on("remove",  function(items)  {            $.each(items,  function()  {                  $("#"  +  this.id).remove()            });      }); // ... }) 13年11⽉月30⽇日星期六
  28. function  Todo(db)  {    var  items  =  [];    this.add

     =  function(name)  {        var  item  =  {  ...  }        items[item.id]  =  item;        this.trigger("add",  item);    }    this.remove  =  function(filter)  {        var  els  =  this.items(filter);        $.each(els,  function()  {            delete  items[this.id]        })        this.trigger("remove",  els);    }    $.observable(this); } $(function() { / 1. Initialize / var  todo  =  new  Todo(); / 2. Listen to user events /      $("#new-­‐todo").keyup(function(e)  {            var  val  =  $.trim(this.value);            if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }      });      $("#clear-­‐completed").click(function()  {          todo.remove("completed");      }) // ... / 3. Listen to model events / // an entry was edited      todo.on("add",  add).on("remove",  function(items)  {            $.each(items,  function()  {                  $("#"  +  this.id).remove()            });      }); // ... }) ࣄ݅ܥ౷ / Observer pattern trigger(eventName, args...) on(eventName, fn) off(eventName) 13年11⽉月30⽇日星期六
  29. (function() { / 1. Initialize / var  todo  =  new

     Todo(); / 2. Listen to user events /    $("#new-­‐todo").keyup(function(e)  {          var  val  =  $.trim(this.value);          if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }    });    $("#clear-­‐completed").click(function()  {        todo.remove("completed");    }) // ... / 3. Listen to model events / // an entry was edited    todo.on("add",  add).on("remove",  function(items)  {          $.each(items,  function()  {                $("#"  +  this.id).remove()          });    }); // ... ) 13年11⽉月30⽇日星期六
  30. (function() { / 1. Initialize / var  todo  =  new

     Todo(); / 2. Listen to user events /    $("#new-­‐todo").keyup(function(e)  {          var  val  =  $.trim(this.value);          if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }    });    $("#clear-­‐completed").click(function()  {        todo.remove("completed");    }) // ... / 3. Listen to model events / // an entry was edited    todo.on("add",  add).on("remove",  function(items)  {          $.each(items,  function()  {                $("#"  +  this.id).remove()          });    }); // ... ) 13年11⽉月30⽇日星期六
  31. (function() { / 1. Initialize / var  todo  =  new

     Todo(); / 2. Listen to user events /    $("#new-­‐todo").keyup(function(e)  {          var  val  =  $.trim(this.value);          if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }    });    $("#clear-­‐completed").click(function()  {        todo.remove("completed");    }) // ... / 3. Listen to model events / // an entry was edited    todo.on("add",  add).on("remove",  function(items)  {          $.each(items,  function()  {                $("#"  +  this.id).remove()          });    }); // ... ) <script  type="html/todo">      <li  id="{id}">      <div  class="view">            <input  class="toggle"  type="checkbox">            <label>{name}</label>            <button  class="destroy"/>      </div>      <input  class="edit"  value="{name}">      </li> </script> 13年11⽉月30⽇日星期六
  32. (function() { / 1. Initialize / var  todo  =  new

     Todo(); / 2. Listen to user events /    $("#new-­‐todo").keyup(function(e)  {          var  val  =  $.trim(this.value);          if  (e.which  ==  13  &&  val)  {            todo.add(val);  this.value  =  "";  }    });    $("#clear-­‐completed").click(function()  {        todo.remove("completed");    }) // ... / 3. Listen to model events / // an entry was edited    todo.on("add",  add).on("remove",  function(items)  {          $.each(items,  function()  {                $("#"  +  this.id).remove()          });    }); // ... ) <script  type="html/todo">      <li  id="{id}">      <div  class="view">            <input  class="toggle"  type="checkbox">            <label>{name}</label>            <button  class="destroy"/>      </div>      <input  class="edit"  value="{name}">      </li> </script> 13年11⽉月30⽇日星期六
  33. Observables are the key to splitting your app into maintainable

    components. It's a classic design pattern to separate the Model from the View. A good event library is the single most important feature in a client-side framework. And this is where Riot places the biggest focus. https://moot.it/blog/technology/riotjs-the-1kb-mvp-framework.html 13年11⽉月30⽇日星期六
  34. No "Backbone way", "Angular way" or "Ember way". Frameworks come

    and go but classic programming skills are forever. https://moot.it/blog/technology/riotjs-the-1kb-mvp-framework.html 13年11⽉月30⽇日星期六
  35. What does Angular say? 13年11⽉月30⽇日星期六

  36. 13年11⽉月30⽇日星期六

  37. ؃ෆ㣛 13年11⽉月30⽇日星期六

  38. AngularJS Angular JS 13年11⽉月30⽇日星期六

  39. 13年11⽉月30⽇日星期六

  40. 13年11⽉月30⽇日星期六

  41. kytu tomchen 13年11⽉月30⽇日星期六

  42. 13年11⽉月30⽇日星期六

  43. ࣗݾఆٛ Element / Attribute 13年11⽉月30⽇日星期六

  44. ࣗݾఆٛ Element / Attribute Dependency Injection 13年11⽉月30⽇日星期六

  45. ࣗݾఆٛ Element / Attribute Dependency Injection Data- binding 13年11⽉月30⽇日星期六

  46. http://www.thinkster.io/pick/GtaQ0oMGIl/ a-better-way-to-learn-angularjs 13年11⽉月30⽇日星期六

  47. 13年11⽉月30⽇日星期六

  48. ࣗݾఆٛ Element / Attribute 13年11⽉月30⽇日星期六

  49. <button ng-click=”post(‘bob’)”></button> 13年11⽉月30⽇日星期六

  50. Current data-binding frameworks promote the use of spaghetti on the

    HTML layer. Suddenly the onclick attribute is back! (I'm looking at you, Angular). Riot takes a puristic approach and does not allow you to do mix any logic inside HTML views. This is also the reason why Riot templating is fast – the templating logic is so simple. 13年11⽉月30⽇日星期六
  51. <button ng-click=”post(‘bob’)”></button> <button onClick=”post(‘bob’);”></button> 1999 2013 13年11⽉月30⽇日星期六

  52. <button ng-click=”post(‘bob’)”></button> <button onClick=”post(‘bob’);”></button> 1999 2013 2002 <button class=”post”></button> $(‘.post’).click(function(){...})

    13年11⽉月30⽇日星期六
  53. Unobtrusive Javascript ≡ Usability - ሣ࢖༻ऀိ㘸௚᧷ ≡ Graceful degration -

    ҆ᯩग़ࡨ ≡ Accessibility - ग़ࡨ࣌နೳఏڙ֩৺ޭೳ ≡ Separation of concern - ෼։+4࿨)5.- http://en.wikipedia.org/wiki/Unobtrusive_JavaScript 13年11⽉月30⽇日星期六
  54. You need a way of saying "Add the behavior here",

    just as CSS requires a way of saying "Use these styles here". In the mouseover example the id="mouseover" fulfills this function, but you could also use other attributes, or add behavior to, say, all <span>s that don’t have a class. -- ppk, quirksmode.org, 2004 13年11⽉月30⽇日星期六
  55. <button ng-click=”post(‘bob’)”></button> 2013 13年11⽉月30⽇日星期六

  56. Views should be Declarative. 13年11⽉月30⽇日星期六

  57. HTML is Declarative. 13年11⽉月30⽇日星期六

  58. http://courses.csail.mit.edu/6.831/archive/2008/lectures/L11-declarative-ui/L11-declarative-ui.html 13年11⽉月30⽇日星期六

  59. Declarative Procedural (Imperative) 13年11⽉月30⽇日星期六

  60. Declarative Procedural (Imperative) Ұൠ programming language 13年11⽉月30⽇日星期六

  61. http://courses.csail.mit.edu/6.831/archive/2008/lectures/L11-declarative-ui/L11-declarative-ui.html 13年11⽉月30⽇日星期六

  62. http://courses.csail.mit.edu/6.831/archive/2008/lectures/L11-declarative-ui/L11-declarative-ui.html 13年11⽉月30⽇日星期六

  63. Declaration is Clear 13年11⽉月30⽇日星期六

  64. Why can’t behavior of an User Interface be Declarative? 13年11⽉月30⽇日星期六

  65. <button class=”post”></button> 13年11⽉月30⽇日星期六

  66. <button class=”post”></button> CSS? Has javascript? 13年11⽉月30⽇日星期六

  67. Discoverability Unobtrusive JS is too “unobtrusive” for declaring interactions. 13年11⽉月30⽇日星期六

  68. <button class=”post”></button> <button ng-click=”post(‘bob’)”></button> CSS? Has javascript? Find post() 13年11⽉月30⽇日星期六

  69. <button ng-click=”post(‘bob’)”></button> <button onClick=”post(‘bob’);”></button> Old Way AngularJS ࣥߦ೚ҙJSᏌड़ context: window

    ᷮ။ࣥߦಛఆexpression context: current scope object 13年11⽉月30⽇日星期六
  70. ࣗఆݩૉሱੑ http://angular-ui.github.io/bootstrap/    <carousel  interval="myInterval">        <slide  ng-­‐repeat="slide

     in  slides"  active="slide.active">            <img  ng-­‐src="{{slide.image}}"  style="margin:auto;">            <div  class="carousel-­‐caption">                <h4>Slide  {{$index}}</h4>                <p>{{slide.text}}</p>            </div>        </slide>    </carousel> 13年11⽉月30⽇日星期六
  71. ࣗఆݩૉሱੑ http://angular-ui.github.io/bootstrap/    <carousel  interval="myInterval">        <slide  ng-­‐repeat="slide

     in  slides"  active="slide.active">            <img  ng-­‐src="{{slide.image}}"  style="margin:auto;">            <div  class="carousel-­‐caption">                <h4>Slide  {{$index}}</h4>                <p>{{slide.text}}</p>            </div>        </slide>    </carousel> ݩૉ 13年11⽉月30⽇日星期六
  72. ࣗఆݩૉሱੑ http://angular-ui.github.io/bootstrap/    <carousel  interval="myInterval">        <slide  ng-­‐repeat="slide

     in  slides"  active="slide.active">            <img  ng-­‐src="{{slide.image}}"  style="margin:auto;">            <div  class="carousel-­‐caption">                <h4>Slide  {{$index}}</h4>                <p>{{slide.text}}</p>            </div>        </slide>    </carousel> ݩૉ ၏᫮ᅲత ሱੑ 13年11⽉月30⽇日星期六
  73. ఆٛ<slide> 13年11⽉月30⽇日星期六

  74. ఆٛ<slide> ≡ ໵ੋ declarative! app.directive('slide',  ['$parse',  function($parse)  {    return

     {        require:  '^carousel',        restrict:  'EA',        transclude:  true,        replace:  true,        templateUrl:  'template/carousel/slide.html',        scope:  {        },        link:  function  (scope,  element,  attrs,  carouselCtrl)  {            //Set  up  optional  'active'  =  binding            scope.$watch(function  parentActiveWatch()  {                var  parentActive  =  getActive(scope.$parent);                if  (parentActive  !==  scope.active)  {                    //  we  are  out  of  sync  and  need  to  copy                    if  (parentActive  !==  lastValue)  { 13年11⽉月30⽇日星期六
  75. ఆٛ<slide> ≡ ໵ੋ declarative! app.directive('slide',  ['$parse',  function($parse)  {    return

     {        require:  '^carousel',        restrict:  'EA',        transclude:  true,        replace:  true,        templateUrl:  'template/carousel/slide.html',        scope:  {        },        link:  function  (scope,  element,  attrs,  carouselCtrl)  {            //Set  up  optional  'active'  =  binding            scope.$watch(function  parentActiveWatch()  {                var  parentActive  =  getActive(scope.$parent);                if  (parentActive  !==  scope.active)  {                    //  we  are  out  of  sync  and  need  to  copy                    if  (parentActive  !==  lastValue)  { ਎ҝҰݸ ݩૉతੑ࣭ 13年11⽉月30⽇日星期六
  76. ఆٛ<slide> ≡ ໵ੋ declarative! app.directive('slide',  ['$parse',  function($parse)  {    return

     {        require:  '^carousel',        restrict:  'EA',        transclude:  true,        replace:  true,        templateUrl:  'template/carousel/slide.html',        scope:  {        },        link:  function  (scope,  element,  attrs,  carouselCtrl)  {            //Set  up  optional  'active'  =  binding            scope.$watch(function  parentActiveWatch()  {                var  parentActive  =  getActive(scope.$parent);                if  (parentActive  !==  scope.active)  {                    //  we  are  out  of  sync  and  need  to  copy                    if  (parentActive  !==  lastValue)  { ਎ҝҰݸ ݩૉతੑ࣭ ֎᧺ަڅ)5.- ʢߦҝ༻OHDMJDL౳ ัଊʣ 13年11⽉月30⽇日星期六
  77. ఆٛ<slide> ≡ ໵ੋ declarative! app.directive('slide',  ['$parse',  function($parse)  {    return

     {        require:  '^carousel',        restrict:  'EA',        transclude:  true,        replace:  true,        templateUrl:  'template/carousel/slide.html',        scope:  {        },        link:  function  (scope,  element,  attrs,  carouselCtrl)  {            //Set  up  optional  'active'  =  binding            scope.$watch(function  parentActiveWatch()  {                var  parentActive  =  getActive(scope.$parent);                if  (parentActive  !==  scope.active)  {                    //  we  are  out  of  sync  and  need  to  copy                    if  (parentActive  !==  lastValue)  { ਎ҝҰݸ ݩૉతੑ࣭ ֎᧺ަڅ)5.- ʢߦҝ༻OHDMJDL౳ ัଊʣ መࡍૢ࡞%0. 13年11⽉月30⽇日星期六
  78. User Interface can be Declarative. 13年11⽉月30⽇日星期六

  79. ݁࿦ 13年11⽉月30⽇日星期六

  80. )5.-ੋจ݅ ୞֘؅Սߏ )5.-ੋ6* ጯඳड़ߦҝ 13年11⽉月30⽇日星期六

  81. ҙࣝܕଶ Өڹ API 13年11⽉月30⽇日星期六

  82. More generally, JavaScript best practices often parallel those in other

    programming languages, such as encapsulation and abstraction layers, avoidance of global variables, meaningful naming conventions, use of appropriate design patterns, and systematic testing. Such principles are essential to large-scale software development, but have not been widely observed in JavaScript programming in the past; their adoption is seen as an essential component of JavaScript's transition from a "toy" language to a tool for serious development. “Unubstrusive Javascript”, Wikipedia 13年11⽉月30⽇日星期六