Skip to content

HTTPS clone URL

Subversion checkout URL

You can clone with HTTPS or Subversion.

Download ZIP
Browse files

More docs, more efficiency, more better.

  • Loading branch information...
commit 7272301d9a193f2f004a7af22df130bbab687165 1 parent a21c640
@ialexi authored
Showing with 1,050 additions and 33 deletions.
  1. +3 −2 apps/hedwig/resources/guide/touch.guide
  2. +1 −1  apps/hedwig/resources/guide/touch.json
  3. +5 −4 apps/hedwig/views/article_view.js
  4. +21 −4 docs/build/articles/touch/a-brief-touch.html
  5. +1 −1  docs/build/articles/touch/a-brief-touch.json
  6. +21 −5 docs/build/articles/touch/a-brief-touch.md
  7. +189 −0 docs/build/articles/touch/multitouch.html
  8. +78 −0 docs/build/articles/touch/multitouch.js
  9. +1 −0  docs/build/articles/touch/multitouch.json
  10. +94 −0 docs/build/articles/touch/multitouch.md
  11. +17 −0 docs/build/articles/touch/touch-demo.js
  12. +115 −0 docs/build/articles/touch/touch-events.html
  13. +1 −1  docs/build/articles/touch/touch-events.json
  14. +113 −1 docs/build/articles/touch/touch-events.md
  15. +3 −2 docs/build/guides/touch.guide
  16. +1 −1  docs/build/guides/touch.json
  17. +21 −5 docs/src/articles/touch/a-brief-touch.md
  18. +78 −0 docs/src/articles/touch/multitouch.js
  19. +94 −0 docs/src/articles/touch/multitouch.md
  20. +17 −0 docs/src/articles/touch/touch-demo.js
  21. +113 −1 docs/src/articles/touch/touch-events.md
  22. +3 −2 docs/src/guides/touch.guide
  23. BIN  themes/pig/resources/images/2.png
  24. +31 −2 themes/pig/resources/theme.css
  25. +24 −0 themes/pig/src/content.css
  26. +4 −0 themes/pig/src/paper/paper.css
  27. BIN  themes/pig/src/tree/bg.jpg
  28. +1 −1  themes/pig/src/tree/tree.css
View
5 apps/hedwig/resources/guide/touch.guide
@@ -1,11 +1,12 @@
{
- "title": "Writing Touch Applications",
+ "title": "Touch Application Guide",
"sections": [
{
"title": "Introduction",
"articles": [
"articles/touch/a-brief-touch",
- "articles/touch/touch-events"
+ "articles/touch/touch-events",
+ "articles/touch/multitouch"
]
},
{
View
2  apps/hedwig/resources/guide/touch.json
@@ -1 +1 @@
-{"title":"Writing Touch Applications","sections":[{"title":"Introduction","articles":[{"content":"<h1>A Brief Touch on Touches</h1>\n\n<p>Some text goes here.</p>\n\n<p>Note that links like the below must appear on empty lines with no preceding whitespace.</p>\n\n<p><a href='touch-demo.js' class='demo'>touch-demo.js</a></p>","errors":[],"demos":{"touch-demo.js":{"ex":"var Dot = SC.View.extend({\n touchStart: function(touch) {\n var f = this.get(\"frame\");\n this._touch = {\n start: { x: touch.pageX, y: touch.pageY },\n ourStart: { x: f.x, y: f.y, width: f.width, height: f.height }\n };\n return YES; // or we won't get touchesDragged\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n this.set(\"layout\", { \n left: t.ourStart.x + evt.pageX - t.start.x,\n top: t.ourStart.y + evt.pageY - t.start.y,\n width: t.ourStart.width,\n height: t.ourStart.height\n });\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"dot1 dot2\".w(),\n dot1: Dot.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 100, height: 100 }\n }),\n dot2: Dot.design({\n backgroundColor: \"blue\",\n layout: { right: 10, bottom: 10, width: 100, height: 100 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n","highlighted":"<span class=\"keyword\">var</span> <span class=\"class\">Dot</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">touchStart</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">f</span> = <span class=\"this\">this</span>.<span class=\"variable\">get</span>(<span class=\"string\">&quot;frame&quot;</span>);\n <span class=\"this\">this</span>.<span class=\"variable\">_touch</span> = {\n <span class=\"variable\">start</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">touch</span>.<span class=\"variable\">pageX</span>, <span class=\"variable\">y</span>: <span class=\"variable\">touch</span>.<span class=\"variable\">pageY</span> },\n <span class=\"variable\">ourStart</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">f</span>.<span class=\"variable\">x</span>, <span class=\"variable\">y</span>: <span class=\"variable\">f</span>.<span class=\"variable\">y</span>, <span class=\"variable\">width</span>: <span class=\"variable\">f</span>.<span class=\"variable\">width</span>, <span class=\"variable\">height</span>: <span class=\"variable\">f</span>.<span class=\"variable\">height</span> }\n };\n <span class=\"keyword\">return</span> <span class=\"class\">YES</span>; <span class=\"comment\">// or we won't get touchesDragged</span>\n },\n \n <span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">t</span> = <span class=\"this\">this</span>.<span class=\"variable\">_touch</span>;\n <span class=\"this\">this</span>.<span class=\"variable\">set</span>(<span class=\"string\">&quot;layout&quot;</span>, { \n <span class=\"variable\">left</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">x</span> + <span class=\"variable\">evt</span>.<span class=\"variable\">pageX</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">x</span>,\n <span class=\"variable\">top</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">y</span> + <span class=\"variable\">evt</span>.<span class=\"variable\">pageY</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">y</span>,\n <span class=\"variable\">width</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">width</span>,\n <span class=\"variable\">height</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">height</span>\n });\n }\n});\n<span class=\"keyword\">var</span> <span class=\"class\">MyExampleView</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;white&quot;</span>,\n <span class=\"variable\">childViews</span>: <span class=\"string\">&quot;dot1 dot2&quot;</span>.<span class=\"variable\">w</span>(),\n <span class=\"variable\">dot1</span>: <span class=\"class\">Dot</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;red&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">left</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">top</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">100</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">100</span> }\n }),\n <span class=\"variable\">dot2</span>: <span class=\"class\">Dot</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;blue&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">right</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">bottom</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">100</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">100</span> }\n })\n});\n\n<span class=\"comment\">// bootstrap code :)</span>\n<span class=\"variable\">exports</span>.<span class=\"variable\">getDemoView</span> = <span class=\"keyword\">function</span>() {\n <span class=\"keyword\">return</span> <span class=\"class\">MyExampleView</span>;\n};\n","original":"var Dot = SC.View.extend({\n touchStart: function(touch) {\n var f = this.get(\"frame\");\n this._touch = {\n start: { x: touch.pageX, y: touch.pageY },\n ourStart: { x: f.x, y: f.y, width: f.width, height: f.height }\n };\n return YES; // or we won't get touchesDragged\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n this.set(\"layout\", { \n left: t.ourStart.x + evt.pageX - t.start.x,\n top: t.ourStart.y + evt.pageY - t.start.y,\n width: t.ourStart.width,\n height: t.ourStart.height\n });\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"dot1 dot2\".w(),\n dot1: Dot.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 100, height: 100 }\n }),\n dot2: Dot.design({\n backgroundColor: \"blue\",\n layout: { right: 10, bottom: 10, width: 100, height: 100 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n"}},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"A Brief Touch on Touches","any":"metadata","goes":"Here","damn":"gruber","this":"is still eye-readable","and":"He is wrong about touch apps."},{"content":"<h1>Touch Events</h1>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Touch Events"}]},{"title":"Built-In Support","articles":[{"content":"<h1>ButtonView Touch Support</h1>","errors":[],"demos":{},"articleDirectory":"articles/controls/button/","outputDirectory":"build/","title":"ButtonView Touch Support"},{"content":"<h1>ScrollView Touch Support</h1>","errors":[],"demos":{},"articleDirectory":"articles/controls/scroll/","outputDirectory":"build/","title":"ScrollView Touch Support"}]},{"title":"Advanced Concepts","articles":[{"content":"<h1>Capturing</h1>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Capturing"},{"content":"<h1>Releasing</h1>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Releasing"}]}],"file":"build/guides/touch"}
+{"title":"Touch Application Guide","sections":[{"title":"Introduction","articles":[{"content":"<h1>A Brief Touch</h1>\n\n<p>It is very possible to build <em>awesome</em> touch-enabled applications in SproutCore.</p>\n\n<p>But, what makes an awesome touch-enabled application? Sure, it must accept touches,\nbut with SproutCore's (constantly growing) touch support, this is now pretty easy:\nmany existing interfaces, if built with newer SproutCore varieties, will function fine (or mostly fine),\non both larger-screened touch devices (such as iPad) and the traditional desktop environment.</p>\n\n<p>But there are many differences between desktop and touch platforms:</p>\n\n<ul>\n<li><strong>Precision.</strong> Touches are less precise than clicks. To compensate, controls should be larger.</li>\n<li><strong>Performance.</strong> Touch-based devices tend to be slow (for now). To get around this just takes \nsome elbow grease: there are many techniques to speed things up... many of which SproutCore will\nhandle for you.</li>\n<li><strong>Animation.</strong> Lack of animation looks okay on desktop (even if animation is cool)... but on\ntouch devices, non-animated interfaces look strange: touch is so much more realistic than mouse,\nbut sudden changes without transitions are not realistic at all.</li>\n<li><strong>Coolness.</strong> Touch-based interfaces are cool. That is all.</li>\n</ul>\n\n<p>In this guide, we go over each of these&emdash;except for the last, which is rather vague; you'll have to figure out\nyour own meaning of \"coolness\".</p>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"A Brief Touch","any":"metadata","goes":"Here","damn":"gruber","this":"is still eye-readable","and":"He is wrong about touch apps."},{"content":"<h1>Touch Events</h1>\n\n<p>SproutCore's touch events have a few great features:</p>\n\n<ul>\n<li>Multiple views can receive touches simultaneously.</li>\n<li>A single view can receive multiple touches.</li>\n<li>A view can capture touches before allowing them to pass through to children.</li>\n<li>Child views can release touches back to parent views that originally captured them.</li>\n</ul>\n\n<p>We won't get into the last two in this article&emdash;they're quite sophisticated!</p>\n\n<h1>Simple Single-Touch Handling</h1>\n\n<p>You may be familiar with this SC.View method signature if you are familiar with SproutCore:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">mouseDown</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>)\n</code></pre>\n\n<p>With <code class='syntax js'><span class=\"variable\">mouseDown</span></code>, you can decide whether or not to accept the mouse event by returning either\n<code class='syntax js'><span class=\"class\">YES</span></code> or <code class='syntax js'><span class=\"class\">NO</span></code>.</p>\n\n<p>Touch events are similar, but work a bit differently:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">touchStart</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>)\n</code></pre>\n\n<p>Instead of being passed a raw event, <code class='syntax js'><span class=\"variable\">touchStart</span></code> is passed an <code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">Touch</span></code> object.\nIf you return YES from touchStart, your view will \"own\" the touch\u2014in SproutCore terms, your\nview will be the <em>touch responder</em>. For more information about what this entails, see\nthe \"Internals\" article. Your view will own the touch until the touch ends.</p>\n\n<p><code class='syntax js'><span class=\"variable\">touchStart</span></code> will be called once for every touch that touches the view.</p>\n\n<p><strong>Note:</strong> By default, views only receive <code class='syntax js'><span class=\"variable\">touchStart</span></code> and touchEnd for a single touch. This is\na feature intended to make it easier to handle such cases, very similar to Cocoa Touch's <code class='syntax js'><span class=\"variable\">acceptsMultitouch</span></code>\nproperty\u2014actually, SC.View uses the same property name! See the \"Multitouch\" article.</p>\n\n<h2>Anatomy of an SC.Touch</h2>\n\n<p>SC.Touch represents the touch from the time the user puts their finger on the screen until the time they lift it.</p>\n\n<p>The touch object acts like an event object in many ways. It has many other useful things, as well:</p>\n\n<ul>\n<li><code class='syntax js'><span class=\"variable\">pageX</span></code> and <code class='syntax js'><span class=\"variable\">pageY</span></code> for the touch</li>\n<li><code class='syntax js'><span class=\"variable\">event</span></code>: if in an event cycle, this contains the event. Otherwise, it is <code class='syntax js'><span class=\"variable\">undefined</span></code>. \nYou will probably never need to access this.</li>\n<li><code class='syntax js'><span class=\"variable\">preventDefault</span></code>: if the touch is connected with an event, this calls <code class='syntax js'><span class=\"variable\">preventDefault</span>()</code> on the event.</li>\n<li><code class='syntax js'><span class=\"variable\">touchesForView</span>(<span class=\"variable\">view</span>)</code>: when supplied a view, will find all touches that the view is the\ntouch responder for. It is a CoreSet; to get an array, call <code class='syntax js'>.<span class=\"variable\">toArray</span></code> on the result.</li>\n<li><code class='syntax js'><span class=\"variable\">averagedTouchesForView</span>(<span class=\"variable\">view</span>)</code>: When supplied a view, averages all the touches on that view,\nreturning both an average position and an average distance from that position.</li>\n</ul>\n\n<p><strong>Note:</strong> If you call <code class='syntax js'><span class=\"variable\">touchesForView</span>(<span class=\"this\">this</span>)</code> from <code class='syntax js'><span class=\"variable\">touchStart</span></code>, the touch supplied will not be in the set\nreturned by <code class='syntax js'><span class=\"variable\">touchesForView</span>(<span class=\"this\">this</span>)</code>: you don't own the touch until you return YES.</p>\n\n<h2>touchEnd</h2>\n\n<p>Knowing when the touch starts is not very useful. At the very least, you will likely want to know when the touch\nends as well.</p>\n\n<p>It is quite simple:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">touchEnd</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>)\n</code></pre>\n\n<p>It works exactly like <code class='syntax js'><span class=\"variable\">touchStart</span></code>. The touch will no longer be in the set of touches for the view,\nso if you call <code class='syntax js'><span class=\"variable\">touch</span>.<span class=\"variable\">touchesForView</span>(<span class=\"this\">this</span>)</code>, you'll only receive any other active touches. If your\nview does not accept multitouch, then the set is guaranteed to have no touches in it\u2014you only receive\n<code class='syntax js'><span class=\"variable\">touchEnd</span></code> for the last touch that ends.</p>\n\n<h2>Tracking Touches</h2>\n\n<p>Tracking touch movement is simple:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>)\n</code></pre>\n\n<p>The <code class='syntax js'><span class=\"variable\">touches</span></code> argument is the set of touches on the view\u2014the same set you get by calling <code class='syntax js'><span class=\"variable\">touchesForView</span>(<span class=\"this\">this</span>)</code>.\nThis will have <em>all</em> touches, regardless of whether or not your view accepts multitouch.</p>\n\n<p>If your view does <em>not</em> accept multitouch, then it is even simpler:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">x</span> = <span class=\"variable\">evt</span>.<span class=\"variable\">pageX</span>;\n<span class=\"variable\">y</span> = <span class=\"variable\">evt</span>.<span class=\"variable\">pageY</span>;\n</code></pre>\n\n<h2>Tip: Cross-Platform-iness</h2>\n\n<p>Did you realize that, assuming you don't use the set of view touches or other touch-specific API,\nyou can do this:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">mouseDown</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchStart</span>(<span class=\"variable\">evt</span>);\n},\n\n<span class=\"variable\">mouseDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchesDragged</span>(<span class=\"variable\">evt</span>);\n},\n\n<span class=\"variable\">mouseUp</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchEnd</span>(<span class=\"variable\">evt</span>);\n}\n</code></pre>\n\n<p>Of course, you can also redirect touch events to mouse events, but that is not as fun.</p>\n\n<h2>Putting it All Together</h2>\n\n<p>Here is a very simple demo that uses the methods described above to allow the user to move two views\naround the screen:</p>\n\n<p><a href='touch-demo.js' class='demo'>touch-demo.js</a></p>","errors":[],"demos":{"touch-demo.js":{"ex":"var Dot = SC.View.extend({\n touchStart: function(touch) {\n var f = this.get(\"frame\");\n this._touch = {\n start: { x: touch.pageX, y: touch.pageY },\n ourStart: { x: f.x, y: f.y, width: f.width, height: f.height }\n };\n return YES; // or we won't get touchesDragged\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n this.set(\"layout\", { \n left: t.ourStart.x + evt.pageX - t.start.x,\n top: t.ourStart.y + evt.pageY - t.start.y,\n width: t.ourStart.width,\n height: t.ourStart.height\n });\n },\n \n touchEnd: function() {\n // actually, we don't need to do anything here...\n },\n \n // and now, redirect mouse events :)\n mouseDown: function(evt) {\n this.touchStart(evt);\n },\n \n mouseDragged: function(evt) {\n this.touchesDragged(evt);\n },\n \n mouseUp: function(evt) {\n this.touchEnd(evt);\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"dot1 dot2\".w(),\n dot1: Dot.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 100, height: 100 }\n }),\n dot2: Dot.design({\n backgroundColor: \"blue\",\n layout: { right: 10, bottom: 10, width: 100, height: 100 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n","highlighted":"<span class=\"keyword\">var</span> <span class=\"class\">Dot</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">touchStart</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">f</span> = <span class=\"this\">this</span>.<span class=\"variable\">get</span>(<span class=\"string\">&quot;frame&quot;</span>);\n <span class=\"this\">this</span>.<span class=\"variable\">_touch</span> = {\n <span class=\"variable\">start</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">touch</span>.<span class=\"variable\">pageX</span>, <span class=\"variable\">y</span>: <span class=\"variable\">touch</span>.<span class=\"variable\">pageY</span> },\n <span class=\"variable\">ourStart</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">f</span>.<span class=\"variable\">x</span>, <span class=\"variable\">y</span>: <span class=\"variable\">f</span>.<span class=\"variable\">y</span>, <span class=\"variable\">width</span>: <span class=\"variable\">f</span>.<span class=\"variable\">width</span>, <span class=\"variable\">height</span>: <span class=\"variable\">f</span>.<span class=\"variable\">height</span> }\n };\n <span class=\"keyword\">return</span> <span class=\"class\">YES</span>; <span class=\"comment\">// or we won't get touchesDragged</span>\n },\n \n <span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">t</span> = <span class=\"this\">this</span>.<span class=\"variable\">_touch</span>;\n <span class=\"this\">this</span>.<span class=\"variable\">set</span>(<span class=\"string\">&quot;layout&quot;</span>, { \n <span class=\"variable\">left</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">x</span> + <span class=\"variable\">evt</span>.<span class=\"variable\">pageX</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">x</span>,\n <span class=\"variable\">top</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">y</span> + <span class=\"variable\">evt</span>.<span class=\"variable\">pageY</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">y</span>,\n <span class=\"variable\">width</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">width</span>,\n <span class=\"variable\">height</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">height</span>\n });\n },\n \n <span class=\"variable\">touchEnd</span>: <span class=\"keyword\">function</span>() {\n <span class=\"comment\">// actually, we don't need to do anything here...</span>\n },\n \n <span class=\"comment\">// and now, redirect mouse events :)</span>\n <span class=\"variable\">mouseDown</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchStart</span>(<span class=\"variable\">evt</span>);\n },\n \n <span class=\"variable\">mouseDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchesDragged</span>(<span class=\"variable\">evt</span>);\n },\n \n <span class=\"variable\">mouseUp</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchEnd</span>(<span class=\"variable\">evt</span>);\n }\n});\n<span class=\"keyword\">var</span> <span class=\"class\">MyExampleView</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;white&quot;</span>,\n <span class=\"variable\">childViews</span>: <span class=\"string\">&quot;dot1 dot2&quot;</span>.<span class=\"variable\">w</span>(),\n <span class=\"variable\">dot1</span>: <span class=\"class\">Dot</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;red&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">left</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">top</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">100</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">100</span> }\n }),\n <span class=\"variable\">dot2</span>: <span class=\"class\">Dot</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;blue&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">right</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">bottom</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">100</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">100</span> }\n })\n});\n\n<span class=\"comment\">// bootstrap code :)</span>\n<span class=\"variable\">exports</span>.<span class=\"variable\">getDemoView</span> = <span class=\"keyword\">function</span>() {\n <span class=\"keyword\">return</span> <span class=\"class\">MyExampleView</span>;\n};\n","original":"var Dot = SC.View.extend({\n touchStart: function(touch) {\n var f = this.get(\"frame\");\n this._touch = {\n start: { x: touch.pageX, y: touch.pageY },\n ourStart: { x: f.x, y: f.y, width: f.width, height: f.height }\n };\n return YES; // or we won't get touchesDragged\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n this.set(\"layout\", { \n left: t.ourStart.x + evt.pageX - t.start.x,\n top: t.ourStart.y + evt.pageY - t.start.y,\n width: t.ourStart.width,\n height: t.ourStart.height\n });\n },\n \n touchEnd: function() {\n // actually, we don't need to do anything here...\n },\n \n // and now, redirect mouse events :)\n mouseDown: function(evt) {\n this.touchStart(evt);\n },\n \n mouseDragged: function(evt) {\n this.touchesDragged(evt);\n },\n \n mouseUp: function(evt) {\n this.touchEnd(evt);\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"dot1 dot2\".w(),\n dot1: Dot.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 100, height: 100 }\n }),\n dot2: Dot.design({\n backgroundColor: \"blue\",\n layout: { right: 10, bottom: 10, width: 100, height: 100 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n"}},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Touch Events"},{"content":"<h1>Multitouch</h1>\n\n<p>Handling single touches is pretty easy&emdash;not that much different from handling\nmouse events. But what about multiple touches?</p>\n\n<h2>Accepting Multiple Touches</h2>\n\n<p>First, you have to tell your view that you do, indeed want to receive multiple\ntouches. By default, views only receive single touches. This is because is easier \nto think in a single-touch model, and most controls only need to track a single touch.</p>\n\n<p>To accept multiple touches, just set the view's <code class='syntax js'><span class=\"variable\">acceptsMultitouch</span></code> property to\n<code class='syntax js'><span class=\"class\">YES</span></code>.</p>\n\n<pre><code class='syntax js'><span class=\"variable\">view</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">acceptsMultitouch</span>: <span class=\"class\">YES</span>\n});\n</code></pre>\n\n<h2>Processing the Individual Touches</h2>\n\n<p>Even without the supplied helper function, processing individual touches is relatively\nsimple:</p>\n\n<ul>\n<li>You get a separate <code class='syntax js'><span class=\"variable\">touchStart</span></code> for each individual touch.</li>\n<li>You get a separate <code class='syntax js'><span class=\"variable\">touchEnd</span></code> for each individual touch.</li>\n<li>You get one <code class='syntax js'><span class=\"variable\">touchesDragged</span></code> each event cycle for all of your touches put together.</li>\n</ul>\n\n<p>So, detecting individual touches starting and ending is simple. Detecting those touches\nmoving is not quite as simple, but still relatively easy.</p>\n\n<p>Remember how <code class='syntax js'><span class=\"variable\">touchesDragged</span></code> works:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>)\n</code></pre>\n\n<p><code class='syntax js'><span class=\"variable\">touches</span></code> is an SC.CoreSet of SC.Touch objects. What can you do with a CoreSet? \nYou can do a couple of things:</p>\n\n<ul>\n<li>Turn it into an array and do whatever.</li>\n<li>Call .forEach to iterate.</li>\n</ul>\n\n<p>But you don't have to use the touches set at all. The <code class='syntax js'><span class=\"variable\">evt</span></code> has some useful properties and methods,\ntoo:</p>\n\n<ul>\n<li>pageX/pageY: the position of the first touch.</li>\n<li>averagedTouchesForView: a method which returns the averaged touch position\nand the average distance of the touches from that position.</li>\n</ul>\n\n<p><code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">ScrollView</span></code>, for instance, makes heavy use of <code class='syntax js'><span class=\"variable\">averagedTouchesForView</span></code>, and never\ndirectly touches the <code class='syntax js'><span class=\"variable\">touches</span></code> set.</p>\n\n<h2>Averaging Touches</h2>\n\n<p>It is often <em>very</em> useful to average the touches. </p>\n\n<p><code class='syntax js'><span class=\"variable\">averagedTouchesForView</span></code> returns an object with four properties:</p>\n\n<ul>\n<li><strong><code class='syntax js'><span class=\"variable\">x</span></code></strong>: The average X position of the touch.</li>\n<li><strong><code class='syntax js'><span class=\"variable\">y</span></code></strong>: The average Y position of the touch.</li>\n<li><strong><code class='syntax js'><span class=\"variable\">d</span></code></strong>: average distance of the all touches from the average x/y position.</li>\n<li><strong><code class='syntax js'><span class=\"variable\">touchCount</span></code></strong> The number of touches averaged.</li>\n</ul>\n\n<p>You can call <code class='syntax js'><span class=\"variable\">averagedTouchesForView</span></code> on two separate objects: an <code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">Event</span></code> object,\nor an <code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">Touch</span></code> object.</p>\n\n<p>The two work identically but for one important difference: when you call it on <code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">Touch</span></code>,\nyou <em>have the option</em> of telling the touch to add itself to the averaged set. Doing so makes no sense\nin most cases: the touch would just be counted twice! But what about <code class='syntax js'><span class=\"variable\">touchStart</span></code>?</p>\n\n<p>Recall that during <code class='syntax js'><span class=\"variable\">touchStart</span></code>, the view does not yet own the touch. So, <code class='syntax js'><span class=\"variable\">averagedTouchesForView</span></code>\nwould not, by default count it.</p>\n\n<pre><code class='syntax js'><span class=\"comment\">// on an event:</span>\n<span class=\"keyword\">var</span> <span class=\"variable\">a</span> = <span class=\"variable\">evt</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>);\n\n<span class=\"comment\">// on a touch</span>\n<span class=\"keyword\">var</span> <span class=\"variable\">a</span> = <span class=\"variable\">touch</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>);\n\n<span class=\"comment\">// on a touch, counting the touch itself</span>\n<span class=\"keyword\">var</span> <span class=\"variable\">a</span> = <span class=\"variable\">touch</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>, <span class=\"class\">YES</span>);\n</code></pre>\n\n<h2>Thinking it Over</h2>\n\n<p>How might you use all of these to produce a good result?</p>\n\n<p>Let's take a simple example: moving and resizing something:</p>\n\n<p><a href='multitouch.js' class='demo'>multitouch.js</a></p>","errors":[],"demos":{"multitouch.js":{"ex":"var Box = SC.View.extend({\n _scale: 1,\n _translateX: 0,\n _translateY: 0,\n acceptsMultitouch: YES,\n touchStart: function(touch) {\n this.recomputeTouchStatus(touch, YES);\n return YES;\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n var avg = evt.averagedTouchesForView(this);\n \n // translation is easy:\n this._translateX = t.ourStart.x + avg.x - t.start.x;\n this._translateY = t.ourStart.y + avg.y - t.start.y;\n \n // mathematically speaking, scale *= the end distance / the start distance\n if (t.start.d > 1) { // but prevent divide-by-0\n this._scale = t.ourStart.scale * (avg.d / t.start.d);\n }\n \n // reposition\n this._reposition();\n },\n \n touchEnd: function(touch) {\n this.recomputeTouchStatus(touch, NO);\n },\n \n /**\n With this, we recompute our touch status--updating the start positioning and scale.\n */\n recomputeTouchStatus: function(touch, considerTouch) {\n var avg = touch.averagedTouchesForView(this, considerTouch);\n this._touch = {\n start: { x: avg.x, y: avg.y, d: avg.d },\n ourStart: { x: this._translateX, y: this._translateY, scale: this._scale }\n };\n },\n \n /**\n Repositions the view.\n */\n _reposition: function() {\n this.get(\"layer\").style.webkitTransform = \n \"translate3d(\" + this._translateX + \"px,\" + this._translateY + \"px, 0px) \" +\n \"scale3d(\" + this._scale + \",\" + this._scale + \",1)\";\n console.error(this._translateX + \" \" + this._translateY + \" \" + this._scale);\n },\n \n // and now, redirect mouse events :)\n mouseDown: function(evt) {\n this.touchStart(evt);\n },\n \n mouseDragged: function(evt) {\n this.touchesDragged(evt);\n },\n \n mouseUp: function(evt) {\n this.touchEnd(evt);\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"box\".w(),\n box: Box.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 200, height: 200 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n","highlighted":"<span class=\"keyword\">var</span> <span class=\"class\">Box</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">_scale</span>: <span class=\"number integer\">1</span>,\n <span class=\"variable\">_translateX</span>: <span class=\"number integer\">0</span>,\n <span class=\"variable\">_translateY</span>: <span class=\"number integer\">0</span>,\n <span class=\"variable\">acceptsMultitouch</span>: <span class=\"class\">YES</span>,\n <span class=\"variable\">touchStart</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">recomputeTouchStatus</span>(<span class=\"variable\">touch</span>, <span class=\"class\">YES</span>);\n <span class=\"keyword\">return</span> <span class=\"class\">YES</span>;\n },\n \n <span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">t</span> = <span class=\"this\">this</span>.<span class=\"variable\">_touch</span>;\n <span class=\"keyword\">var</span> <span class=\"variable\">avg</span> = <span class=\"variable\">evt</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>);\n \n <span class=\"comment\">// translation is easy:</span>\n <span class=\"this\">this</span>.<span class=\"variable\">_translateX</span> = <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">x</span> + <span class=\"variable\">avg</span>.<span class=\"variable\">x</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">x</span>;\n <span class=\"this\">this</span>.<span class=\"variable\">_translateY</span> = <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">y</span> + <span class=\"variable\">avg</span>.<span class=\"variable\">y</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">y</span>;\n \n <span class=\"comment\">// mathematically speaking, scale *= the end distance / the start distance</span>\n <span class=\"keyword\">if</span> (<span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">d</span> &gt; <span class=\"number integer\">1</span>) { <span class=\"comment\">// but prevent divide-by-0</span>\n <span class=\"this\">this</span>.<span class=\"variable\">_scale</span> = <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">scale</span> * (<span class=\"variable\">avg</span>.<span class=\"variable\">d</span> / <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">d</span>);\n }\n \n <span class=\"comment\">// reposition</span>\n <span class=\"this\">this</span>.<span class=\"variable\">_reposition</span>();\n },\n \n <span class=\"variable\">touchEnd</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">recomputeTouchStatus</span>(<span class=\"variable\">touch</span>, <span class=\"class\">NO</span>);\n },\n \n <span class=\"multiline comment\">/**\n With this, we recompute our touch status--updating the start positioning and scale.\n */</span>\n <span class=\"variable\">recomputeTouchStatus</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>, <span class=\"variable\">considerTouch</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">avg</span> = <span class=\"variable\">touch</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>, <span class=\"variable\">considerTouch</span>);\n <span class=\"this\">this</span>.<span class=\"variable\">_touch</span> = {\n <span class=\"variable\">start</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">avg</span>.<span class=\"variable\">x</span>, <span class=\"variable\">y</span>: <span class=\"variable\">avg</span>.<span class=\"variable\">y</span>, <span class=\"variable\">d</span>: <span class=\"variable\">avg</span>.<span class=\"variable\">d</span> },\n <span class=\"variable\">ourStart</span>: { <span class=\"variable\">x</span>: <span class=\"this\">this</span>.<span class=\"variable\">_translateX</span>, <span class=\"variable\">y</span>: <span class=\"this\">this</span>.<span class=\"variable\">_translateY</span>, <span class=\"variable\">scale</span>: <span class=\"this\">this</span>.<span class=\"variable\">_scale</span> }\n };\n },\n \n <span class=\"multiline comment\">/**\n Repositions the view.\n */</span>\n <span class=\"variable\">_reposition</span>: <span class=\"keyword\">function</span>() {\n <span class=\"this\">this</span>.<span class=\"variable\">get</span>(<span class=\"string\">&quot;layer&quot;</span>).<span class=\"variable\">style</span>.<span class=\"variable\">webkitTransform</span> = \n <span class=\"string\">&quot;translate3d(&quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_translateX</span> + <span class=\"string\">&quot;px,&quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_translateY</span> + <span class=\"string\">&quot;px, 0px) &quot;</span> +\n <span class=\"string\">&quot;scale3d(&quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_scale</span> + <span class=\"string\">&quot;,&quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_scale</span> + <span class=\"string\">&quot;,1)&quot;</span>;\n <span class=\"variable\">console</span>.<span class=\"variable\">error</span>(<span class=\"this\">this</span>.<span class=\"variable\">_translateX</span> + <span class=\"string\">&quot; &quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_translateY</span> + <span class=\"string\">&quot; &quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_scale</span>);\n },\n \n <span class=\"comment\">// and now, redirect mouse events :)</span>\n <span class=\"variable\">mouseDown</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchStart</span>(<span class=\"variable\">evt</span>);\n },\n \n <span class=\"variable\">mouseDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchesDragged</span>(<span class=\"variable\">evt</span>);\n },\n \n <span class=\"variable\">mouseUp</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchEnd</span>(<span class=\"variable\">evt</span>);\n }\n});\n<span class=\"keyword\">var</span> <span class=\"class\">MyExampleView</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;white&quot;</span>,\n <span class=\"variable\">childViews</span>: <span class=\"string\">&quot;box&quot;</span>.<span class=\"variable\">w</span>(),\n <span class=\"variable\">box</span>: <span class=\"class\">Box</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;red&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">left</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">top</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">200</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">200</span> }\n })\n});\n\n<span class=\"comment\">// bootstrap code :)</span>\n<span class=\"variable\">exports</span>.<span class=\"variable\">getDemoView</span> = <span class=\"keyword\">function</span>() {\n <span class=\"keyword\">return</span> <span class=\"class\">MyExampleView</span>;\n};\n","original":"var Box = SC.View.extend({\n _scale: 1,\n _translateX: 0,\n _translateY: 0,\n acceptsMultitouch: YES,\n touchStart: function(touch) {\n this.recomputeTouchStatus(touch, YES);\n return YES;\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n var avg = evt.averagedTouchesForView(this);\n \n // translation is easy:\n this._translateX = t.ourStart.x + avg.x - t.start.x;\n this._translateY = t.ourStart.y + avg.y - t.start.y;\n \n // mathematically speaking, scale *= the end distance / the start distance\n if (t.start.d > 1) { // but prevent divide-by-0\n this._scale = t.ourStart.scale * (avg.d / t.start.d);\n }\n \n // reposition\n this._reposition();\n },\n \n touchEnd: function(touch) {\n this.recomputeTouchStatus(touch, NO);\n },\n \n /**\n With this, we recompute our touch status--updating the start positioning and scale.\n */\n recomputeTouchStatus: function(touch, considerTouch) {\n var avg = touch.averagedTouchesForView(this, considerTouch);\n this._touch = {\n start: { x: avg.x, y: avg.y, d: avg.d },\n ourStart: { x: this._translateX, y: this._translateY, scale: this._scale }\n };\n },\n \n /**\n Repositions the view.\n */\n _reposition: function() {\n this.get(\"layer\").style.webkitTransform = \n \"translate3d(\" + this._translateX + \"px,\" + this._translateY + \"px, 0px) \" +\n \"scale3d(\" + this._scale + \",\" + this._scale + \",1)\";\n console.error(this._translateX + \" \" + this._translateY + \" \" + this._scale);\n },\n \n // and now, redirect mouse events :)\n mouseDown: function(evt) {\n this.touchStart(evt);\n },\n \n mouseDragged: function(evt) {\n this.touchesDragged(evt);\n },\n \n mouseUp: function(evt) {\n this.touchEnd(evt);\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"box\".w(),\n box: Box.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 200, height: 200 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n"}},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Multitouch"}]},{"title":"Built-In Support","articles":[{"content":"<h1>ButtonView Touch Support</h1>","errors":[],"demos":{},"articleDirectory":"articles/controls/button/","outputDirectory":"build/","title":"ButtonView Touch Support"},{"content":"<h1>ScrollView Touch Support</h1>","errors":[],"demos":{},"articleDirectory":"articles/controls/scroll/","outputDirectory":"build/","title":"ScrollView Touch Support"}]},{"title":"Advanced Concepts","articles":[{"content":"<h1>Capturing</h1>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Capturing"},{"content":"<h1>Releasing</h1>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Releasing"}]}],"file":"build/guides/touch"}
View
9 apps/hedwig/views/article_view.js
@@ -4,7 +4,7 @@ Hedwig.ArticleView = SC.View.extend(Hedwig.TouchHelper, {
mouseDown: function(evt) {
this._mouseStart = { x: evt.pageX, y: evt.pageY };
- return YES;
+ return NO;
},
mouseUp: function(evt) {
@@ -52,9 +52,9 @@ Hedwig.ArticleView = SC.View.extend(Hedwig.TouchHelper, {
borderStyle: SC.BORDER_NONE,
contentView: SC.StaticContentView.design({
contentBinding: "Hedwig.articleController.html",
- contentDidChange: function() {
+ didUpdateLayer: function() {
sc_super();
- this.invokeLater("processContent", 1);
+ this.processContent();
}.observes("content"),
processContent: function() {
@@ -65,7 +65,7 @@ Hedwig.ArticleView = SC.View.extend(Hedwig.TouchHelper, {
touchStart: function(touch) { return this.mouseDown(touch); },
mouseDown: function(evt) {
- evt.preventDefault();
+
var el = document.elementFromPoint(evt.pageX, evt.pageY), demoNode = null;
while (el) {
@@ -79,6 +79,7 @@ Hedwig.ArticleView = SC.View.extend(Hedwig.TouchHelper, {
if (demoNode) {
Hedwig.sendAction("openDemo", demoNode.getAttribute("href"));
+ evt.preventDefault();
return YES;
}
View
25 docs/build/articles/touch/a-brief-touch.html
@@ -88,11 +88,28 @@
</style>
</head><body><div class="header"><a href="../../index.html" class="img"><img src="../../resources/logo.png" /></a><a href="../../index.html" class="here">Documentation
</a><a href="../../reference/index.html" class="item">SproutCore Reference
-</a></div><div class="content"><h1>A Brief Touch on Touches</h1>
+</a></div><div class="content"><h1>A Brief Touch</h1>
-<p>Some text goes here.</p>
+<p>It is very possible to build <em>awesome</em> touch-enabled applications in SproutCore.</p>
-<p>Note that links like the below must appear on empty lines with no preceding whitespace.</p>
+<p>But, what makes an awesome touch-enabled application? Sure, it must accept touches,
+but with SproutCore's (constantly growing) touch support, this is now pretty easy:
+many existing interfaces, if built with newer SproutCore varieties, will function fine (or mostly fine),
+on both larger-screened touch devices (such as iPad) and the traditional desktop environment.</p>
-<p><a href='touch-demo.js' class='demo'>touch-demo.js</a></p>
+<p>But there are many differences between desktop and touch platforms:</p>
+
+<ul>
+<li><strong>Precision.</strong> Touches are less precise than clicks. To compensate, controls should be larger.</li>
+<li><strong>Performance.</strong> Touch-based devices tend to be slow (for now). To get around this just takes
+some elbow grease: there are many techniques to speed things up... many of which SproutCore will
+handle for you.</li>
+<li><strong>Animation.</strong> Lack of animation looks okay on desktop (even if animation is cool)... but on
+touch devices, non-animated interfaces look strange: touch is so much more realistic than mouse,
+but sudden changes without transitions are not realistic at all.</li>
+<li><strong>Coolness.</strong> Touch-based interfaces are cool. That is all.</li>
+</ul>
+
+<p>In this guide, we go over each of these&emdash;except for the last, which is rather vague; you'll have to figure out
+your own meaning of "coolness".</p>
</div><div class="footer"></div></body></html>
View
2  docs/build/articles/touch/a-brief-touch.json
@@ -1 +1 @@
-{"content":"<h1>A Brief Touch on Touches</h1>\n\n<p>Some text goes here.</p>\n\n<p>Note that links like the below must appear on empty lines with no preceding whitespace.</p>\n\n<p><a href='touch-demo.js' class='demo'>touch-demo.js</a></p>","errors":[],"demos":{"touch-demo.js":{"ex":"var Dot = SC.View.extend({\n touchStart: function(touch) {\n var f = this.get(\"frame\");\n this._touch = {\n start: { x: touch.pageX, y: touch.pageY },\n ourStart: { x: f.x, y: f.y, width: f.width, height: f.height }\n };\n return YES; // or we won't get touchesDragged\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n this.set(\"layout\", { \n left: t.ourStart.x + evt.pageX - t.start.x,\n top: t.ourStart.y + evt.pageY - t.start.y,\n width: t.ourStart.width,\n height: t.ourStart.height\n });\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"dot1 dot2\".w(),\n dot1: Dot.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 100, height: 100 }\n }),\n dot2: Dot.design({\n backgroundColor: \"blue\",\n layout: { right: 10, bottom: 10, width: 100, height: 100 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n","highlighted":"<span class=\"keyword\">var</span> <span class=\"class\">Dot</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">touchStart</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">f</span> = <span class=\"this\">this</span>.<span class=\"variable\">get</span>(<span class=\"string\">&quot;frame&quot;</span>);\n <span class=\"this\">this</span>.<span class=\"variable\">_touch</span> = {\n <span class=\"variable\">start</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">touch</span>.<span class=\"variable\">pageX</span>, <span class=\"variable\">y</span>: <span class=\"variable\">touch</span>.<span class=\"variable\">pageY</span> },\n <span class=\"variable\">ourStart</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">f</span>.<span class=\"variable\">x</span>, <span class=\"variable\">y</span>: <span class=\"variable\">f</span>.<span class=\"variable\">y</span>, <span class=\"variable\">width</span>: <span class=\"variable\">f</span>.<span class=\"variable\">width</span>, <span class=\"variable\">height</span>: <span class=\"variable\">f</span>.<span class=\"variable\">height</span> }\n };\n <span class=\"keyword\">return</span> <span class=\"class\">YES</span>; <span class=\"comment\">// or we won't get touchesDragged</span>\n },\n \n <span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">t</span> = <span class=\"this\">this</span>.<span class=\"variable\">_touch</span>;\n <span class=\"this\">this</span>.<span class=\"variable\">set</span>(<span class=\"string\">&quot;layout&quot;</span>, { \n <span class=\"variable\">left</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">x</span> + <span class=\"variable\">evt</span>.<span class=\"variable\">pageX</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">x</span>,\n <span class=\"variable\">top</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">y</span> + <span class=\"variable\">evt</span>.<span class=\"variable\">pageY</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">y</span>,\n <span class=\"variable\">width</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">width</span>,\n <span class=\"variable\">height</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">height</span>\n });\n }\n});\n<span class=\"keyword\">var</span> <span class=\"class\">MyExampleView</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;white&quot;</span>,\n <span class=\"variable\">childViews</span>: <span class=\"string\">&quot;dot1 dot2&quot;</span>.<span class=\"variable\">w</span>(),\n <span class=\"variable\">dot1</span>: <span class=\"class\">Dot</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;red&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">left</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">top</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">100</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">100</span> }\n }),\n <span class=\"variable\">dot2</span>: <span class=\"class\">Dot</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;blue&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">right</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">bottom</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">100</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">100</span> }\n })\n});\n\n<span class=\"comment\">// bootstrap code :)</span>\n<span class=\"variable\">exports</span>.<span class=\"variable\">getDemoView</span> = <span class=\"keyword\">function</span>() {\n <span class=\"keyword\">return</span> <span class=\"class\">MyExampleView</span>;\n};\n","original":"var Dot = SC.View.extend({\n touchStart: function(touch) {\n var f = this.get(\"frame\");\n this._touch = {\n start: { x: touch.pageX, y: touch.pageY },\n ourStart: { x: f.x, y: f.y, width: f.width, height: f.height }\n };\n return YES; // or we won't get touchesDragged\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n this.set(\"layout\", { \n left: t.ourStart.x + evt.pageX - t.start.x,\n top: t.ourStart.y + evt.pageY - t.start.y,\n width: t.ourStart.width,\n height: t.ourStart.height\n });\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"dot1 dot2\".w(),\n dot1: Dot.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 100, height: 100 }\n }),\n dot2: Dot.design({\n backgroundColor: \"blue\",\n layout: { right: 10, bottom: 10, width: 100, height: 100 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n"}},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"A Brief Touch on Touches","any":"metadata","goes":"Here","damn":"gruber","this":"is still eye-readable","and":"He is wrong about touch apps."}
+{"content":"<h1>A Brief Touch</h1>\n\n<p>It is very possible to build <em>awesome</em> touch-enabled applications in SproutCore.</p>\n\n<p>But, what makes an awesome touch-enabled application? Sure, it must accept touches,\nbut with SproutCore's (constantly growing) touch support, this is now pretty easy:\nmany existing interfaces, if built with newer SproutCore varieties, will function fine (or mostly fine),\non both larger-screened touch devices (such as iPad) and the traditional desktop environment.</p>\n\n<p>But there are many differences between desktop and touch platforms:</p>\n\n<ul>\n<li><strong>Precision.</strong> Touches are less precise than clicks. To compensate, controls should be larger.</li>\n<li><strong>Performance.</strong> Touch-based devices tend to be slow (for now). To get around this just takes \nsome elbow grease: there are many techniques to speed things up... many of which SproutCore will\nhandle for you.</li>\n<li><strong>Animation.</strong> Lack of animation looks okay on desktop (even if animation is cool)... but on\ntouch devices, non-animated interfaces look strange: touch is so much more realistic than mouse,\nbut sudden changes without transitions are not realistic at all.</li>\n<li><strong>Coolness.</strong> Touch-based interfaces are cool. That is all.</li>\n</ul>\n\n<p>In this guide, we go over each of these&emdash;except for the last, which is rather vague; you'll have to figure out\nyour own meaning of \"coolness\".</p>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"A Brief Touch","any":"metadata","goes":"Here","damn":"gruber","this":"is still eye-readable","and":"He is wrong about touch apps."}
View
26 docs/build/articles/touch/a-brief-touch.md
@@ -6,10 +6,26 @@
"and": "He is wrong about touch apps."
}
-A Brief Touch on Touches
-========================
-Some text goes here.
+A Brief Touch
+=============
-Note that links like the below must appear on empty lines with no preceding whitespace.
+It is very possible to build _awesome_ touch-enabled applications in SproutCore.
-{{demo:sc|touch-demo.js}}
+But, what makes an awesome touch-enabled application? Sure, it must accept touches,
+but with SproutCore's (constantly growing) touch support, this is now pretty easy:
+many existing interfaces, if built with newer SproutCore varieties, will function fine (or mostly fine),
+on both larger-screened touch devices (such as iPad) and the traditional desktop environment.
+
+But there are many differences between desktop and touch platforms:
+
+- **Precision.** Touches are less precise than clicks. To compensate, controls should be larger.
+- **Performance.** Touch-based devices tend to be slow (for now). To get around this just takes
+ some elbow grease: there are many techniques to speed things up... many of which SproutCore will
+ handle for you.
+- **Animation.** Lack of animation looks okay on desktop (even if animation is cool)... but on
+ touch devices, non-animated interfaces look strange: touch is so much more realistic than mouse,
+ but sudden changes without transitions are not realistic at all.
+- **Coolness.** Touch-based interfaces are cool. That is all.
+
+In this guide, we go over each of these—except for the last, which is rather vague; you'll have to figure out
+your own meaning of "coolness".
View
189 docs/build/articles/touch/multitouch.html
@@ -0,0 +1,189 @@
+<html><head><title>Docs
+</title><meta http-equiv: 'Content-Type' content="text/html; charset=utf-8" />
+<style type="text/css">
+ body {
+ font-family: "Lucida Sans", "Lucida Grande", Verdana, Arial, sans-serif;
+ margin: 0px;
+ margin-bottom:1em;
+ font-family: sans-serif;
+ font-size: 10px;
+ line-height:1.2;
+ }
+
+ .content { font-size: 14px; }
+
+ code {
+ font-family: Monaco, Inconsolata, Courier, fixed-width;
+ font-size: 12px;
+ }
+
+ pre code {
+ margin-right: 1em;
+ border: 1px solid #a0b0a0;
+ overflow-y: hidden;
+ overflow-x: auto;
+ background: #f5f9f5;
+ display:block;
+ padding: 1em;
+ }
+
+ a { text-decoration: none; color: rgb(50, 50, 155); }
+
+
+ .header {
+ background-color: rgb(38, 43, 50);
+ height: 60px;
+ padding-top:17px;
+ padding-bottom:17px;
+ padding-left:2em;
+ }
+
+ .header a.img {
+ float: left;
+ }
+
+ .header .here {
+ float:left;
+ margin-top:27px;
+ margin-left:5px;
+ color: rgb(200, 255, 200);
+ font-size: 25px;
+ }
+
+ .header span.here {
+ margin-top:34px;
+ font-size:15px;
+ color: white;
+ }
+
+ .header a.item {
+ float:right;
+ margin-top: 34px;
+ margin-right:10px;
+ color: white;
+ font-size:15px;
+ }
+
+ .header a.item:hover {
+ text-decoration: underline;
+ }
+
+ .content {
+ padding-top: 1em;
+ padding-left: 2em;
+ padding-right: 2em;
+ }
+
+ img { margin-left: auto; margin-right: auto; display: block; }
+
+
+ h1, h2, h3 { color: rgb(100, 155, 100); }
+
+ code .class { color: rgb(0, 0, 150); }
+ /*code .variable { color: rgb(10, 70, 10); }*/
+ code .comment { color: rgb(100, 150, 200); }
+ code .string { color: rgb(0, 100, 10); }
+ code .number { color: rgb(0, 0, 255); }
+ code .keyword, code .this { color: rgb(25, 110, 25); font-weight: bold; }
+</style>
+</head><body><div class="header"><a href="../../index.html" class="img"><img src="../../resources/logo.png" /></a><a href="../../index.html" class="here">Documentation
+</a><a href="../../reference/index.html" class="item">SproutCore Reference
+</a></div><div class="content"><h1>Multitouch</h1>
+
+<p>Handling single touches is pretty easy&emdash;not that much different from handling
+mouse events. But what about multiple touches?</p>
+
+<h2>Accepting Multiple Touches</h2>
+
+<p>First, you have to tell your view that you do, indeed want to receive multiple
+touches. By default, views only receive single touches. This is because is easier
+to think in a single-touch model, and most controls only need to track a single touch.</p>
+
+<p>To accept multiple touches, just set the view's <code class='syntax js'><span class="variable">acceptsMultitouch</span></code> property to
+<code class='syntax js'><span class="class">YES</span></code>.</p>
+
+<pre><code class='syntax js'><span class="variable">view</span> = <span class="class">SC</span>.<span class="class">View</span>.<span class="variable">extend</span>({
+ <span class="variable">acceptsMultitouch</span>: <span class="class">YES</span>
+});
+</code></pre>
+
+<h2>Processing the Individual Touches</h2>
+
+<p>Even without the supplied helper function, processing individual touches is relatively
+simple:</p>
+
+<ul>
+<li>You get a separate <code class='syntax js'><span class="variable">touchStart</span></code> for each individual touch.</li>
+<li>You get a separate <code class='syntax js'><span class="variable">touchEnd</span></code> for each individual touch.</li>
+<li>You get one <code class='syntax js'><span class="variable">touchesDragged</span></code> each event cycle for all of your touches put together.</li>
+</ul>
+
+<p>So, detecting individual touches starting and ending is simple. Detecting those touches
+moving is not quite as simple, but still relatively easy.</p>
+
+<p>Remember how <code class='syntax js'><span class="variable">touchesDragged</span></code> works:</p>
+
+<pre><code class='syntax js'><span class="variable">touchesDragged</span>: <span class="keyword">function</span>(<span class="variable">evt</span>, <span class="variable">touches</span>)
+</code></pre>
+
+<p><code class='syntax js'><span class="variable">touches</span></code> is an SC.CoreSet of SC.Touch objects. What can you do with a CoreSet?
+You can do a couple of things:</p>
+
+<ul>
+<li>Turn it into an array and do whatever.</li>
+<li>Call .forEach to iterate.</li>
+</ul>
+
+<p>But you don't have to use the touches set at all. The <code class='syntax js'><span class="variable">evt</span></code> has some useful properties and methods,
+too:</p>
+
+<ul>
+<li>pageX/pageY: the position of the first touch.</li>
+<li>averagedTouchesForView: a method which returns the averaged touch position
+and the average distance of the touches from that position.</li>
+</ul>
+
+<p><code class='syntax js'><span class="class">SC</span>.<span class="class">ScrollView</span></code>, for instance, makes heavy use of <code class='syntax js'><span class="variable">averagedTouchesForView</span></code>, and never
+directly touches the <code class='syntax js'><span class="variable">touches</span></code> set.</p>
+
+<h2>Averaging Touches</h2>
+
+<p>It is often <em>very</em> useful to average the touches. </p>
+
+<p><code class='syntax js'><span class="variable">averagedTouchesForView</span></code> returns an object with four properties:</p>
+
+<ul>
+<li><strong><code class='syntax js'><span class="variable">x</span></code></strong>: The average X position of the touch.</li>
+<li><strong><code class='syntax js'><span class="variable">y</span></code></strong>: The average Y position of the touch.</li>
+<li><strong><code class='syntax js'><span class="variable">d</span></code></strong>: average distance of the all touches from the average x/y position.</li>
+<li><strong><code class='syntax js'><span class="variable">touchCount</span></code></strong> The number of touches averaged.</li>
+</ul>
+
+<p>You can call <code class='syntax js'><span class="variable">averagedTouchesForView</span></code> on two separate objects: an <code class='syntax js'><span class="class">SC</span>.<span class="class">Event</span></code> object,
+or an <code class='syntax js'><span class="class">SC</span>.<span class="class">Touch</span></code> object.</p>
+
+<p>The two work identically but for one important difference: when you call it on <code class='syntax js'><span class="class">SC</span>.<span class="class">Touch</span></code>,
+you <em>have the option</em> of telling the touch to add itself to the averaged set. Doing so makes no sense
+in most cases: the touch would just be counted twice! But what about <code class='syntax js'><span class="variable">touchStart</span></code>?</p>
+
+<p>Recall that during <code class='syntax js'><span class="variable">touchStart</span></code>, the view does not yet own the touch. So, <code class='syntax js'><span class="variable">averagedTouchesForView</span></code>
+would not, by default count it.</p>
+
+<pre><code class='syntax js'><span class="comment">// on an event:</span>
+<span class="keyword">var</span> <span class="variable">a</span> = <span class="variable">evt</span>.<span class="variable">averagedTouchesForView</span>(<span class="this">this</span>);
+
+<span class="comment">// on a touch</span>
+<span class="keyword">var</span> <span class="variable">a</span> = <span class="variable">touch</span>.<span class="variable">averagedTouchesForView</span>(<span class="this">this</span>);
+
+<span class="comment">// on a touch, counting the touch itself</span>
+<span class="keyword">var</span> <span class="variable">a</span> = <span class="variable">touch</span>.<span class="variable">averagedTouchesForView</span>(<span class="this">this</span>, <span class="class">YES</span>);
+</code></pre>
+
+<h2>Thinking it Over</h2>
+
+<p>How might you use all of these to produce a good result?</p>
+
+<p>Let's take a simple example: moving and resizing something:</p>
+
+<p><a href='multitouch.js' class='demo'>multitouch.js</a></p>
+</div><div class="footer"></div></body></html>
View
78 docs/build/articles/touch/multitouch.js
@@ -0,0 +1,78 @@
+var Box = SC.View.extend({
+ _scale: 1,
+ _translateX: 0,
+ _translateY: 0,
+ acceptsMultitouch: YES,
+ touchStart: function(touch) {
+ this.recomputeTouchStatus(touch, YES);
+ return YES;
+ },
+
+ touchesDragged: function(evt, touches) {
+ var t = this._touch;
+ var avg = evt.averagedTouchesForView(this);
+
+ // translation is easy:
+ this._translateX = t.ourStart.x + avg.x - t.start.x;
+ this._translateY = t.ourStart.y + avg.y - t.start.y;
+
+ // mathematically speaking, scale *= the end distance / the start distance
+ if (t.start.d > 1) { // but prevent divide-by-0
+ this._scale = t.ourStart.scale * (avg.d / t.start.d);
+ }
+
+ // reposition
+ this._reposition();
+ },
+
+ touchEnd: function(touch) {
+ this.recomputeTouchStatus(touch, NO);
+ },
+
+ /**
+ With this, we recompute our touch status--updating the start positioning and scale.
+ */
+ recomputeTouchStatus: function(touch, considerTouch) {
+ var avg = touch.averagedTouchesForView(this, considerTouch);
+ this._touch = {
+ start: { x: avg.x, y: avg.y, d: avg.d },
+ ourStart: { x: this._translateX, y: this._translateY, scale: this._scale }
+ };
+ },
+
+ /**
+ Repositions the view.
+ */
+ _reposition: function() {
+ this.get("layer").style.webkitTransform =
+ "translate3d(" + this._translateX + "px," + this._translateY + "px, 0px) " +
+ "scale3d(" + this._scale + "," + this._scale + ",1)";
+ console.error(this._translateX + " " + this._translateY + " " + this._scale);
+ },
+
+ // and now, redirect mouse events :)
+ mouseDown: function(evt) {
+ this.touchStart(evt);
+ },
+
+ mouseDragged: function(evt) {
+ this.touchesDragged(evt);
+ },
+
+ mouseUp: function(evt) {
+ this.touchEnd(evt);
+ }
+});
+var MyExampleView = SC.View.extend({
+ backgroundColor: "white",
+ childViews: "box".w(),
+ box: Box.design({
+ backgroundColor: "red",
+ layout: { left: 10, top: 10, width: 200, height: 200 }
+ })
+});
+
+// bootstrap code :)
+exports.getDemoView = function() {
+ return MyExampleView;
+};
View
1  docs/build/articles/touch/multitouch.json
@@ -0,0 +1 @@
+{"content":"<h1>Multitouch</h1>\n\n<p>Handling single touches is pretty easy&emdash;not that much different from handling\nmouse events. But what about multiple touches?</p>\n\n<h2>Accepting Multiple Touches</h2>\n\n<p>First, you have to tell your view that you do, indeed want to receive multiple\ntouches. By default, views only receive single touches. This is because is easier \nto think in a single-touch model, and most controls only need to track a single touch.</p>\n\n<p>To accept multiple touches, just set the view's <code class='syntax js'><span class=\"variable\">acceptsMultitouch</span></code> property to\n<code class='syntax js'><span class=\"class\">YES</span></code>.</p>\n\n<pre><code class='syntax js'><span class=\"variable\">view</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">acceptsMultitouch</span>: <span class=\"class\">YES</span>\n});\n</code></pre>\n\n<h2>Processing the Individual Touches</h2>\n\n<p>Even without the supplied helper function, processing individual touches is relatively\nsimple:</p>\n\n<ul>\n<li>You get a separate <code class='syntax js'><span class=\"variable\">touchStart</span></code> for each individual touch.</li>\n<li>You get a separate <code class='syntax js'><span class=\"variable\">touchEnd</span></code> for each individual touch.</li>\n<li>You get one <code class='syntax js'><span class=\"variable\">touchesDragged</span></code> each event cycle for all of your touches put together.</li>\n</ul>\n\n<p>So, detecting individual touches starting and ending is simple. Detecting those touches\nmoving is not quite as simple, but still relatively easy.</p>\n\n<p>Remember how <code class='syntax js'><span class=\"variable\">touchesDragged</span></code> works:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>)\n</code></pre>\n\n<p><code class='syntax js'><span class=\"variable\">touches</span></code> is an SC.CoreSet of SC.Touch objects. What can you do with a CoreSet? \nYou can do a couple of things:</p>\n\n<ul>\n<li>Turn it into an array and do whatever.</li>\n<li>Call .forEach to iterate.</li>\n</ul>\n\n<p>But you don't have to use the touches set at all. The <code class='syntax js'><span class=\"variable\">evt</span></code> has some useful properties and methods,\ntoo:</p>\n\n<ul>\n<li>pageX/pageY: the position of the first touch.</li>\n<li>averagedTouchesForView: a method which returns the averaged touch position\nand the average distance of the touches from that position.</li>\n</ul>\n\n<p><code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">ScrollView</span></code>, for instance, makes heavy use of <code class='syntax js'><span class=\"variable\">averagedTouchesForView</span></code>, and never\ndirectly touches the <code class='syntax js'><span class=\"variable\">touches</span></code> set.</p>\n\n<h2>Averaging Touches</h2>\n\n<p>It is often <em>very</em> useful to average the touches. </p>\n\n<p><code class='syntax js'><span class=\"variable\">averagedTouchesForView</span></code> returns an object with four properties:</p>\n\n<ul>\n<li><strong><code class='syntax js'><span class=\"variable\">x</span></code></strong>: The average X position of the touch.</li>\n<li><strong><code class='syntax js'><span class=\"variable\">y</span></code></strong>: The average Y position of the touch.</li>\n<li><strong><code class='syntax js'><span class=\"variable\">d</span></code></strong>: average distance of the all touches from the average x/y position.</li>\n<li><strong><code class='syntax js'><span class=\"variable\">touchCount</span></code></strong> The number of touches averaged.</li>\n</ul>\n\n<p>You can call <code class='syntax js'><span class=\"variable\">averagedTouchesForView</span></code> on two separate objects: an <code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">Event</span></code> object,\nor an <code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">Touch</span></code> object.</p>\n\n<p>The two work identically but for one important difference: when you call it on <code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">Touch</span></code>,\nyou <em>have the option</em> of telling the touch to add itself to the averaged set. Doing so makes no sense\nin most cases: the touch would just be counted twice! But what about <code class='syntax js'><span class=\"variable\">touchStart</span></code>?</p>\n\n<p>Recall that during <code class='syntax js'><span class=\"variable\">touchStart</span></code>, the view does not yet own the touch. So, <code class='syntax js'><span class=\"variable\">averagedTouchesForView</span></code>\nwould not, by default count it.</p>\n\n<pre><code class='syntax js'><span class=\"comment\">// on an event:</span>\n<span class=\"keyword\">var</span> <span class=\"variable\">a</span> = <span class=\"variable\">evt</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>);\n\n<span class=\"comment\">// on a touch</span>\n<span class=\"keyword\">var</span> <span class=\"variable\">a</span> = <span class=\"variable\">touch</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>);\n\n<span class=\"comment\">// on a touch, counting the touch itself</span>\n<span class=\"keyword\">var</span> <span class=\"variable\">a</span> = <span class=\"variable\">touch</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>, <span class=\"class\">YES</span>);\n</code></pre>\n\n<h2>Thinking it Over</h2>\n\n<p>How might you use all of these to produce a good result?</p>\n\n<p>Let's take a simple example: moving and resizing something:</p>\n\n<p><a href='multitouch.js' class='demo'>multitouch.js</a></p>","errors":[],"demos":{"multitouch.js":{"ex":"var Box = SC.View.extend({\n _scale: 1,\n _translateX: 0,\n _translateY: 0,\n acceptsMultitouch: YES,\n touchStart: function(touch) {\n this.recomputeTouchStatus(touch, YES);\n return YES;\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n var avg = evt.averagedTouchesForView(this);\n \n // translation is easy:\n this._translateX = t.ourStart.x + avg.x - t.start.x;\n this._translateY = t.ourStart.y + avg.y - t.start.y;\n \n // mathematically speaking, scale *= the end distance / the start distance\n if (t.start.d > 1) { // but prevent divide-by-0\n this._scale = t.ourStart.scale * (avg.d / t.start.d);\n }\n \n // reposition\n this._reposition();\n },\n \n touchEnd: function(touch) {\n this.recomputeTouchStatus(touch, NO);\n },\n \n /**\n With this, we recompute our touch status--updating the start positioning and scale.\n */\n recomputeTouchStatus: function(touch, considerTouch) {\n var avg = touch.averagedTouchesForView(this, considerTouch);\n this._touch = {\n start: { x: avg.x, y: avg.y, d: avg.d },\n ourStart: { x: this._translateX, y: this._translateY, scale: this._scale }\n };\n },\n \n /**\n Repositions the view.\n */\n _reposition: function() {\n this.get(\"layer\").style.webkitTransform = \n \"translate3d(\" + this._translateX + \"px,\" + this._translateY + \"px, 0px) \" +\n \"scale3d(\" + this._scale + \",\" + this._scale + \",1)\";\n console.error(this._translateX + \" \" + this._translateY + \" \" + this._scale);\n },\n \n // and now, redirect mouse events :)\n mouseDown: function(evt) {\n this.touchStart(evt);\n },\n \n mouseDragged: function(evt) {\n this.touchesDragged(evt);\n },\n \n mouseUp: function(evt) {\n this.touchEnd(evt);\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"box\".w(),\n box: Box.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 200, height: 200 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n","highlighted":"<span class=\"keyword\">var</span> <span class=\"class\">Box</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">_scale</span>: <span class=\"number integer\">1</span>,\n <span class=\"variable\">_translateX</span>: <span class=\"number integer\">0</span>,\n <span class=\"variable\">_translateY</span>: <span class=\"number integer\">0</span>,\n <span class=\"variable\">acceptsMultitouch</span>: <span class=\"class\">YES</span>,\n <span class=\"variable\">touchStart</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">recomputeTouchStatus</span>(<span class=\"variable\">touch</span>, <span class=\"class\">YES</span>);\n <span class=\"keyword\">return</span> <span class=\"class\">YES</span>;\n },\n \n <span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">t</span> = <span class=\"this\">this</span>.<span class=\"variable\">_touch</span>;\n <span class=\"keyword\">var</span> <span class=\"variable\">avg</span> = <span class=\"variable\">evt</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>);\n \n <span class=\"comment\">// translation is easy:</span>\n <span class=\"this\">this</span>.<span class=\"variable\">_translateX</span> = <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">x</span> + <span class=\"variable\">avg</span>.<span class=\"variable\">x</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">x</span>;\n <span class=\"this\">this</span>.<span class=\"variable\">_translateY</span> = <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">y</span> + <span class=\"variable\">avg</span>.<span class=\"variable\">y</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">y</span>;\n \n <span class=\"comment\">// mathematically speaking, scale *= the end distance / the start distance</span>\n <span class=\"keyword\">if</span> (<span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">d</span> &gt; <span class=\"number integer\">1</span>) { <span class=\"comment\">// but prevent divide-by-0</span>\n <span class=\"this\">this</span>.<span class=\"variable\">_scale</span> = <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">scale</span> * (<span class=\"variable\">avg</span>.<span class=\"variable\">d</span> / <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">d</span>);\n }\n \n <span class=\"comment\">// reposition</span>\n <span class=\"this\">this</span>.<span class=\"variable\">_reposition</span>();\n },\n \n <span class=\"variable\">touchEnd</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">recomputeTouchStatus</span>(<span class=\"variable\">touch</span>, <span class=\"class\">NO</span>);\n },\n \n <span class=\"multiline comment\">/**\n With this, we recompute our touch status--updating the start positioning and scale.\n */</span>\n <span class=\"variable\">recomputeTouchStatus</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>, <span class=\"variable\">considerTouch</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">avg</span> = <span class=\"variable\">touch</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>, <span class=\"variable\">considerTouch</span>);\n <span class=\"this\">this</span>.<span class=\"variable\">_touch</span> = {\n <span class=\"variable\">start</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">avg</span>.<span class=\"variable\">x</span>, <span class=\"variable\">y</span>: <span class=\"variable\">avg</span>.<span class=\"variable\">y</span>, <span class=\"variable\">d</span>: <span class=\"variable\">avg</span>.<span class=\"variable\">d</span> },\n <span class=\"variable\">ourStart</span>: { <span class=\"variable\">x</span>: <span class=\"this\">this</span>.<span class=\"variable\">_translateX</span>, <span class=\"variable\">y</span>: <span class=\"this\">this</span>.<span class=\"variable\">_translateY</span>, <span class=\"variable\">scale</span>: <span class=\"this\">this</span>.<span class=\"variable\">_scale</span> }\n };\n },\n \n <span class=\"multiline comment\">/**\n Repositions the view.\n */</span>\n <span class=\"variable\">_reposition</span>: <span class=\"keyword\">function</span>() {\n <span class=\"this\">this</span>.<span class=\"variable\">get</span>(<span class=\"string\">&quot;layer&quot;</span>).<span class=\"variable\">style</span>.<span class=\"variable\">webkitTransform</span> = \n <span class=\"string\">&quot;translate3d(&quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_translateX</span> + <span class=\"string\">&quot;px,&quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_translateY</span> + <span class=\"string\">&quot;px, 0px) &quot;</span> +\n <span class=\"string\">&quot;scale3d(&quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_scale</span> + <span class=\"string\">&quot;,&quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_scale</span> + <span class=\"string\">&quot;,1)&quot;</span>;\n <span class=\"variable\">console</span>.<span class=\"variable\">error</span>(<span class=\"this\">this</span>.<span class=\"variable\">_translateX</span> + <span class=\"string\">&quot; &quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_translateY</span> + <span class=\"string\">&quot; &quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_scale</span>);\n },\n \n <span class=\"comment\">// and now, redirect mouse events :)</span>\n <span class=\"variable\">mouseDown</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchStart</span>(<span class=\"variable\">evt</span>);\n },\n \n <span class=\"variable\">mouseDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchesDragged</span>(<span class=\"variable\">evt</span>);\n },\n \n <span class=\"variable\">mouseUp</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchEnd</span>(<span class=\"variable\">evt</span>);\n }\n});\n<span class=\"keyword\">var</span> <span class=\"class\">MyExampleView</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;white&quot;</span>,\n <span class=\"variable\">childViews</span>: <span class=\"string\">&quot;box&quot;</span>.<span class=\"variable\">w</span>(),\n <span class=\"variable\">box</span>: <span class=\"class\">Box</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;red&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">left</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">top</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">200</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">200</span> }\n })\n});\n\n<span class=\"comment\">// bootstrap code :)</span>\n<span class=\"variable\">exports</span>.<span class=\"variable\">getDemoView</span> = <span class=\"keyword\">function</span>() {\n <span class=\"keyword\">return</span> <span class=\"class\">MyExampleView</span>;\n};\n","original":"var Box = SC.View.extend({\n _scale: 1,\n _translateX: 0,\n _translateY: 0,\n acceptsMultitouch: YES,\n touchStart: function(touch) {\n this.recomputeTouchStatus(touch, YES);\n return YES;\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n var avg = evt.averagedTouchesForView(this);\n \n // translation is easy:\n this._translateX = t.ourStart.x + avg.x - t.start.x;\n this._translateY = t.ourStart.y + avg.y - t.start.y;\n \n // mathematically speaking, scale *= the end distance / the start distance\n if (t.start.d > 1) { // but prevent divide-by-0\n this._scale = t.ourStart.scale * (avg.d / t.start.d);\n }\n \n // reposition\n this._reposition();\n },\n \n touchEnd: function(touch) {\n this.recomputeTouchStatus(touch, NO);\n },\n \n /**\n With this, we recompute our touch status--updating the start positioning and scale.\n */\n recomputeTouchStatus: function(touch, considerTouch) {\n var avg = touch.averagedTouchesForView(this, considerTouch);\n this._touch = {\n start: { x: avg.x, y: avg.y, d: avg.d },\n ourStart: { x: this._translateX, y: this._translateY, scale: this._scale }\n };\n },\n \n /**\n Repositions the view.\n */\n _reposition: function() {\n this.get(\"layer\").style.webkitTransform = \n \"translate3d(\" + this._translateX + \"px,\" + this._translateY + \"px, 0px) \" +\n \"scale3d(\" + this._scale + \",\" + this._scale + \",1)\";\n console.error(this._translateX + \" \" + this._translateY + \" \" + this._scale);\n },\n \n // and now, redirect mouse events :)\n mouseDown: function(evt) {\n this.touchStart(evt);\n },\n \n mouseDragged: function(evt) {\n this.touchesDragged(evt);\n },\n \n mouseUp: function(evt) {\n this.touchEnd(evt);\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"box\".w(),\n box: Box.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 200, height: 200 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n"}},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Multitouch"}
View
94 docs/build/articles/touch/multitouch.md
@@ -0,0 +1,94 @@
+Multitouch
+===========
+Handling single touches is pretty easy—not that much different from handling
+mouse events. But what about multiple touches?
+
+Accepting Multiple Touches
+---------------------------
+First, you have to tell your view that you do, indeed want to receive multiple
+touches. By default, views only receive single touches. This is because is easier
+to think in a single-touch model, and most controls only need to track a single touch.
+
+To accept multiple touches, just set the view's `#js:acceptsMultitouch` property to
+`#js:YES`.
+
+ #js
+ view = SC.View.extend({
+ acceptsMultitouch: YES
+ });
+
+Processing the Individual Touches
+-----------------------------------
+Even without the supplied helper function, processing individual touches is relatively
+simple:
+
+- You get a separate `#js:touchStart` for each individual touch.
+- You get a separate `#js:touchEnd` for each individual touch.
+- You get one `#js:touchesDragged` each event cycle for all of your touches put together.
+
+So, detecting individual touches starting and ending is simple. Detecting those touches
+moving is not quite as simple, but still relatively easy.
+
+Remember how `#js:touchesDragged` works:
+
+ #js
+ touchesDragged: function(evt, touches)
+
+`#js:touches` is an SC.CoreSet of SC.Touch objects. What can you do with a CoreSet?
+You can do a couple of things:
+
+- Turn it into an array and do whatever.
+- Call .forEach to iterate.
+
+But you don't have to use the touches set at all. The `#js:evt` has some useful properties and methods,
+too:
+
+- pageX/pageY: the position of the first touch.
+- averagedTouchesForView: a method which returns the averaged touch position
+ and the average distance of the touches from that position.
+
+`#js:SC.ScrollView`, for instance, makes heavy use of `#js:averagedTouchesForView`, and never
+directly touches the `#js:touches` set.
+
+
+Averaging Touches
+------------------
+It is often _very_ useful to average the touches.
+
+`#js:averagedTouchesForView` returns an object with four properties:
+
+- **`#js:x`**: The average X position of the touch.
+- **`#js:y`**: The average Y position of the touch.
+- **`#js:d`**: average distance of the all touches from the average x/y position.
+- **`#js:touchCount`** The number of touches averaged.
+
+You can call `#js:averagedTouchesForView` on two separate objects: an `#js:SC.Event` object,
+or an `#js:SC.Touch` object.
+
+The two work identically but for one important difference: when you call it on `#js:SC.Touch`,
+you _have the option_ of telling the touch to add itself to the averaged set. Doing so makes no sense
+in most cases: the touch would just be counted twice! But what about `#js:touchStart`?
+
+Recall that during `#js:touchStart`, the view does not yet own the touch. So, `#js:averagedTouchesForView`
+would not, by default count it.
+
+ #js
+ // on an event:
+ var a = evt.averagedTouchesForView(this);
+
+ // on a touch
+ var a = touch.averagedTouchesForView(this);
+
+ // on a touch, counting the touch itself
+ var a = touch.averagedTouchesForView(this, YES);
+
+Thinking it Over
+------------------
+How might you use all of these to produce a good result?
+
+Let's take a simple example: moving and resizing something:
+
+{{demo:sc|multitouch.js}}
+
+
+
View
17 docs/build/articles/touch/touch-demo.js
@@ -16,6 +16,23 @@ var Dot = SC.View.extend({
width: t.ourStart.width,
height: t.ourStart.height
});
+ },
+
+ touchEnd: function() {
+ // actually, we don't need to do anything here...
+ },
+
+ // and now, redirect mouse events :)
+ mouseDown: function(evt) {
+ this.touchStart(evt);
+ },
+
+ mouseDragged: function(evt) {
+ this.touchesDragged(evt);
+ },
+
+ mouseUp: function(evt) {
+ this.touchEnd(evt);
}
});
var MyExampleView = SC.View.extend({
View
115 docs/build/articles/touch/touch-events.html
@@ -89,4 +89,119 @@
</head><body><div class="header"><a href="../../index.html" class="img"><img src="../../resources/logo.png" /></a><a href="../../index.html" class="here">Documentation
</a><a href="../../reference/index.html" class="item">SproutCore Reference
</a></div><div class="content"><h1>Touch Events</h1>
+
+<p>SproutCore's touch events have a few great features:</p>
+
+<ul>
+<li>Multiple views can receive touches simultaneously.</li>
+<li>A single view can receive multiple touches.</li>
+<li>A view can capture touches before allowing them to pass through to children.</li>
+<li>Child views can release touches back to parent views that originally captured them.</li>
+</ul>
+
+<p>We won't get into the last two in this article&emdash;they're quite sophisticated!</p>
+
+<h1>Simple Single-Touch Handling</h1>
+
+<p>You may be familiar with this SC.View method signature if you are familiar with SproutCore:</p>
+
+<pre><code class='syntax js'><span class="variable">mouseDown</span>: <span class="keyword">function</span>(<span class="variable">evt</span>)
+</code></pre>
+
+<p>With <code class='syntax js'><span class="variable">mouseDown</span></code>, you can decide whether or not to accept the mouse event by returning either
+<code class='syntax js'><span class="class">YES</span></code> or <code class='syntax js'><span class="class">NO</span></code>.</p>
+
+<p>Touch events are similar, but work a bit differently:</p>
+
+<pre><code class='syntax js'><span class="variable">touchStart</span>: <span class="keyword">function</span>(<span class="variable">touch</span>)
+</code></pre>
+
+<p>Instead of being passed a raw event, <code class='syntax js'><span class="variable">touchStart</span></code> is passed an <code class='syntax js'><span class="class">SC</span>.<span class="class">Touch</span></code> object.
+If you return YES from touchStart, your view will "own" the touch—in SproutCore terms, your
+view will be the <em>touch responder</em>. For more information about what this entails, see
+the "Internals" article. Your view will own the touch until the touch ends.</p>
+
+<p><code class='syntax js'><span class="variable">touchStart</span></code> will be called once for every touch that touches the view.</p>
+
+<p><strong>Note:</strong> By default, views only receive <code class='syntax js'><span class="variable">touchStart</span></code> and touchEnd for a single touch. This is
+a feature intended to make it easier to handle such cases, very similar to Cocoa Touch's <code class='syntax js'><span class="variable">acceptsMultitouch</span></code>
+property—actually, SC.View uses the same property name! See the "Multitouch" article.</p>
+
+<h2>Anatomy of an SC.Touch</h2>
+
+<p>SC.Touch represents the touch from the time the user puts their finger on the screen until the time they lift it.</p>
+
+<p>The touch object acts like an event object in many ways. It has many other useful things, as well:</p>
+
+<ul>
+<li><code class='syntax js'><span class="variable">pageX</span></code> and <code class='syntax js'><span class="variable">pageY</span></code> for the touch</li>
+<li><code class='syntax js'><span class="variable">event</span></code>: if in an event cycle, this contains the event. Otherwise, it is <code class='syntax js'><span class="variable">undefined</span></code>.
+You will probably never need to access this.</li>
+<li><code class='syntax js'><span class="variable">preventDefault</span></code>: if the touch is connected with an event, this calls <code class='syntax js'><span class="variable">preventDefault</span>()</code> on the event.</li>
+<li><code class='syntax js'><span class="variable">touchesForView</span>(<span class="variable">view</span>)</code>: when supplied a view, will find all touches that the view is the
+touch responder for. It is a CoreSet; to get an array, call <code class='syntax js'>.<span class="variable">toArray</span></code> on the result.</li>
+<li><code class='syntax js'><span class="variable">averagedTouchesForView</span>(<span class="variable">view</span>)</code>: When supplied a view, averages all the touches on that view,
+returning both an average position and an average distance from that position.</li>
+</ul>
+
+<p><strong>Note:</strong> If you call <code class='syntax js'><span class="variable">touchesForView</span>(<span class="this">this</span>)</code> from <code class='syntax js'><span class="variable">touchStart</span></code>, the touch supplied will not be in the set
+returned by <code class='syntax js'><span class="variable">touchesForView</span>(<span class="this">this</span>)</code>: you don't own the touch until you return YES.</p>
+
+<h2>touchEnd</h2>
+
+<p>Knowing when the touch starts is not very useful. At the very least, you will likely want to know when the touch
+ends as well.</p>
+
+<p>It is quite simple:</p>
+
+<pre><code class='syntax js'><span class="variable">touchEnd</span>: <span class="keyword">function</span>(<span class="variable">touch</span>)
+</code></pre>
+
+<p>It works exactly like <code class='syntax js'><span class="variable">touchStart</span></code>. The touch will no longer be in the set of touches for the view,
+so if you call <code class='syntax js'><span class="variable">touch</span>.<span class="variable">touchesForView</span>(<span class="this">this</span>)</code>, you'll only receive any other active touches. If your
+view does not accept multitouch, then the set is guaranteed to have no touches in it—you only receive
+<code class='syntax js'><span class="variable">touchEnd</span></code> for the last touch that ends.</p>
+
+<h2>Tracking Touches</h2>
+
+<p>Tracking touch movement is simple:</p>
+
+<pre><code class='syntax js'><span class="variable">touchesDragged</span>: <span class="keyword">function</span>(<span class="variable">evt</span>, <span class="variable">touches</span>)
+</code></pre>
+
+<p>The <code class='syntax js'><span class="variable">touches</span></code> argument is the set of touches on the view—the same set you get by calling <code class='syntax js'><span class="variable">touchesForView</span>(<span class="this">this</span>)</code>.
+This will have <em>all</em> touches, regardless of whether or not your view accepts multitouch.</p>
+
+<p>If your view does <em>not</em> accept multitouch, then it is even simpler:</p>
+
+<pre><code class='syntax js'><span class="variable">x</span> = <span class="variable">evt</span>.<span class="variable">pageX</span>;
+<span class="variable">y</span> = <span class="variable">evt</span>.<span class="variable">pageY</span>;
+</code></pre>
+
+<h2>Tip: Cross-Platform-iness</h2>
+
+<p>Did you realize that, assuming you don't use the set of view touches or other touch-specific API,
+you can do this:</p>
+
+<pre><code class='syntax js'><span class="variable">mouseDown</span>: <span class="keyword">function</span>(<span class="variable">evt</span>) {
+ <span class="this">this</span>.<span class="variable">touchStart</span>(<span class="variable">evt</span>);
+},
+
+<span class="variable">mouseDragged</span>: <span class="keyword">function</span>(<span class="variable">evt</span>) {
+ <span class="this">this</span>.<span class="variable">touchesDragged</span>(<span class="variable">evt</span>);
+},
+
+<span class="variable">mouseUp</span>: <span class="keyword">function</span>(<span class="variable">evt</span>) {
+ <span class="this">this</span>.<span class="variable">touchEnd</span>(<span class="variable">evt</span>);
+}
+</code></pre>
+
+<p>Of course, you can also redirect touch events to mouse events, but that is not as fun.</p>
+
+<h2>Putting it All Together</h2>
+
+<p>Here is a very simple demo that uses the methods described above to allow the user to move two views
+around the screen:</p>
+
+<p><a href='touch-demo.js' class='demo'>touch-demo.js</a></p>
</div><div class="footer"></div></body></html>
View
2  docs/build/articles/touch/touch-events.json
@@ -1 +1 @@
-{"content":"<h1>Touch Events</h1>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Touch Events"}
+{"content":"<h1>Touch Events</h1>\n\n<p>SproutCore's touch events have a few great features:</p>\n\n<ul>\n<li>Multiple views can receive touches simultaneously.</li>\n<li>A single view can receive multiple touches.</li>\n<li>A view can capture touches before allowing them to pass through to children.</li>\n<li>Child views can release touches back to parent views that originally captured them.</li>\n</ul>\n\n<p>We won't get into the last two in this article&emdash;they're quite sophisticated!</p>\n\n<h1>Simple Single-Touch Handling</h1>\n\n<p>You may be familiar with this SC.View method signature if you are familiar with SproutCore:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">mouseDown</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>)\n</code></pre>\n\n<p>With <code class='syntax js'><span class=\"variable\">mouseDown</span></code>, you can decide whether or not to accept the mouse event by returning either\n<code class='syntax js'><span class=\"class\">YES</span></code> or <code class='syntax js'><span class=\"class\">NO</span></code>.</p>\n\n<p>Touch events are similar, but work a bit differently:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">touchStart</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>)\n</code></pre>\n\n<p>Instead of being passed a raw event, <code class='syntax js'><span class=\"variable\">touchStart</span></code> is passed an <code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">Touch</span></code> object.\nIf you return YES from touchStart, your view will \"own\" the touch\u2014in SproutCore terms, your\nview will be the <em>touch responder</em>. For more information about what this entails, see\nthe \"Internals\" article. Your view will own the touch until the touch ends.</p>\n\n<p><code class='syntax js'><span class=\"variable\">touchStart</span></code> will be called once for every touch that touches the view.</p>\n\n<p><strong>Note:</strong> By default, views only receive <code class='syntax js'><span class=\"variable\">touchStart</span></code> and touchEnd for a single touch. This is\na feature intended to make it easier to handle such cases, very similar to Cocoa Touch's <code class='syntax js'><span class=\"variable\">acceptsMultitouch</span></code>\nproperty\u2014actually, SC.View uses the same property name! See the \"Multitouch\" article.</p>\n\n<h2>Anatomy of an SC.Touch</h2>\n\n<p>SC.Touch represents the touch from the time the user puts their finger on the screen until the time they lift it.</p>\n\n<p>The touch object acts like an event object in many ways. It has many other useful things, as well:</p>\n\n<ul>\n<li><code class='syntax js'><span class=\"variable\">pageX</span></code> and <code class='syntax js'><span class=\"variable\">pageY</span></code> for the touch</li>\n<li><code class='syntax js'><span class=\"variable\">event</span></code>: if in an event cycle, this contains the event. Otherwise, it is <code class='syntax js'><span class=\"variable\">undefined</span></code>. \nYou will probably never need to access this.</li>\n<li><code class='syntax js'><span class=\"variable\">preventDefault</span></code>: if the touch is connected with an event, this calls <code class='syntax js'><span class=\"variable\">preventDefault</span>()</code> on the event.</li>\n<li><code class='syntax js'><span class=\"variable\">touchesForView</span>(<span class=\"variable\">view</span>)</code>: when supplied a view, will find all touches that the view is the\ntouch responder for. It is a CoreSet; to get an array, call <code class='syntax js'>.<span class=\"variable\">toArray</span></code> on the result.</li>\n<li><code class='syntax js'><span class=\"variable\">averagedTouchesForView</span>(<span class=\"variable\">view</span>)</code>: When supplied a view, averages all the touches on that view,\nreturning both an average position and an average distance from that position.</li>\n</ul>\n\n<p><strong>Note:</strong> If you call <code class='syntax js'><span class=\"variable\">touchesForView</span>(<span class=\"this\">this</span>)</code> from <code class='syntax js'><span class=\"variable\">touchStart</span></code>, the touch supplied will not be in the set\nreturned by <code class='syntax js'><span class=\"variable\">touchesForView</span>(<span class=\"this\">this</span>)</code>: you don't own the touch until you return YES.</p>\n\n<h2>touchEnd</h2>\n\n<p>Knowing when the touch starts is not very useful. At the very least, you will likely want to know when the touch\nends as well.</p>\n\n<p>It is quite simple:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">touchEnd</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>)\n</code></pre>\n\n<p>It works exactly like <code class='syntax js'><span class=\"variable\">touchStart</span></code>. The touch will no longer be in the set of touches for the view,\nso if you call <code class='syntax js'><span class=\"variable\">touch</span>.<span class=\"variable\">touchesForView</span>(<span class=\"this\">this</span>)</code>, you'll only receive any other active touches. If your\nview does not accept multitouch, then the set is guaranteed to have no touches in it\u2014you only receive\n<code class='syntax js'><span class=\"variable\">touchEnd</span></code> for the last touch that ends.</p>\n\n<h2>Tracking Touches</h2>\n\n<p>Tracking touch movement is simple:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>)\n</code></pre>\n\n<p>The <code class='syntax js'><span class=\"variable\">touches</span></code> argument is the set of touches on the view\u2014the same set you get by calling <code class='syntax js'><span class=\"variable\">touchesForView</span>(<span class=\"this\">this</span>)</code>.\nThis will have <em>all</em> touches, regardless of whether or not your view accepts multitouch.</p>\n\n<p>If your view does <em>not</em> accept multitouch, then it is even simpler:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">x</span> = <span class=\"variable\">evt</span>.<span class=\"variable\">pageX</span>;\n<span class=\"variable\">y</span> = <span class=\"variable\">evt</span>.<span class=\"variable\">pageY</span>;\n</code></pre>\n\n<h2>Tip: Cross-Platform-iness</h2>\n\n<p>Did you realize that, assuming you don't use the set of view touches or other touch-specific API,\nyou can do this:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">mouseDown</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchStart</span>(<span class=\"variable\">evt</span>);\n},\n\n<span class=\"variable\">mouseDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchesDragged</span>(<span class=\"variable\">evt</span>);\n},\n\n<span class=\"variable\">mouseUp</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchEnd</span>(<span class=\"variable\">evt</span>);\n}\n</code></pre>\n\n<p>Of course, you can also redirect touch events to mouse events, but that is not as fun.</p>\n\n<h2>Putting it All Together</h2>\n\n<p>Here is a very simple demo that uses the methods described above to allow the user to move two views\naround the screen:</p>\n\n<p><a href='touch-demo.js' class='demo'>touch-demo.js</a></p>","errors":[],"demos":{"touch-demo.js":{"ex":"var Dot = SC.View.extend({\n touchStart: function(touch) {\n var f = this.get(\"frame\");\n this._touch = {\n start: { x: touch.pageX, y: touch.pageY },\n ourStart: { x: f.x, y: f.y, width: f.width, height: f.height }\n };\n return YES; // or we won't get touchesDragged\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n this.set(\"layout\", { \n left: t.ourStart.x + evt.pageX - t.start.x,\n top: t.ourStart.y + evt.pageY - t.start.y,\n width: t.ourStart.width,\n height: t.ourStart.height\n });\n },\n \n touchEnd: function() {\n // actually, we don't need to do anything here...\n },\n \n // and now, redirect mouse events :)\n mouseDown: function(evt) {\n this.touchStart(evt);\n },\n \n mouseDragged: function(evt) {\n this.touchesDragged(evt);\n },\n \n mouseUp: function(evt) {\n this.touchEnd(evt);\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"dot1 dot2\".w(),\n dot1: Dot.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 100, height: 100 }\n }),\n dot2: Dot.design({\n backgroundColor: \"blue\",\n layout: { right: 10, bottom: 10, width: 100, height: 100 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n","highlighted":"<span class=\"keyword\">var</span> <span class=\"class\">Dot</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">touchStart</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">f</span> = <span class=\"this\">this</span>.<span class=\"variable\">get</span>(<span class=\"string\">&quot;frame&quot;</span>);\n <span class=\"this\">this</span>.<span class=\"variable\">_touch</span> = {\n <span class=\"variable\">start</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">touch</span>.<span class=\"variable\">pageX</span>, <span class=\"variable\">y</span>: <span class=\"variable\">touch</span>.<span class=\"variable\">pageY</span> },\n <span class=\"variable\">ourStart</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">f</span>.<span class=\"variable\">x</span>, <span class=\"variable\">y</span>: <span class=\"variable\">f</span>.<span class=\"variable\">y</span>, <span class=\"variable\">width</span>: <span class=\"variable\">f</span>.<span class=\"variable\">width</span>, <span class=\"variable\">height</span>: <span class=\"variable\">f</span>.<span class=\"variable\">height</span> }\n };\n <span class=\"keyword\">return</span> <span class=\"class\">YES</span>; <span class=\"comment\">// or we won't get touchesDragged</span>\n },\n \n <span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">t</span> = <span class=\"this\">this</span>.<span class=\"variable\">_touch</span>;\n <span class=\"this\">this</span>.<span class=\"variable\">set</span>(<span class=\"string\">&quot;layout&quot;</span>, { \n <span class=\"variable\">left</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">x</span> + <span class=\"variable\">evt</span>.<span class=\"variable\">pageX</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">x</span>,\n <span class=\"variable\">top</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">y</span> + <span class=\"variable\">evt</span>.<span class=\"variable\">pageY</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">y</span>,\n <span class=\"variable\">width</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">width</span>,\n <span class=\"variable\">height</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">height</span>\n });\n },\n \n <span class=\"variable\">touchEnd</span>: <span class=\"keyword\">function</span>() {\n <span class=\"comment\">// actually, we don't need to do anything here...</span>\n },\n \n <span class=\"comment\">// and now, redirect mouse events :)</span>\n <span class=\"variable\">mouseDown</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchStart</span>(<span class=\"variable\">evt</span>);\n },\n \n <span class=\"variable\">mouseDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchesDragged</span>(<span class=\"variable\">evt</span>);\n },\n \n <span class=\"variable\">mouseUp</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchEnd</span>(<span class=\"variable\">evt</span>);\n }\n});\n<span class=\"keyword\">var</span> <span class=\"class\">MyExampleView</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;white&quot;</span>,\n <span class=\"variable\">childViews</span>: <span class=\"string\">&quot;dot1 dot2&quot;</span>.<span class=\"variable\">w</span>(),\n <span class=\"variable\">dot1</span>: <span class=\"class\">Dot</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;red&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">left</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">top</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">100</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">100</span> }\n }),\n <span class=\"variable\">dot2</span>: <span class=\"class\">Dot</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;blue&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">right</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">bottom</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">100</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">100</span> }\n })\n});\n\n<span class=\"comment\">// bootstrap code :)</span>\n<span class=\"variable\">exports</span>.<span class=\"variable\">getDemoView</span> = <span class=\"keyword\">function</span>() {\n <span class=\"keyword\">return</span> <span class=\"class\">MyExampleView</span>;\n};\n","original":"var Dot = SC.View.extend({\n touchStart: function(touch) {\n var f = this.get(\"frame\");\n this._touch = {\n start: { x: touch.pageX, y: touch.pageY },\n ourStart: { x: f.x, y: f.y, width: f.width, height: f.height }\n };\n return YES; // or we won't get touchesDragged\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n this.set(\"layout\", { \n left: t.ourStart.x + evt.pageX - t.start.x,\n top: t.ourStart.y + evt.pageY - t.start.y,\n width: t.ourStart.width,\n height: t.ourStart.height\n });\n },\n \n touchEnd: function() {\n // actually, we don't need to do anything here...\n },\n \n // and now, redirect mouse events :)\n mouseDown: function(evt) {\n this.touchStart(evt);\n },\n \n mouseDragged: function(evt) {\n this.touchesDragged(evt);\n },\n \n mouseUp: function(evt) {\n this.touchEnd(evt);\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"dot1 dot2\".w(),\n dot1: Dot.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 100, height: 100 }\n }),\n dot2: Dot.design({\n backgroundColor: \"blue\",\n layout: { right: 10, bottom: 10, width: 100, height: 100 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n"}},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Touch Events"}
View
114 docs/build/articles/touch/touch-events.md
@@ -1,2 +1,114 @@
Touch Events
-=============
+=============
+
+SproutCore's touch events have a few great features:
+
+- Multiple views can receive touches simultaneously.
+- A single view can receive multiple touches.
+- A view can capture touches before allowing them to pass through to children.
+- Child views can release touches back to parent views that originally captured them.
+
+We won't get into the last two in this article—they're quite sophisticated!
+
+Simple Single-Touch Handling
+=====================================================
+You may be familiar with this SC.View method signature if you are familiar with SproutCore:
+
+ #js
+ mouseDown: function(evt)
+
+With `#js:mouseDown`, you can decide whether or not to accept the mouse event by returning either
+`#js:YES` or `#js:NO`.
+
+Touch events are similar, but work a bit differently:
+
+ #js
+ touchStart: function(touch)
+
+Instead of being passed a raw event, `#js:touchStart` is passed an `#js:SC.Touch` object.
+If you return YES from touchStart, your view will "own" the touch—in SproutCore terms, your
+view will be the _touch responder_. For more information about what this entails, see
+the "Internals" article. Your view will own the touch until the touch ends.
+
+`#js:touchStart` will be called once for every touch that touches the view.
+
+**Note:** By default, views only receive `#js:touchStart` and touchEnd for a single touch. This is
+a feature intended to make it easier to handle such cases, very similar to Cocoa Touch's `#js:acceptsMultitouch`
+property—actually, SC.View uses the same property name! See the "Multitouch" article.
+
+Anatomy of an SC.Touch
+------------------------
+SC.Touch represents the touch from the time the user puts their finger on the screen until the time they lift it.
+
+The touch object acts like an event object in many ways. It has many other useful things, as well:
+
+- `#js:pageX` and `#js:pageY` for the touch
+- `#js:event`: if in an event cycle, this contains the event. Otherwise, it is `#js:undefined`.
+ You will probably never need to access this.
+- `#js:preventDefault`: if the touch is connected with an event, this calls `#js:preventDefault()` on the event.
+- `#js:touchesForView(view)`: when supplied a view, will find all touches that the view is the
+ touch responder for. It is a CoreSet; to get an array, call `#js:.toArray` on the result.
+- `#js:averagedTouchesForView(view)`: When supplied a view, averages all the touches on that view,
+ returning both an average position and an average distance from that position.
+
+**Note:** If you call `#js:touchesForView(this)` from `#js:touchStart`, the touch supplied will not be in the set
+returned by `#js:touchesForView(this)`: you don't own the touch until you return YES.
+
+touchEnd
+-----------------
+Knowing when the touch starts is not very useful. At the very least, you will likely want to know when the touch
+ends as well.
+
+It is quite simple:
+
+ #js
+ touchEnd: function(touch)
+
+It works exactly like `#js:touchStart`. The touch will no longer be in the set of touches for the view,
+so if you call `#js:touch.touchesForView(this)`, you'll only receive any other active touches. If your
+view does not accept multitouch, then the set is guaranteed to have no touches in it—you only receive
+`#js:touchEnd` for the last touch that ends.
+
+Tracking Touches
+------------------
+Tracking touch movement is simple:
+
+ #js
+ touchesDragged: function(evt, touches)
+
+The `#js:touches` argument is the set of touches on the view—the same set you get by calling `#js:touchesForView(this)`.
+This will have _all_ touches, regardless of whether or not your view accepts multitouch.
+
+If your view does _not_ accept multitouch, then it is even simpler:
+
+ #js
+ x = evt.pageX;
+ y = evt.pageY;
+
+Tip: Cross-Platform-iness
+--------------------------
+Did you realize that, assuming you don't use the set of view touches or other touch-specific API,
+you can do this:
+
+ #js
+ mouseDown: function(evt) {
+ this.touchStart(evt);
+ },
+
+ mouseDragged: function(evt) {
+ this.touchesDragged(evt);
+ },
+
+ mouseUp: function(evt) {
+ this.touchEnd(evt);
+ }
+
+Of course, you can also redirect touch events to mouse events, but that is not as fun.
+
+Putting it All Together
+------------------------
+Here is a very simple demo that uses the methods described above to allow the user to move two views
+around the screen:
+
+{{demo:sc|touch-demo.js}}
+
View
5 docs/build/guides/touch.guide
@@ -1,11 +1,12 @@
{
- "title": "Writing Touch Applications",
+ "title": "Touch Application Guide",
"sections": [
{
"title": "Introduction",
"articles": [
"articles/touch/a-brief-touch",
- "articles/touch/touch-events"
+ "articles/touch/touch-events",
+ "articles/touch/multitouch"
]
},
{
View
2  docs/build/guides/touch.json
@@ -1 +1 @@
-{"title":"Writing Touch Applications","sections":[{"title":"Introduction","articles":[{"content":"<h1>A Brief Touch on Touches</h1>\n\n<p>Some text goes here.</p>\n\n<p>Note that links like the below must appear on empty lines with no preceding whitespace.</p>\n\n<p><a href='touch-demo.js' class='demo'>touch-demo.js</a></p>","errors":[],"demos":{"touch-demo.js":{"ex":"var Dot = SC.View.extend({\n touchStart: function(touch) {\n var f = this.get(\"frame\");\n this._touch = {\n start: { x: touch.pageX, y: touch.pageY },\n ourStart: { x: f.x, y: f.y, width: f.width, height: f.height }\n };\n return YES; // or we won't get touchesDragged\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n this.set(\"layout\", { \n left: t.ourStart.x + evt.pageX - t.start.x,\n top: t.ourStart.y + evt.pageY - t.start.y,\n width: t.ourStart.width,\n height: t.ourStart.height\n });\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"dot1 dot2\".w(),\n dot1: Dot.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 100, height: 100 }\n }),\n dot2: Dot.design({\n backgroundColor: \"blue\",\n layout: { right: 10, bottom: 10, width: 100, height: 100 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n","highlighted":"<span class=\"keyword\">var</span> <span class=\"class\">Dot</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">touchStart</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">f</span> = <span class=\"this\">this</span>.<span class=\"variable\">get</span>(<span class=\"string\">&quot;frame&quot;</span>);\n <span class=\"this\">this</span>.<span class=\"variable\">_touch</span> = {\n <span class=\"variable\">start</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">touch</span>.<span class=\"variable\">pageX</span>, <span class=\"variable\">y</span>: <span class=\"variable\">touch</span>.<span class=\"variable\">pageY</span> },\n <span class=\"variable\">ourStart</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">f</span>.<span class=\"variable\">x</span>, <span class=\"variable\">y</span>: <span class=\"variable\">f</span>.<span class=\"variable\">y</span>, <span class=\"variable\">width</span>: <span class=\"variable\">f</span>.<span class=\"variable\">width</span>, <span class=\"variable\">height</span>: <span class=\"variable\">f</span>.<span class=\"variable\">height</span> }\n };\n <span class=\"keyword\">return</span> <span class=\"class\">YES</span>; <span class=\"comment\">// or we won't get touchesDragged</span>\n },\n \n <span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">t</span> = <span class=\"this\">this</span>.<span class=\"variable\">_touch</span>;\n <span class=\"this\">this</span>.<span class=\"variable\">set</span>(<span class=\"string\">&quot;layout&quot;</span>, { \n <span class=\"variable\">left</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">x</span> + <span class=\"variable\">evt</span>.<span class=\"variable\">pageX</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">x</span>,\n <span class=\"variable\">top</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">y</span> + <span class=\"variable\">evt</span>.<span class=\"variable\">pageY</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">y</span>,\n <span class=\"variable\">width</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">width</span>,\n <span class=\"variable\">height</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">height</span>\n });\n }\n});\n<span class=\"keyword\">var</span> <span class=\"class\">MyExampleView</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;white&quot;</span>,\n <span class=\"variable\">childViews</span>: <span class=\"string\">&quot;dot1 dot2&quot;</span>.<span class=\"variable\">w</span>(),\n <span class=\"variable\">dot1</span>: <span class=\"class\">Dot</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;red&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">left</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">top</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">100</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">100</span> }\n }),\n <span class=\"variable\">dot2</span>: <span class=\"class\">Dot</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;blue&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">right</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">bottom</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">100</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">100</span> }\n })\n});\n\n<span class=\"comment\">// bootstrap code :)</span>\n<span class=\"variable\">exports</span>.<span class=\"variable\">getDemoView</span> = <span class=\"keyword\">function</span>() {\n <span class=\"keyword\">return</span> <span class=\"class\">MyExampleView</span>;\n};\n","original":"var Dot = SC.View.extend({\n touchStart: function(touch) {\n var f = this.get(\"frame\");\n this._touch = {\n start: { x: touch.pageX, y: touch.pageY },\n ourStart: { x: f.x, y: f.y, width: f.width, height: f.height }\n };\n return YES; // or we won't get touchesDragged\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n this.set(\"layout\", { \n left: t.ourStart.x + evt.pageX - t.start.x,\n top: t.ourStart.y + evt.pageY - t.start.y,\n width: t.ourStart.width,\n height: t.ourStart.height\n });\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"dot1 dot2\".w(),\n dot1: Dot.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 100, height: 100 }\n }),\n dot2: Dot.design({\n backgroundColor: \"blue\",\n layout: { right: 10, bottom: 10, width: 100, height: 100 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n"}},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"A Brief Touch on Touches","any":"metadata","goes":"Here","damn":"gruber","this":"is still eye-readable","and":"He is wrong about touch apps."},{"content":"<h1>Touch Events</h1>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Touch Events"}]},{"title":"Built-In Support","articles":[{"content":"<h1>ButtonView Touch Support</h1>","errors":[],"demos":{},"articleDirectory":"articles/controls/button/","outputDirectory":"build/","title":"ButtonView Touch Support"},{"content":"<h1>ScrollView Touch Support</h1>","errors":[],"demos":{},"articleDirectory":"articles/controls/scroll/","outputDirectory":"build/","title":"ScrollView Touch Support"}]},{"title":"Advanced Concepts","articles":[{"content":"<h1>Capturing</h1>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Capturing"},{"content":"<h1>Releasing</h1>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Releasing"}]}],"file":"build/guides/touch"}
+{"title":"Touch Application Guide","sections":[{"title":"Introduction","articles":[{"content":"<h1>A Brief Touch</h1>\n\n<p>It is very possible to build <em>awesome</em> touch-enabled applications in SproutCore.</p>\n\n<p>But, what makes an awesome touch-enabled application? Sure, it must accept touches,\nbut with SproutCore's (constantly growing) touch support, this is now pretty easy:\nmany existing interfaces, if built with newer SproutCore varieties, will function fine (or mostly fine),\non both larger-screened touch devices (such as iPad) and the traditional desktop environment.</p>\n\n<p>But there are many differences between desktop and touch platforms:</p>\n\n<ul>\n<li><strong>Precision.</strong> Touches are less precise than clicks. To compensate, controls should be larger.</li>\n<li><strong>Performance.</strong> Touch-based devices tend to be slow (for now). To get around this just takes \nsome elbow grease: there are many techniques to speed things up... many of which SproutCore will\nhandle for you.</li>\n<li><strong>Animation.</strong> Lack of animation looks okay on desktop (even if animation is cool)... but on\ntouch devices, non-animated interfaces look strange: touch is so much more realistic than mouse,\nbut sudden changes without transitions are not realistic at all.</li>\n<li><strong>Coolness.</strong> Touch-based interfaces are cool. That is all.</li>\n</ul>\n\n<p>In this guide, we go over each of these&emdash;except for the last, which is rather vague; you'll have to figure out\nyour own meaning of \"coolness\".</p>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"A Brief Touch","any":"metadata","goes":"Here","damn":"gruber","this":"is still eye-readable","and":"He is wrong about touch apps."},{"content":"<h1>Touch Events</h1>\n\n<p>SproutCore's touch events have a few great features:</p>\n\n<ul>\n<li>Multiple views can receive touches simultaneously.</li>\n<li>A single view can receive multiple touches.</li>\n<li>A view can capture touches before allowing them to pass through to children.</li>\n<li>Child views can release touches back to parent views that originally captured them.</li>\n</ul>\n\n<p>We won't get into the last two in this article&emdash;they're quite sophisticated!</p>\n\n<h1>Simple Single-Touch Handling</h1>\n\n<p>You may be familiar with this SC.View method signature if you are familiar with SproutCore:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">mouseDown</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>)\n</code></pre>\n\n<p>With <code class='syntax js'><span class=\"variable\">mouseDown</span></code>, you can decide whether or not to accept the mouse event by returning either\n<code class='syntax js'><span class=\"class\">YES</span></code> or <code class='syntax js'><span class=\"class\">NO</span></code>.</p>\n\n<p>Touch events are similar, but work a bit differently:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">touchStart</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>)\n</code></pre>\n\n<p>Instead of being passed a raw event, <code class='syntax js'><span class=\"variable\">touchStart</span></code> is passed an <code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">Touch</span></code> object.\nIf you return YES from touchStart, your view will \"own\" the touch\u2014in SproutCore terms, your\nview will be the <em>touch responder</em>. For more information about what this entails, see\nthe \"Internals\" article. Your view will own the touch until the touch ends.</p>\n\n<p><code class='syntax js'><span class=\"variable\">touchStart</span></code> will be called once for every touch that touches the view.</p>\n\n<p><strong>Note:</strong> By default, views only receive <code class='syntax js'><span class=\"variable\">touchStart</span></code> and touchEnd for a single touch. This is\na feature intended to make it easier to handle such cases, very similar to Cocoa Touch's <code class='syntax js'><span class=\"variable\">acceptsMultitouch</span></code>\nproperty\u2014actually, SC.View uses the same property name! See the \"Multitouch\" article.</p>\n\n<h2>Anatomy of an SC.Touch</h2>\n\n<p>SC.Touch represents the touch from the time the user puts their finger on the screen until the time they lift it.</p>\n\n<p>The touch object acts like an event object in many ways. It has many other useful things, as well:</p>\n\n<ul>\n<li><code class='syntax js'><span class=\"variable\">pageX</span></code> and <code class='syntax js'><span class=\"variable\">pageY</span></code> for the touch</li>\n<li><code class='syntax js'><span class=\"variable\">event</span></code>: if in an event cycle, this contains the event. Otherwise, it is <code class='syntax js'><span class=\"variable\">undefined</span></code>. \nYou will probably never need to access this.</li>\n<li><code class='syntax js'><span class=\"variable\">preventDefault</span></code>: if the touch is connected with an event, this calls <code class='syntax js'><span class=\"variable\">preventDefault</span>()</code> on the event.</li>\n<li><code class='syntax js'><span class=\"variable\">touchesForView</span>(<span class=\"variable\">view</span>)</code>: when supplied a view, will find all touches that the view is the\ntouch responder for. It is a CoreSet; to get an array, call <code class='syntax js'>.<span class=\"variable\">toArray</span></code> on the result.</li>\n<li><code class='syntax js'><span class=\"variable\">averagedTouchesForView</span>(<span class=\"variable\">view</span>)</code>: When supplied a view, averages all the touches on that view,\nreturning both an average position and an average distance from that position.</li>\n</ul>\n\n<p><strong>Note:</strong> If you call <code class='syntax js'><span class=\"variable\">touchesForView</span>(<span class=\"this\">this</span>)</code> from <code class='syntax js'><span class=\"variable\">touchStart</span></code>, the touch supplied will not be in the set\nreturned by <code class='syntax js'><span class=\"variable\">touchesForView</span>(<span class=\"this\">this</span>)</code>: you don't own the touch until you return YES.</p>\n\n<h2>touchEnd</h2>\n\n<p>Knowing when the touch starts is not very useful. At the very least, you will likely want to know when the touch\nends as well.</p>\n\n<p>It is quite simple:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">touchEnd</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>)\n</code></pre>\n\n<p>It works exactly like <code class='syntax js'><span class=\"variable\">touchStart</span></code>. The touch will no longer be in the set of touches for the view,\nso if you call <code class='syntax js'><span class=\"variable\">touch</span>.<span class=\"variable\">touchesForView</span>(<span class=\"this\">this</span>)</code>, you'll only receive any other active touches. If your\nview does not accept multitouch, then the set is guaranteed to have no touches in it\u2014you only receive\n<code class='syntax js'><span class=\"variable\">touchEnd</span></code> for the last touch that ends.</p>\n\n<h2>Tracking Touches</h2>\n\n<p>Tracking touch movement is simple:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>)\n</code></pre>\n\n<p>The <code class='syntax js'><span class=\"variable\">touches</span></code> argument is the set of touches on the view\u2014the same set you get by calling <code class='syntax js'><span class=\"variable\">touchesForView</span>(<span class=\"this\">this</span>)</code>.\nThis will have <em>all</em> touches, regardless of whether or not your view accepts multitouch.</p>\n\n<p>If your view does <em>not</em> accept multitouch, then it is even simpler:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">x</span> = <span class=\"variable\">evt</span>.<span class=\"variable\">pageX</span>;\n<span class=\"variable\">y</span> = <span class=\"variable\">evt</span>.<span class=\"variable\">pageY</span>;\n</code></pre>\n\n<h2>Tip: Cross-Platform-iness</h2>\n\n<p>Did you realize that, assuming you don't use the set of view touches or other touch-specific API,\nyou can do this:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">mouseDown</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchStart</span>(<span class=\"variable\">evt</span>);\n},\n\n<span class=\"variable\">mouseDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchesDragged</span>(<span class=\"variable\">evt</span>);\n},\n\n<span class=\"variable\">mouseUp</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchEnd</span>(<span class=\"variable\">evt</span>);\n}\n</code></pre>\n\n<p>Of course, you can also redirect touch events to mouse events, but that is not as fun.</p>\n\n<h2>Putting it All Together</h2>\n\n<p>Here is a very simple demo that uses the methods described above to allow the user to move two views\naround the screen:</p>\n\n<p><a href='touch-demo.js' class='demo'>touch-demo.js</a></p>","errors":[],"demos":{"touch-demo.js":{"ex":"var Dot = SC.View.extend({\n touchStart: function(touch) {\n var f = this.get(\"frame\");\n this._touch = {\n start: { x: touch.pageX, y: touch.pageY },\n ourStart: { x: f.x, y: f.y, width: f.width, height: f.height }\n };\n return YES; // or we won't get touchesDragged\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n this.set(\"layout\", { \n left: t.ourStart.x + evt.pageX - t.start.x,\n top: t.ourStart.y + evt.pageY - t.start.y,\n width: t.ourStart.width,\n height: t.ourStart.height\n });\n },\n \n touchEnd: function() {\n // actually, we don't need to do anything here...\n },\n \n // and now, redirect mouse events :)\n mouseDown: function(evt) {\n this.touchStart(evt);\n },\n \n mouseDragged: function(evt) {\n this.touchesDragged(evt);\n },\n \n mouseUp: function(evt) {\n this.touchEnd(evt);\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"dot1 dot2\".w(),\n dot1: Dot.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 100, height: 100 }\n }),\n dot2: Dot.design({\n backgroundColor: \"blue\",\n layout: { right: 10, bottom: 10, width: 100, height: 100 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n","highlighted":"<span class=\"keyword\">var</span> <span class=\"class\">Dot</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">touchStart</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">f</span> = <span class=\"this\">this</span>.<span class=\"variable\">get</span>(<span class=\"string\">&quot;frame&quot;</span>);\n <span class=\"this\">this</span>.<span class=\"variable\">_touch</span> = {\n <span class=\"variable\">start</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">touch</span>.<span class=\"variable\">pageX</span>, <span class=\"variable\">y</span>: <span class=\"variable\">touch</span>.<span class=\"variable\">pageY</span> },\n <span class=\"variable\">ourStart</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">f</span>.<span class=\"variable\">x</span>, <span class=\"variable\">y</span>: <span class=\"variable\">f</span>.<span class=\"variable\">y</span>, <span class=\"variable\">width</span>: <span class=\"variable\">f</span>.<span class=\"variable\">width</span>, <span class=\"variable\">height</span>: <span class=\"variable\">f</span>.<span class=\"variable\">height</span> }\n };\n <span class=\"keyword\">return</span> <span class=\"class\">YES</span>; <span class=\"comment\">// or we won't get touchesDragged</span>\n },\n \n <span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">t</span> = <span class=\"this\">this</span>.<span class=\"variable\">_touch</span>;\n <span class=\"this\">this</span>.<span class=\"variable\">set</span>(<span class=\"string\">&quot;layout&quot;</span>, { \n <span class=\"variable\">left</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">x</span> + <span class=\"variable\">evt</span>.<span class=\"variable\">pageX</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">x</span>,\n <span class=\"variable\">top</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">y</span> + <span class=\"variable\">evt</span>.<span class=\"variable\">pageY</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">y</span>,\n <span class=\"variable\">width</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">width</span>,\n <span class=\"variable\">height</span>: <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">height</span>\n });\n },\n \n <span class=\"variable\">touchEnd</span>: <span class=\"keyword\">function</span>() {\n <span class=\"comment\">// actually, we don't need to do anything here...</span>\n },\n \n <span class=\"comment\">// and now, redirect mouse events :)</span>\n <span class=\"variable\">mouseDown</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchStart</span>(<span class=\"variable\">evt</span>);\n },\n \n <span class=\"variable\">mouseDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchesDragged</span>(<span class=\"variable\">evt</span>);\n },\n \n <span class=\"variable\">mouseUp</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchEnd</span>(<span class=\"variable\">evt</span>);\n }\n});\n<span class=\"keyword\">var</span> <span class=\"class\">MyExampleView</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;white&quot;</span>,\n <span class=\"variable\">childViews</span>: <span class=\"string\">&quot;dot1 dot2&quot;</span>.<span class=\"variable\">w</span>(),\n <span class=\"variable\">dot1</span>: <span class=\"class\">Dot</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;red&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">left</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">top</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">100</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">100</span> }\n }),\n <span class=\"variable\">dot2</span>: <span class=\"class\">Dot</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;blue&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">right</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">bottom</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">100</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">100</span> }\n })\n});\n\n<span class=\"comment\">// bootstrap code :)</span>\n<span class=\"variable\">exports</span>.<span class=\"variable\">getDemoView</span> = <span class=\"keyword\">function</span>() {\n <span class=\"keyword\">return</span> <span class=\"class\">MyExampleView</span>;\n};\n","original":"var Dot = SC.View.extend({\n touchStart: function(touch) {\n var f = this.get(\"frame\");\n this._touch = {\n start: { x: touch.pageX, y: touch.pageY },\n ourStart: { x: f.x, y: f.y, width: f.width, height: f.height }\n };\n return YES; // or we won't get touchesDragged\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n this.set(\"layout\", { \n left: t.ourStart.x + evt.pageX - t.start.x,\n top: t.ourStart.y + evt.pageY - t.start.y,\n width: t.ourStart.width,\n height: t.ourStart.height\n });\n },\n \n touchEnd: function() {\n // actually, we don't need to do anything here...\n },\n \n // and now, redirect mouse events :)\n mouseDown: function(evt) {\n this.touchStart(evt);\n },\n \n mouseDragged: function(evt) {\n this.touchesDragged(evt);\n },\n \n mouseUp: function(evt) {\n this.touchEnd(evt);\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"dot1 dot2\".w(),\n dot1: Dot.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 100, height: 100 }\n }),\n dot2: Dot.design({\n backgroundColor: \"blue\",\n layout: { right: 10, bottom: 10, width: 100, height: 100 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n"}},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Touch Events"},{"content":"<h1>Multitouch</h1>\n\n<p>Handling single touches is pretty easy&emdash;not that much different from handling\nmouse events. But what about multiple touches?</p>\n\n<h2>Accepting Multiple Touches</h2>\n\n<p>First, you have to tell your view that you do, indeed want to receive multiple\ntouches. By default, views only receive single touches. This is because is easier \nto think in a single-touch model, and most controls only need to track a single touch.</p>\n\n<p>To accept multiple touches, just set the view's <code class='syntax js'><span class=\"variable\">acceptsMultitouch</span></code> property to\n<code class='syntax js'><span class=\"class\">YES</span></code>.</p>\n\n<pre><code class='syntax js'><span class=\"variable\">view</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">acceptsMultitouch</span>: <span class=\"class\">YES</span>\n});\n</code></pre>\n\n<h2>Processing the Individual Touches</h2>\n\n<p>Even without the supplied helper function, processing individual touches is relatively\nsimple:</p>\n\n<ul>\n<li>You get a separate <code class='syntax js'><span class=\"variable\">touchStart</span></code> for each individual touch.</li>\n<li>You get a separate <code class='syntax js'><span class=\"variable\">touchEnd</span></code> for each individual touch.</li>\n<li>You get one <code class='syntax js'><span class=\"variable\">touchesDragged</span></code> each event cycle for all of your touches put together.</li>\n</ul>\n\n<p>So, detecting individual touches starting and ending is simple. Detecting those touches\nmoving is not quite as simple, but still relatively easy.</p>\n\n<p>Remember how <code class='syntax js'><span class=\"variable\">touchesDragged</span></code> works:</p>\n\n<pre><code class='syntax js'><span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>)\n</code></pre>\n\n<p><code class='syntax js'><span class=\"variable\">touches</span></code> is an SC.CoreSet of SC.Touch objects. What can you do with a CoreSet? \nYou can do a couple of things:</p>\n\n<ul>\n<li>Turn it into an array and do whatever.</li>\n<li>Call .forEach to iterate.</li>\n</ul>\n\n<p>But you don't have to use the touches set at all. The <code class='syntax js'><span class=\"variable\">evt</span></code> has some useful properties and methods,\ntoo:</p>\n\n<ul>\n<li>pageX/pageY: the position of the first touch.</li>\n<li>averagedTouchesForView: a method which returns the averaged touch position\nand the average distance of the touches from that position.</li>\n</ul>\n\n<p><code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">ScrollView</span></code>, for instance, makes heavy use of <code class='syntax js'><span class=\"variable\">averagedTouchesForView</span></code>, and never\ndirectly touches the <code class='syntax js'><span class=\"variable\">touches</span></code> set.</p>\n\n<h2>Averaging Touches</h2>\n\n<p>It is often <em>very</em> useful to average the touches. </p>\n\n<p><code class='syntax js'><span class=\"variable\">averagedTouchesForView</span></code> returns an object with four properties:</p>\n\n<ul>\n<li><strong><code class='syntax js'><span class=\"variable\">x</span></code></strong>: The average X position of the touch.</li>\n<li><strong><code class='syntax js'><span class=\"variable\">y</span></code></strong>: The average Y position of the touch.</li>\n<li><strong><code class='syntax js'><span class=\"variable\">d</span></code></strong>: average distance of the all touches from the average x/y position.</li>\n<li><strong><code class='syntax js'><span class=\"variable\">touchCount</span></code></strong> The number of touches averaged.</li>\n</ul>\n\n<p>You can call <code class='syntax js'><span class=\"variable\">averagedTouchesForView</span></code> on two separate objects: an <code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">Event</span></code> object,\nor an <code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">Touch</span></code> object.</p>\n\n<p>The two work identically but for one important difference: when you call it on <code class='syntax js'><span class=\"class\">SC</span>.<span class=\"class\">Touch</span></code>,\nyou <em>have the option</em> of telling the touch to add itself to the averaged set. Doing so makes no sense\nin most cases: the touch would just be counted twice! But what about <code class='syntax js'><span class=\"variable\">touchStart</span></code>?</p>\n\n<p>Recall that during <code class='syntax js'><span class=\"variable\">touchStart</span></code>, the view does not yet own the touch. So, <code class='syntax js'><span class=\"variable\">averagedTouchesForView</span></code>\nwould not, by default count it.</p>\n\n<pre><code class='syntax js'><span class=\"comment\">// on an event:</span>\n<span class=\"keyword\">var</span> <span class=\"variable\">a</span> = <span class=\"variable\">evt</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>);\n\n<span class=\"comment\">// on a touch</span>\n<span class=\"keyword\">var</span> <span class=\"variable\">a</span> = <span class=\"variable\">touch</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>);\n\n<span class=\"comment\">// on a touch, counting the touch itself</span>\n<span class=\"keyword\">var</span> <span class=\"variable\">a</span> = <span class=\"variable\">touch</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>, <span class=\"class\">YES</span>);\n</code></pre>\n\n<h2>Thinking it Over</h2>\n\n<p>How might you use all of these to produce a good result?</p>\n\n<p>Let's take a simple example: moving and resizing something:</p>\n\n<p><a href='multitouch.js' class='demo'>multitouch.js</a></p>","errors":[],"demos":{"multitouch.js":{"ex":"var Box = SC.View.extend({\n _scale: 1,\n _translateX: 0,\n _translateY: 0,\n acceptsMultitouch: YES,\n touchStart: function(touch) {\n this.recomputeTouchStatus(touch, YES);\n return YES;\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n var avg = evt.averagedTouchesForView(this);\n \n // translation is easy:\n this._translateX = t.ourStart.x + avg.x - t.start.x;\n this._translateY = t.ourStart.y + avg.y - t.start.y;\n \n // mathematically speaking, scale *= the end distance / the start distance\n if (t.start.d > 1) { // but prevent divide-by-0\n this._scale = t.ourStart.scale * (avg.d / t.start.d);\n }\n \n // reposition\n this._reposition();\n },\n \n touchEnd: function(touch) {\n this.recomputeTouchStatus(touch, NO);\n },\n \n /**\n With this, we recompute our touch status--updating the start positioning and scale.\n */\n recomputeTouchStatus: function(touch, considerTouch) {\n var avg = touch.averagedTouchesForView(this, considerTouch);\n this._touch = {\n start: { x: avg.x, y: avg.y, d: avg.d },\n ourStart: { x: this._translateX, y: this._translateY, scale: this._scale }\n };\n },\n \n /**\n Repositions the view.\n */\n _reposition: function() {\n this.get(\"layer\").style.webkitTransform = \n \"translate3d(\" + this._translateX + \"px,\" + this._translateY + \"px, 0px) \" +\n \"scale3d(\" + this._scale + \",\" + this._scale + \",1)\";\n console.error(this._translateX + \" \" + this._translateY + \" \" + this._scale);\n },\n \n // and now, redirect mouse events :)\n mouseDown: function(evt) {\n this.touchStart(evt);\n },\n \n mouseDragged: function(evt) {\n this.touchesDragged(evt);\n },\n \n mouseUp: function(evt) {\n this.touchEnd(evt);\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"box\".w(),\n box: Box.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 200, height: 200 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n","highlighted":"<span class=\"keyword\">var</span> <span class=\"class\">Box</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">_scale</span>: <span class=\"number integer\">1</span>,\n <span class=\"variable\">_translateX</span>: <span class=\"number integer\">0</span>,\n <span class=\"variable\">_translateY</span>: <span class=\"number integer\">0</span>,\n <span class=\"variable\">acceptsMultitouch</span>: <span class=\"class\">YES</span>,\n <span class=\"variable\">touchStart</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">recomputeTouchStatus</span>(<span class=\"variable\">touch</span>, <span class=\"class\">YES</span>);\n <span class=\"keyword\">return</span> <span class=\"class\">YES</span>;\n },\n \n <span class=\"variable\">touchesDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>, <span class=\"variable\">touches</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">t</span> = <span class=\"this\">this</span>.<span class=\"variable\">_touch</span>;\n <span class=\"keyword\">var</span> <span class=\"variable\">avg</span> = <span class=\"variable\">evt</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>);\n \n <span class=\"comment\">// translation is easy:</span>\n <span class=\"this\">this</span>.<span class=\"variable\">_translateX</span> = <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">x</span> + <span class=\"variable\">avg</span>.<span class=\"variable\">x</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">x</span>;\n <span class=\"this\">this</span>.<span class=\"variable\">_translateY</span> = <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">y</span> + <span class=\"variable\">avg</span>.<span class=\"variable\">y</span> - <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">y</span>;\n \n <span class=\"comment\">// mathematically speaking, scale *= the end distance / the start distance</span>\n <span class=\"keyword\">if</span> (<span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">d</span> &gt; <span class=\"number integer\">1</span>) { <span class=\"comment\">// but prevent divide-by-0</span>\n <span class=\"this\">this</span>.<span class=\"variable\">_scale</span> = <span class=\"variable\">t</span>.<span class=\"variable\">ourStart</span>.<span class=\"variable\">scale</span> * (<span class=\"variable\">avg</span>.<span class=\"variable\">d</span> / <span class=\"variable\">t</span>.<span class=\"variable\">start</span>.<span class=\"variable\">d</span>);\n }\n \n <span class=\"comment\">// reposition</span>\n <span class=\"this\">this</span>.<span class=\"variable\">_reposition</span>();\n },\n \n <span class=\"variable\">touchEnd</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">recomputeTouchStatus</span>(<span class=\"variable\">touch</span>, <span class=\"class\">NO</span>);\n },\n \n <span class=\"multiline comment\">/**\n With this, we recompute our touch status--updating the start positioning and scale.\n */</span>\n <span class=\"variable\">recomputeTouchStatus</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">touch</span>, <span class=\"variable\">considerTouch</span>) {\n <span class=\"keyword\">var</span> <span class=\"variable\">avg</span> = <span class=\"variable\">touch</span>.<span class=\"variable\">averagedTouchesForView</span>(<span class=\"this\">this</span>, <span class=\"variable\">considerTouch</span>);\n <span class=\"this\">this</span>.<span class=\"variable\">_touch</span> = {\n <span class=\"variable\">start</span>: { <span class=\"variable\">x</span>: <span class=\"variable\">avg</span>.<span class=\"variable\">x</span>, <span class=\"variable\">y</span>: <span class=\"variable\">avg</span>.<span class=\"variable\">y</span>, <span class=\"variable\">d</span>: <span class=\"variable\">avg</span>.<span class=\"variable\">d</span> },\n <span class=\"variable\">ourStart</span>: { <span class=\"variable\">x</span>: <span class=\"this\">this</span>.<span class=\"variable\">_translateX</span>, <span class=\"variable\">y</span>: <span class=\"this\">this</span>.<span class=\"variable\">_translateY</span>, <span class=\"variable\">scale</span>: <span class=\"this\">this</span>.<span class=\"variable\">_scale</span> }\n };\n },\n \n <span class=\"multiline comment\">/**\n Repositions the view.\n */</span>\n <span class=\"variable\">_reposition</span>: <span class=\"keyword\">function</span>() {\n <span class=\"this\">this</span>.<span class=\"variable\">get</span>(<span class=\"string\">&quot;layer&quot;</span>).<span class=\"variable\">style</span>.<span class=\"variable\">webkitTransform</span> = \n <span class=\"string\">&quot;translate3d(&quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_translateX</span> + <span class=\"string\">&quot;px,&quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_translateY</span> + <span class=\"string\">&quot;px, 0px) &quot;</span> +\n <span class=\"string\">&quot;scale3d(&quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_scale</span> + <span class=\"string\">&quot;,&quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_scale</span> + <span class=\"string\">&quot;,1)&quot;</span>;\n <span class=\"variable\">console</span>.<span class=\"variable\">error</span>(<span class=\"this\">this</span>.<span class=\"variable\">_translateX</span> + <span class=\"string\">&quot; &quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_translateY</span> + <span class=\"string\">&quot; &quot;</span> + <span class=\"this\">this</span>.<span class=\"variable\">_scale</span>);\n },\n \n <span class=\"comment\">// and now, redirect mouse events :)</span>\n <span class=\"variable\">mouseDown</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchStart</span>(<span class=\"variable\">evt</span>);\n },\n \n <span class=\"variable\">mouseDragged</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchesDragged</span>(<span class=\"variable\">evt</span>);\n },\n \n <span class=\"variable\">mouseUp</span>: <span class=\"keyword\">function</span>(<span class=\"variable\">evt</span>) {\n <span class=\"this\">this</span>.<span class=\"variable\">touchEnd</span>(<span class=\"variable\">evt</span>);\n }\n});\n<span class=\"keyword\">var</span> <span class=\"class\">MyExampleView</span> = <span class=\"class\">SC</span>.<span class=\"class\">View</span>.<span class=\"variable\">extend</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;white&quot;</span>,\n <span class=\"variable\">childViews</span>: <span class=\"string\">&quot;box&quot;</span>.<span class=\"variable\">w</span>(),\n <span class=\"variable\">box</span>: <span class=\"class\">Box</span>.<span class=\"variable\">design</span>({\n <span class=\"variable\">backgroundColor</span>: <span class=\"string\">&quot;red&quot;</span>,\n <span class=\"variable\">layout</span>: { <span class=\"variable\">left</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">top</span>: <span class=\"number integer\">10</span>, <span class=\"variable\">width</span>: <span class=\"number integer\">200</span>, <span class=\"variable\">height</span>: <span class=\"number integer\">200</span> }\n })\n});\n\n<span class=\"comment\">// bootstrap code :)</span>\n<span class=\"variable\">exports</span>.<span class=\"variable\">getDemoView</span> = <span class=\"keyword\">function</span>() {\n <span class=\"keyword\">return</span> <span class=\"class\">MyExampleView</span>;\n};\n","original":"var Box = SC.View.extend({\n _scale: 1,\n _translateX: 0,\n _translateY: 0,\n acceptsMultitouch: YES,\n touchStart: function(touch) {\n this.recomputeTouchStatus(touch, YES);\n return YES;\n },\n \n touchesDragged: function(evt, touches) {\n var t = this._touch;\n var avg = evt.averagedTouchesForView(this);\n \n // translation is easy:\n this._translateX = t.ourStart.x + avg.x - t.start.x;\n this._translateY = t.ourStart.y + avg.y - t.start.y;\n \n // mathematically speaking, scale *= the end distance / the start distance\n if (t.start.d > 1) { // but prevent divide-by-0\n this._scale = t.ourStart.scale * (avg.d / t.start.d);\n }\n \n // reposition\n this._reposition();\n },\n \n touchEnd: function(touch) {\n this.recomputeTouchStatus(touch, NO);\n },\n \n /**\n With this, we recompute our touch status--updating the start positioning and scale.\n */\n recomputeTouchStatus: function(touch, considerTouch) {\n var avg = touch.averagedTouchesForView(this, considerTouch);\n this._touch = {\n start: { x: avg.x, y: avg.y, d: avg.d },\n ourStart: { x: this._translateX, y: this._translateY, scale: this._scale }\n };\n },\n \n /**\n Repositions the view.\n */\n _reposition: function() {\n this.get(\"layer\").style.webkitTransform = \n \"translate3d(\" + this._translateX + \"px,\" + this._translateY + \"px, 0px) \" +\n \"scale3d(\" + this._scale + \",\" + this._scale + \",1)\";\n console.error(this._translateX + \" \" + this._translateY + \" \" + this._scale);\n },\n \n // and now, redirect mouse events :)\n mouseDown: function(evt) {\n this.touchStart(evt);\n },\n \n mouseDragged: function(evt) {\n this.touchesDragged(evt);\n },\n \n mouseUp: function(evt) {\n this.touchEnd(evt);\n }\n});\nvar MyExampleView = SC.View.extend({\n backgroundColor: \"white\",\n childViews: \"box\".w(),\n box: Box.design({\n backgroundColor: \"red\",\n layout: { left: 10, top: 10, width: 200, height: 200 }\n })\n});\n\n// bootstrap code :)\nexports.getDemoView = function() {\n return MyExampleView;\n};\n"}},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Multitouch"}]},{"title":"Built-In Support","articles":[{"content":"<h1>ButtonView Touch Support</h1>","errors":[],"demos":{},"articleDirectory":"articles/controls/button/","outputDirectory":"build/","title":"ButtonView Touch Support"},{"content":"<h1>ScrollView Touch Support</h1>","errors":[],"demos":{},"articleDirectory":"articles/controls/scroll/","outputDirectory":"build/","title":"ScrollView Touch Support"}]},{"title":"Advanced Concepts","articles":[{"content":"<h1>Capturing</h1>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Capturing"},{"content":"<h1>Releasing</h1>","errors":[],"demos":{},"articleDirectory":"articles/touch/","outputDirectory":"build/","title":"Releasing"}]}],"file":"build/guides/touch"}
View
26 docs/src/articles/touch/a-brief-touch.md
@@ -6,10 +6,26 @@
"and": "He is wrong about touch apps."
}
-A Brief Touch on Touches
-========================
-Some text goes here.
+A Brief Touch
+=============
-Note that links like the below must appear on empty lines with no preceding whitespace.
+It is very possible to build _awesome_ touch-enabled applications in SproutCore.
-{{demo:sc|touch-demo.js}}
+But, what makes an awesome touch-enabled application? Sure, it must accept touches,
+but with SproutCore's (constantly growing) touch support, this is now pretty easy:
+many existing interfaces, if built with newer SproutCore varieties, will function fine (or mostly fine),
+on both larger-screened touch devices (such as iPad) and the traditional desktop environment.
+
+But there are many differences between desktop and touch platforms:
+
+- **Precision.** Touches are less precise than clicks. To compensate, controls should be larger.
+- **Performance.** Touch-based devices tend to be slow (for now). To get around this just takes
+ some elbow grease: there are many techniques to speed things up... many of which SproutCore will
+ handle for you.
+- **Animation.** Lack of animation looks okay on desktop (even if animation is cool)... but on
+ touch devices, non-animated interfaces look strange: touch is so much more realistic than mouse,
+ but sudden changes without transitions are not realistic at all.
+- **Coolness.** Touch-based interfaces are cool. That is all.
+
+In this guide, we go over each of these—except for the last, which is rather vague; you'll have to figure out
+your own meaning of "coolness".
View
78 docs/src/articles/touch/multitouch.js
@@ -0,0 +1,78 @@
+var Box = SC.View.extend({
+ _scale: 1,
+ _translateX: 0,
+ _translateY: 0,
+ acceptsMultitouch: YES,
+ touchStart: function(touch) {
+ this.recomputeTouchStatus(touch, YES);
+ return YES;
+ },
+
+ touchesDragged: function(evt, touches) {
+ var t = this._touch;
+ var avg = evt.averagedTouchesForView(this);
+
+ // translation is easy:
+ this._translateX = t.ourStart.x + avg.x - t.start.x;
+ this._translateY = t.ourStart.y + avg.y - t.start.y;
+
+ // mathematically speaking, scale *= the end distance / the start distance
+ if (t.start.d > 1) { // but prevent divide-by-0
+ this._scale = t.ourStart.scale * (avg.d / t.start.d);
+ }
+
+ // reposition
+ this._reposition();
+ },
+
+ touchEnd: function(touch) {
+ this.recomputeTouchStatus(touch, NO);
+ },
+
+ /**
+ With this, we recompute our touch status--updating the start positioning and scale.
+ */
+ recomputeTouchStatus: function(touch, considerTouch) {
+ var avg = touch.averagedTouchesForView(this, considerTouch);
+ this._touch = {
+ start: { x: avg.x, y: avg.y, d: avg.d },
+ ourStart: { x: this._translateX, y: this._translateY, scale: this._scale }
+ };
+ },
+
+ /**
+ Repositions the view.
+ */
+ _reposition: function() {
+ this.get("layer").style.webkitTransform =
+ "translate3d(" + this._translateX + "px," + this._translateY + "px, 0px) " +
+ "scale3d(" + this._scale + "," + this._scale + ",1)";
+ console.error(this._translateX + " " + this._translateY + " " + this._scale);
+ },
+
+ // and now, redirect mouse events :)
+ mouseDown: function(evt) {
+ this.touchStart(evt);
+ },
+
+ mouseDragged: function(evt) {
+ this.touchesDragged(evt);
+ },
+
+ mouseUp: function(evt) {
+ this.touchEnd(evt);
+ }
+});
+var MyExampleView = SC.View.extend({
+ backgroundColor: "white",
+ childViews: "box".w(),
+ box: Box.design({
+ backgroundColor: "red",
+ layout: { left: 10, top: 10, width: 200, height: 200 }
+ })
+});
+
+// bootstrap code :)
+exports.getDemoView = function() {
+ return MyExampleView;
+};
View
94 docs/src/articles/touch/multitouch.md
@@ -0,0 +1,94 @@
+Multitouch
+===========
+Handling single touches is pretty easy—not that much different from handling
+mouse events. But what about multiple touches?
+
+Accepting Multiple Touches
+---------------------------
+First, you have to tell your view that you do, indeed want to receive multiple
+touches. By default, views only receive single touches. This is because is easier
+to think in a single-touch model, and most controls only need to track a single touch.
+
+To accept multiple touches, just set the view's `#js:acceptsMultitouch` property to
+`#js:YES`.
+
+ #js
+ view = SC.View.extend({
+ acceptsMultitouch: YES
+ });
+
+Processing the Individual Touches
+-----------------------------------
+Even without the supplied helper function, processing individual touches is relatively
+simple:
+
+- You get a separate `#js:touchStart` for each individual touch.
+- You get a separate `#js:touchEnd` for each individual touch.
+- You get one `#js:touchesDragged` each event cycle for all of your touches put together.
+
+So, detecting individual touches starting and ending is simple. Detecting those touches
+moving is not quite as simple, but still relatively easy.
+
+Remember how `#js:touchesDragged` works:
+
+ #js
+ touchesDragged: function(evt, touches)
+
+`#js:touches` is an SC.CoreSet of SC.Touch objects. What can you do with a CoreSet?
+You can do a couple of things:
+
+- Turn it into an array and do whatever.
+- Call .forEach to iterate.
+
+But you don't have to use the touches set at all. The `#js:evt` has some useful properties and methods,
+too:
+
+- pageX/pageY: the position of the first touch.
+- averagedTouchesForView: a method which returns the averaged touch position
+ and the average distance of the touches from that position.
+
+`#js:SC.ScrollView`, for instance, makes heavy use of `#js:averagedTouchesForView`, and never
+directly touches the `#js:touches` set.
+
+
+Averaging Touches
+------------------
+It is often _very_ useful to average the touches.
+
+`#js:averagedTouchesForView` returns an object with four properties:
+
+- **`#js:x`**: The average X position of the touch.
+- **`#js:y`**: The average Y position of the touch.
+- **`#js:d`**: average distance of the all touches from the average x/y position.
+- **`#js:touchCount`** The number of touches averaged.
+
+You can call `#js:averagedTouchesForView` on two separate objects: an `#js:SC.Event` object,
+or an `#js:SC.Touch` object.
+
+The two work identically but for one important difference: when you call it on `#js:SC.Touch`,
+you _have the option_ of telling the touch to add itself to the averaged set. Doing so makes no sense
+in most cases: the touch would just be counted twice! But what about `#js:touchStart`?
+
+Recall that during `#js:touchStart`, the view does not yet own the touch. So, `#js:averagedTouchesForView`
+would not, by default count it.
+
+ #js
+ // on an event:
+ var a = evt.averagedTouchesForView(this);
+
+ // on a touch
+ var a = touch.averagedTouchesForView(this);
+
+ // on a touch, counting the touch itself
+ var a = touch.averagedTouchesForView(this, YES);
+
+Thinking it Over
+------------------
+How might you use all of these to produce a good result?
+
+Let's take a simple example: moving and resizing something:
+
+{{demo:sc|multitouch.js}}
+
+
+
View
17 docs/src/articles/touch/touch-demo.js
@@ -16,6 +16,23 @@ var Dot = SC.View.extend({
width: t.ourStart.width,
height: t.ourStart.height
});
+ },
+
+ touchEnd: function() {
+ // actually, we don't need to do anything here...
+ },
+
+ // and now, redirect mouse events :)
+ mouseDown: function(evt) {
+ this.touchStart(evt);
+ },
+
+ mouseDragged: function(evt) {
+ this.touchesDragged(evt);
+ },
+
+ mouseUp: function(evt) {
+ this.touchEnd(evt);
}
});
var MyExampleView = SC.View.extend({
View
114 docs/src/articles/touch/touch-events.md
@@ -1,2 +1,114 @@
Touch Events
-=============
+=============
+
+SproutCore's touch events have a few great features:
+
+- Multiple views can receive touches simultaneously.
+- A single view can receive multiple touches.
+- A view can capture touches before allowing them to pass through to children.
+- Child views can release touches back to parent views that originally captured them.
+
+We won't get into the last two in this article—they're quite sophisticated!
+
+Simple Single-Touch Handling
+=====================================================
+You may be familiar with this SC.View method signature if you are familiar with SproutCore:
+
+ #js
+ mouseDown: function(evt)
+
+With `#js:mouseDown`, you can decide whether or not to accept the mouse event by returning either
+`#js:YES` or `#js:NO`.
+
+Touch events are similar, but work a bit differently:
+
+ #js
+ touchStart: function(touch)
+
+Instead of being passed a raw event, `#js:touchStart` is passed an `#js:SC.Touch` object.
+If you return YES from touchStart, your view will "own" the touch—in SproutCore terms, your
+view will be the _touch responder_. For more information about what this entails, see
+the "Internals" article. Your view will own the touch until the touch ends.
+
+`#js:touchStart` will be called once for every touch that touches the view.
+
+**Note:** By default, views only receive `#js:touchStart` and touchEnd for a single touch. This is
+a feature intended to make it easier to handle such cases, very similar to Cocoa Touch's `#js:acceptsMultitouch`
+property—actually, SC.View uses the same property name! See the "Multitouch" article.
+
+Anatomy of an SC.Touch
+------------------------
+SC.Touch represents the touch from the time the user puts their finger on the screen until the time they lift it.
+
+The touch object acts like an event object in many ways. It has many other useful things, as well:
+
+- `#js:pageX` and `#js:pageY` for the touch
+- `#js:event`: if in an event cycle, this contains the event. Otherwise, it is `#js:undefined`.
+ You will probably never need to access this.
+- `#js:preventDefault`: if the touch is connected with an event, this calls `#js:preventDefault()` on the event.
+- `#js:touchesForView(view)`: when supplied a view, will find all touches that the view is the
+ touch responder for. It is a CoreSet; to get an array, call `#js:.toArray` on the result.
+- `#js:averagedTouchesForView(view)`: When supplied a view, averages all the touches on that view,
+ returning both an average position and an average distance from that position.
+
+**Note:** If you call `#js:touchesForView(this)` from `#js:touchStart`, the touch supplied will not be in the set
+returned by `#js:touchesForView(this)`: you don't own the touch until you return YES.
+
+touchEnd
+-----------------
+Knowing when the touch starts is not very useful. At the very least, you will likely want to know when the touch
+ends as well.
+
+It is quite simple:
+
+ #js
+ touchEnd: function(touch)
+
+It works exactly like `#js:touchStart`. The touch will no longer be in the set of touches for the view,
+so if you call `#js:touch.touchesForView(this)`, you'll only receive any other active touches. If your
+view does not accept multitouch, then the set is guaranteed to have no touches in it—you only receive
+`#js:touchEnd` for the last touch that ends.
+
+Tracking Touches
+------------------
+Tracking touch movement is simple:
+
+ #js
+ touchesDragged: function(evt, touches)
+
+The `#js:touches` argument is the set of touches on the view—the same set you get by calling `#js:touchesForView(this)`.
+This will have _all_ touches, regardless of whether or not your view accepts multitouch.
+
+If your view does _not_ accept multitouch, then it is even simpler:
+
+ #js
+ x = evt.pageX;
+ y = evt.pageY;
+
+Tip: Cross-Platform-iness
+--------------------------
+Did you realize that, assuming you don't use the set of view touches or other touch-specific API,
+you can do this:
+
+ #js
+ mouseDown: function(evt) {
+ this.touchStart(evt);
+ },
+
+ mouseDragged: function(evt) {
+ this.touchesDragged(evt);
+ },
+
+ mouseUp: function(evt) {
+ this.touchEnd(evt);
+ }
+
+Of course, you can also redirect touch events to mouse events, but that is not as fun.
+
+Putting it All Together
+------------------------
+Here is a very simple demo that uses the methods described above to allow the user to move two views
+around the screen:
+
+{{demo:sc|touch-demo.js}}
+
View
5 docs/src/guides/touch.guide
@@ -1,11 +1,12 @@
{
- "title": "Writing Touch Applications",
+ "title": "Touch Application Guide",
"sections": [
{
"title": "Introduction",
"articles": [
"articles/touch/a-brief-touch",
- "articles/touch/touch-events"
+ "articles/touch/touch-events",
+ "articles/touch/multitouch"
]
},
{
View
BIN  themes/pig/resources/images/2.png
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
View
33 themes/pig/resources/theme.css
@@ -1,4 +1,4 @@
-.sc-theme .sc-view.ace.light.pig.sc-scroll-view.desk { background: url('') repeat-y; }
+.sc-theme .sc-view.ace.light.pig.sc-scroll-view.desk { background: url('') repeat-y; }
.sc-theme .sc-view.ace.light.pig.sc-collection-view.sc-list-view.sc-source-list.desk {
z-index: 0;
background: none;
@@ -70,6 +70,7 @@
background: white;
-webkit-box-shadow: 0px 5px 25px white;
}
+.sc-theme .sc-view.ace.light.pig.paper.sc-scroller-view { z-index: 20; }
.sc-theme .sc-view.ace.light.pig.sc-toolbar-view { -webkit-box-shadow: none; }
.sc-theme .sc-view.ace.light.pig.sc-view.demo { background: none; }
.sc-theme .sc-view.ace.light.pig.sc-view.demo .middle { display: none; }
@@ -93,12 +94,40 @@
height: 150px;
border: 1px solid gray;
}
-.sc-theme .sc-view.ace.light.pig.sc-static-content-view { font-size: 1.2em; }
+.sc-theme .sc-view.ace.light.pig.sc-static-content-view {
+ -webkit-user-select: text;
+ font-size: 1.2em;
+}
.sc-theme .sc-view.ace.light.pig.sc-static-content-view h1 { font-size: 1.5em; }
.sc-theme .sc-view.ace.light.pig.sc-static-content-view h1 { color: #5078aa; }
.sc-theme .sc-view.ace.light.pig.sc-static-content-view h2 { color: #5078aa; }
.sc-theme .sc-view.ace.light.pig.sc-static-content-view h3 { color: #5078aa; }
.sc-theme .sc-view.ace.light.pig.sc-static-content-view h4 { color: #5078aa; }
+.sc-theme .sc-view.ace.light.pig.sc-static-content-view code {
+ font-family: Monaco, Inconsolata, Courier, fixed-width;
+ font-size: 12px;
+}
+.sc-theme .sc-view.ace.light.pig.sc-static-content-view pre code {
+ margin-right: 1em;
+ border: 1px solid #a0b0a0;
+ overflow-y: hidden;
+ overflow-x: auto;
+ background: #f5f9f5;
+ display: block;
+ padding: 1em;
+}
+.sc-theme .sc-view.ace.light.pig.sc-static-content-view code .class { color: #000096; }
+.sc-theme .sc-view.ace.light.pig.sc-static-content-view code .comment { color: #6496c8; }
+.sc-theme .sc-view.ace.light.pig.sc-static-content-view code .string { color: #00640a; }
+.sc-theme .sc-view.ace.light.pig.sc-static-content-view code .number { color: #0000ff; }
+.sc-theme .sc-view.ace.light.pig.sc-static-content-view code .keyword {
+ color: #196e19;
+ font-weight: bold;
+}
+.sc-theme .sc-view.ace.light.pig.sc-static-content-view code .this {
+ color: #196e19;
+ font-weight: bold;
+}
.sc-theme .sc-view.ace.light.pig.sc-scroll-view.source {
background-color: white;
-webkit-transform: translate3d(0px,0px,0px);
View
24 themes/pig/src/content.css
@@ -4,6 +4,7 @@
}
@view(sc-static-content-view) {
+ -webkit-user-select: text;
font-size: 1.2em;
h1 {
font-size: 1.5em;
@@ -13,4 +14,27 @@
h1, h2, h3, h4 {
color: rgb(80,120,170);
}
+
+ code {
+ font-family: Monaco, Inconsolata, Courier, fixed-width;
+ font-size: 12px;
+ }
+
+ pre code {
+ margin-right: 1em;
+ border: 1px solid #a0b0a0;
+ overflow-y: hidden;
+ overflow-x: auto;
+ background: #f5f9f5;
+ display:block;
+ padding: 1em;
+ }
+
+ code .class { color: rgb(0, 0, 150); }
+
+ /*code .variable { color: rgb(10, 70, 10); }*/
+ code .comment { color: rgb(100, 150, 200); }
+ code .string { color: rgb(0, 100, 10); }
+ code .number { color: rgb(0, 0, 255); }
+ code .keyword, code .this { color: rgb(25, 110, 25); font-weight: bold; }
}
View
4 themes/pig/src/paper/paper.css
@@ -12,4 +12,8 @@
-webkit-box-shadow: 0px 5px 25px white;
}
+@view(sc-scroller-view) {
+ z-index: 20; /* Sit above toolbar */
+}
+
@end
View
BIN  themes/pig/src/tree/bg.jpg
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
View
2  themes/pig/src/tree/tree.css
@@ -1,5 +1,5 @@
@view(sc-scroll-view).desk {
- background: sprite("bg.png" repeat-y);
+ background: sprite("bg.jpg" repeat-y);
}
@view(sc-collection-view.sc-list-view.sc-source-list).desk {
Please sign in to comment.
Something went wrong with that request. Please try again.