Skip to content

Instantly share code, notes, and snippets.

@asnowfix
Created September 30, 2013 09:51
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save asnowfix/6761624 to your computer and use it in GitHub Desktop.
Save asnowfix/6761624 to your computer and use it in GitHub Desktop.
large form-data exhibit issue refered to by https://github.com/mscdex/busboy/issues/1
This file has been truncated, but you can view the full file.
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="README-CORDOVA-WEBOS.md"
Content-Type: application/octet-stream; x-encoding=base64
Q29yZG92YSB3ZWJPUyBJbnN0cnVjdGlvbnM6Ci0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKLSBTZXQgdXAgeW91ciBwcm9qZWN0IHRoZSBzYW1lIHdheSB5b3Ugd291bGQgc2V0IHVwIGFueSBvdGhlciBib290cGxhdGUgcHJvamVjdC4KLSBNb2RpZnkgeW91ciBpbmRleC5odG1sIGFuZCBKYXZhU2NyaXB0IGZpbGVzIGFzIG91dGxpbmVkIGluIHRoaXMgYmxvZyBwb3N0OiBbSmF2YVNjcmlwdCBBcHBzIGZvciBPcGVuIHdlYk9TIFdpdGggRW55byBhbmQgQ29yZG92YV0oaHR0cDovL2Jsb2cub3BlbndlYm9zcHJvamVjdC5vcmcvcG9zdC8zOTI3ODYxODI5OS9qYXZhc2NyaXB0LWFwcHMtZm9yLW9wZW4td2Vib3Mtd2l0aC1lbnlvLWFuZC1jb3Jkb3ZhKQotIENvcHkgYXBwaW5mby5qc29uIGZyb20gdGhlIFBob25lR2FwICdsaWIvd2Vib3MvZnJhbWV3b3JrJyBkaXJlY3RvcnkgYW5kIGEgY29yZG92YSBKYXZhU2NyaXB0IGZpbGUgZnJvbSB0aGUgUGhvbmVHYXAgJ2xpYi93ZWJvcy9saWInIGRpcmVjdG9yeQotIE1vZGlmeSBhcHBpbmZvLmpzb24gZm9yIHlvdXIgcHJvamVjdCBhbmQgYWRkIGFuIGljb24KLSBydW4gdG9vbHNcZGVwbG95LmJhdCAtLWNvcmRvdmEtd2Vib3Mgb3IgdG9vbHMvZGVwbG95LnNoIC0tY29yZG92YS13ZWJvcyB0byBkZXBsb3kgeW91ciBhcHAuCg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="README.md"
Content-Type: application/octet-stream; x-encoding=base64
VGlwQ2FsYwo9PT09PT09CgpCYXNlZCBvbiBib290cGxhdGUKClJlZmVyIHRvIHRoZSBbV2lraV0oaHR0cHM6Ly9naXRodWIuY29tL2VueW9qcy9lbnlvL3dpa2kvQm9vdHBsYXRlKSBmb3IgaG93IHRvIGdldCBzdGFydGVkLgoKCgpCdWlsZGluZyBmb3Igd2ViT1MKPT09PT09PT09PT09PT09PT09CgotIC4vdG9vbHMvZGVwbG95LnNoCi0gY3AgYXBwaW5mby5qc29uIGRlcGxveS9UaXBDYWxjCi0gY3AgZnJhbWV3b3JrX2NvbmZpZy5qc29uIGRlcGxveS9UaXBDYWxjCi0gY2QgZGVwbG95Ci0gcGFsbS1wYWNrYWdlIFRpcENhbGMKLSBwYWxtLWluc3RhbGwgY29tLnlkbS50aXBjYWxjXzEuMC4wX2FsbC5pcGsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="appinfo.json"
Content-Type: application/octet-stream; x-encoding=base64
ewogICJpZCI6ICJjb20uZXhhbXBsZXMuYXBwcy5teWFwcCIsCiAgInZlcnNpb24iOiAiMC4wLjEiLAogICJ2ZW5kb3IiOiAiTXkgQ29tcGFueSIsCiAgInR5cGUiOiAid2ViIiwKICAibWFpbiI6ICJpbmRleC5odG1sIiwKICAidGl0bGUiOiAiRXhhbXBsZTogTXkgQXBwbGljYXRpb24iLAogICJpY29uIjogImljb24ucG5nIiwKICAidWlSZXZpc2lvbiI6IDIKfQ==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="assets/enyo.png"
Content-Type: application/octet-stream; x-encoding=base64
iVBORw0KGgoAAAANSUhEUgAAAIAAAACACAYAAADDPmHLAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAG21JREFUeNrsXWl0FNeZvVWtDSEQYCMWIRASYpVAIAwGBZABY+wEg3HsxB4nASeZnNgniX1m8iMzP2LPn/zImbGTcRzHcxLLx0kcExODdyCA2AQItIAWQGySQCCEEBJol7pr3lZVr3rTQsvdUteTy9XdVFdVv+9+97vfe6/eUzRNg13Ct6h2FdgAsIsNALvYALBLWJaIcPvBc7MeSia7TWRbJfa07CDbAbqvKNxfFU71oYRDFkCMnkl23yNbDtkyezm8hGx5ZHuXgKHEBsDQNPgYYeyNwsvHeI1/iorIqCh0d3fB5XJ5O6RJsMNOCgoCiCYbAEOL2j1/MPmDotD/4HA4EB0zgrxWQOvB6exBT3c3AUS3r68Pu1AxpAHQV2pXiKdTg6uqyl4zDJA/1aEyAPA60MCrQmMw6enphrOnB91dXfBRR8MiVAwpAAhql718jM/0Rhib7rnHE4Mrwvs5FRAGiCAAiDEMbOxJOKCvXWSj3+khQKCAoGDoJVTo7NBkAyDw1L5ReLpPaldVhXu7QxhfGJx9blA/3xhISAiIio42zmHUhaZxALh0ZtAMw1NwMEAQ3eAnVOQJ3RDyoSIkAUCMLgu4ZN/UToypqkzMKarC9+wz3dCqYXBFkQAAAYAIogGiY7iRBfub4YBsLrrnbCBvDAyi3rp7DxVVupAkYMizAXCP1M5pXRhX1aldf80NzmgfMhBkzzcZgIrAqJgY0+vF3vB6HQT0Txjd5XJ5gIHu6TmdIlT0dPcwQTkUQkXQANAfaje9WvJ0i/FVQ+jx14oZ/yWm4GEAQgNQoBAGGOEJAG9agLxhIYEb3mWwAweJy8oOIpzQjIICgbJDqIaKoACAGP+yX2q3GF2icpXHdd3Iqmp+Lh+nWt7Dg/41egy9DksDY9w0gGaAwGMzDK2DwWUJCTobyHtdcHJ28B8qCAimf9W2CFZTcLJPozOWNmO5TO+qZHgYhlY9gKBKHq9/rulMIIUCmgZGREZ6ZQBNAIIb3eUBCpf4nP67DgbFxd/T+3Hpe3EOKjgjyb1SzUH/jbIDBYQUKpKDYYiQ6AtQ9dxckTxelendNDb73KB88dqdGej5xPctws8Qg3o7gAOOiAiL+teZQN9AwWN4PTewHlY0jRhYIYZX6ef83C56PPV80bjEQEGOpwmFC2Z4iSDXpeCj52xvaw1a3QcdAIZBDMNzY3LD8jjtbni69wUICwikcCCngBp0EUjO45CqwNIeYFK4y/BsUwO4e7smgKBSRnADAn1P/kcDrskoLvN6wSxBBYCi/0mijQFAGN70bhkMYi8Mz71dBoSVDYxwophhBlIW4CsE0L1ucMWhCaqnhnMxz+deT0Hi5KCi1xNAgMIB4hJ6QyBdfMbpQFN48qmEMwAECsx4bwBAtbw2vJ4YjLGDw2EFjGIeyypa5ccpvP3XmgpSo7CwIzSAxACWNgABBiPtI2BQDXHn5J7OPlOYBmBsoTiZt+tez/ai4ZAa3wk+AIOdjwKFogBaeAPAI+67GZ9StCq8nhqMGpsCQE/j3EEDWTSK14Di0SCkM4AZAkz6VyQdoKouqU2As4Ci0zsVgU56TonqCQg0/Rq6qCVnpDhQ6XnofeoaI/j2D5EBIYpngw03qMOgeVWIOub9zOhEwAnjOxgg3MKELgBlBhBA0EQ4cDAR6JBCsQQCIxRwEciMZwg+Huep8GPGpmBwOg2jE41vDLXSDa9p4l6ELpCBGL4AUOT0T8/brR5tbNT4RDnLwFDEa0WEB/eGIYsAZGkhb0nURL2zLEAwgCYJQQ2mDjCyAWE4lQJC5UxAQ4HLyRlG0xuXaM5PqtUpgUDTHBwEvOHF1DzgYcHWAAIEqUmJqLp+wxMIwvMNPeD2mrGA6jBUPwUFdDbgXYNubQASAAio4N4bKOf8uiYQ52Yq3iU3KXPP11h2wXN6lWKHVC1lAoWFEQIFF00ZNcyYkojzNVeY84dCL0xE0K0vMcCTG9dj8uSJ2Lb9c5ScPc9jv2RkrgMcjLp5r5+DGV5xYwVjk8OKlCrqxSExgGF4PR3UeP6ux31qYKYDFM4AoJSv6X0O5Bgn92WX4jTTO+L5Ootkpc/B05sfRS0B+K/f+KOlQSrMRSCkBhoF05KT8PN/+xGuXLmG7Tt24dS5i1aqNwwsDO6g3h9hEYOQj3PrEdSkziA9dECSAJrRFCwagBTViPssjVNcPJ5TYUdCgIMpfNOMjOZJ2qir/czZs7F5w1okJU1i/36NMZxi9YHwbQfQWwM86yIpaTJe+slW1NZex45P9+N05SVhVIdhfO71Dt7/rzoMzzeOs7QDmINCNHEtSv8yALj4A2/okUMBE3sqS/MUSuXE8JqDm9sFSLFe1w4OzJ+VisfXr0Ri4kSPtg9LVhLuIlDfFB+KODFxEl780bNobr6L3fuOouDUWXSTvFtnBYcOCKELYGkTkPoJhPCC0fIoMgu3lkCzMcjs8WPxXTTiOIS7u2jod5g/gZYoAqgHMmZh3aolGB0f18vvDn4GEBohAEqf4mF8/Cg89cQ6PLJ6GfYeKsTJskp09biYDjBEoR4q3ECg6I1HUsoJlgYK8WjQtyY6A2n8V1njjyaafDWhI2isV3XWomGffD0qKhKL56ZhdXYmRo2O61vDV/DZP4TaAQwa6P3I0QQIT3wjB2tXPYC8/BIUVlwijOBiniwbXpFFodRVLDfOqJa+ADkTUIX3c/XP2/VpykevA9HoQwwfGYmsOdOxcul8jBoV25+mT2lvA8AUg/3wiVGjRmLDI9nIWZ6JwwXlKDpXjS6nyzS+QzUahQz6F20B0PNwB9+8hgBFDwMKS+X0oaC0HycyMgKZs6Yhe9GcfhjeLeopNgNYKHGghQLh0TVLsGJpOvILz6H4fA26nZolFJipoMPIOngWwEWkRQQCRv8/c3OVv6bSMYbE+AVpU7Ascxbi4mICgnrNZgBFpoABl7i4WCK+FrLt5OmLOFZRhdaObtPwqpsQBERfg6o7P78TpgEUo8OGvo6LjsKSudOwKH16YH6upRk43NsBBuGci+ensq24vAoFZ6+gpbPbMDzbiwvz5mOHgT+9DYA28ND7GhkTgwdmTUXmnKTA9nsgdEpIaAAzRw/seRfOS2ZbCQHByXO1bkCA2ZEkZwDkb2Q0UfUzE0kuP2XQfq8NAKv6G1Q6zJydxLbTlbU4eb4OrQQITOvrjUUiHR05IgpZaZOQnjb5KwF92ANArgT6etfew0ibmYKxY+MH5XrzqVeTrfTCdZRVN6CptZPpg3FE0KVPT8C81ImD+nubbjdjz/58C9CDDYNgDQtnF+VDsqIQFRXFmmUjyeuRsbEkr87CNx7NwZhBAoJert5oQjS5dtKksYN6neamO/hs90EcOFaE1rY29jg6GxVMtq7OTrS23GXHVRTuV8KKAbyVjq4u7D16AqfOXsQjOcuwdvWyQbtW0sSxiCYKfzDLvgMnsOvAMTQ0NrKnhhRbBPaukOn/m1tasf3LPBwrrsDG9auQMW8GhlIpO3MJO3cfxrX6m3AKw8tax9YAHhrQsz2AVtINEjf/tO1zTEuciGc2rsGECfeFtOFv1Dfi758dxOXa6+xpoFBS/CHMALySpk+eiCv1DUbroD5oWmPx+hZee+cj3tu2chFrAQyl0tLSjn8eKUFB6TnxtI/i0bZlZByKDQCvGHj8sdUkNYvAPz7bi4Y7Laz51TKYg2xFZy/hbHUdHpiXinWrFoXErf/zyGmcLL+AtvYOjw4nBQiZAaAhCgDFkg4tXpzBtj378vFF3nH06KOrpEfHugi1Hi27iAoChBULZyErIzUod15Udhn5py+g6W4r6ya2jjm0jvxVvLZ72ADwWR5evRxLsjLwz4MncLiw3PJvemi429aJ3QVnUFF1A9kLUpEydcJXcm9VV28iv/Qyakm853G+byDv/bOwBoBnhdBBIE9uWI3sBzKwjQira7eaLQ9T6QM7rt26g38cKkPKpDrkLErFuDFxg3KHt5tbcbD4Ii7XNTJlb9y24r9Dy4z9tB/CaTNAf8vEiePx0+8/iTPnqvBx3km0d/VYcSMquOrGbby7uxiZKZOwdG4SYmOjA3L99vYuFJy5gtLLdWxQiHdjKhgVG0OA4SBA8c0KoaYEhtRUsXNmJbNt94EiFFXWMH2geMkpS6tv4nxdM+YnJ+DB9HvrySsov4qy6np0dnXDVxNOdEQEli5Iw4olc/H7P39GAHB3IIRnA6Cvhar/5YvnYPeRUlSSeGytVy6+OnucKLxUT4BwB1mpCZg7fXy/rnGm6iaKL9bjbnunMSeAV1BOm4S1D87DyJExQ7Eqh+5k0XEjR2DzuiWorbuFfSfP40az90kWWjq6cPDMNcYIWanjMSXBf/9Cbf1dFBPgXL/dAn/9JJPvi8fKzBRMnjhuYD9AswEQkJI48T585xv3oeRMDU5U0j7/HmNol1zqmtrwefFVTLu/GQ/OTED8KKvHNt/tIN+/gepbLT49no4ViBsRjcUzJ/scK6BAsxkgGCVzzlTMnJZAtME1lNU0oNsHa9fcasWV41XInDoW6cn3sef3y6sbUVrT6MdLNURFqCSMTMaCmYmIHRHVi3Nrlsk/LM8c2gAYvBJLVPjXCC3Pm56Aw2VXUdPQYg705BbgQ79cGkpqbuPs9bsYEeVAl0CLSzKPPG9Q0v2jsXxeEsbGx97jHWohFwNCCACBq5Cx8XHYkD0bNddvI//sddxu7YLiEP0K9LEvhY/77yZC0SEyB934xhw+9Dwjo7F01iRMnTQmcMbWbAbwWVGH8guRlha4EUFTJ41lW9G56yipbmDPDfAn+QQI2Hh/yetF7I9yqFhAsobMtP6PEKKzgSkGebhM+g9RBITMmkG0noorKvHKr95AYVFZQM+9iHjxMytnY96UcfzJDjfjyMafkzgGT6+YNSDj6z9Ek6aDM8KJ5b3NAD5LU0sLfpf7ARYXluNfnn6MNQcHooyIicSKjClInzYOB8uvo6653ZgIkpYJ8SOwMn2KR3YwUDB7M3QoTswdIgDQDPWsvy05U4nzv76Kbz3+MB5cMj9w+mB0LDYuS8XFq40ouHATMUQELpk5AalT7gsQk5kzjUFiAqvxNRsAnnSoWSZqpoWOD/zLzj0oKq3Ed55ej1GjAtfJk0rCwZSE0YgfPTLgYJZnGnP/TfI0tLYG8ONFchp2rvoqfvW7v+LQ0VMBvU5MdGTA772jq9t8shjSjCOQdIAWOiEhpACQvWg+xsTFGXWk6U/lkjftnV34eP8xvPHODtTfbAw50NbfvI033/sE9bebTerXNAvbx5PfRn9jKIUANZQ8fvGiDPzXf/wEy2klSVO2Gt5DPOvKjQa89dcvsO9I6KzTdOBYKf6wbRe5t1vM++VFJ/Rt+ULy237xIrIWzrN4f1g/HeyeHtHqoA+D/PD5p7G4uBzv79iDFuL5bH5e1Zyrv51og33HT+NCdR02rVuKhPvHBuX+bzY0YefeE6ipuwmXs0ekli7L+gLxcbH41oY1WLhgTkhmAxGhwwC8EUUvC4mnpKRMxc7P89hIW0UYn863p1fwlfpbePP9XVieORvrVmR+pfe85/ApHD1dCafTaU4lb6wzxLcl8+dg0/qV1vmCQqw/IPhZgJ/UiLYBfPeZDVhG0sBtH+9DQ3MLm8EbYqpWPjmzivxTlbhwpR4bchYiafL4Qb3fq9ca8OnBEtQ3NpkLStD7EYtH0Iam8WNH46mv52BGSpL3H2zoAw1hP1k0IFeG95KWOg3/+fJWfPRpHg4XV7BJmumUrXziZj7r9k0ivnI/PozlGTOwJjsj4HfZ2tqB46cvsNHIlO4hrRoiLyS1IisdGx9d0QfA82JPF6+ZTtFboZNDLZg3A9u/OISGO618Aid98mVwRsgvvYDyqut4LHs+ZiQH5mnfC1V1+DK/FM2tbdAo5YvZQ+lQcDpGkHr/eMJWm9dlI2V6Yp/aCTS39gG7JbCPmjhl+hT8/IVnsCfvBA4XVaDbxefodzAw8MWgmu+24oM9BVg4cypWLp7FRg8N1OsPFZ1DceUVPjkkvRbdM+rnQIh0qFi9ZAHWrMjCUCwhoAGkNXv74REP5zyA+XNS8OEXh3HtVhMDAe/ng7EvrqzBxav1eHQAbEC9ftfRMtwhXk/nCGYqX1A9fRaAAmDy+LF4ct1yTJgwrn+MJzUUhTcANFMUDYQO6YOiL27ZiL0Hi3Ck5CyfL1AYX5/vv7mljbFBChGHG3Iye2UD6vWfEJF3ubbeXCFMV/qU8qnXR6jIzpzLJoYcCOD1hapDAQQRwba/USX3UBlrVi5CxpxkfLb/JC5du8knambz8YPFZ0Vz4GLtDbz1931YvywD6bOmej1P2bka7D5Wjo7ubonmzRXD6JaSmIDHVmUhYfzA2x40Wf2H+4ohhidornvqKE8YPw5bn16H4tLz+OxAEbqo10LM9s2Wb1PQToz50f5CnCKhYePqLIyK40O8WqjXk88vkRSP34vTnDJWpHqREQ58PScLmekz7vkHW0VgWKeBbqtyBKAuFmakkXg/GR9+mY9LhMZZu4GimEAgry9eqcPv3t9NaHwmmyLmKMkcOulK4B6LR3MAUK/fvG5Zv2cF9Q/44Bs/JDQAF4F8Sta33/0b4uJGImPB3Hs6LZ07YOtTDzM22JN/ij3coS/spM870N7hwr6CcsSIKWI0Y0yguVj06NgYrF02/969XpTS0rP4v/e2Ge0GoaACgzpJlALFWOhJFTN308kbn3zsYTz33JMBudZdkhLuOVTM5hWAPGk0xOrhFAAeS8a4sGj2dKzNXhgQr6flL3/dge2f72HGp83HTtGeQK/V1sofagnGJFFBBQAt8pp/fDkYPsnzjKlT8Iufv4Cx4wLT0XO2sho79h5HS3uH0QZHQWcyAOfnOOL1G1cvxey0qQG57u3GJvzqv9/ChZqrRhahG58WOmNYZ0dH+AKA3YSi8iXgHPoqIApbyyc2JgZbn9mMNWtXBIwNdh8sxMmKiwYDREsMsGhuKtatyAqY1+/dewS5H+xAa3u7YXy9BZGyUWc7nTKu2zg+nABAE+h3yGYk0vo6fopYHJKu56cvErVySRa+8+ymgLFBVfU1IhKPoLH5LqJiolnnDW3GnT5tUsC8/s/vf4yDJwoNb9dpn4LNSdLLdkL7Ut3TwQ1bCQBKwgIAAgT0aYtfku0l+XOHviikQ+gCsRBE4oTx+NHWZzAvfVbA7uHDT/IwenQcHn8kO2DnLC+rJGL2A9TW3zTonq0v2MO9nk4M2dnRLn/ldbK9SozfFDYi0A0ImwQbjJF1gcwA+gJRFBSbH12Db397Y2B+PDFIICeK/GDbp9j+5V7RSWRSvs4C7W1tYgYxVpqE1+8IuyzACwiSBQhyLCEhgq/rp8rrCBNWSE+bgZ++8N17DgmBAgCdA/i3v38PZecvitVEnWZvIanf7q4udJB4L9V1njB+VVimgX6A8IoIC5aQwJeMFRmCAASdy//Hzz+LBx9cFFQAFBwvwZu5fyNCr8NifD3et4u5gaVC6f6VUKlzJdSeVhEC8SOyJcuGokwg6wJ9QSgqEH/y4pagAOAN4vUHC4q44V0uq+eTPZ0EWqpf6u1PBEPoDSkASALxNbJZLKuDwKHrArFkXOL4+/HDLd/ut0AcKAAqKs7j7dxtQuhZjU7rs6uzg1C+Rejlku3lYAm9IQcAvwKRrvIREclDg8NcT5iGhG9ueAQbNjw8qAD45NO9+PDT3Wjr6LCkeC4xNpC26oWa0BuyAJAE4kcebQZi2VeHw1xClgJi3owU/IyEhL4IxP4AgOb2//vWeyglQk9zp3tSh3Tu//Y2j9z+iVAQekMaAH4Folj920wV+TYuPh7PP/fNXgViXwFwnAi9P/1lO27ducMHh7g0k/rJ1hHiQm9YAECAIEeEBEkgqoiIMBnAWG6eGHd9TjZ+8P1n7wkAf3znA3x5IN+geLk510lz+9YW9l4SepTy84ZKnQ4pAEgCkYJgk/QzECGtIUyBwBeHdmBywv3495/+gC1L3x8AVFddxf+88SfU3mzwMD5lgK4uD6G3Qxi/aSjV55ADgASELSJTsApEh7y8PN9GREfjqcfXY9Om9X0CwM6P9+Dvn+xiQk8fGCLn9h3WTpwmofBzh2I9DlkASG0GHp1KagRfVp63FYh1gsnrxRlz8eN/fc4QiO4AaLx1G2//8W8oKC2XHvzQDAbocToZ5YdCJ44NACsQKBO85C4QVbFkLB9zwIEwbvQovPzC82zUkQwAOlrn9Tdz0UiEnjEaWHg+HS1A++xpR45UXieGf3mo192wAIAkED+SQ4IxzkBiAkWs5rF+1dfwwgtbGAD+8Paf8UXeEeP5BP2RLz48DGhruSsLvSaR3uUNh3obNgDwJxBVNuxMlQDAV/JImzoFMTEjcOFqLTtSHghKC13mjbblD3WhFzYAkIDwkhCIFsGnSkvJa2IASnRMjNtUrhrTEe6jdYTQe3241dWwBIAvgahnCvpKI7QVMYoCAHxyR/6AiitkRut8FUXFMC3CYA+Bj7gxipHLa2ZOr89M0dPdReK9ReXT7z40XI0/rBnAjQ08OpV4phCB6BExjPJpo85Q6sSxATAwgUizhBwzHNDnAqJZw45U8oTKbwqHegkbAPgTiMNd6NkA8C4Q5VFHVQjB0To2AAY/JLwmeX5TONZD2ALALsM8DbSLDQC72ACwiw0Au9gAsIvvEtLrBoo+fl+lyVfeLvL8Mb0dK1JBubOoytswbrf78HWuZKldodd7tAHg3/CvkN3P3IzoXvLAO3vk770kvpfs5Zw0z/8N+EgePeenxt8vHfYq2V7xcq39vq4r+hle83ZNP9e1Q4Af49PK/mUvxvf2vXf8GUKcj553v/D8QNzrFrg9x+jvurYG6L1CqQfnDOB71Au39PFw6vUvBeBeqdHf6cdXMgWz2SHAT/me23vaMdPs49gqP9/bIWhXLzmwPlX0PR9U35+yxUtIotdsGuTrDmsAyIIsV8RkXxPyNrnRrF5KSKx9wt04xPumSUZLDsC9rnJ7760LeTCuGzZZQLUXkeZXBHoBhvv5ejt2gY+MotfiR+BV21lAiBaaogmVrjPIJvK+2A0Y7gDYabcDDK/yKqwDRPx5fJMITcOmhHJLIKVjfWCnt+3lALHA6300KjX+Q76o3k9qucBmgL6XEskDN4lKr/aTBZQECARbiQHfFSo9x02s5Qnaz3Uz/gG3lJW2L7iHh2mwPKRiyVxsAHgpv3HLrf3l9nmBpGPxqFee+0QU5POHfHwl1y3Fy+wlfNDyrh0C/Bshd6jEWNFnsLUfX8kNxVlD1BCs2K1CmFUNARDkCj2S5+cw+jteFb8r5Io9JjDMiz0ewAaAXWwA2MUGgF3Cs/y/AAMA3Vr2WgCSYnYAAAAASUVORK5CYII=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="assets/favicon.ico"
Content-Type: application/octet-stream; x-encoding=base64
AAABAAEAEBAAAAEAIABoBAAAFgAAACgAAAAQAAAAIAAAAAEAIAAAAAAAQAQAABMLAAATCwAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA2sm9/1MyGf/ayb3/AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA2sm9/1MyGf+SYj7/UzIZ/9rJvf8AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA2sm9/1MyGf+WZ0T/ropv/864qf9TMhn/2sm9/wAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA2sm9/1MyGf+WZ0T/ropv///////OuKn/zrip/1MyGf/ayb3/AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA2sm9/1MyGf+WZ0T/ropv///////Eqpf//////864qf/OuKn/UzIZ/9rJvf8AAAAAAAAAAAAAAAAAAAAAAAAAAFMyGf+WZ0T/ropv///////Eqpf/ropv/864qf//////zrip/864qf9TMhn/AAAAAAAAAAAAAAAAAAAAAAAAAABTMhn/lmdE///////Eqpf/ropv///////OuKn/zrip///////OuKn/UzIZ/wAAAAAAAAAAAAAAAAAAAAAAAAAAUzIZ/5ZnRP/Eqpf/ropv///////Eqpf//////864qf/OuKn/zrip/1MyGf8AAAAAAAAAAAAAAAAAAAAAAAAAAFMyGf+WZ0T/ropv///////Eqpf/ropv/864qf//////zrip/864qf9TMhn/AAAAAAAAAAAAAAAAAAAAAAAAAABTMhn/lmdE///////Eqpf/ropv///////OuKn/zrip/864qf/OuKn/UzIZ/wAAAAAAAAAAAAAAAAAAAAAAAAAAUzIZ/5ZnRP/Eqpf/ropv//////+uim///////864qf/OuKn/zrip/1MyGf8AAAAAAAAAAAAAAAAAAAAAAAAAAFMyGf+WZ0T/ropv//////+uim//kmI+/864qf//////zrip/864qf9TMhn/AAAAAAAAAAAAAAAAAAAAAAAAAABTMhn/lmdE//////+uim//kmI+/5JiPv/OuKn/zrip///////OuKn/UzIZ/wAAAAAAAAAAAAAAAAAAAAAAAAAAUzIZ/5NlQf+uim//lmdE/5ZnRP+WZ0T/zrip/864qf/OuKn/zrip/1MyGf8AAAAAAAAAAAAAAAAAAAAAAAAAANrJvf9TMhn/UzIZ/5hrSP+WZ0T/lmdE/864qf/OuKn/UzIZ/1MyGf/Eqpf/AAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAANrJvf9TMhn/UzIZ/1MyGf9TMhn/UzIZ/8Sql/8AAAAAAAAAAAAAAAAAAAAA/j8AAPwfAAD4DwAA8AcAAOADAADgAwAA4AMAAOADAADgAwAA4AMAAOADAADgAwAA4AMAAOADAADgAwAA+A8AAA==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="assets/resources/en.json"
Content-Type: application/octet-stream; x-encoding=base64
ewoJIkVYQUNUIjogIkVYQUNUIiwKCSJVUCI6ICJVUCIsCgkiRE9XTiI6ICJET1dOIgp9
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="assets/resources/fr.json"
Content-Type: application/octet-stream; x-encoding=base64
ewoJIkVYQUNUIjogIkV4YWN0IiwKCSJVUCI6ICJIYXV0IiwKCSJET1dOIjogIkJhcyIKfQ==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="debug.html"
Content-Type: application/octet-stream; x-encoding=base64
PCFET0NUWVBFIGh0bWw+CjxodG1sPgoJPGhlYWQ+CgkJPG1ldGEgaHR0cC1lcXVpdj0iWC1VQS1Db21wYXRpYmxlIiBjb250ZW50PSJJRT1lZGdlLGNocm9tZT0xIj4KCQk8dGl0bGU+VGlwQ2FsYzwvdGl0bGU+CgkJPGxpbmsgcmVsPSJzaG9ydGN1dCBpY29uIiBocmVmPSJhc3NldHMvZmF2aWNvbi5pY28iLz4KCQk8IS0tIC0tPgoJCTxtZXRhIGh0dHAtZXF1aXY9IkNvbnRlbnQtVHlwZSIgY29udGVudD0idGV4dC9odG1sOyBjaGFyc2V0PXV0ZjgiLz4KCQk8bWV0YSBuYW1lPSJhcHBsZS1tb2JpbGUtd2ViLWFwcC1jYXBhYmxlIiBjb250ZW50PSJ5ZXMiLz4KCQk8bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWF4aW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iLz4KCQk8IS0tIExlc3MuanMgKHVuY29tbWVudCBmb3IgY2xpZW50LXNpZGUgcmVuZGVyaW5nIG9mIGxlc3Mgc3R5bGVzaGVldHM7IGxlYXZlIGNvbW1lbnRlZCB0byB1c2Ugb25seSBDU1MpIC0tPgoJCTwhLS0gPHNjcmlwdCBzcmM9ImVueW8vdG9vbHMvbWluaWZpZXIvbm9kZV9tb2R1bGVzL2xlc3MvZGlzdC9sZXNzLTEuMy4wZS5taW4uanMiPjwvc2NyaXB0PiAtLT4KCQk8IS0tIGVueW8gKGRlYnVnKSAtLT4KCQk8c2NyaXB0IHNyYz0iZW55by9lbnlvLmpzIj48L3NjcmlwdD4KCQk8IS0tIGFwcGxpY2F0aW9uIChkZWJ1ZykgLS0+CgkJPHNjcmlwdCBzcmM9InNvdXJjZS9wYWNrYWdlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPC9oZWFkPgoJPGJvZHkgY2xhc3M9ImVueW8tdW5zZWxlY3RhYmxlIj4KCQk8c2NyaXB0PgoJCQluZXcgRXguVGlwQ2FsYygpLnJlbmRlckludG8oZG9jdW1lbnQuYm9keSk7CiAgICAgICAgICAgICAgICAgICAgICAgIGlmICh3aW5kb3cuUGFsbVN5c3RlbSkgewogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHdpbmRvdy5QYWxtU3lzdGVtLnN0YWdlUmVhZHkoKTsKICAgICAgICAgICAgICAgICAgICAgICAgfQoJCTwvc2NyaXB0PgoJPC9ib2R5Pgo8L2h0bWw+Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="deploy.json"
Content-Type: application/octet-stream; x-encoding=base64
ewoJImVueW8iOiAiZW55byIsCgkic291cmNlIjogIi4iLAoJImFzc2V0cyI6IFsiaWNvbi5wbmciLCAiaW5kZXguaHRtbCIsICJhc3NldHMiXSwKCSJsaWJzIjogWyJsaWIvb255eCIsICJsaWIvbGF5b3V0Il0KfQ==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="deploy.md"
Content-Type: application/octet-stream; x-encoding=base64
IyBNaW5pZmljYXRpb24gYW5kIERlcGxveW1lbnQKIyMgVW5kZXJzdGFuZGluZyBhbmQgVXNpbmcgTWluaWZpY2F0aW9uCgpFbnlvIGNvbWVzIHdpdGggYSBtaW5pZmljYXRpb24gdG9vbCBiYXNlZCBvbiBVZ2xpZnlKUywgcnVuIGJ5IE5vZGUuanMuCgpUaGlzIHRvb2wgY2FuIGJlIHVzZWQgdG8gY29tcHJlc3MgdGhlIGZyYW1ld29yaywgb3RoZXIgbGlicmFyaWVzLCBhbmQgYXBwbGljYXRpb25zLCBhbmQgd2lsbCBrZWVwIGxvYWQgb3JkZXIgaW50YWN0IGFzIHdlbGwgYXMgY29ycmVjdCB1cmwgcGF0aHMgaW4gY3NzLgoKIyMjIFdoeSBjb21wcmVzcz8KCkNvbXByZXNzaW5nIGVueW8gYXBwcyBncmVhdGx5IHJlZHVjZXMgbG9hZCB0aW1lcyBvZiBhcHBsaWNhdGlvbnMsIGFzIHdlbGwgYXMgcmVkdWNpbmcgb3ZlcmFsbCBjb2RlIHNpemUuCgpUaGlzIHdheSwgeW91IGNhbiBiZSB2ZXJ5IHZlcmJvc2UgaW4gdGhlIGRvY3VtZW50YXRpb24gb2YgeW91ciBzb3VyY2UgY29kZSwgd2l0aG91dCB0aGF0IGltcGFjdGluZyB0aGUgcGVyZm9ybWFuY2Ugb2YgeW91ciBhcHBsaWNhdGlvbiBpbiBwcm9kdWN0aW9uLgoKIyMjIFdoYXQgaXMgY29tcHJlc3NlZAoKRm9yIGVueW8sIHRoZSBsaWJyYXJpZXMsIGFuZCB5b3VyIGNvZGU6ICoqZXh0ZXJuYWwgYXNzZXRzIHN1Y2ggYXMgaW1hZ2VzIHdpbGwgbm90IGJlIGNvcGllZCBvciBtb3ZlZCoqLgoKSW5zdGVhZCwgdGhlIENTUyB1cmwgcGF0aHMgYXJlIGZpeGVkIHVwIHRvIHJlZmVyZW5jZSB0aGUgbmV3IHBhdGggZnJvbSB0aGUgYnVpbGQgbG9jYXRpb24uCgojIyMgSG93IHRvIGNvbXByZXNzCgpUbyBjb21wcmVzcyB5b3VyIGFwcGxpY2F0aW9uLCB5b3UgbXVzdCBydW4gdGhlIE5vZGUuanMgc2NyaXB0IG5hbWVkIGBkZXBsb3kuanNgIHRoYXQgY29tZXMgd2l0aCBFbnlvLCBhcyBgZW55by90b29scy9kZXBsb3kuanNgLgoKICAgICQgbm9kZSBlbnlvL3Rvb2xzL2RlcGxveS5qcyAtaAoKVGhpcyBzY3JpcHQgd2lsbCBydW4gdGhlIG1pbmlmaWNhdGlvbiB0b29sIGxvY2F0ZWQgaW4gYGVueW8vdG9vbHMvbWluaWZpZXIvbWluaWZ5LmpzYCwgYW5kIG1ha2UgYSBidWlsZCBvZiBlbnlvLCB0aGVuIGEgYnVpbGQgb2YgeW91ciBhcHAuCgpBbnkgbGlicmFyaWVzIHJlZmVyZW5jZWQgaW4geW91ciBgcGFja2FnZS5qc2AgbWFuaWZlc3Qgd2lsbCBiZSBidWlsdCBpbnRvIHlvdXIgYXBwJ3MgYnVpbHQgY29kZS4KCioqTk9URToqKiBgZGVwbG95LmpzYCBleHBlY3RzIHRvIGZpbmQgYSBgcGFja2FnZS5qc2AgaW4gdGhlIHJvb3QtZm9sZGVyIG9mIHlvdXIgYXBwbGljYXRpb24uIEl0IG9ubHkgcmVmZXJlbmNlcyB5b3VyIGFwcCdzIGBwYWNrYWdlLmpzYCB0byBrZWVwIHBhdGhzIGNvcnJlY3QuIERvIG5vdCBtb2RpZnkgdGhpcy4KCiMjIyBXaGF0IGNvbWVzIG91dD8KCkFmdGVyIHJ1bm5pbmcgdGhlIGBkZXBsb3kuanNgIHNjcmlwdCwgYSBuZXcgZm9sZGVyIGBidWlsZGAgKHlvdSBjaGFuZ2UgdGhpcyB1c2luZyB0aGUgYC1iYCBmbGFnIG9mIHRoZSBgZGVwbG95LmpzYCBzY3JpcHQpIHdpbGwgYmUgbG9jYXRlZCBuZXh0IHRvIHlvdXIgYHNvdXJjZWAgZGlyZWN0b3J5LgoKSW4gaXQgd2lsbCBiZSA0IGZpbGVzOgotIGVueW8uY3NzCi0gZW55by5qcwotIGFwcC5jc3MKLSBhcHAuanMKClRoZXNlIGZpbGVzIHdpbGwgYmUgbG9hZGVkIGluIHRoZSBnaXZlbiBvcmRlciBieSBgaW5kZXguaHRtbGAuCgpUaGUgb3V0cHV0IG9mIHRoZSBgZGVwbG95YCBzY3JpcHRzIHdpbGwgbWluaWZ5IHlvdXIgYXBwbGlhY3Rpb24sIGFuZCBjb3B5IHRoZSBuZWNlc3NhcnkgZmlsZXMgaW50byBgZGVwbG95LzxhcHBuYW1lPi9gLgoKSWYgdGhlIGxpYnJhcmllcyBoYXZlIGEgY29tcGF0aWJsZSBgZGVwbG95LnNoYCBvciBgZGVwbG95LmJhdGAgc2NyaXB0LCB0aGV5IHdpbGwgYmUgZXhlY3V0ZWQsIGFuZCBhIG1pbmltYWwgY29weSB3aWxsIGJlIHBsYWNlZCBpbiB0aGUgZGVwbG95bWVudCdzIGxpYiBmb2xkZXIuCgpJZiBubyBgZGVwbG95LihzaHxiYXQpYCBzY3JpcHQgaXMgZm91bmQgZm9yIHRoZSBsaWJyYXJ5LCBhbGwgb2YgdGhlIGxpYnJhcnkgaXMgY29waWVkIGludG8gdGhlIGxpYiBmb2xkZXIgdG8gcHJvdmlkZSBtYXhpbXVtIHNhZmV0eS4KCklmIHlvdSBhcmUgYWRkaW5nIGEgbGlicmFyeSwgcGxlYXNlIGFkZCBhIGBkZXBsb3kuc2hgIGZpbGUgYW5kIGBkZXBsb3kuYmF0YCBmaWxlIHNpbWlsYXIgdG8gdGhlIG9uZXMgaW4gYGxpYi9vbnl4YC4KCklmIG5vIGltYWdlcyBvciBmaWxlcyBhcmUgbmVlZGVkIGZyb20gdGhlIGxpYnJhcnksIGp1c3QgaW5jbHVkZSBibGFuayAoYW5kIGV4ZWN1dGFibGUpIGNvcGllcyBvZiB0aGUgZGVwbG95IHNjcmlwdHMuCg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/CONTRIBUTING.md"
Content-Type: application/octet-stream; x-encoding=base64
IyBDb250cmlidXRpb25zCgpDb250cmlidXRpb25zIGFyZSB3ZWxjb21lIGZvciBFbnlvIGFuZCBpdHMgYXNzb2NpYXRlZCBsaWJyYXJpZXMgaW5jbHVkaW5nIG9ueXggYW5kIGxheW91dC4KClBsZWFzZSBzZWUgW0NvbnRyaWJ1dGluZyB0byBFbnlvXShodHRwOi8vZW55b2pzLmNvbS9jb21tdW5pdHkvY29udHJpYnV0ZS8pIGZvciBkZXRhaWxzCm9uIG91ciBjb250cmlidXRpb24gcG9saWN5IGFuZCBndWlkZWxpbmVzIGZvciB1c2Ugb2YgdGhlIEVueW8tRENPLTEuMS1TaWduZWQtb2ZmLWJ5CmxpbmUgaW4geW91ciBjb21taXRzIGFuZCBwdWxsIHJlcXVlc3RzLgoKV2UgaGF2ZSBhIGJyaWVmCltFbnlvIEpTIHN0eWxlIGd1aWRlXShodHRwczovL2dpdGh1Yi5jb20vZW55b2pzL2VueW8vd2lraS9TdHlsZS1HdWlkZSkKZm9yIGNvZGUgZ29pbmcgaW50byB0aGUgcmVwby4gV2UndmUgcmVjZW50bHkgYWRkZWQgLmpzaGludHJjCmZpbGVzIHRvIHRoZSByb290IG9mIHRoZSB2YXJpb3VzIGNvZGUgYmFzZXMsIGFuZCBhc2sgdGhhdCB5b3UgbWFrZSB5b3VyIGNvZGUKcmVhc29uYWJsZSBjbGVhbiB0byB0aGUgW0pTIEhpbnRdKGh0dHA6Ly9qc2hpbnQuY29tLykgdG9vbCBiZWZvcmUgc3VibWl0dGluZy4KV2Uga25vdyB0aGF0IHRoZXJlIGFyZSBpc3N1ZXMgc29tZXRpbWVzIHdpdGggaG93IGpzaGludCBoYW5kbGVzIGluZGVudGF0aW9uCmxldmVscyB3aXRoIEpTIGRhdGEgc3RydWN0dXJlcyB0aGF0IGFmZmVjdCBraW5kIGRlZmluaXRpb25zLCBzbyB3ZSBnaXZlIGEKbGl0dGxlIGxlZXdheS4KCklmIHlvdSdyZSBpbnRlcmVzdGVkIGluIGludHJvZHVjaW5nIG5ldyBraW5kcywgeW91IG1pZ2h0IGFsc28gY29uc2lkZXIgaG9zdGluZwp5b3VyIG93biByZXBvIGFuZCBjb250cmlidXRpbmcgdG8gdGhlIFtFbnlvIGNvbW11bml0eSBnYWxsZXJ5XShodHRwOi8vZW55b2pzLmNvbS9nYWxsZXJ5KS4K
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/LICENSE-2.0.txt"
Content-Type: application/octet-stream; x-encoding=base64

                                 Apache License
                           Version 2.0, January 2004
                        http://www.apache.org/licenses/

   TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION

   1. Definitions.

      "License" shall mean the terms and conditions for use, reproduction,
      and distribution as defined by Sections 1 through 9 of this document.

      "Licensor" shall mean the copyright owner or entity authorized by
      the copyright owner that is granting the License.

      "Legal Entity" shall mean the union of the acting entity and all
      other entities that control, are controlled by, or are under common
      control with that entity. For the purposes of this definition,
      "control" means (i) the power, direct or indirect, to cause the
      direction or management of such entity, whether by contract or
      otherwise, or (ii) ownership of fifty percent (50%) or more of the
      outstanding shares, or (iii) beneficial ownership of such entity.

      "You" (or "Your") shall mean an individual or Legal Entity
      exercising permissions granted by this License.

      "Source" form shall mean the preferred form for making modifications,
      including but not limited to software source code, documentation
      source, and configuration files.

      "Object" form shall mean any form resulting from mechanical
      transformation or translation of a Source form, including but
      not limited to compiled object code, generated documentation,
      and conversions to other media types.

      "Work" shall mean the work of authorship, whether in Source or
      Object form, made available under the License, as indicated by a
      copyright notice that is included in or attached to the work
      (an example is provided in the Appendix below).

      "Derivative Works" shall mean any work, whether in Source or Object
      form, that is based on (or derived from) the Work and for which the
      editorial revisions, annotations, elaborations, or other modifications
      represent, as a whole, an original work of authorship. For the purposes
      of this License, Derivative Works shall not include works that remain
      separable from, or merely link (or bind by name) to the interfaces of,
      the Work and Derivative Works thereof.

      "Contribution" shall mean any work of authorship, including
      the original version of the Work and any modifications or additions
      to that Work or Derivative Works thereof, that is intentionally
      submitted to Licensor for inclusion in the Work by the copyright owner
      or by an individual or Legal Entity authorized to submit on behalf of
      the copyright owner. For the purposes of this definition, "submitted"
      means any form of electronic, verbal, or written communication sent
      to the Licensor or its representatives, including but not limited to
      communication on electronic mailing lists, source code control systems,
      and issue tracking systems that are managed by, or on behalf of, the
      Licensor for the purpose of discussing and improving the Work, but
      excluding communication that is conspicuously marked or otherwise
      designated in writing by the copyright owner as "Not a Contribution."

      "Contributor" shall mean Licensor and any individual or Legal Entity
      on behalf of whom a Contribution has been received by Licensor and
      subsequently incorporated within the Work.

   2. Grant of Copyright License. Subject to the terms and conditions of
      this License, each Contributor hereby grants to You a perpetual,
      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
      copyright license to reproduce, prepare Derivative Works of,
      publicly display, publicly perform, sublicense, and distribute the
      Work and such Derivative Works in Source or Object form.

   3. Grant of Patent License. Subject to the terms and conditions of
      this License, each Contributor hereby grants to You a perpetual,
      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
      (except as stated in this section) patent license to make, have made,
      use, offer to sell, sell, import, and otherwise transfer the Work,
      where such license applies only to those patent claims licensable
      by such Contributor that are necessarily infringed by their
      Contribution(s) alone or by combination of their Contribution(s)
      with the Work to which such Contribution(s) was submitted. If You
      institute patent litigation against any entity (including a
      cross-claim or counterclaim in a lawsuit) alleging that the Work
      or a Contribution incorporated within the Work constitutes direct
      or contributory patent infringement, then any patent licenses
      granted to You under this License for that Work shall terminate
      as of the date such litigation is filed.

   4. Redistribution. You may reproduce and distribute copies of the
      Work or Derivative Works thereof in any medium, with or without
      modifications, and in Source or Object form, provided that You
      meet the following conditions:

      (a) You must give any other recipients of the Work or
          Derivative Works a copy of this License; and

      (b) You must cause any modified files to carry prominent notices
          stating that You changed the files; and

      (c) You must retain, in the Source form of any Derivative Works
          that You distribute, all copyright, patent, trademark, and
          attribution notices from the Source form of the Work,
          excluding those notices that do not pertain to any part of
          the Derivative Works; and

      (d) If the Work includes a "NOTICE" text file as part of its
          distribution, then any Derivative Works that You distribute must
          include a readable copy of the attribution notices contained
          within such NOTICE file, excluding those notices that do not
          pertain to any part of the Derivative Works, in at least one
          of the following places: within a NOTICE text file distributed
          as part of the Derivative Works; within the Source form or
          documentation, if provided along with the Derivative Works; or,
          within a display generated by the Derivative Works, if and
          wherever such third-party notices normally appear. The contents
          of the NOTICE file are for informational purposes only and
          do not modify the License. You may add Your own attribution
          notices within Derivative Works that You distribute, alongside
          or as an addendum to the NOTICE text from the Work, provided
          that such additional attribution notices cannot be construed
          as modifying the License.

      You may add Your own copyright statement to Your modifications and
      may provide additional or different license terms and conditions
      for use, reproduction, or distribution of Your modifications, or
      for any such Derivative Works as a whole, provided Your use,
      reproduction, and distribution of the Work otherwise complies with
      the conditions stated in this License.

   5. Submission of Contributions. Unless You explicitly state otherwise,
      any Contribution intentionally submitted for inclusion in the Work
      by You to the Licensor shall be under the terms and conditions of
      this License, without any additional terms or conditions.
      Notwithstanding the above, nothing herein shall supersede or modify
      the terms of any separate license agreement you may have executed
      with Licensor regarding such Contributions.

   6. Trademarks. This License does not grant permission to use the trade
      names, trademarks, service marks, or product names of the Licensor,
      except as required for reasonable and customary use in describing the
      origin of the Work and reproducing the content of the NOTICE file.

   7. Disclaimer of Warranty. Unless required by applicable law or
      agreed to in writing, Licensor provides the Work (and each
      Contributor provides its Contributions) on an "AS IS" BASIS,
      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
      implied, including, without limitation, any warranties or conditions
      of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
      PARTICULAR PURPOSE. You are solely responsible for determining the
      appropriateness of using or redistributing the Work and assume any
      risks associated with Your exercise of permissions under this License.

   8. Limitation of Liability. In no event and under no legal theory,
      whether in tort (including negligence), contract, or otherwise,
      unless required by applicable law (such as deliberate and grossly
      negligent acts) or agreed to in writing, shall any Contributor be
      liable to You for damages, including any direct, indirect, special,
      incidental, or consequential damages of any character arising as a
      result of this License or out of the use or inability to use the
      Work (including but not limited to damages for loss of goodwill,
      work stoppage, computer failure or malfunction, or any and all
      other commercial damages or losses), even if such Contributor
      has been advised of the possibility of such damages.

   9. Accepting Warranty or Additional Liability. While redistributing
      the Work or Derivative Works thereof, You may choose to offer,
      and charge a fee for, acceptance of support, warranty, indemnity,
      or other liability obligations and/or rights consistent with this
      License. However, in accepting such obligations, You may act only
      on Your own behalf and on Your sole responsibility, not on behalf
      of any other Contributor, and only if You agree to indemnify,
      defend, and hold each Contributor harmless for any liability
      incurred by, or claims asserted against, such Contributor by reason
      of your accepting any such warranty or additional liability.

   END OF TERMS AND CONDITIONS

   APPENDIX: How to apply the Apache License to your work.

      To apply the Apache License to your work, attach the following
      boilerplate notice, with the fields enclosed by brackets "[]"
      replaced with your own identifying information. (Don't include
      the brackets!)  The text should be enclosed in the appropriate
      comment syntax for the file format. We also recommend that a
      file or class name and description of purpose be included on the
      same "printed page" as the copyright notice for easier
      identification within third-party archives.

   Copyright [yyyy] [name of copyright owner]

   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at

       http://www.apache.org/licenses/LICENSE-2.0

   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/README.md"
Content-Type: application/octet-stream; x-encoding=base64
### Looking for the issue tracker?
It's moved to [https://enyojs.atlassian.net](https://enyojs.atlassian.net).

---

# Quick Info

## Core

This repository contains Enyo core. We've pared it down to the essentials, so folks can work at the metal. Widget libraries, g11n code, and other fancy bits are optional packages.

## Lib

Packages should go in a folder named _lib_ (e.g. the _extra_ or _canvas_ repositories on GitHub). _lib_ is a magic name that enyo uses to work with add-on packages. It's recommended you create a _lib_ folder as sibling to _enyo_ and keep your packages there, but you can make as many _lib_ folders as you like and put them anywhere.

## Warning about file://

_Note_: in Chrome, various samples will not work from file:// because of Chrome's security policy. Run from a local http server, use the --allow-file-access-from-files in Chrome, or use the online versions at http://enyojs.com.

# What Is Enyo

Enyo is an object-oriented JavaScript application framework emphasizing modularity and encapsulation. Enyo is suitable for small and large-scale applications.

Enyo up to 1.x was the underlying framework used to develop applications for HP's TouchPad tablet. Enyo as shipped on the TouchPad included an complete set of user interface components and service wrappers. What you will find here is Enyo 2, what we informally call _core_: the primary infrastructure needed to support any number of Enyo-based libraries. Not to worry, Enyo 1.x itself is open-source licensed, and work is progressing on packaging up those controls and goodies to work with Enyo 2.

Enyo was designed from the beginning to be highly extensible. This repository reflects a small working set of code, that can be expanded with any number of libraries or plugins.

Enyo 2 is lightweight, easy to digest, and powerful.

# What Do I Get

The core code includes the Enyo kernel, the DOM extensions, some Ajax (XHR) tools, and basic wrapper kinds for a lot of DOM form elements. These things are actually separable (it's easy to make micro-builds of Enyo), but we believe this is a useful working set.

The Enyo 2 kernel provides a modularity concept (Component) and a view concept (UiComponent). The DOM aspect includes a widget concept (Control) and an extensible event system (Dispatcher). The Ajax package includes basic xhr functionality and an implementation of xhr as a Component (Ajax).  The touch package provides platform-optimized scrollers, while the UI package provides base kinds for common controls like buttons and popups.

Just these pieces are sufficient to create large applications using the Enyo encapsulation model. Developers that want only this low-level code are encouraged to roll-their-own. For those that want a richer set of tools, there are some pre-built libraries already available, and much more on the way.

# Why Do I Care

First is our emphasis on cross-platform: Enyo core works on both desktop and mobile browsers.

Second is Enyo's building block approach to applications. Each piece of an application is a Component, and Components are constructed out of other Components.

For example, it's easy to define a combination of an `<input>` tag with a `<label>` tag into one _LabeledInput_ Component.

Now I can use (and re-use) LabeledInput as one atomic piece.

But that's just the beginning. Ultimately, large pieces of functionality can be exposed as single Components, for example a fancy report generator, or a color picker, or an entire painting application.

Use the Enyo encapsulation model to divide and conquer large projects. No particular piece of an application need be especially complex. Because combining pieces is central, it's natural to factor complex sections into smaller pieces. And because of the modularity, all these pieces tend to be reusable, in the same project, in other projects, or even for the public at large.

This is all part of our strategy to allow developers to focus on creativity and avoid Repeating Themselves.

# That's a Lot of Talk

The core Enyo design was proven out by the complex applications HP developed for the TouchPad platform. We don't claim this was particularly easy, there were a lot of hardworking developers on the apps teams, but we are confident the Enyo principles are effective on a large scale.

In any case, roll your sleeves up and try it for yourself.

# Give me the Basics

Here is an Enyo Hello World:

```html
<!doctype html>
<html>
<head>
	<title>Enyo</title>
	<script src="enyojs/2.2.0/enyo.js" type="text/javascript"></script>
	<link href="enyojs/2.2.0/enyo.css" rel="stylesheet" type="text/css" />
</head>
<body>
	<script type="text/javascript">
		new enyo.Control({content: "Hello From Enyo"}).renderInto(document.body);
	</script>
</body>
</html>
```

This example loads an enyo.js build from _enyojs/2.2.0/_. If you downloaded the SDK you have a versioned build file. If you pulled from GitHub, you can either make your own build using a minify script in _enyo/source/minify_ (requires Node), or you can load directly from the source (_enyo/source/enyo.js_). Loading from source is also called 'debug loading' because the modules are loaded as individual files, which is easier for debugging, but much less efficient.

The base enyo.Control works much like an HTML tag. You can assign _classes_ and _attributes_ and give it a _style_. E.g.

```javascript
new enyo.Control({content: "Hello From Enyo", classes: "foo", style: "color: red",
      attributes: {tabIndex: 0}}).renderInto(document.body);
```

produces

```html
<div class="foo" style="color: red;" tabIndex="0">Hello From Enyo</div>
```

Now, the good parts start when you combine more than one Control, e.g.

```javascript
new enyo.Control({
	components: [
		{content: "Hello From Enyo"},
		{tag: "hr"}
	]
}).renderInto(document.body);
```

This Control now encapsulates two Controls into one scope (we can encapsulate any type of Component, that's why the property is called _components_. Controls are one kind of Component.) The outer Control is responsible for the encapsulated components: it manages their lifecycle, listens to their messages, and maintains references to them. For example:

```javascript
new enyo.Control({
	components: [
		{name: "hello", content: "Hello From Enyo", ontap: "helloTap"},
		{tag: "hr"}
	],
	helloTap: function() {
		this.$.hello.addStyles("color: red");
	}
}).renderInto(document.body);
```

Here we've given one of the components a name ('hello') and told it to send a 'helloTap' message when it's tapped (tap is basically the same as the DOM click event, but it works in both mouse and touch environments). The '$' property is a hash that references all the sub-components (we don't store these references directly on _this_ to avoid name conflicts). Notice there is no add/remove machinery to listen to this event, that's all taken care of.

The main point is that 'hello' and the 'hr', their references and behavior, are completely contained inside the outer control. Now, to re-use this, we need to make it a prototype.

Enyo contains a constructor/prototype-generator that we call enyo.kind. Constructors that enyo.kind produces are called _kinds_. Kinds are not magic, they are just regular old JavaScript constructors. Using the enyo.kind factory however allows us to remove boilerplate from our prototype generation (DRY) and have compact syntax. We can convert the Control above to a kind like so:

```javascript
enyo.kind({
	name: "Hello",
	kind: enyo.Control,
	components: [
		{name: "hello", content: "Hello From Enyo", ontap: "helloTap"},
		{tag: "hr"}
	],
	helloTap: function() {
		this.$.hello.addStyles("color: red");
	}
});
// make two, they're small
new enyo.Control({
      components: [ {kind: "Hello"}, {kind: "Hello"} ]
  }).renderInto(document.body);
```

The code above creates a new kind called "Hello" derived from enyo.Control. It contains some components and some behavior. I can create as many "Hello" objects as I want, each instance is independent, and the user of a "Hello" doesn't need to know anything about its internals.

This ability to define encapsulated objects and behavior (Components) and to re-use those encapsulations as prototypes (kinds) is money.

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/enyo.js"
Content-Type: application/octet-stream; x-encoding=base64
LyogZ2xvYmFsIGVueW86dHJ1ZSAqLwooZnVuY3Rpb24oKSB7CgkvLyBlbnlvIGNhbiB1c2UgaW5mb3JtYXRpb24gZnJvbSB0aGUgc2NyaXB0IHRhZyB0aGF0IGxvYWRzIHRoaXMgYm9vdHN0cmFwIGZpbGUKCXZhciB0aGlzU2NyaXB0ID0gImVueW8uanMiOwoKCWVueW8gPSB3aW5kb3cuZW55byB8fCB7fTsKCgllbnlvLmxvY2F0ZVNjcmlwdCA9IGZ1bmN0aW9uKGluTmFtZSkgewoJCXZhciBzY3JpcHRzID0gZG9jdW1lbnQuZ2V0RWxlbWVudHNCeVRhZ05hbWUoInNjcmlwdCIpOwoJCWZvciAodmFyIGk9c2NyaXB0cy5sZW5ndGgtMSwgcywgc3JjLCBsPWluTmFtZS5sZW5ndGg7IChpPj0wKSAmJiAocz1zY3JpcHRzW2ldKTsgaS0tKSB7CgkJCWlmICghcy5sb2NhdGVkKSB7CgkJCQlzcmMgPSBzLmdldEF0dHJpYnV0ZSgic3JjIikgfHwgIiI7CgkJCQlpZiAoc3JjLnNsaWNlKC1sKSA9PSBpbk5hbWUpIHsKCQkJCQlzLmxvY2F0ZWQgPSB0cnVlOwoJCQkJCXJldHVybiB7cGF0aDogc3JjLnNsaWNlKDAsIE1hdGgubWF4KDAsIHNyYy5sYXN0SW5kZXhPZigiLyIpKSksIG5vZGU6IHN9OwoJCQkJfQoJCQl9CgkJfQoJfTsKCgllbnlvLmFyZ3MgPSBlbnlvLmFyZ3MgfHwge307CgoJdmFyIHRhZyA9IGVueW8ubG9jYXRlU2NyaXB0KHRoaXNTY3JpcHQpOwoJaWYgKHRhZykgewoJCS8vIGluZmVyIHRoZSBmcmFtZXdvcmsgcGF0aCBmcm9tIHRoZSBkb2N1bWVudCwgdW5sZXNzIHRoZSB1c2VyIGhhcyBzcGVjaWZpZWQgb25lIGV4cGxpY2l0bHkKCQllbnlvLmFyZ3Mucm9vdCA9IChlbnlvLmFyZ3Mucm9vdCB8fCB0YWcucGF0aCk7CgkJLy8gYWxsIGF0dHJpYnV0ZXMgb2YgdGhlIGJvb3RzdHJhcCBzY3JpcHQgdGFnIGJlY29tZSBlbnlvLmFyZ3MKCQlmb3IgKHZhciBpPTAsIGFsID0gdGFnLm5vZGUuYXR0cmlidXRlcy5sZW5ndGgsIGl0OyAoaSA8IGFsKSAmJiAoaXQgPSB0YWcubm9kZS5hdHRyaWJ1dGVzLml0ZW0oaSkpOyBpKyspIHsKCQkJZW55by5hcmdzW2l0Lm5vZGVOYW1lXSA9IGl0LnZhbHVlOwoJCX0KCX0KCgl2YXIgcm9vdCA9IGVueW8uYXJncy5yb290OwoKCXZhciBzY3JpcHQgPSBmdW5jdGlvbihpblNyYykgewoJCS8qIGpzaGludCBldmlsOiB0cnVlICovCgkJZG9jdW1lbnQud3JpdGUoJzxzY3JpJyArICdwdCBzcmM9IicgKyByb290ICsgIi9zb3VyY2UvYm9vdC8iICsgaW5TcmMgKyAnIj48L3NjcmknICsgJ3B0PicpOwoJCS8qIGpzaGludCBldmlsOiBmYWxzZSAqLwoJfTsKCglzY3JpcHQoInJlYWR5LmpzIik7CglzY3JpcHQoIi4uLy4uL2xvYWRlci5qcyIpOwoJc2NyaXB0KCJib290LmpzIik7CglzY3JpcHQoIi4uL3BhY2thZ2UuanMiKTsKfSkoKTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/kind-template.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCgkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCS8vIFBVQkxJQyBQUk9QRVJUSUVTCgoJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkvLyBQUk9URUNURUQgUFJPUEVSVElFUwoKCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJLy8gQ09NUFVURUQgUFJPUEVSVElFUwoKCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJLy8gUFVCTElDIE1FVEhPRFMKCgkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCS8vIFBST1RFQ1RFRCBNRVRIT0RTCgoJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkvLyBPQlNFUlZFUlMKCn0pOw==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/loader.js"
Content-Type: application/octet-stream; x-encoding=base64
/* global enyo:true */
(function() {
	enyo = window.enyo || {};

	enyo.pathResolverFactory = function() {
		this.paths = {};
		this.pathNames = [];
	};

	enyo.pathResolverFactory.prototype = {
		addPath: function(inName, inPath) {
			this.paths[inName] = inPath;
			this.pathNames.push(inName);
			this.pathNames.sort(function(a, b) {
				return b.length - a.length;
			});
			return inPath;
		},
		addPaths: function(inPaths) {
			if (inPaths) {
				for (var n in inPaths) {
					this.addPath(n, inPaths[n]);
				}
			}
		},
		includeTrailingSlash: function(inPath) {
			return (inPath && inPath.slice(-1) !== "/") ? inPath + "/" : inPath;
		},
		// replace macros of the form $pathname with the mapped value of paths.pathname
		rewrite: function (inPath) {
			var working, its = this.includeTrailingSlash, paths = this.paths;
			var fn = function(macro, name) {
				working = true;
				return its(paths[name]) || '';
			};
			var result = inPath;
			do {
				working = false;
				for (var i=0; i<this.pathNames.length; i++) {
					var regex = new RegExp("\\$(" + this.pathNames[i] + ")(\\/)?", "g");
					result = result.replace(regex, fn);
				}
			} while (working);
			return result;
		}
	};

	enyo.path = new enyo.pathResolverFactory();

	enyo.loaderFactory = function(inMachine, inPathResolver) {
		this.machine = inMachine;
		// package information
		this.packages = [];
		// module information
		this.modules = [];
		// stylesheet paths
		this.sheets = [];
		// designer metadata paths
		this.designs = [];
		// (protected) internal dependency stack
		this.stack = [];
		this.pathResolver = inPathResolver || enyo.path;
		this.packageName = "";
		this.packageFolder = "";
		this.finishCallbacks = {};
	};

	enyo.loaderFactory.prototype  = {
		verbose: false,
		loadScript: function(inScript, success, failure) {
			this.machine.script(inScript, success, failure);
		},
		loadSheet: function(inSheet) {
			this.machine.sheet(inSheet);
		},
		loadPackage: function(inPackage) {
			this.machine.script(inPackage);
		},
		report: function() {
		},
		//
		load: function(/*<inDependency0, inDependency1 ...>*/) {
			// begin processing dependencies
			this.more({
				index: 0,
				depends: arguments || []
			});
		},
		more: function(inBlock) {
			// a 'block' is a dependency list with a bookmark
			// the bookmark (index) allows us to interrupt
			// processing and then continue asynchronously.
			if (inBlock) {
				// returns true if this block has asynchronous requirements
				// in that case, we unwind the stack. The asynchronous loader
				// must provide the continuation (by calling 'more' again).
				if (this.continueBlock(inBlock)) {
					return;
				}
			}
			// A package is now complete. Pop the block that was interrupted for that package (if any).
			var block = this.stack.pop();
			if (block) {
				// propagate failed scripts to queued block
				if(enyo.runtimeLoading && inBlock.failed) {
					block.failed = block.failed || [];
					block.failed.push.apply(block.failed, inBlock.failed);
				}

				// block.packageName is the name of the package that interrupted us
				//this.report("finished package", block.packageName);
				if (this.verbose) {
					window.console.groupEnd("* finish package (" + (block.packageName || "anon") + ")");
				}
				// cache the folder for the currently processing package
				this.packageFolder = block.folder;
				// no current package
				this.packageName = "";
				// process this new block
				this.more(block);
			} else {
				this.finish(inBlock);
			}
		},
		finish: function(inBlock) {
			this.packageFolder = "";
			if (this.verbose) {
				window.console.log("-------------- fini");
			}
			for (var i in this.finishCallbacks) {
				if (this.finishCallbacks[i]) {
					var callback = this.finishCallbacks[i];
					this.finishCallbacks[i] = null;
					callback(inBlock);
				}
			}
		},
		continueBlock: function(inBlock) {
			while (inBlock.index < inBlock.depends.length) {
				var d = inBlock.depends[inBlock.index++];
				if (d) {
					if (typeof d == "string") {
						if (this.require(d, inBlock)) {
							// return true to indicate we need to interrupt
							// processing until asynchronous file load completes
							// the load process itself must provide the
							// continuation
							return true;
						}
					} else {
						this.pathResolver.addPaths(d);
					}
				}
			}
		},
		require: function(inPath, inBlock) {
			// process aliases
			var path = this.pathResolver.rewrite(inPath);
			// get path root
			var prefix = this.getPathPrefix(inPath);
			// assemble path
			path = prefix + path;
			// process path
			if ((path.slice(-4) == ".css") || (path.slice(-5) == ".less")) {
				if (this.verbose) {
					window.console.log("+ stylesheet: [" + prefix + "][" + inPath + "]");
				}
				this.requireStylesheet(path);
			} else if (path.slice(-3) == ".js" && path.slice(-10) != "package.js") {
				if (this.verbose) {
					window.console.log("+ module: [" + prefix + "][" + inPath + "]");
				}

				return this.requireScript(inPath, path, inBlock);
			} else if (path.slice(-7) == ".design") {
				if (this.verbose) {
					window.console.log("+ design metadata: [" + prefix + "][" + inPath + "]");
				}
				this.requireDesign(path);
			} else {
				// package
				this.requirePackage(path, inBlock);
				// return true to indicate a package was located and
				// we need to interrupt further processing until it's completed
				return true;
			}
		},
		getPathPrefix: function(inPath) {
			var delim = inPath.slice(0, 1);
			if ((delim != "/") && (delim != "\\") && (delim != "$") && !/^https?:/i.test(inPath)) {
				return this.packageFolder;
			}
			return "";
		},
		requireStylesheet: function(inPath) {
			// stylesheet
			this.sheets.push(inPath);
			this.loadSheet(inPath);
		},
		requireScript: function(inRawPath, inPath, inBlock) {
			// script file
			this.modules.push({
				packageName: this.packageName,
				rawPath: inRawPath,
				path: inPath
			});

			if(enyo.runtimeLoading) {
				var _this = this;
				var success = function() {
					_this.more(inBlock);
				};

				var failure = function() {
					inBlock.failed = inBlock.failed || [];
					inBlock.failed.push(inPath);
					_this.more(inBlock);
				};

				this.loadScript(inPath, success, failure);
			} else {
				this.loadScript(inPath);
			}

			return enyo.runtimeLoading;
		},
		requireDesign: function(inPath) {
			// designer metadata (no loading here)
			this.designs.push({
				packageName: this.packageName,
				path: inPath
			});
		},
		decodePackagePath: function(inPath) {
			// A package path can be encoded in two ways:
			//
			//	1. [folder]
			//	2. [folder]/[*package.js]
			//
			// Note: manifest file name must end in "package.js"
			//
			var folder = '', manifest = 'package.js';
			// convert back slashes to forward slashes, remove double slashes, split on slash
			var parts = inPath.replace(/\\/g, "/").replace(/\/\//g, "/").replace(/:\//, "://").split("/");
			if (parts.length) {
				// if inPath has a trailing slash, parts has an empty string which we pop off and ignore
				var name = parts.pop() || parts.pop() || "";
				// test if name includes the manifest tag
				if (name.slice(-manifest.length) !== manifest) {
					// if not a manifest name, it's part of the folder path
					parts.push(name);
				} else {
					// otherwise this is the manifest name
					manifest = name;
				}
				//
				folder = parts.join("/");
				folder = (folder ? folder + "/" : "");
				manifest = folder + manifest;
			}
			return {
				folder: folder,
				manifest: manifest
			};
		},
		aliasPackage: function(inPath) {
			var parts = this.decodePackagePath(inPath);
			// cache manifest path
			this.manifest = parts.manifest;
		},
		requirePackage: function(inPath, inBlock) {
			// cache the interrupted packageFolder
			inBlock.folder = this.packageFolder;
			// set new manifest/packageFolder
			var parts = this.decodePackagePath(inPath);
			this.manifest = parts.manifest;
			this.packageFolder = parts.folder;
			// cache the name of the package 'inBlock' is loading now
			inBlock.packageName = this.packageName;
			// push inBlock on the continuation stack
			this.stack.push(inBlock);
			// console/user reporting
			this.report("loading package", this.packageName);
			if (this.verbose) {
				window.console.group("* start package [" + this.packageName + "]");
			}
			// load the actual package. the package MUST call a continuation function
			// or the process will halt.
			this.loadPackage(this.manifest);
		}
	};
})();

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/minify/minify.bat"
Content-Type: application/octet-stream; x-encoding=base64
QEVDSE8gT0ZGClJFTSBGSVhNRTogbWluaWZ5IGhhcyBhIHByb2JsZW0gaWYgJ3BhY2thZ2UuanMnIGlzIG5vdCBpbiBDV0QsIGhlbmNlIHB1c2gsIGNkLCBwb3AKUFVTSEQgIiVDRCUiCkNEICIlfmRwMCIKQ0FMTCAuLlx0b29sc1xtaW5pZnkuYmF0IHBhY2thZ2UuanMgLW5vLWFsaWFzIC1vdXRwdXQgLi5cLi5cYnVpbGRcZW55byAlKgpQT1BECg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/minify/minify.sh"
Content-Type: application/octet-stream; x-encoding=base64
IyEvYmluL3NoCmNkICQoZGlybmFtZSAkMCkKLi4vdG9vbHMvbWluaWZ5LnNoIHBhY2thZ2UuanMgLW5vLWFsaWFzIC1vdXRwdXQgLi4vLi4vYnVpbGQvZW55bwo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/minify/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJIi4uL3NvdXJjZS9ib290IiwKCSIuLi9zb3VyY2UiCik7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/package.json"
Content-Type: application/octet-stream; x-encoding=base64
ewogICAgIm5hbWUiOiAiZW55b2pzIiwKICAgICJmaWxlbmFtZSI6ICJlbnlvLmpzIiwKICAgICJ2ZXJzaW9uIjogIjIuMy4wIiwKICAgICJkZXNjcmlwdGlvbiI6IkVueW8gaXMgYW4gb3BlbiBzb3VyY2Ugb2JqZWN0LW9yaWVudGVkIEphdmFTY3JpcHQgZnJhbWV3b3JrIGVtcGhhc2l6aW5nIGVuY2Fwc3VsYXRpb24gYW5kIG1vZHVsYXJpdHkuIEVueW8gY29udGFpbnMgZXZlcnl0aGluZyB5b3UgbmVlZCB0byBjcmVhdGUgYSBmYXN0LCBzY2FsYWJsZSBtb2JpbGUgb3Igd2ViIGFwcGxpY2F0aW9uLiIsCiAgICAiaG9tZXBhZ2UiOiAiaHR0cDovL2VueW9qcy5jb20vIiwKICAgICJidWdzIjogImh0dHA6Ly9qaXJhLmVueW9qcy5jb20vIiwKICAgICJrZXl3b3JkcyI6IFsgImZyYW1ld29yayIsICJ0b29sa2l0IiwgImNvbXBvbmVudHMiLCAibW9iaWxlIiwgIndlYk9TIiBdLAogICAgIm1haW50YWluZXJzIjogW3sKICAgICAgICAibmFtZSI6ICJFbnlvIEpTIEZyYW1ld29yayBUZWFtIiwKICAgICAgICAid2ViIjogImh0dHA6Ly9lbnlvanMuY29tLyIKICAgIH1dLAogICAgImxpY2Vuc2VzIjogW3sKICAgICAgICAidHlwZSI6ICJBcGFjaGUtMi4wIiwgInVybCI6ICJodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAiCiAgICB9XSwKICAgICJidWdzIjogImh0dHBzOi8vamlyYS5lbnlvanMuY29tLyIsCiAgICAicmVwb3NpdG9yeSI6IHsKICAgICAgICAidHlwZSI6ICJnaXQiLCAidXJsIjogImh0dHA6Ly9naXRodWIuY29tL2VueW9qcy9lbnlvIgogICAgfSwKICAgICJkZXBlbmRlbmNpZXMiOiBbCiAgICAgICAgeyJsZXNzIjogIjEuMy4zIn0sCiAgICAgICAgeyJ1Z2xpZnktanMiOiAiMi4yLjUifSwKICAgICAgICB7InNoZWxsanMiOiAiMC4wLjgifSwKICAgICAgICB7Im5vcHQiOiAiMi4wLjAifQogICAgXQp9Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/AjaxSample.css"
Content-Type: application/octet-stream; x-encoding=base64
LmFqYXgtc2FtcGxlIHsKCXBhZGRpbmc6IDE1cHg7Cn0KCi5hamF4LXNhbXBsZS1zb3VyY2UgewoJcGFkZGluZzoxNXB4OwoJZm9udC1mYW1pbHk6IG1vbm9zcGFjZTsKCWZvbnQtc2l6ZTogMTJweDsKCXdoaXRlLXNwYWNlOiBwcmUtd3JhcDsKCWJveC1zaXppbmc6IGJvcmRlci1ib3g7Cgl3aWR0aDoxMDAlOyAKCWhlaWdodDoxMDAlOwoJbWFyZ2luLXRvcDoxNXB4Owp9Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/AjaxSample.html"
Content-Type: application/octet-stream; x-encoding=base64
PCFET0NUWVBFIGh0bWw+CjxodG1sPgo8aGVhZD4KCTxtZXRhIGh0dHAtZXF1aXY9IlgtVUEtQ29tcGF0aWJsZSIgY29udGVudD0iSUU9ZWRnZSxjaHJvbWU9MSIvPgoJPG1ldGEgaHR0cC1lcXVpdj0iQ29udGVudC1UeXBlIiBjb250ZW50PSJ0ZXh0L2h0bWw7IGNoYXJzZXQ9dXRmLTgiPgoJPG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+Cgk8bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWF4aW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iLz4KCTxtZXRhIG5hbWU9ImZvcm1hdC1kZXRlY3Rpb24iIGNvbnRlbnQ9InRlbGVwaG9uZT1ubyIvPgoJPG1ldGEgbmFtZT0iZ29vZ2xlIiB2YWx1ZT0ibm90cmFuc2xhdGUiLz4KCTx0aXRsZT5BSkFYIFNhbXBsZTwvdGl0bGU+Cgk8IS0tIC0tPgoJPHNjcmlwdCBzcmM9Ii4uLy4uL2VueW8vZW55by5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTxzY3JpcHQgc3JjPSIuLi8uLi9saWIvbGF5b3V0L3BhY2thZ2UuanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8c2NyaXB0IHNyYz0iLi4vLi4vbGliL29ueXgvcGFja2FnZS5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTwhLS0gLS0+Cgk8bGluayBocmVmPSJBamF4U2FtcGxlLmNzcyIgcmVsPSJzdHlsZXNoZWV0Ij4KCTxzY3JpcHQgc3JjPSJBamF4U2FtcGxlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPCEtLSAtLT4KPC9oZWFkPgo8Ym9keSBjbGFzcz0ib255eCI+CjxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij4KCW5ldyBlbnlvLnNhbXBsZS5BamF4U2FtcGxlKCkucmVuZGVySW50byhkb2N1bWVudC5ib2R5KTsKPC9zY3JpcHQ+CjwvYm9keT4KPC9odG1sPg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/AjaxSample.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLnNhbXBsZS5BamF4U2FtcGxlIiwKCWtpbmQ6ICJGaXR0YWJsZVJvd3MiLAoJY2xhc3NlczogImVueW8tZml0IGFqYXgtc2FtcGxlIiwKCWNvbXBvbmVudHM6IFsKCQl7a2luZDogIkZpdHRhYmxlQ29sdW1ucyIsIGNsYXNzZXM6Im9ueXgtdG9vbGJhci1pbmxpbmUiLCBjb21wb25lbnRzOiBbCgkJCXtjb250ZW50OiAiWVFMOiAifSwKCQkJe2tpbmQ6ICJvbnl4LklucHV0IiwgbmFtZToicXVlcnkiLCBmaXQ6dHJ1ZSwgdmFsdWU6J3NlbGVjdCAqIGZyb20gdXBjb21pbmcuZXZlbnRzIHdoZXJlIHdvZWlkIGluIChzZWxlY3Qgd29laWQgZnJvbSBnZW8ucGxhY2VzIHdoZXJlIHRleHQ9IlN1bm55dmFsZSwgQ0EiKSd9LAoJCQl7a2luZDogIm9ueXguQnV0dG9uIiwgY29udGVudDoiRmV0Y2giLCBvbnRhcDoiZmV0Y2gifQoJCV19LAoJCXtraW5kOiAiRml0dGFibGVDb2x1bW5zIiwgY2xhc3Nlczoib255eC10b29sYmFyLWlubGluZSIsIGNvbXBvbmVudHM6IFsKCQkJe2NvbnRlbnQ6ICJVUkw6ICJ9LAoJCQl7a2luZDogIm9ueXguSW5wdXQiLCBuYW1lOiJiYXNlVXJsIiwgZml0OnRydWUsIHZhbHVlOidodHRwOi8vcXVlcnkueWFob29hcGlzLmNvbS92MS9wdWJsaWMveXFsP2Zvcm1hdD1qc29uJ30KCQldfSwKCQl7a2luZDogIm9ueXguVGV4dEFyZWEiLCBmaXQ6dHJ1ZSwgY2xhc3NlczoiYWpheC1zYW1wbGUtc291cmNlIn0KCV0sCglmZXRjaDogZnVuY3Rpb24oKSB7CgkJdmFyIGFqYXggPSBuZXcgZW55by5BamF4KHsKCQkJdXJsOiB0aGlzLiQuYmFzZVVybC5nZXRWYWx1ZSgpCgkJfSk7CgkJLy8gc2VuZCBwYXJhbWV0ZXJzIHRoZSByZW1vdGUgc2VydmljZSB1c2luZyB0aGUgJ2dvKCknIG1ldGhvZAoJCWFqYXguZ28oewoJCQlxOiB0aGlzLiQucXVlcnkuZ2V0VmFsdWUoKQoJCX0pOwoJCS8vIGF0dGFjaCByZXNwb25kZXJzIHRvIHRoZSB0cmFuc2FjdGlvbiBvYmplY3QKCQlhamF4LnJlc3BvbnNlKHRoaXMsICJwcm9jZXNzUmVzcG9uc2UiKTsKCQkvLyBoYW5kbGUgZXJyb3IKCQlhamF4LmVycm9yKHRoaXMsICJwcm9jZXNzRXJyb3IiKTsKCX0sCglwcm9jZXNzUmVzcG9uc2U6IGZ1bmN0aW9uKGluU2VuZGVyLCBpblJlc3BvbnNlKSB7CgkJLy8gZG8gc29tZXRoaW5nIHdpdGggaXQKCQl0aGlzLiQudGV4dEFyZWEuc2V0VmFsdWUoSlNPTi5zdHJpbmdpZnkoaW5SZXNwb25zZSwgbnVsbCwgMikpOwoJfSwKCXByb2Nlc3NFcnJvcjogZnVuY3Rpb24oaW5TZW5kZXIsIGluUmVzcG9uc2UpIHsKCQl3aW5kb3cuYWxlcnQoIkVycm9yISIpOwoJfQp9KTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/AudioSample.css"
Content-Type: application/octet-stream; x-encoding=base64
LmF1ZGlvLXNhbXBsZSB7CglwYWRkaW5nOiAxNXB4Owp9CgouYXVkaW8tc2FtcGxlLWRpdmlkZXIgewoJY29sb3I6ICNGNDkyMDA7Cgl0ZXh0LXRyYW5zZm9ybTogdXBwZXJjYXNlOwoJZm9udC1mYW1pbHk6IFNlZ29lIFVJLCBQcmVsdWRlIE1lZGl1bSwgSGVsdmV0aWNhLCBWZXJkYW5hLCBzYW5zLXNlcmlmOwoJZm9udC1zaXplOiAxNHB4OwoJZm9udC13ZWlnaHQ6IGJvbGQ7CgltYXJnaW4tYm90dG9tOiA4cHg7Cn0=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/AudioSample.html"
Content-Type: application/octet-stream; x-encoding=base64
PCFET0NUWVBFIGh0bWw+CjxodG1sPgo8aGVhZD4KCTxtZXRhIGh0dHAtZXF1aXY9IlgtVUEtQ29tcGF0aWJsZSIgY29udGVudD0iSUU9ZWRnZSxjaHJvbWU9MSIvPgoJPG1ldGEgaHR0cC1lcXVpdj0iQ29udGVudC1UeXBlIiBjb250ZW50PSJ0ZXh0L2h0bWw7IGNoYXJzZXQ9dXRmLTgiPgoJPG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+Cgk8bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWF4aW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iLz4KCTxtZXRhIG5hbWU9ImZvcm1hdC1kZXRlY3Rpb24iIGNvbnRlbnQ9InRlbGVwaG9uZT1ubyIvPgoJPG1ldGEgbmFtZT0iZ29vZ2xlIiB2YWx1ZT0ibm90cmFuc2xhdGUiLz4KCTx0aXRsZT5BdWRpbyBTYW1wbGU8L3RpdGxlPgoJPCEtLSAtLT4KCTxzY3JpcHQgc3JjPSIuLi8uLi9lbnlvL2VueW8uanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8c2NyaXB0IHNyYz0iLi4vLi4vbGliL29ueXgvcGFja2FnZS5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTwhLS0gLS0+Cgk8bGluayBocmVmPSJBdWRpb1NhbXBsZS5jc3MiIHJlbD0ic3R5bGVzaGVldCI+Cgk8c2NyaXB0IHNyYz0iQXVkaW9TYW1wbGUuanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8IS0tIC0tPgo8L2hlYWQ+Cjxib2R5IGNsYXNzPSJvbnl4Ij4KPHNjcmlwdCB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPgoJbmV3IGVueW8uc2FtcGxlLkF1ZGlvU2FtcGxlKCkucmVuZGVySW50byhkb2N1bWVudC5ib2R5KTsKPC9zY3JpcHQ+CjwvYm9keT4KPC9odG1sPg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/AudioSample.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLnNhbXBsZS5BdWRpb1NhbXBsZSIsCgljbGFzc2VzOiAiYXVkaW8tc2FtcGxlIiwKCWNvbXBvbmVudHM6IFsKCQl7a2luZDogImVueW8uQXVkaW8iLCBvbkVuZGVkOiAiZW5kZWQifSwKCQl7Y29udGVudDogIlNvdW5kcyIsIGNsYXNzZXM6ImF1ZGlvLXNhbXBsZS1kaXZpZGVyIn0sCgkJe2NsYXNzZXM6ICJvbnl4LXRvb2xiYXItaW5saW5lIiwgY29tcG9uZW50czogWwoJCQl7a2luZDogIm9ueXguUGlja2VyRGVjb3JhdG9yIiwgY29tcG9uZW50czogWwoJCQkJe30sCgkJCQl7a2luZDogIm9ueXguUGlja2VyIiwgb25TZWxlY3Q6ICJpdGVtU2VsZWN0ZWQiLCBjb21wb25lbnRzOiBbCgkJCQkJe2NvbnRlbnQ6ICJCaXJkcyIsIGFjdGl2ZTogdHJ1ZX0sCgkJCQkJe2NvbnRlbnQ6ICJDYXQifSwKCQkJCQl7Y29udGVudDogIkNvdyJ9LAoJCQkJCXtjb250ZW50OiAiR3VpdGFyIn0KCQkJCV19CgkJCV19LAoJCQl7a2luZDogIm9ueXguQnV0dG9uIiwgY29udGVudDogIlBsYXkiLCBvbnRhcDogInRvZ2dsZVBsYXkifQoJCV19LAoJCXt0YWc6ICJiciJ9LAoJCXtuYW1lOiAiY29uc29sZSJ9CgldLAoJc291bmRzOiBbCgkJe2xhYmVsOiAiQmlyZHMiLCBzcmM6ICJodHRwOi8vd3d3LnVuaXZlcnNhbC1zb3VuZGJhbmsuY29tL21wMy9zb3VuZHMvMTI1OTEubXAzIn0sCgkJe2xhYmVsOiAiQ2F0Iiwgc3JjOiAiaHR0cDovL3d3dy51bml2ZXJzYWwtc291bmRiYW5rLmNvbS9tcDMvc291bmRzLzk4Ni5tcDMifSwKCQl7bGFiZWw6ICJDb3ciLCBzcmM6ICJodHRwOi8vd3d3LnVuaXZlcnNhbC1zb3VuZGJhbmsuY29tL21wMy9zb3VuZHMvMTAxLm1wMyJ9LAoJCXtsYWJlbDogIkd1aXRhciIsIHNyYzogImh0dHA6Ly93d3cudW5pdmVyc2FsLXNvdW5kYmFuay5jb20vbXAzL3NvdW5kcy8xODI2NS5tcDMifQoJXSwKCXJlbmRlcmVkOiBmdW5jdGlvbigpIHsKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCXRoaXMubG9hZEF1ZGlvKDApOwoJfSwKCWxvYWRBdWRpbzogZnVuY3Rpb24oaW5JbmRleCkgewoJCXRoaXMuJC5hdWRpby5zZXRTcmModGhpcy5zb3VuZHNbaW5JbmRleF0uc3JjKTsKCQl0aGlzLiQuY29uc29sZS5zZXRDb250ZW50KCIiKTsKCQl0aGlzLiQuYnV0dG9uLnNldENvbnRlbnQoIlBsYXkiKTsKCX0sCglwbGF5QXVkaW86IGZ1bmN0aW9uKCkgewoJCXRoaXMuJC5hdWRpby5wbGF5KCk7CgkJdGhpcy4kLmJ1dHRvbi5zZXRDb250ZW50KCJQYXVzZSIpOwoJCXRoaXMuJC5jb25zb2xlLnNldENvbnRlbnQoIkF1ZGlvIHBsYXlpbmciKTsKCX0sCglwYXVzZUF1ZGlvOiBmdW5jdGlvbigpIHsKCQl0aGlzLiQuYXVkaW8ucGF1c2UoKTsKCQl0aGlzLiQuYnV0dG9uLnNldENvbnRlbnQoIlBsYXkiKTsKCQl0aGlzLiQuY29uc29sZS5zZXRDb250ZW50KCJBdWRpbyBwYXVzZWQiKTsKCX0sCgl0b2dnbGVQbGF5OiBmdW5jdGlvbihpblNlbmRlciwgaW5SZXNwb25zZSkgewoJCWlmICh0aGlzLiQuYXVkaW8uZ2V0UGF1c2VkKCkpIHsKCQkJdGhpcy5wbGF5QXVkaW8oKTsKCQl9IGVsc2UgewoJCQl0aGlzLnBhdXNlQXVkaW8oKTsKCQl9Cgl9LAoJZW5kZWQ6IGZ1bmN0aW9uKGluU2VuZGVyLCBpblJlc3BvbnNlKSB7CgkJdGhpcy4kLmNvbnNvbGUuc2V0Q29udGVudCgiQXVkaW8gZW5kZWQiKTsKCQl0aGlzLiQuYnV0dG9uLnNldENvbnRlbnQoIlBsYXkiKTsKCX0sCglpdGVtU2VsZWN0ZWQ6IGZ1bmN0aW9uKGluSXRlbSkgewoJCXZhciBjb250ZW50ID0gaW5JdGVtLnNlbGVjdGVkLmdldENvbnRlbnQoKTsKCQl2YXIgc291bmRDb3VudCA9IHRoaXMuc291bmRzLmxlbmd0aDsKCQlmb3IgKHZhciBpPTA7IGk8c291bmRDb3VudDsgaSsrKSB7CgkJCWlmIChjb250ZW50ID09PSB0aGlzLnNvdW5kc1tpXS5sYWJlbCkgewoJCQkJdGhpcy5sb2FkQXVkaW8oaSk7CgkJCQlicmVhazsKCQkJfQoJCX0KCX0KfSk7
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/DrawerSample.css"
Content-Type: application/octet-stream; x-encoding=base64
LmRyYXdlci1zYW1wbGUgewoJcGFkZGluZzogMTBweDsKfQouZHJhd2VyLXNhbXBsZS1kaXZpZGVyIHsKCWNvbG9yOiAjRjQ5MjAwOwoJdGV4dC10cmFuc2Zvcm06IHVwcGVyY2FzZTsKCWZvbnQtZmFtaWx5OiBTZWdvZSBVSSwgUHJlbHVkZSBNZWRpdW0sIEhlbHZldGljYSwgVmVyZGFuYSwgc2Fucy1zZXJpZjsKCWZvbnQtc2l6ZTogMTRweDsKCWZvbnQtd2VpZ2h0OiBib2xkOwoJbWFyZ2luLWJvdHRvbTogOHB4Owp9Ci5kcmF3ZXItc2FtcGxlLWxhYmVsIHsKCWNvbG9yOiAjMTg3OUNEOwoJZm9udC1zaXplOiAxOHB4OwoJdGV4dC10cmFuc2Zvcm06IHVwcGVyY2FzZTsKfQouZHJhd2VyLXNhbXBsZS1ib3ggewoJYm9yZGVyOiAycHggc29saWQgbGlnaHRibHVlOwoJcGFkZGluZzogNHB4OwoJd2hpdGUtc3BhY2U6IG5vd3JhcDsKCW92ZXJmbG93OmhpZGRlbjsKfQouZHJhd2VyLXNhbXBsZS1tdGIgewoJbWFyZ2luLXRvcDogM3B4OwoJbWFyZ2luLWJvdHRvbTogM3B4Owp9Ci5kcmF3ZXItc2FtcGxlLW1sciB7CgltYXJnaW4tbGVmdDogNnB4OwoJbWFyZ2luLXJpZ2h0OiA2cHg7Cn0KLmRyYXdlci1zYW1wbGUtbyB7Cglib3JkZXItY29sb3I6IG9yYW5nZTsKfQ==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/DrawerSample.html"
Content-Type: application/octet-stream; x-encoding=base64
PCFET0NUWVBFIGh0bWw+CjxodG1sPgo8aGVhZD4KCTxtZXRhIGh0dHAtZXF1aXY9IlgtVUEtQ29tcGF0aWJsZSIgY29udGVudD0iSUU9ZWRnZSxjaHJvbWU9MSIvPgoJPG1ldGEgaHR0cC1lcXVpdj0iQ29udGVudC1UeXBlIiBjb250ZW50PSJ0ZXh0L2h0bWw7IGNoYXJzZXQ9dXRmLTgiPgoJPG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+Cgk8bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWF4aW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iLz4KCTxtZXRhIG5hbWU9ImZvcm1hdC1kZXRlY3Rpb24iIGNvbnRlbnQ9InRlbGVwaG9uZT1ubyIvPgoJPG1ldGEgbmFtZT0iZ29vZ2xlIiB2YWx1ZT0ibm90cmFuc2xhdGUiLz4KCTx0aXRsZT5EcmF3ZXIgU2FtcGxlPC90aXRsZT4KCTwhLS0gLS0+Cgk8c2NyaXB0IHNyYz0iLi4vLi4vZW55by9lbnlvLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPHNjcmlwdCBzcmM9Ii4uLy4uL2xpYi9sYXlvdXQvcGFja2FnZS5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTwhLS0gLS0+Cgk8bGluayBocmVmPSJEcmF3ZXJTYW1wbGUuY3NzIiByZWw9InN0eWxlc2hlZXQiPgoJPHNjcmlwdCBzcmM9IkRyYXdlclNhbXBsZS5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTwhLS0gLS0+CjwvaGVhZD4KPGJvZHk+CjxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij4KCW5ldyBlbnlvLnNhbXBsZS5EcmF3ZXJTYW1wbGUoe2NsYXNzZXM6ICJlbnlvLXVuc2VsZWN0YWJsZSJ9KS5yZW5kZXJJbnRvKGRvY3VtZW50LmJvZHkpOwo8L3NjcmlwdD4KPC9ib2R5Pgo8L2h0bWw+
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/DrawerSample.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLnNhbXBsZS5EcmF3ZXJTYW1wbGUiLAoJY2xhc3NlczogImRyYXdlci1zYW1wbGUiLAoJY29tcG9uZW50czogWwoJCXtjb250ZW50OiAiRHJhd2VycyIsIGNsYXNzZXM6ImRyYXdlci1zYW1wbGUtZGl2aWRlciJ9LAoJCXtjb250ZW50OiAiQWN0aXZhdGUgKFYpIiwgY2xhc3NlczogImRyYXdlci1zYW1wbGUtYm94IGRyYXdlci1zYW1wbGUtbXRiIiwgb250YXA6ImFjdGl2YXRlRHJhd2VyIn0sCgkJe25hbWU6ICJkcmF3ZXIiLCBraW5kOiAiZW55by5EcmF3ZXIiLCBjb21wb25lbnRzOiBbCgkJCXtjb250ZW50OiAiVmVydGljYWwgRHJhd2VyPGJyPlZlcnRpY2FsIERyYXdlcjxicj5WZXJ0aWNhbCBEcmF3ZXI8YnI+VmVydGljYWwgRHJhd2VyIiwgYWxsb3dIdG1sOiB0cnVlLCBjbGFzc2VzOiAiZHJhd2VyLXNhbXBsZS1ib3ggZHJhd2VyLXNhbXBsZS1tdGIifSwKCQkJe2NvbXBvbmVudHM6IFsKCQkJCXtraW5kOiAiZW55by5DaGVja2JveCIsIG5hbWU6ICJhbmltYXRlU2V0dGluZyIsIHZhbHVlOiB0cnVlfSwKCQkJCXtjb250ZW50OiJBbmltYXRlZCIsIGNsYXNzZXM6ImVueW8taW5saW5lIG9ueXgtc2FtcGxlLWFuaW1hdGUtbGFiZWwifQoJCQldfSwKCQkJe2NvbnRlbnQ6ICJBY3RpdmF0ZSAoVikgKFRvZ2dsZWQgQW5pbWF0aW9uKSIsIGNsYXNzZXM6ICJkcmF3ZXItc2FtcGxlLWJveCBkcmF3ZXItc2FtcGxlLW10YiIsIG9udGFwOiJhY3RpdmF0ZURyYXdlcjIifSwKCQkJe25hbWU6ICJkcmF3ZXIyIiwga2luZDogImVueW8uRHJhd2VyIiwgYW5pbWF0ZWQ6IHRydWUsIGNvbXBvbmVudHM6IFsKCQkJCXtjb250ZW50OiAiVmVydGljYWwgRHJhd2VyPGJyPlZlcnRpY2FsIERyYXdlcjxicj5WZXJ0aWNhbCBEcmF3ZXI8YnI+VmVydGljYWwgRHJhd2VyIiwgYWxsb3dIdG1sOiB0cnVlLCBjbGFzc2VzOiAiZHJhd2VyLXNhbXBsZS1ib3ggZHJhd2VyLXNhbXBsZS1tdGIifQoJCQldfSwKCQkJe2NvbnRlbnQ6ICJWZXJ0aWNhbCBEcmF3ZXI8YnI+VmVydGljYWwgRHJhd2VyPGJyPlZlcnRpY2FsIERyYXdlcjxicj5WZXJ0aWNhbCBEcmF3ZXIiLCBhbGxvd0h0bWw6IHRydWUsIGNsYXNzZXM6ICJkcmF3ZXItc2FtcGxlLWJveCBkcmF3ZXItc2FtcGxlLW10YiJ9CgkJXX0sCgkJe2NvbnRlbnQ6ICJGb288YnI+Rm9vIiwgYWxsb3dIdG1sOiB0cnVlLCBjbGFzc2VzOiAiZHJhd2VyLXNhbXBsZS1ib3ggZHJhd2VyLXNhbXBsZS1tdGIifSwKCQl7a2luZDogIkZpdHRhYmxlQ29sdW1ucyIsIGZpdDogdHJ1ZSwgb250YXA6ICJhY3RpdmF0ZUNvbHVtbnNEcmF3ZXIiLCBjbGFzc2VzOiAiZHJhd2VyLXNhbXBsZS1ib3ggZHJhd2VyLXNhbXBsZS1tdGIgZHJhd2VyLXNhbXBsZS1vIiwgY29tcG9uZW50czogWwoJCQl7Y29udGVudDogIkFjdGl2YXRlIChIKSIsIGNsYXNzZXM6ICJkcmF3ZXItc2FtcGxlLWJveCBkcmF3ZXItc2FtcGxlLW1sciJ9LAoJCQl7bmFtZTogImNvbHVtbnNEcmF3ZXIiLCBvcmllbnQ6ICJoIiwga2luZDogImVueW8uRHJhd2VyIiwgb3BlbjogZmFsc2UsIGNvbXBvbmVudHM6IFsKCQkJCXtjb250ZW50OiAiSC1EcmF3ZXIiLCBjbGFzc2VzOiAiZHJhd2VyLXNhbXBsZS1ib3ggZHJhd2VyLXNhbXBsZS1tbHIifQoJCQldfSwKCQkJe2NvbnRlbnQ6ICJGb28iLCBmaXQ6IHRydWUsIGNsYXNzZXM6ICJkcmF3ZXItc2FtcGxlLWJveCBkcmF3ZXItc2FtcGxlLW1sciBkcmF3ZXItc2FtcGxlLW8ifSwKCQkJe2NvbnRlbnQ6ICJGb28iLCBjbGFzc2VzOiAiZHJhd2VyLXNhbXBsZS1ib3ggZHJhd2VyLXNhbXBsZS1tbHIifQoJCV19LAoJCXtjb250ZW50OiAiRm9vIiwgY2xhc3NlczogImRyYXdlci1zYW1wbGUtYm94IGRyYXdlci1zYW1wbGUtbXRiIn0KCV0sCglhY3RpdmF0ZURyYXdlcjogZnVuY3Rpb24oKSB7CgkJdGhpcy4kLmRyYXdlci5zZXRPcGVuKCF0aGlzLiQuZHJhd2VyLm9wZW4pOwoJfSwKCWFjdGl2YXRlRHJhd2VyMjogZnVuY3Rpb24oKSB7CgkJdGhpcy4kLmRyYXdlcjIuc2V0QW5pbWF0ZWQodGhpcy4kLmFuaW1hdGVTZXR0aW5nLmdldFZhbHVlKCkpOwoJCXRoaXMuJC5kcmF3ZXIyLnNldE9wZW4oIXRoaXMuJC5kcmF3ZXIyLm9wZW4pOwoJfSwKCWFjdGl2YXRlQ29sdW1uc0RyYXdlcjogZnVuY3Rpb24oKSB7CgkJdGhpcy4kLmNvbHVtbnNEcmF3ZXIuc2V0T3BlbighdGhpcy4kLmNvbHVtbnNEcmF3ZXIub3Blbik7CgkJcmV0dXJuIHRydWU7Cgl9Cn0pOwo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/FullscreenSample.css"
Content-Type: application/octet-stream; x-encoding=base64
LmZ1bGxzY3JlZW4tc2FtcGxlIHsKCXBhZGRpbmc6IDE1cHg7Cn0KCi5mdWxsc2NyZWVuLXNhbXBsZSAuZnVsbHNjcmVlbi1zYW1wbGUtY29udGFpbmVyIHsKCXdpZHRoOiA0MDBweDsKfQoKLmZ1bGxzY3JlZW4tc2FtcGxlIC5mdWxsc2NyZWVuLXNhbXBsZS1jb250YWluZXIgLmZ1bGxzY3JlZW4tc2FtcGxlLXNjcm9sbGVyIHsKCXBhZGRpbmc6IDEwcHg7CgloZWlnaHQ6IDMwMHB4OwoJYm9yZGVyOiAxcHggc29saWQgI2FhYTsKfQoKLmZ1bGxzY3JlZW4tc2FtcGxlIGJ1dHRvbiB7CgltYXJnaW4tdG9wOiAyMHB4Owp9CgovKiBGdWxsc2NyZWVuIFN0eWxlcyAtIE5vdGU6IFRoZXNlIHJ1bGVzIGNhbid0IGJlIGNvbW1hLXNlcGFyYXRlZCAqLwouZnVsbHNjcmVlbi1zYW1wbGUgLmZ1bGxzY3JlZW4tc2FtcGxlLWNvbnRhaW5lcjotd2Via2l0LWZ1bGwtc2NyZWVuIC5mdWxsc2NyZWVuLXNhbXBsZS1zY3JvbGxlciB7Cglib3JkZXI6IG5vbmU7Cn0KLmZ1bGxzY3JlZW4tc2FtcGxlIC5mdWxsc2NyZWVuLXNhbXBsZS1jb250YWluZXI6LW1vei1mdWxsLXNjcmVlbiAuZnVsbHNjcmVlbi1zYW1wbGUtc2Nyb2xsZXIgewoJYm9yZGVyOiBub25lOwoJYmFja2dyb3VuZDogI2RkZDsKfQouZnVsbHNjcmVlbi1zYW1wbGUgLmZ1bGxzY3JlZW4tc2FtcGxlLWNvbnRhaW5lcjotbXMtZnVsbC1zY3JlZW4gLmZ1bGxzY3JlZW4tc2FtcGxlLXNjcm9sbGVyIHsKCWJvcmRlcjogbm9uZTsKfQouZnVsbHNjcmVlbi1zYW1wbGUgLmZ1bGxzY3JlZW4tc2FtcGxlLWNvbnRhaW5lcjotby1mdWxsLXNjcmVlbiAuZnVsbHNjcmVlbi1zYW1wbGUtc2Nyb2xsZXIgewoJYm9yZGVyOiBub25lOwp9Ci5mdWxsc2NyZWVuLXNhbXBsZSAuZnVsbHNjcmVlbi1zYW1wbGUtY29udGFpbmVyOi1mdWxsLXNjcmVlbiAuZnVsbHNjcmVlbi1zYW1wbGUtc2Nyb2xsZXIgewoJYm9yZGVyOiBub25lOwp9Ci8qIEZhbGxiYWNrIEZ1bGxzY3JlZW4gU3R5bGUgKGZvciBicm93c2VycyB3aXRob3V0IEhUTUw1IGZ1bGxzY3JlZW4gc3VwcG9ydCkgKi8KYm9keSAuZW55by1mdWxsc2NyZWVuLmZ1bGxzY3JlZW4tc2FtcGxlLWNvbnRhaW5lciB7CgliYWNrZ3JvdW5kOiAjZGRkOwp9
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/FullscreenSample.html"
Content-Type: application/octet-stream; x-encoding=base64
PCFET0NUWVBFIGh0bWw+CjxodG1sPgo8aGVhZD4KCTxtZXRhIGh0dHAtZXF1aXY9IlgtVUEtQ29tcGF0aWJsZSIgY29udGVudD0iSUU9ZWRnZSxjaHJvbWU9MSIvPgoJPG1ldGEgaHR0cC1lcXVpdj0iQ29udGVudC1UeXBlIiBjb250ZW50PSJ0ZXh0L2h0bWw7IGNoYXJzZXQ9dXRmLTgiPgoJPG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+Cgk8bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWF4aW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iLz4KCTxtZXRhIG5hbWU9ImZvcm1hdC1kZXRlY3Rpb24iIGNvbnRlbnQ9InRlbGVwaG9uZT1ubyIvPgoJPG1ldGEgbmFtZT0iZ29vZ2xlIiB2YWx1ZT0ibm90cmFuc2xhdGUiLz4KCTx0aXRsZT5GdWxsc2NyZWVuIFNhbXBsZTwvdGl0bGU+Cgk8IS0tIC0tPgoJPHNjcmlwdCBzcmM9Ii4uLy4uL2VueW8vZW55by5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTxzY3JpcHQgc3JjPSIuLi8uLi9saWIvbGF5b3V0L3BhY2thZ2UuanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8c2NyaXB0IHNyYz0iLi4vLi4vbGliL29ueXgvcGFja2FnZS5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTwhLS0gLS0+Cgk8bGluayBocmVmPSJGdWxsc2NyZWVuU2FtcGxlLmNzcyIgcmVsPSJzdHlsZXNoZWV0Ij4KCTxzY3JpcHQgc3JjPSJGdWxsc2NyZWVuU2FtcGxlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPCEtLSAtLT4KPC9oZWFkPgo8Ym9keSBjbGFzcz0ib255eCI+CjxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij4KCW5ldyBlbnlvLnNhbXBsZS5GdWxsc2NyZWVuU2FtcGxlKCkucmVuZGVySW50byhkb2N1bWVudC5ib2R5KTsKPC9zY3JpcHQ+CjwvYm9keT4KPC9odG1sPg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/FullscreenSample.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLnNhbXBsZS5GdWxsc2NyZWVuU2FtcGxlIiwKCWNsYXNzZXM6ICJmdWxsc2NyZWVuLXNhbXBsZSBlbnlvLWZpdCBlbnlvLXVuc2VsZWN0YWJsZSIsCgljb21wb25lbnRzOiBbCgkJe25hbWU6ICJzYW1wbGVDb250ZW50IiwgY2xhc3NlczogImZ1bGxzY3JlZW4tc2FtcGxlLWNvbnRhaW5lciIsIGNvbXBvbmVudHM6IFsKCQkJe2tpbmQ6ICJvbnl4Lkdyb3VwYm94SGVhZGVyIiwgY29udGVudDogIkZ1bGxzY3JlZW4gU2FtcGxlIn0sCgkJCXtraW5kOiAiZW55by5TY3JvbGxlciIsIGNsYXNzZXM6ICJmdWxsc2NyZWVuLXNhbXBsZS1zY3JvbGxlciIsIGNvbXBvbmVudHM6IFsKCQkJCXtjb250ZW50OiAiTG9yZW0gaXBzdW0gZG9sb3Igc2l0IGFtZXQsIGNvbnNlY3RldHVyIGFkaXBpc2ljaW5nIGVsaXQsIHNlZCBkbyBlaXVzbW9kIHRlbXBvciBpbmNpZGlkdW50IHV0IGxhYm9yZSBldCBkb2xvcmUgbWFnbmEgYWxpcXVhLiBVdCBlbmltIGFkIG1pbmltIHZlbmlhbSwgcXVpcyBub3N0cnVkIGV4ZXJjaXRhdGlvbiB1bGxhbWNvIGxhYm9yaXMgbmlzaSB1dCBhbGlxdWlwIGV4IGVhIGNvbW1vZG8gY29uc2VxdWF0LiBEdWlzIGF1dGUgaXJ1cmUgZG9sb3IgaW4gcmVwcmVoZW5kZXJpdCBpbiB2b2x1cHRhdGUgdmVsaXQgZXNzZSBjaWxsdW0gZG9sb3JlIGV1IGZ1Z2lhdCBudWxsYSBwYXJpYXR1ci4gRXhjZXB0ZXVyIHNpbnQgb2NjYWVjYXQgY3VwaWRhdGF0IG5vbiBwcm9pZGVudCwgc3VudCBpbiBjdWxwYSBxdWkgb2ZmaWNpYSBkZXNlcnVudCBtb2xsaXQgYW5pbSBpZCBlc3QgbGFib3J1bS4gU2VkIHV0IHBlcnNwaWNpYXRpcyB1bmRlIG9tbmlzIGlzdGUgbmF0dXMgZXJyb3Igc2l0IHZvbHVwdGF0ZW0gYWNjdXNhbnRpdW0gZG9sb3JlbXF1ZSBsYXVkYW50aXVtLCB0b3RhbSByZW0gYXBlcmlhbSwgZWFxdWUgaXBzYSBxdWFlIGFiIGlsbG8gaW52ZW50b3JlIHZlcml0YXRpcyBldCBxdWFzaSBhcmNoaXRlY3RvIGJlYXRhZSB2aXRhZSBkaWN0YSBzdW50IGV4cGxpY2Fiby4gTmVtbyBlbmltIGlwc2FtIHZvbHVwdGF0ZW0gcXVpYSB2b2x1cHRhcyBzaXQgYXNwZXJuYXR1ciBhdXQgb2RpdCBhdXQgZnVnaXQsIHNlZCBxdWlhIGNvbnNlcXV1bnR1ciBtYWduaSBkb2xvcmVzIGVvcyBxdWkgcmF0aW9uZSB2b2x1cHRhdGVtIHNlcXVpIG5lc2NpdW50LiBOZXF1ZSBwb3JybyBxdWlzcXVhbSBlc3QsIHF1aSBkb2xvcmVtIGlwc3VtIHF1aWEgZG9sb3Igc2l0IGFtZXQsIGNvbnNlY3RldHVyLCBhZGlwaXNjaSB2ZWxpdCwgc2VkIHF1aWEgbm9uIG51bXF1YW0gZWl1cyBtb2RpIHRlbXBvcmEgaW5jaWR1bnQgdXQgbGFib3JlIGV0IGRvbG9yZSBtYWduYW0gYWxpcXVhbSBxdWFlcmF0IHZvbHVwdGF0ZW0uIFV0IGVuaW0gYWQgbWluaW1hIHZlbmlhbSwgcXVpcyBub3N0cnVtIGV4ZXJjaXRhdGlvbmVtIHVsbGFtIGNvcnBvcmlzIHN1c2NpcGl0IGxhYm9yaW9zYW0sIG5pc2kgdXQgYWxpcXVpZCBleCBlYSBjb21tb2RpIGNvbnNlcXVhdHVyPyBRdWlzIGF1dGVtIHZlbCBldW0gaXVyZSByZXByZWhlbmRlcml0IHF1aSBpbiBlYSB2b2x1cHRhdGUgdmVsaXQgZXNzZSBxdWFtIG5paGlsIG1vbGVzdGlhZSBjb25zZXF1YXR1ciwgdmVsIGlsbHVtIHF1aSBkb2xvcmVtIGV1bSBmdWdpYXQgcXVvIHZvbHVwdGFzIG51bGxhIHBhcmlhdHVyPyJ9CgkJCV19LAoJCQl7a2luZDogIm9ueXguQnV0dG9uIiwgY29udGVudDogIlRvZ2dsZSBGdWxsc2NyZWVuIiwgb250YXA6ICJ0b2dnbGVGdWxsc2NyZWVuIn0KCQldfQoJXSwKCS8vKiBTZXQvdW5zZXQgX3RoaXMuJC5zYW1wbGVDb250ZW50XyBhcyBmdWxsc2NyZWVuIGNvbnRyb2wKCXRvZ2dsZUZ1bGxzY3JlZW46IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJdmFyIHRhcmdldENvbnRyb2wgPSB0aGlzLiQuc2FtcGxlQ29udGVudDsKCQkKCQkvLyBJZiBfdGFyZ2V0Q29udHJvbF8gaXMgY3VycmVudGx5IGZ1bGxzY3JlZW4sIGNhbmNlbCBmdWxsc2NyZWVuCgkJaWYgKHRhcmdldENvbnRyb2wuaXNGdWxsc2NyZWVuKCkpIHsKCQkJdGFyZ2V0Q29udHJvbC5jYW5jZWxGdWxsc2NyZWVuKCk7CgkJCgkJLy8gSWYgX3RhcmdldENvbnRyb2xfIGlzIG5vdCBjdXJyZW50bHkgZnVsbHNjcmVlbiwgcmVxdWVzdCBmdWxsc2NyZWVuCgkJfSBlbHNlIHsKCQkJdGFyZ2V0Q29udHJvbC5yZXF1ZXN0RnVsbHNjcmVlbigpOwoJCX0KCX0KfSk7
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/GestureSample.css"
Content-Type: application/octet-stream; x-encoding=base64
Lmdlc3R1cmUtc2FtcGxlIHsKCXBhZGRpbmc6IDE1cHg7Cn0KCi5nZXN0dXJlLXNhbXBsZSA+ICogewoJbWFyZ2luLWJvdHRvbToxMHB4Owp9CgouZ2VzdHVyZS1zYW1wbGUtcGFkIHsKCWJvcmRlci1yYWRpdXM6IDVweDsKCWJhY2tncm91bmQtY29sb3I6ICM4ODg7Cgljb2xvcjogIzY4Njg2ODsKIAl0ZXh0LXNoYWRvdzojQUFBIDBweCAxcHggMCwgIzAwMCAwIC0xcHggMDsKIAl0ZXh0LWFsaWduOmNlbnRlcjsKCXBhZGRpbmctdG9wOjE1JTsKCWZvbnQtc2l6ZTozMHB4OwoJb3ZlcmZsb3c6aGlkZGVuOwp9CgouZ2VzdHVyZS1zYW1wbGUtbm90ZSB7Cglmb250LXNpemU6MTVweDsgCgl0ZXh0LXNoYWRvdzpub25lOwp9CgouZ2VzdHVyZS1zYW1wbGUtcGFkZGVkID4gKiB7CglwYWRkaW5nOjVweDsKfQoKLmdlc3R1cmUtc2FtcGxlLWxlZnQgPiAqIHsKCXRleHQtYWxpZ246bGVmdDsKfQoKLmdlc3R1cmUtc2FtcGxlLXNldHRpbmcgewoJZm9udC1zaXplOiAxNXB4OwoJcGFkZGluZzogMTBweDsKCW1pbi1oZWlnaHQ6IDMwcHg7CglsaW5lLWhlaWdodDogMzBweDsKfQouZ2VzdHVyZS1zYW1wbGUtc2V0dGluZyA+ICogewoJZGlzcGxheTogaW5saW5lLWJsb2NrOwp9CgouZ2VzdHVyZS1zYW1wbGUtc2V0dGluZyA+IDpsYXN0LWNoaWxkIHsKCWZsb2F0OiByaWdodDsKfQoKQG1lZGlhIGFsbCBhbmQgKG1heC1oZWlnaHQ6NjAwcHgpIHsKCS5nZXN0dXJlLXNhbXBsZS10cnVuY2F0ZSB7CgkJd2lkdGg6MTAwJTsKCQl3b3JkLXdyYXA6bm9ybWFsOwoJCXdoaXRlLXNwYWNlOm5vd3JhcDsKCQlvdmVyZmxvdzpoaWRkZW47CgkJLW8tdGV4dC1vdmVyZmxvdzplbGxpcHNpczsKCQl0ZXh0LW92ZXJmbG93OmVsbGlwc2lzOwoJfQp9Cgo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/GestureSample.html"
Content-Type: application/octet-stream; x-encoding=base64
PCFET0NUWVBFIGh0bWw+CjxodG1sPgo8aGVhZD4KCTxtZXRhIGh0dHAtZXF1aXY9IlgtVUEtQ29tcGF0aWJsZSIgY29udGVudD0iSUU9ZWRnZSxjaHJvbWU9MSIvPgoJPG1ldGEgaHR0cC1lcXVpdj0iQ29udGVudC1UeXBlIiBjb250ZW50PSJ0ZXh0L2h0bWw7IGNoYXJzZXQ9dXRmLTgiPgoJPG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+Cgk8bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWF4aW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iLz4KCTxtZXRhIG5hbWU9ImZvcm1hdC1kZXRlY3Rpb24iIGNvbnRlbnQ9InRlbGVwaG9uZT1ubyIvPgoJPG1ldGEgbmFtZT0iZ29vZ2xlIiB2YWx1ZT0ibm90cmFuc2xhdGUiLz4KCTx0aXRsZT5HZXN0dXJlIFNhbXBsZTwvdGl0bGU+Cgk8IS0tIC0tPgoJPHNjcmlwdCBzcmM9Ii4uLy4uL2VueW8vZW55by5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTxzY3JpcHQgc3JjPSIuLi8uLi9saWIvbGF5b3V0L3BhY2thZ2UuanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8c2NyaXB0IHNyYz0iLi4vLi4vbGliL29ueXgvcGFja2FnZS5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTwhLS0gLS0+Cgk8bGluayBocmVmPSJHZXN0dXJlU2FtcGxlLmNzcyIgcmVsPSJzdHlsZXNoZWV0Ij4KCTxzY3JpcHQgc3JjPSJHZXN0dXJlU2FtcGxlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPCEtLSAtLT4KPC9oZWFkPgo8Ym9keSBjbGFzcz0ib255eCI+CjxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij4KCW5ldyBlbnlvLnNhbXBsZS5HZXN0dXJlU2FtcGxlKCkucmVuZGVySW50byhkb2N1bWVudC5ib2R5KTsKPC9zY3JpcHQ+CjwvYm9keT4KPC9odG1sPg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/GestureSample.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLnNhbXBsZS5HZXN0dXJlU2FtcGxlIiwKCWtpbmQ6ICJGaXR0YWJsZVJvd3MiLAoJY2xhc3NlczogImdlc3R1cmUtc2FtcGxlIGVueW8tZml0IGVueW8tdW5zZWxlY3RhYmxlIiwKCWNvbXBvbmVudHM6IFsKCQl7CgkJCWNsYXNzZXM6Imdlc3R1cmUtc2FtcGxlLXBhZCIsCgkJCWZpdDp0cnVlLAoJCQlvbmRvd246ICJoYW5kbGVFdmVudCIsCgkJCW9udXA6ICJoYW5kbGVFdmVudCIsCgkJCW9udGFwOiAiaGFuZGxlRXZlbnQiLAoJCQlvbm1vdmU6ICJoYW5kbGVFdmVudCIsCgkJCW9uZW50ZXI6ICJoYW5kbGVFdmVudCIsCgkJCW9ubGVhdmU6ICJoYW5kbGVFdmVudCIsCgkJCW9uZHJhZ3N0YXJ0OiAiaGFuZGxlRXZlbnQiLAoJCQlvbmRyYWc6ICJoYW5kbGVFdmVudCIsCgkJCW9uZHJhZ292ZXI6ICJoYW5kbGVFdmVudCIsCgkJCW9uaG9sZDogImhhbmRsZUV2ZW50IiwKCQkJb25yZWxlYXNlOiAiaGFuZGxlRXZlbnQiLAoJCQlvbmhvbGRwdWxzZTogImhhbmRsZUV2ZW50IiwKCQkJb25mbGljazogImhhbmRsZUV2ZW50IiwKCQkJb25nZXN0dXJlc3RhcnQ6ICJoYW5kbGVFdmVudCIsCgkJCW9uZ2VzdHVyZWNoYW5nZTogImhhbmRsZUV2ZW50IiwKCQkJb25nZXN0dXJlZW5kOiAiaGFuZGxlRXZlbnQiLAoJCQljb21wb25lbnRzOiBbCgkJCQl7Y29udGVudDogIlBlcmZvcm0gZ2VzdHVyZXMgaGVyZSJ9LAoJCQkJe2NsYXNzZXM6ICJnZXN0dXJlLXNhbXBsZS1ub3RlIiwgY29udGVudDoiKHRhcCBiZWxvdyBmb3Igb3B0aW9ucykifQoJCQldCgkJfSwKCQl7a2luZDogIm9ueXguR3JvdXBib3giLCBvbnRhcDoidG9nZ2xlU2V0dGluZ3MiLCBjb21wb25lbnRzOiBbCgkJCXtraW5kOiAib255eC5Hcm91cGJveEhlYWRlciIsIGNvbnRlbnQ6ICJFdmVudHMifSwKCQkJe25hbWU6ICJldmVudExpc3QiLCBzdHlsZToiZm9udC1zaXplOjEycHg7Iiwgb25Eb25lOiJyZW1vdmVFdmVudCIsIGNvbXBvbmVudHM6IFsKCQkJCXtuYW1lOiJ3YWl0aW5nIiwgY29udGVudDogIldhaXRpbmcgZm9yIGV2ZW50cy4uLiIsIHN0eWxlOiJwYWRkaW5nOjRweDtmb250LXN0eWxlOml0YWxpYztjb2xvcjpncmF5OyJ9CgkJCV19CgkJXX0sCgkJe29udGFwOiJ0b2dnbGVTZXR0aW5ncyIsIG5hbWU6InNldHRpbmdzIiwgc2hvd2luZzpmYWxzZSwgY29tcG9uZW50czogWwoJCQl7a2luZDogIm9ueXguR3JvdXBib3giLCBjbGFzc2VzOiJnZXN0dXJlLXNhbXBsZS1wYWRkZWQiLCBjb21wb25lbnRzOiBbCgkJCQl7a2luZDogIm9ueXguR3JvdXBib3hIZWFkZXIiLCBjb250ZW50OiAiT3B0aW9ucyJ9LAoJCQkJe2NsYXNzZXM6Imdlc3R1cmUtc2FtcGxlLXNldHRpbmciLCBjb21wb25lbnRzOiBbCgkJCQkJe2NvbnRlbnQ6IlRydW5jYXRlIGRldGFpbCBvbiBzbWFsbCBzY3JlZW46ICJ9LAoJCQkJCXtuYW1lOiJ0cnVuY2F0ZURldGFpbCIsIG9uY2hhbmdlOiJ0cnVuY2F0ZUNoYW5nZWQiLCBvbnRhcDoicHJldmVudERlZmF1bHQiLCBraW5kOiJvbnl4LkNoZWNrYm94IiwgY2hlY2tlZDp0cnVlfQoJCQkJXX0sCgkJCQl7Y2xhc3NlczoiZ2VzdHVyZS1zYW1wbGUtc2V0dGluZyIsIHN0eWxlOiJtaW4taGVpZ2h0OjQwcHg7IiwgY29tcG9uZW50czogWwoJCQkJCXtjb250ZW50OiJNb25pdG9yIGV2ZW50OiAifSwKCQkJCQl7a2luZDoib255eC5QaWNrZXJEZWNvcmF0b3IiLCBvbnRhcDoicHJldmVudERlZmF1bHQiLCBvblNlbGVjdDoibW9uaXRvckV2ZW50U2VsZWN0ZWQiLCBjb21wb25lbnRzOiBbCgkJCQkJCXtjb250ZW50OiJTZWxlY3QgZXZlbnQiLCBzdHlsZToid2lkdGg6MTQwcHg7IG1hcmdpbi1ib3R0b206NXB4OyJ9LAoJCQkJCQl7bmFtZToiZXZlbnRQaWNrZXIiLCBraW5kOiJvbnl4LlBpY2tlciIsIGNsYXNzZXM6Imdlc3R1cmUtc2FtcGxlLWxlZnQifQoJCQkJCV19CgkJCQldfQoJCQldfQoJCV19CgldLAoJY3JlYXRlOiBmdW5jdGlvbigpIHsKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCXRoaXMuZXZlbnRMaXN0ID0ge307CgkJdGhpcy5ldmVudENvdW50ID0gMDsKCQllbnlvLmZvckVhY2goCgkJCVsiQWxsIGV2ZW50cyIsImRvd24iLCJ1cCIsInRhcCIsIm1vdmUiLCJlbnRlciIsImxlYXZlIiwiZHJhZ3N0YXJ0IiwiZHJhZyIsImRyYWdvdmVyIiwiaG9sZCIsInJlbGVhc2UiLAoJCQkJImhvbGRwdWxzZSIsImZsaWNrIiwiZ2VzdHVyZXN0YXJ0IiwiZ2VzdHVyZWNoYW5nZSIsImdlc3R1cmVlbmQiXSwKCQkJdGhpcy5iaW5kU2FmZWx5KGZ1bmN0aW9uKGV2ZW50KSB7CgkJCQl0aGlzLiQuZXZlbnRQaWNrZXIuY3JlYXRlQ29tcG9uZW50KHtjb250ZW50OmV2ZW50LCBzdHlsZToidGV4dC1hbGlnbjpsZWZ0In0pOwoJCQl9KSk7Cgl9LAoJaGFuZGxlRXZlbnQ6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJdmFyIGV2ZW50ID0gZW55by5jbG9uZShpbkV2ZW50KTsKCQlpZiAodGhpcy5tb25pdG9yRXZlbnQgJiYgKGV2ZW50LnR5cGUgIT0gdGhpcy5tb25pdG9yRXZlbnQpKSB7CgkJCXJldHVybiB0cnVlOwoJCX0KCQl2YXIgZXZlbnRJdGVtID0gdGhpcy5ldmVudExpc3RbZXZlbnQudHlwZV07CgkJaWYgKGV2ZW50SXRlbSkgewoJCQlldmVudEl0ZW0uc2V0KCJldmVudCIsIGV2ZW50LCB0cnVlKTsKCQl9IGVsc2UgewoJCQl0aGlzLmV2ZW50Q291bnQrKzsKCQkJZXZlbnRJdGVtID0gdGhpcy4kLmV2ZW50TGlzdC5jcmVhdGVDb21wb25lbnQoewoJCQkJa2luZDogImVueW8uc2FtcGxlLkV2ZW50SXRlbSIsCgkJCQlldmVudDpldmVudCwKCQkJCXRydW5jYXRlOiB0aGlzLiQudHJ1bmNhdGVEZXRhaWwuZ2V0KCJ2YWx1ZSIpLAoJCQkJcGVyc2lzdDogdGhpcy5tb25pdG9yRXZlbnQKCQkJfSk7CgkJCXRoaXMuZXZlbnRMaXN0W2V2ZW50LnR5cGVdID0gZXZlbnRJdGVtOwoJCX0KCQlldmVudEl0ZW0ucmVuZGVyKCk7CgkJdGhpcy4kLndhaXRpbmcuaGlkZSgpOwoJCXRoaXMucmVmbG93KCk7CgkJcmV0dXJuIHRydWU7Cgl9LAoJdHJ1bmNhdGVDaGFuZ2VkOiBmdW5jdGlvbigpIHsKCQlmb3IgKHZhciBpIGluIHRoaXMuZXZlbnRMaXN0KSB7CgkJCXRoaXMuZXZlbnRMaXN0W2ldLnNldCgidHJ1bmNhdGUiLCB0aGlzLiQudHJ1bmNhdGVEZXRhaWwuZ2V0KCJ2YWx1ZSIpKTsKCQl9CgkJdGhpcy5yZWZsb3coKTsKCQlyZXR1cm4gZmFsc2U7Cgl9LAoJcmVtb3ZlRXZlbnQ6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJdGhpcy5ldmVudENvdW50LS07CgkJdGhpcy5ldmVudExpc3RbaW5FdmVudC50eXBlXS5kZXN0cm95KCk7CgkJZGVsZXRlIHRoaXMuZXZlbnRMaXN0W2luRXZlbnQudHlwZV07CgkJaWYgKHRoaXMuZXZlbnRDb3VudCA9PT0gMCkgewoJCQl0aGlzLiQud2FpdGluZy5zaG93KCk7CgkJfQoJCXRoaXMucmVmbG93KCk7CgkJcmV0dXJuIHRydWU7Cgl9LAoJcmVtb3ZlQWxsRXZlbnRzOiBmdW5jdGlvbigpIHsKCQlmb3IgKHZhciBpIGluIHRoaXMuZXZlbnRMaXN0KSB7CgkJCXRoaXMuZXZlbnRMaXN0W2ldLmRlc3Ryb3koKTsKCQkJZGVsZXRlIHRoaXMuZXZlbnRMaXN0W2ldOwoJCX0KCQl0aGlzLmV2ZW50Q291bnQgPSAwOwoJCXRoaXMuJC53YWl0aW5nLnNob3coKTsKCQl0aGlzLnJlZmxvdygpOwoJfSwKCXRvZ2dsZVNldHRpbmdzOiBmdW5jdGlvbigpIHsKCQl0aGlzLiQuc2V0dGluZ3Muc2V0KCJzaG93aW5nIiwgIXRoaXMuJC5zZXR0aW5ncy5nZXQoInNob3dpbmciKSk7CgkJdGhpcy5yZWZsb3coKTsKCX0sCglwcmV2ZW50RGVmYXVsdDogZnVuY3Rpb24oKSB7CgkJcmV0dXJuIHRydWU7Cgl9LAoJbW9uaXRvckV2ZW50U2VsZWN0ZWQ6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJdGhpcy5yZW1vdmVBbGxFdmVudHMoKTsKCQlpZiAoaW5FdmVudC5vcmlnaW5hdG9yLmNvbnRlbnQgPT0gIkFsbCBldmVudHMiKSB7CgkJCXRoaXMubW9uaXRvckV2ZW50ID0gbnVsbDsKCQl9IGVsc2UgewoJCQl0aGlzLm1vbml0b3JFdmVudCA9IGluRXZlbnQub3JpZ2luYXRvci5jb250ZW50OwoJCX0KCX0KfSk7CgplbnlvLmtpbmQoewoJbmFtZToiZW55by5zYW1wbGUuRXZlbnRJdGVtIiwKCXB1Ymxpc2hlZDogewoJCWV2ZW50OiIiLAoJCXRydW5jYXRlOiB0cnVlLAoJCXBlcnNpc3Q6IGZhbHNlCgl9LAoJc3R5bGU6InBhZGRpbmc6NHB4OyIsCglldmVudHM6IHsKCQlvbkRvbmU6IiIKCX0sCgljb21wb25lbnRzOiBbCgkJe25hbWU6ImV2ZW50UHJvcHMiLCBhbGxvd0h0bWw6dHJ1ZX0sCgkJe2tpbmQ6IkFuaW1hdG9yIiwgZHVyYXRpb246MTAwMCwgc3RhcnRWYWx1ZTowLCBlbmRWYWx1ZToyNTUsIG9uU3RlcDoic3RlcEFuaW1hdGlvbiIsIG9uRW5kOiJhbmltYXRpb25FbmRlZCJ9CgldLAoJY3JlYXRlOiBmdW5jdGlvbigpIHsKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCXRoaXMuZXZlbnRDaGFuZ2VkKCk7CgkJdGhpcy50cnVuY2F0ZUNoYW5nZWQoKTsKCX0sCgl0cnVuY2F0ZUNoYW5nZWQ6IGZ1bmN0aW9uKCkgewoJCXRoaXMuJC5ldmVudFByb3BzLmFkZFJlbW92ZUNsYXNzKCJnZXN0dXJlLXNhbXBsZS10cnVuY2F0ZSIsIHRoaXMudHJ1bmNhdGUpOwoJfSwKCWV2ZW50Q2hhbmdlZDogZnVuY3Rpb24oaW5PbGQpIHsKCQlpZiAodGhpcy5ldmVudCkgewoJCQlpZiAodGhpcy50aW1lb3V0KSB7CgkJCQljbGVhclRpbWVvdXQodGhpcy50aW1lb3V0KTsKCQkJCXRoaXMudGltZW91dCA9IG51bGw7CgkJCX0KCQkJdGhpcy4kLmFuaW1hdG9yLnN0b3AoKTsKCQkJdGhpcy4kLmV2ZW50UHJvcHMuc2V0KCJjb250ZW50IiwgdGhpcy5nZXRQcm9wc1N0cmluZygpKTsKCQkJdGhpcy4kLmFuaW1hdG9yLnBsYXkoKTsKCQl9Cgl9LAoJc3RlcEFuaW1hdGlvbjogZnVuY3Rpb24oaW5TZW5kZXIsIGluRXZlbnQpIHsKCQl2YXIgdiA9IE1hdGguZmxvb3IoaW5TZW5kZXIudmFsdWUpOwoJCXRoaXMuYXBwbHlTdHlsZSgiYmFja2dyb3VuZC1jb2xvciIsICJyZ2IoIiArIHYgKyAiLDI1NSwiICsgdiArICIpOyIpOwoJCXJldHVybiB0cnVlOwoJfSwKCWFuaW1hdGlvbkVuZGVkOiBmdW5jdGlvbigpIHsKCQlpZiAoIXRoaXMucGVyc2lzdCkgewoJCQl0aGlzLnRpbWVvdXQgPSBzZXRUaW1lb3V0KHRoaXMuYmluZFNhZmVseShmdW5jdGlvbigpIHsKCQkJCXRoaXMuZG9Eb25lKHt0eXBlOnRoaXMuZXZlbnQudHlwZX0pOwoJCQl9KSwgMjAwMCk7CgkJfQoJCXJldHVybiB0cnVlOwoJfSwKCWRlc3Ryb3k6IGZ1bmN0aW9uKCkgewoJCWlmICh0aGlzLnRpbWVvdXQpIHsKCQkJY2xlYXJUaW1lb3V0KHRoaXMudGltZW91dCk7CgkJCXRoaXMudGltZW91dCA9IG51bGw7CgkJfQoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7Cgl9LAoJZ2V0UHJvcHNTdHJpbmc6IGZ1bmN0aW9uKCkgewoJCXZhciBwcm9wcyA9IFtdOwoJCWZvciAodmFyIGkgaW4gdGhpcy5ldmVudCkgewoJCQlpZiAoKHRoaXMuZXZlbnRbaV0gIT09IHVuZGVmaW5lZCkgJiYKCQkJCSh0aGlzLmV2ZW50W2ldICE9PSBudWxsKSAmJgoJCQkJISh0aGlzLmV2ZW50W2ldIGluc3RhbmNlb2YgT2JqZWN0KSAmJgoJCQkJKGkgIT0gInR5cGUiKSkgewoJCQkJcHJvcHMucHVzaChpICsgIjogIiArIHRoaXMuZXZlbnRbaV0pOwoJCQl9CgkJfQoJCWlmICh0aGlzLmV2ZW50LnNyY0V2ZW50ICYmIHRoaXMuZXZlbnQuc3JjRXZlbnQudHlwZSkgewoJCQlwcm9wcy5wdXNoKCJzcmNFdmVudC50eXBlOiAiICsgdGhpcy5ldmVudC5zcmNFdmVudC50eXBlKTsKCQl9CgkJcmV0dXJuICI8Yj4iICsgdGhpcy5ldmVudC50eXBlICsgIjwvYj46IHsgIiArIHByb3BzLmpvaW4oIiwgIikgKyAiIH0iOwoJfQp9KTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/JsonpSample.css"
Content-Type: application/octet-stream; x-encoding=base64
Lmpzb25wLXNhbXBsZSB7CglwYWRkaW5nOiAxNXB4Owp9CgouanNvbnAtc2FtcGxlLXNvdXJjZSB7CglwYWRkaW5nOjE1cHg7Cglmb250LWZhbWlseTogbW9ub3NwYWNlOwoJZm9udC1zaXplOiAxMnB4OwoJd2hpdGUtc3BhY2U6IHByZS13cmFwOwoJYm94LXNpemluZzogYm9yZGVyLWJveDsKCXdpZHRoOjEwMCU7IAoJaGVpZ2h0OjEwMCU7CgltYXJnaW4tdG9wOjE1cHg7Cn0K
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/JsonpSample.html"
Content-Type: application/octet-stream; x-encoding=base64
PCFET0NUWVBFIGh0bWw+CjxodG1sPgo8aGVhZD4KCTxtZXRhIGh0dHAtZXF1aXY9IlgtVUEtQ29tcGF0aWJsZSIgY29udGVudD0iSUU9ZWRnZSxjaHJvbWU9MSIvPgoJPG1ldGEgaHR0cC1lcXVpdj0iQ29udGVudC1UeXBlIiBjb250ZW50PSJ0ZXh0L2h0bWw7IGNoYXJzZXQ9dXRmLTgiPgoJPG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+Cgk8bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWF4aW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iLz4KCTxtZXRhIG5hbWU9ImZvcm1hdC1kZXRlY3Rpb24iIGNvbnRlbnQ9InRlbGVwaG9uZT1ubyIvPgoJPG1ldGEgbmFtZT0iZ29vZ2xlIiB2YWx1ZT0ibm90cmFuc2xhdGUiLz4KCTx0aXRsZT5KU09OUCBTYW1wbGU8L3RpdGxlPgoJPCEtLSAtLT4KCTxzY3JpcHQgc3JjPSIuLi8uLi9lbnlvL2VueW8uanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8c2NyaXB0IHNyYz0iLi4vLi4vbGliL2xheW91dC9wYWNrYWdlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPHNjcmlwdCBzcmM9Ii4uLy4uL2xpYi9vbnl4L3BhY2thZ2UuanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8IS0tIC0tPgoJPGxpbmsgaHJlZj0iSnNvbnBTYW1wbGUuY3NzIiByZWw9InN0eWxlc2hlZXQiPgoJPHNjcmlwdCBzcmM9Ikpzb25wU2FtcGxlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPCEtLSAtLT4KPC9oZWFkPgo8Ym9keSBjbGFzcz0ib255eCI+CjxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij4KCW5ldyBlbnlvLnNhbXBsZS5Kc29ucFNhbXBsZSgpLnJlbmRlckludG8oZG9jdW1lbnQuYm9keSk7Cjwvc2NyaXB0Pgo8L2JvZHk+CjwvaHRtbD4=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/JsonpSample.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLnNhbXBsZS5Kc29ucFNhbXBsZSIsCglraW5kOiAiRml0dGFibGVSb3dzIiwKCWNsYXNzZXM6ICJlbnlvLWZpdCBqc29ucC1zYW1wbGUiLAoJY29tcG9uZW50czogWwoJCXtraW5kOiAiRml0dGFibGVDb2x1bW5zIiwgY2xhc3Nlczoib255eC10b29sYmFyLWlubGluZSIsIGNvbXBvbmVudHM6IFsKCQkJe2NvbnRlbnQ6ICJZUUw6ICJ9LAoJCQl7a2luZDogIm9ueXguSW5wdXQiLCBuYW1lOiJxdWVyeSIsIGZpdDp0cnVlLCB2YWx1ZTonc2VsZWN0ICogZnJvbSB1cGNvbWluZy5ldmVudHMgd2hlcmUgd29laWQgaW4gKHNlbGVjdCB3b2VpZCBmcm9tIGdlby5wbGFjZXMgd2hlcmUgdGV4dD0iU3Vubnl2YWxlLCBDQSIpJ30sCgkJCXtraW5kOiAib255eC5CdXR0b24iLCBjb250ZW50OiJGZXRjaCIsIG9udGFwOiJmZXRjaCJ9CgkJXX0sCgkJe2tpbmQ6ICJvbnl4LlRleHRBcmVhIiwgZml0OnRydWUsIGNsYXNzZXM6Impzb25wLXNhbXBsZS1zb3VyY2UifQoJXSwKCWZldGNoOiBmdW5jdGlvbigpIHsKCQl2YXIganNvbnAgPSBuZXcgZW55by5Kc29ucFJlcXVlc3QoewoJCQl1cmw6ICJodHRwOi8vcXVlcnkueWFob29hcGlzLmNvbS92MS9wdWJsaWMveXFsP2Zvcm1hdD1qc29uIiwKCQkJY2FsbGJhY2tOYW1lOiAiY2FsbGJhY2siCgkJfSk7CgkJLy8gc2VuZCBwYXJhbWV0ZXJzIHRoZSByZW1vdGUgc2VydmljZSB1c2luZyB0aGUgJ2dvKCknIG1ldGhvZAoJCWpzb25wLmdvKHsKCQkJcTogdGhpcy4kLnF1ZXJ5LmdldFZhbHVlKCkKCQl9KTsKCQkvLyBhdHRhY2ggcmVzcG9uZGVycyB0byB0aGUgdHJhbnNhY3Rpb24gb2JqZWN0CgkJanNvbnAucmVzcG9uc2UodGhpcywgInByb2Nlc3NSZXNwb25zZSIpOwoJfSwKCXByb2Nlc3NSZXNwb25zZTogZnVuY3Rpb24oaW5TZW5kZXIsIGluUmVzcG9uc2UpIHsKCQkvLyBkbyBzb21ldGhpbmcgd2l0aCBpdAoJCXRoaXMuJC50ZXh0QXJlYS5zZXRWYWx1ZShKU09OLnN0cmluZ2lmeShpblJlc3BvbnNlLCBudWxsLCAyKSk7Cgl9Cn0pOw==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/NestedRepeaterSample.css"
Content-Type: application/octet-stream; x-encoding=base64
Lm5lc3RlZC1yZXBlYXRlci1zYW1wbGUgewoJcGFkZGluZzoxNXB4Owp9Ci5uZXN0ZWQtcmVwZWF0ZXItc2FtcGxlLWl0ZW0gewoJcGFkZGluZzo0cHg7CgltYXJnaW46MnB4OwoJYm9yZGVyOjFweCBzb2xpZCBncmF5OwoJYmFja2dyb3VuZC1jb2xvcjogbGlnaHRncmF5Owp9
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/NestedRepeaterSample.html"
Content-Type: application/octet-stream; x-encoding=base64
PCFET0NUWVBFIGh0bWw+CjxodG1sPgo8aGVhZD4KCTxtZXRhIGh0dHAtZXF1aXY9IlgtVUEtQ29tcGF0aWJsZSIgY29udGVudD0iSUU9ZWRnZSxjaHJvbWU9MSIvPgoJPG1ldGEgaHR0cC1lcXVpdj0iQ29udGVudC1UeXBlIiBjb250ZW50PSJ0ZXh0L2h0bWw7IGNoYXJzZXQ9dXRmLTgiPgoJPG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+Cgk8bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWF4aW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iLz4KCTxtZXRhIG5hbWU9ImZvcm1hdC1kZXRlY3Rpb24iIGNvbnRlbnQ9InRlbGVwaG9uZT1ubyIvPgoJPG1ldGEgbmFtZT0iZ29vZ2xlIiB2YWx1ZT0ibm90cmFuc2xhdGUiLz4KCTx0aXRsZT5OZXN0ZWQgUmVwZWF0ZXIgU2FtcGxlPC90aXRsZT4KCTwhLS0gLS0+Cgk8c2NyaXB0IHNyYz0iLi4vLi4vZW55by9lbnlvLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPHNjcmlwdCBzcmM9Ii4uLy4uL2xpYi9sYXlvdXQvcGFja2FnZS5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTxzY3JpcHQgc3JjPSIuLi8uLi9saWIvb255eC9wYWNrYWdlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPCEtLSAtLT4KCTxsaW5rIGhyZWY9Ik5lc3RlZFJlcGVhdGVyU2FtcGxlLmNzcyIgcmVsPSJzdHlsZXNoZWV0Ij4KCTxzY3JpcHQgc3JjPSJOZXN0ZWRSZXBlYXRlclNhbXBsZS5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTwhLS0gLS0+CjwvaGVhZD4KPGJvZHkgY2xhc3M9Im9ueXgiPgo8c2NyaXB0IHR5cGU9InRleHQvamF2YXNjcmlwdCI+CgluZXcgZW55by5TY3JvbGxlcih7CgkJY29tcG9uZW50czogWyB7IGtpbmQ6ICJlbnlvLnNhbXBsZS5OZXN0ZWRSZXBlYXRlclNhbXBsZSIgfSBdCgl9KS5yZW5kZXJJbnRvKGRvY3VtZW50LmJvZHkpOwo8L3NjcmlwdD4KPC9ib2R5Pgo8L2h0bWw+
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/NestedRepeaterSample.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLnNhbXBsZS5OZXN0ZWRSZXBlYXRlclNhbXBsZSIsCgljbGFzc2VzOiAiZW55by1maXQgbmVzdGVkLXJlcGVhdGVyLXNhbXBsZSIsCgljb21wb25lbnRzOiBbCgkJe2tpbmQ6ICJSZXBlYXRlciIsIG5hbWU6ICJvdXRlciIsIG9uU2V0dXBJdGVtOiJzZXR1cEdyb3VwIiwgY291bnQ6IDMsIGNvbXBvbmVudHM6IFsKCQkJe2tpbmQ6ICJSZXBlYXRlciIsIG5hbWU6ICJpbm5lciIsIG9uU2V0dXBJdGVtOiJzZXR1cEl0ZW0iLCBjb21wb25lbnRzOiBbCgkJCQl7bmFtZToiaXRlbSIsIGNsYXNzZXM6Im5lc3RlZC1yZXBlYXRlci1zYW1wbGUtaXRlbSIsIGNvbXBvbmVudHM6IFsKCQkJCQl7dGFnOiJzcGFuIiwgbmFtZTogInBlcnNvbk51bWJlciJ9LAoJCQkJCXt0YWc6InNwYW4iLCBuYW1lOiAicGVyc29uTmFtZSJ9CgkJCQldfQoJCQldfQoJCV19CgldLAoJY3JlYXRlOiBmdW5jdGlvbigpIHsKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJfSwKCXNldHVwR3JvdXA6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJdmFyIGl0ZW0gPSBpbkV2ZW50Lml0ZW07CgkJaXRlbS4kLmlubmVyLnNldENvdW50KHRoaXMucGVvcGxlLmxlbmd0aCk7CgkJcmV0dXJuIHRydWU7Cgl9LAoJc2V0dXBJdGVtOiBmdW5jdGlvbihpblNlbmRlciwgaW5FdmVudCkgewoJCXZhciBncm91cCA9IGluRXZlbnQuaW5kaWNlc1sxXTsKCQl2YXIgaW5kZXggPSBpbkV2ZW50LmluZGV4OwoJCXZhciBpdGVtID0gaW5FdmVudC5pdGVtOwoJCXZhciBwZXJzb24gPSB0aGlzLnBlb3BsZVtpbmRleF07CgkJaXRlbS4kLnBlcnNvbk51bWJlci5zZXRDb250ZW50KChncm91cCArIDEpICsgIjoiICsgKGluZGV4KzEpICsgIi4gIik7CgkJaXRlbS4kLnBlcnNvbk5hbWUuc2V0Q29udGVudChwZXJzb24ubmFtZSk7CgkJaXRlbS4kLnBlcnNvbk5hbWUuYXBwbHlTdHlsZSgiY29sb3IiLCBwZXJzb24uc2V4ID09ICJtYWxlIiA/ICJkb2RnZXJibHVlIiA6ICJkZWVwcGluayIpOwoJCS8qIHN0b3AgcHJvcGFnYXRpb24gKi8KCQlyZXR1cm4gdHJ1ZTsKCX0sCglwZW9wbGU6IFsKCQl7bmFtZTogIkFuZHJldyIsIHNleDoibWFsZSJ9LAoJCXtuYW1lOiAiQmV0dHkiLCBzZXg6ImZlbWFsZSJ9LAoJCXtuYW1lOiAiQ2hyaXN0b3BoZXIiLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIkRvbm5hIiwgc2V4OiJmZW1hbGUifSwKCQl7bmFtZTogIkVwaHJhaW0iLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIkZyYW5raWUiLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIkdlcmFsZCIsIHNleDoibWFsZSJ9LAoJCXtuYW1lOiAiSGVhdGhlciIsIHNleDoiZmVtYWxlIn0sCgkJe25hbWU6ICJJbmdyZWQiLCBzZXg6ImZlbWFsZSJ9LAoJCXtuYW1lOiAiSmFjayIsIHNleDoibWFsZSJ9LAoJCXtuYW1lOiAiS2V2aW4iLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIkx1Y3kiLCBzZXg6ImZlbWFsZSJ9LAoJCXtuYW1lOiAiTWF0dGhldyIsIHNleDoibWFsZSJ9LAoJCXtuYW1lOiAiTm9yZWVuIiwgc2V4OiJmZW1hbGUifSwKCQl7bmFtZTogIk9zY2FyIiwgc2V4OiJtYWxlIn0sCgkJe25hbWU6ICJQZWRybyIsIHNleDoibWFsZSJ9LAoJCXtuYW1lOiAiUXVlbnRpbiIsIHNleDoibWFsZSJ9LAoJCXtuYW1lOiAiUmFscGgiLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIlN0ZXZlbiIsIHNleDoibWFsZSJ9LAoJCXtuYW1lOiAiVHJhY3kiLCBzZXg6ImZlbWFsZSJ9LAoJCXtuYW1lOiAiVW1hIiwgc2V4OiJmZW1hbGUifSwKCQl7bmFtZTogIlZpY3RvciIsIHNleDoibWFsZSJ9LAoJCXtuYW1lOiAiV2VuZHkiLCBzZXg6ImZlbWFsZSJ9LAoJCXtuYW1lOiAiWGluIiwgc2V4OiJtYWxlIn0sCgkJe25hbWU6ICJZdWxpYSIsIHNleDoiZmVtYWxlIn0sCgkJe25hbWU6ICJab2x0YW4ifQoJXQp9KTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/PlatformSample.css"
Content-Type: application/octet-stream; x-encoding=base64
LnBsYXRmb3JtLXNhbXBsZSB7CglwYWRkaW5nOiAxNXB4Owp9Ci5wbGF0Zm9ybS1zYW1wbGUtZGl2aWRlciB7Cgljb2xvcjogI0Y0OTIwMDsKCXRleHQtdHJhbnNmb3JtOiB1cHBlcmNhc2U7Cglmb250LWZhbWlseTogU2Vnb2UgVUksIFByZWx1ZGUgTWVkaXVtLCBIZWx2ZXRpY2EsIFZlcmRhbmEsIHNhbnMtc2VyaWY7Cglmb250LXNpemU6IDE0cHg7Cglmb250LXdlaWdodDogYm9sZDsKCW1hcmdpbi1ib3R0b206IDhweDsKfQ==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/PlatformSample.html"
Content-Type: application/octet-stream; x-encoding=base64
PCFET0NUWVBFIGh0bWw+CjxodG1sPgo8aGVhZD4KCTxtZXRhIGh0dHAtZXF1aXY9IlgtVUEtQ29tcGF0aWJsZSIgY29udGVudD0iSUU9ZWRnZSxjaHJvbWU9MSIvPgoJPG1ldGEgaHR0cC1lcXVpdj0iQ29udGVudC1UeXBlIiBjb250ZW50PSJ0ZXh0L2h0bWw7IGNoYXJzZXQ9dXRmLTgiPgoJPG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+Cgk8bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWF4aW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iLz4KCTxtZXRhIG5hbWU9ImZvcm1hdC1kZXRlY3Rpb24iIGNvbnRlbnQ9InRlbGVwaG9uZT1ubyIvPgoJPG1ldGEgbmFtZT0iZ29vZ2xlIiB2YWx1ZT0ibm90cmFuc2xhdGUiLz4KCTx0aXRsZT5QbGF0Zm9ybSBTYW1wbGU8L3RpdGxlPgoJPCEtLSAtLT4KCTxzY3JpcHQgc3JjPSIuLi8uLi9lbnlvL2VueW8uanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8c2NyaXB0IHNyYz0iLi4vLi4vbGliL2xheW91dC9wYWNrYWdlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPHNjcmlwdCBzcmM9Ii4uLy4uL2xpYi9vbnl4L3BhY2thZ2UuanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8IS0tIC0tPgoJPGxpbmsgaHJlZj0iUGxhdGZvcm1TYW1wbGUuY3NzIiByZWw9InN0eWxlc2hlZXQiPgoJPHNjcmlwdCBzcmM9IlBsYXRmb3JtU2FtcGxlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPCEtLSAtLT4KPC9oZWFkPgo8Ym9keSBjbGFzcz0ib255eCI+CjxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij4KCW5ldyBlbnlvLnNhbXBsZS5QbGF0Zm9ybVNhbXBsZSgpLnJlbmRlckludG8oZG9jdW1lbnQuYm9keSk7Cjwvc2NyaXB0Pgo8L2JvZHk+CjwvaHRtbD4=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/PlatformSample.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLnNhbXBsZS5QbGF0Zm9ybVNhbXBsZSIsCglraW5kOiAiRml0dGFibGVSb3dzIiwKCWNsYXNzZXM6ICJlbnlvLWZpdCBwbGF0Zm9ybS1zYW1wbGUiLAoJY29tcG9uZW50czogWwoJCXtjbGFzc2VzOiAicGxhdGZvcm0tc2FtcGxlLWRpdmlkZXIiLCBjb250ZW50OiAiRW55byBQbGF0Zm9ybSBEZXRlY3Rpb24ifSwKCQl7a2luZDogIm9ueXguR3JvdXBib3giLCBjb21wb25lbnRzOiBbCgkJCXtraW5kOiAib255eC5Hcm91cGJveEhlYWRlciIsIGNvbnRlbnQ6ICJVc2VyLUFnZW50IFN0cmluZyJ9LAoJCQl7bmFtZTogInVhU3RyaW5nIiwgY29udGVudDogIiIsIHN0eWxlOiAicGFkZGluZzogOHB4OyJ9CgkJXX0sCgkJe3RhZzogImJyIn0sCgkJe2tpbmQ6ICJvbnl4Lkdyb3VwYm94IiwgY29tcG9uZW50czogWwoJCQl7a2luZDogIm9ueXguR3JvdXBib3hIZWFkZXIiLCBjb250ZW50OiAiV2luZG93In0sCgkJCXtuYW1lOiAid2luZG93QXR0ciIsIGNvbnRlbnQ6ICIiLCBzdHlsZTogInBhZGRpbmc6IDhweDsifQoJCV19LAoJCXt0YWc6ICJiciJ9LAoJCXtraW5kOiAib255eC5Hcm91cGJveCIsIGNvbXBvbmVudHM6IFsKCQkJe2tpbmQ6ICJvbnl4Lkdyb3VwYm94SGVhZGVyIiwgY29udGVudDogImVueW8ucGxhdGZvcm0ifSwKCQkJe25hbWU6ICJlbnlvUGxhdGZvcm1KU09OIiwgY29udGVudDogIiIsIHN0eWxlOiAicGFkZGluZzogOHB4OyJ9CgkJXX0KCV0sCgl1cGRhdGVXaW5kb3dTaXplOiBmdW5jdGlvbigpIHsKCQl2YXIgd2lkdGggPSB3aW5kb3cuaW5uZXJXaWR0aDsKCQlpZiAod2lkdGggPT09IHVuZGVmaW5lZCkgewoJCQl3aWR0aCA9IGRvY3VtZW50LmRvY3VtZW50RWxlbWVudC5jbGllbnRXaWR0aDsKCQl9CgkJdmFyIGhlaWdodCA9IHdpbmRvdy5pbm5lckhlaWdodDsKCQlpZiAoaGVpZ2h0ID09PSB1bmRlZmluZWQpIHsKCQkJaGVpZ2h0ID0gZG9jdW1lbnQuZG9jdW1lbnRFbGVtZW50LmNsaWVudEhlaWdodDsKCQl9CgkJdGhpcy4kLndpbmRvd0F0dHIuc2V0Q29udGVudCgic2l6ZTogIiArIHdpZHRoICsgIngiICsgaGVpZ2h0ICsKCQkJIiwgZGV2aWNlUGl4ZWxSYXRpbzogIiArIHdpbmRvdy5kZXZpY2VQaXhlbFJhdGlvKTsKCX0sCgljcmVhdGU6IGZ1bmN0aW9uKCkgewoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7CgkJdGhpcy4kLnVhU3RyaW5nLnNldENvbnRlbnQobmF2aWdhdG9yLnVzZXJBZ2VudCk7CgkJdGhpcy4kLmVueW9QbGF0Zm9ybUpTT04uc2V0Q29udGVudChKU09OLnN0cmluZ2lmeShlbnlvLnBsYXRmb3JtLCBudWxsLCAxKSk7CgkJdGhpcy51cGRhdGVXaW5kb3dTaXplKCk7Cgl9LAoJcmVzaXplSGFuZGxlcjogZnVuY3Rpb24oKSB7CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCQl0aGlzLnVwZGF0ZVdpbmRvd1NpemUoKTsKCX0KfSk7
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/RepeaterSample.css"
Content-Type: application/octet-stream; x-encoding=base64
LnJlcGVhdGVyLXNhbXBsZSB7CglwYWRkaW5nOjE1cHg7Cn0KLnJlcGVhdGVyLXNhbXBsZS1pdGVtIHsKCXBhZGRpbmc6NHB4OwoJbWFyZ2luOjJweDsKCWJvcmRlcjoxcHggc29saWQgZ3JheTsKCWJhY2tncm91bmQtY29sb3I6IGxpZ2h0Z3JheTsKfQ==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/RepeaterSample.html"
Content-Type: application/octet-stream; x-encoding=base64
PCFET0NUWVBFIGh0bWw+CjxodG1sPgo8aGVhZD4KCTxtZXRhIGh0dHAtZXF1aXY9IlgtVUEtQ29tcGF0aWJsZSIgY29udGVudD0iSUU9ZWRnZSxjaHJvbWU9MSIvPgoJPG1ldGEgaHR0cC1lcXVpdj0iQ29udGVudC1UeXBlIiBjb250ZW50PSJ0ZXh0L2h0bWw7IGNoYXJzZXQ9dXRmLTgiPgoJPG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+Cgk8bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWF4aW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iLz4KCTxtZXRhIG5hbWU9ImZvcm1hdC1kZXRlY3Rpb24iIGNvbnRlbnQ9InRlbGVwaG9uZT1ubyIvPgoJPG1ldGEgbmFtZT0iZ29vZ2xlIiB2YWx1ZT0ibm90cmFuc2xhdGUiLz4KCTx0aXRsZT5SZXBlYXRlciBTYW1wbGU8L3RpdGxlPgoJPCEtLSAtLT4KCTxzY3JpcHQgc3JjPSIuLi8uLi9lbnlvL2VueW8uanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8c2NyaXB0IHNyYz0iLi4vLi4vbGliL2xheW91dC9wYWNrYWdlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPHNjcmlwdCBzcmM9Ii4uLy4uL2xpYi9vbnl4L3BhY2thZ2UuanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8IS0tIC0tPgoJPGxpbmsgaHJlZj0iUmVwZWF0ZXJTYW1wbGUuY3NzIiByZWw9InN0eWxlc2hlZXQiPgoJPHNjcmlwdCBzcmM9IlJlcGVhdGVyU2FtcGxlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPCEtLSAtLT4KPC9oZWFkPgo8Ym9keSBjbGFzcz0ib255eCI+CjxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij4KCW5ldyBlbnlvLlNjcm9sbGVyKHsKCQljb21wb25lbnRzOiBbIHsga2luZDogImVueW8uc2FtcGxlLlJlcGVhdGVyU2FtcGxlIiB9IF0KCX0pLnJlbmRlckludG8oZG9jdW1lbnQuYm9keSk7Cjwvc2NyaXB0Pgo8L2JvZHk+CjwvaHRtbD4=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/RepeaterSample.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLnNhbXBsZS5SZXBlYXRlclNhbXBsZSIsCgljbGFzc2VzOiAiZW55by1maXQgcmVwZWF0ZXItc2FtcGxlIiwKCWNvbXBvbmVudHM6IFsKCQl7a2luZDogIlJlcGVhdGVyIiwgb25TZXR1cEl0ZW06InNldHVwSXRlbSIsIGNvbXBvbmVudHM6IFsKCQkJe25hbWU6Iml0ZW0iLCBjbGFzc2VzOiJyZXBlYXRlci1zYW1wbGUtaXRlbSIsIGNvbXBvbmVudHM6IFsKCQkJCXt0YWc6InNwYW4iLCBuYW1lOiAicGVyc29uTnVtYmVyIn0sCgkJCQl7dGFnOiJzcGFuIiwgbmFtZTogInBlcnNvbk5hbWUifQoJCQldfQoJCV19CgldLAoJY3JlYXRlOiBmdW5jdGlvbigpIHsKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCXRoaXMuJC5yZXBlYXRlci5zZXRDb3VudCh0aGlzLnBlb3BsZS5sZW5ndGgpOwoJfSwKCXNldHVwSXRlbTogZnVuY3Rpb24oaW5TZW5kZXIsIGluRXZlbnQpIHsKCQl2YXIgaW5kZXggPSBpbkV2ZW50LmluZGV4OwoJCXZhciBpdGVtID0gaW5FdmVudC5pdGVtOwoJCXZhciBwZXJzb24gPSB0aGlzLnBlb3BsZVtpbmRleF07CgkJaXRlbS4kLnBlcnNvbk51bWJlci5zZXRDb250ZW50KChpbmRleCsxKSArICIuICIpOwoJCWl0ZW0uJC5wZXJzb25OYW1lLnNldENvbnRlbnQocGVyc29uLm5hbWUpOwoJCWl0ZW0uJC5wZXJzb25OYW1lLmFwcGx5U3R5bGUoImNvbG9yIiwgcGVyc29uLnNleCA9PSAibWFsZSIgPyAiZG9kZ2VyYmx1ZSIgOiAiZGVlcHBpbmsiKTsKCQkvKiBzdG9wIHByb3BhZ2F0aW9uICovCgkJcmV0dXJuIHRydWU7Cgl9LAoJcGVvcGxlOiBbCgkJe25hbWU6ICJBbmRyZXciLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIkJldHR5Iiwgc2V4OiJmZW1hbGUifSwKCQl7bmFtZTogIkNocmlzdG9waGVyIiwgc2V4OiJtYWxlIn0sCgkJe25hbWU6ICJEb25uYSIsIHNleDoiZmVtYWxlIn0sCgkJe25hbWU6ICJFcGhyYWltIiwgc2V4OiJtYWxlIn0sCgkJe25hbWU6ICJGcmFua2llIiwgc2V4OiJtYWxlIn0sCgkJe25hbWU6ICJHZXJhbGQiLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIkhlYXRoZXIiLCBzZXg6ImZlbWFsZSJ9LAoJCXtuYW1lOiAiSW5ncmVkIiwgc2V4OiJmZW1hbGUifSwKCQl7bmFtZTogIkphY2siLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIktldmluIiwgc2V4OiJtYWxlIn0sCgkJe25hbWU6ICJMdWN5Iiwgc2V4OiJmZW1hbGUifSwKCQl7bmFtZTogIk1hdHRoZXciLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIk5vcmVlbiIsIHNleDoiZmVtYWxlIn0sCgkJe25hbWU6ICJPc2NhciIsIHNleDoibWFsZSJ9LAoJCXtuYW1lOiAiUGVkcm8iLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIlF1ZW50aW4iLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIlJhbHBoIiwgc2V4OiJtYWxlIn0sCgkJe25hbWU6ICJTdGV2ZW4iLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIlRyYWN5Iiwgc2V4OiJmZW1hbGUifSwKCQl7bmFtZTogIlVtYSIsIHNleDoiZmVtYWxlIn0sCgkJe25hbWU6ICJWaWN0b3IiLCBzZXg6Im1hbGUifSwKCQl7bmFtZTogIldlbmR5Iiwgc2V4OiJmZW1hbGUifSwKCQl7bmFtZTogIlhpbiIsIHNleDoibWFsZSJ9LAoJCXtuYW1lOiAiWXVsaWEiLCBzZXg6ImZlbWFsZSJ9LAoJCXtuYW1lOiAiWm9sdGFuIn0KCV0KfSk7
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/ScrollerSample.css"
Content-Type: application/octet-stream; x-encoding=base64
LnNjcm9sbGVyLXNhbXBsZS1wYW5lbHMgewoJbWFyZ2luOjE1cHg7Cn0KLnNjcm9sbGVyLXNhbXBsZS1jb250ZW50IHsKCXdpZHRoOjQwMDBweDsKfQouc2Nyb2xsZXItc2FtcGxlLXNjcm9sbGVyIHsKCWJvcmRlcjogMXB4IHNvbGlkIG9yYW5nZTsKfQ==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/ScrollerSample.html"
Content-Type: application/octet-stream; x-encoding=base64
PCFET0NUWVBFIGh0bWw+CjxodG1sPgo8aGVhZD4KCTxtZXRhIGh0dHAtZXF1aXY9IlgtVUEtQ29tcGF0aWJsZSIgY29udGVudD0iSUU9ZWRnZSxjaHJvbWU9MSIvPgoJPG1ldGEgaHR0cC1lcXVpdj0iQ29udGVudC1UeXBlIiBjb250ZW50PSJ0ZXh0L2h0bWw7IGNoYXJzZXQ9dXRmLTgiPgoJPG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+Cgk8bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWF4aW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iLz4KCTxtZXRhIG5hbWU9ImZvcm1hdC1kZXRlY3Rpb24iIGNvbnRlbnQ9InRlbGVwaG9uZT1ubyIvPgoJPG1ldGEgbmFtZT0iZ29vZ2xlIiB2YWx1ZT0ibm90cmFuc2xhdGUiLz4KCTx0aXRsZT5TY3JvbGxlciBTYW1wbGU8L3RpdGxlPgoJPCEtLSAtLT4KCTxzY3JpcHQgc3JjPSIuLi8uLi9lbnlvL2VueW8uanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8c2NyaXB0IHNyYz0iLi4vLi4vbGliL2xheW91dC9wYWNrYWdlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPHNjcmlwdCBzcmM9Ii4uLy4uL2xpYi9vbnl4L3BhY2thZ2UuanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8IS0tIC0tPgoJPGxpbmsgaHJlZj0iU2Nyb2xsZXJTYW1wbGUuY3NzIiByZWw9InN0eWxlc2hlZXQiPgoJPHNjcmlwdCBzcmM9IlNjcm9sbGVyU2FtcGxlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPCEtLSAtLT4KPC9oZWFkPgo8Ym9keSBjbGFzcz0ib255eCI+CjxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij4KCW5ldyBlbnlvLnNhbXBsZS5TY3JvbGxlclNhbXBsZSgpLnJlbmRlckludG8oZG9jdW1lbnQuYm9keSk7Cjwvc2NyaXB0Pgo8L2JvZHk+CjwvaHRtbD4=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/ScrollerSample.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLnNhbXBsZS5TY3JvbGxlclNhbXBsZSIsCglraW5kOiAiRml0dGFibGVSb3dzIiwKCWNsYXNzZXM6ICJlbnlvLWZpdCAgZW55by11bnNlbGVjdGFibGUiLAoJY29tcG9uZW50czogWwoJCXtraW5kOiAib255eC5Ub29sYmFyIiwgY29tcG9uZW50czogWwoJCQl7a2luZDogIm9ueXguUGlja2VyRGVjb3JhdG9yIiwgY29tcG9uZW50czogWwoJCQkJe2NvbnRlbnQ6IkNob29zZSBTY3JvbGxlciIsIHN0eWxlOiJ3aWR0aDoyNTBweDsifSwKCQkJCXtraW5kOiAib255eC5QaWNrZXIiLCBmbG9hdGluZzp0cnVlLCBtYXhIZWlnaHQ6IDMwMCwgb25TZWxlY3Q6InNhbXBsZUNoYW5nZWQiLCBjb21wb25lbnRzOiBbCgkJCQkJe2NvbnRlbnQ6IkRlZmF1bHQgc2Nyb2xsZXIiLCBhY3RpdmU6dHJ1ZX0sCgkJCQkJe2NvbnRlbnQ6IkZvcmNlIHRvdWNoIHNjcm9sbGVyIn0sCgkJCQkJe2NvbnRlbnQ6Ikhvcml6b250YWwgb25seSJ9LAoJCQkJCXtjb250ZW50OiJWZXJ0aWNhbCBvbmx5In0sCgkJCQkJe2NvbnRlbnQ6IkZvcmNlIFRvdWNoU2Nyb2xsU3RyYXRlZ3kifSwKCQkJCQl7Y29udGVudDoiRm9yY2UgVHJhbnNpdGlvblNjcm9sbFN0cmF0ZWd5In0sCgkJCQkJe2NvbnRlbnQ6IkZvcmNlIFRyYW5zbGF0ZVNjcm9sbFN0cmF0ZWd5In0KCQkJCV19CgkJCV19CgkJXX0sCgkJe2tpbmQ6ICJQYW5lbHMiLCBmaXQ6IHRydWUsIGRyYWdnYWJsZTogZmFsc2UsIGNsYXNzZXM6ICJzY3JvbGxlci1zYW1wbGUtcGFuZWxzIiwgY29tcG9uZW50czogWwoJCQkvLyBEZWZhdWx0IHNjcm9sbGVyIChjaG9vc2VzIGJlc3Qgc2Nyb2xsaW5nIG1ldGhvZCBmb3IgcGxhdGZvcm0pCgkJCXtraW5kOiAiU2Nyb2xsZXIiLCBjbGFzc2VzOiAic2Nyb2xsZXItc2FtcGxlLXNjcm9sbGVyIGVueW8tZml0In0sCgkJCS8vIEZvcmNlcyB0b3VjaCBzY3JvbGxpbmcsIGV2ZW4gb24gZGVza3RvcAoJCQl7a2luZDogIlNjcm9sbGVyIiwgdG91Y2g6dHJ1ZSwgY2xhc3NlczogInNjcm9sbGVyLXNhbXBsZS1zY3JvbGxlciBlbnlvLWZpdCJ9LAoJCQkvLyBIb3Jpem9udGFsLW9ubHkgc2Nyb2xsaW5nCgkJCXtraW5kOiAiU2Nyb2xsZXIiLCB2ZXJ0aWNhbDoiaGlkZGVuIiwgY2xhc3NlczogInNjcm9sbGVyLXNhbXBsZS1zY3JvbGxlciBlbnlvLWZpdCJ9LAoJCQkvLyBWZXJ0aWNhbC1vbmx5IHNjcm9sbGluZwoJCQl7a2luZDogIlNjcm9sbGVyIiwgaG9yaXpvbnRhbDoiaGlkZGVuIiwgY2xhc3NlczogInNjcm9sbGVyLXNhbXBsZS1zY3JvbGxlciBlbnlvLWZpdCIsIG9ubW91c2Vkb3duOiAibW91c2VEb3duIiwgb25kcmFnc3RhcnQ6ICJkcmFnU3RhcnQifSwKCQkJLy8gRm9yY2Ugc3BlY2lmaWMgc3RyYXRlZ2llcwoJCQl7a2luZDogIlNjcm9sbGVyIiwgY2xhc3NlczogInNjcm9sbGVyLXNhbXBsZS1zY3JvbGxlciBlbnlvLWZpdCIsIHN0cmF0ZWd5S2luZDogIlRvdWNoU2Nyb2xsU3RyYXRlZ3kifSwKCQkJe2tpbmQ6ICJTY3JvbGxlciIsIGNsYXNzZXM6ICJzY3JvbGxlci1zYW1wbGUtc2Nyb2xsZXIgZW55by1maXQiLCBzdHJhdGVneUtpbmQ6ICJUcmFuc2l0aW9uU2Nyb2xsU3RyYXRlZ3kifSwKCQkJe2tpbmQ6ICJTY3JvbGxlciIsIGNsYXNzZXM6ICJzY3JvbGxlci1zYW1wbGUtc2Nyb2xsZXIgZW55by1maXQiLCBzdHJhdGVneUtpbmQ6ICJUcmFuc2xhdGVTY3JvbGxTdHJhdGVneSJ9CgkJXX0KCV0sCgljcmVhdGU6IGZ1bmN0aW9uKCkgewoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7CgkJdmFyIHNjcm9sbGVycyA9IHRoaXMuJC5wYW5lbHMuZ2V0UGFuZWxzKCk7CgkJZm9yICh2YXIgaSBpbiBzY3JvbGxlcnMpIHsKCQkJc2Nyb2xsZXJzW2ldLmNyZWF0ZUNvbXBvbmVudCh7CgkJCQlhbGxvd0h0bWw6dHJ1ZSwKCQkJCWNvbnRlbnQ6dGhpcy50ZXh0LAoJCQkJY2xhc3Nlczoic2Nyb2xsZXItc2FtcGxlLWNvbnRlbnQiCgkJCX0pOwoJCX0KCX0sCglzYW1wbGVDaGFuZ2VkOiBmdW5jdGlvbihpblNlbmRlciwgaW5FdmVudCkgewoJCXRoaXMuJC5wYW5lbHMuc2V0SW5kZXgoaW5FdmVudC5zZWxlY3RlZC5pbmRleEluQ29udGFpbmVyKCktMSk7Cgl9LAoJdGV4dDogIkZvbzxicj5CYXI8YnI+QmFyPGJyPkJvb20gYm9vbSBwb3c8YnI+Rm9vPGJyPkJhcjxicj5Cb29tIGJvb20gcG93PGJyPkZvbzxicj5CYXI8YnI+Qm9vbSBib29tIHBvdzxicj4iICsKCQkiRm9vPGJyPkJhcjxicj5Cb29tIGJvb20gcG93PGJyPkZvbzxicj5CYXI8YnI+Qm9vbSBib29tIHBvdzxicj5Gb288YnI+QmFyPGJyPkJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuICIgKwoJCSJCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiAiICsKCQkiQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gIiArCgkJIkJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIDxicj5Gb288YnI+QmFyPGJyPkJhcjxicj4iICsKCQkiQm9vbSBib29tIHBvdzxicj5Gb288YnI+QmFyPGJyPkJvb20gYm9vbSBwb3c8YnI+Rm9vPGJyPkJhcjxicj5Cb29tIGJvb20gcG93PGJyPkZvbzxicj5CYXI8YnI+Qm9vbSBib29tIHBvdzxicj4iICsKCQkiRm9vPGJyPkJhcjxicj5Cb29tIGJvb20gcG93PGJyPkZvbzxicj5CYXI8YnI+Qm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gIiArCgkJIkJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuICIgKwoJCSJCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiAiICsKCQkiQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gPGJyPkZvbzxicj5CYXI8YnI+QmFyPGJyPkJvb20gYm9vbSBwb3c8YnI+Rm9vPGJyPkJhcjxicj4iICsKCQkiQm9vbSBib29tIHBvdzxicj5Gb288YnI+QmFyPGJyPkJvb20gYm9vbSBwb3c8YnI+Rm9vPGJyPkJhcjxicj5Cb29tIGJvb20gcG93PGJyPkZvbzxicj5CYXI8YnI+Qm9vbSBib29tIHBvdzxicj4iICsKCQkiRm9vPGJyPkJhcjxicj5Cb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiAiICsKCQkiQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gIiArCgkJIkJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuICIgKwoJCSJCb29tIGJvb20gcG93LiA8YnI+Rm9vPGJyPkJhcjxicj5CYXI8YnI+Qm9vbSBib29tIHBvdzxicj5Gb288YnI+QmFyPGJyPkJvb20gYm9vbSBwb3c8YnI+Rm9vPGJyPkJhcjxicj4iICsKCQkiQm9vbSBib29tIHBvdzxicj5Gb288YnI+QmFyPGJyPkJvb20gYm9vbSBwb3c8YnI+Rm9vPGJyPkJhcjxicj5Cb29tIGJvb20gcG93PGJyPkZvbzxicj5CYXI8YnI+Qm9vbSBib29tIHBvdy4gIiArCgkJIkJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuIEJvb20gYm9vbSBwb3cuICIgKwoJCSJCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiBCb29tIGJvb20gcG93LiAiICsKCQkiQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gQm9vbSBib29tIHBvdy4gPGJyPiIsCgkvLyBUaGUgZm9sbG93aW5nIGFyZSB1c2VkIHdoZW4gdGhpcyBzYW1wbGUgaXMgY2FsbGVkIGZyb20gdGhlIFNhbXBsZXIgYXBwCgltb3VzZURvd246IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJaW5FdmVudC5wcmV2ZW50RGVmYXVsdCgpOwoJfSwKCWRyYWdTdGFydDogZnVuY3Rpb24oaW5TZW5kZXIsIGluRXZlbnQpIHsKCQlpZiAoaW5FdmVudC5ob3Jpem9udGFsKSB7CgkJCS8vIFByZXZlbnQgZHJhZyBwcm9wYWdhdGlvbiBvbiBob3Jpem9udGFsIGRyYWcgZXZlbnRzCgkJCXJldHVybiB0cnVlOwoJCX0KCX0KfSk7
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/WebServiceSample.css"
Content-Type: application/octet-stream; x-encoding=base64
LndlYnNlcnZpY2Utc2FtcGxlIHsKCXBhZGRpbmc6IDE1cHg7Cn0KCi53ZWJzZXJ2aWNlLXNhbXBsZS1zb3VyY2UgewoJcGFkZGluZzoxNXB4OwoJZm9udC1mYW1pbHk6IG1vbm9zcGFjZTsKCWZvbnQtc2l6ZTogMTJweDsKCXdoaXRlLXNwYWNlOiBwcmUtd3JhcDsKCWJveC1zaXppbmc6IGJvcmRlci1ib3g7Cgl3aWR0aDoxMDAlOyAKCWhlaWdodDoxMDAlOwoJbWFyZ2luLXRvcDoxNXB4Owp9Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/WebServiceSample.html"
Content-Type: application/octet-stream; x-encoding=base64
PCFET0NUWVBFIGh0bWw+CjxodG1sPgo8aGVhZD4KCTxtZXRhIGh0dHAtZXF1aXY9IlgtVUEtQ29tcGF0aWJsZSIgY29udGVudD0iSUU9ZWRnZSxjaHJvbWU9MSIvPgoJPG1ldGEgaHR0cC1lcXVpdj0iQ29udGVudC1UeXBlIiBjb250ZW50PSJ0ZXh0L2h0bWw7IGNoYXJzZXQ9dXRmLTgiPgoJPG1ldGEgbmFtZT0iYXBwbGUtbW9iaWxlLXdlYi1hcHAtY2FwYWJsZSIgY29udGVudD0ieWVzIi8+Cgk8bWV0YSBuYW1lPSJ2aWV3cG9ydCIgY29udGVudD0id2lkdGg9ZGV2aWNlLXdpZHRoLCBpbml0aWFsLXNjYWxlPTEuMCwgbWF4aW11bS1zY2FsZT0xLjAsIHVzZXItc2NhbGFibGU9bm8iLz4KCTxtZXRhIG5hbWU9ImZvcm1hdC1kZXRlY3Rpb24iIGNvbnRlbnQ9InRlbGVwaG9uZT1ubyIvPgoJPG1ldGEgbmFtZT0iZ29vZ2xlIiB2YWx1ZT0ibm90cmFuc2xhdGUiLz4KCTx0aXRsZT5XZWJTZXJ2aWNlIFNhbXBsZTwvdGl0bGU+Cgk8IS0tIC0tPgoJPHNjcmlwdCBzcmM9Ii4uLy4uL2VueW8vZW55by5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTxzY3JpcHQgc3JjPSIuLi8uLi9saWIvbGF5b3V0L3BhY2thZ2UuanMiIHR5cGU9InRleHQvamF2YXNjcmlwdCI+PC9zY3JpcHQ+Cgk8c2NyaXB0IHNyYz0iLi4vLi4vbGliL29ueXgvcGFja2FnZS5qcyIgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij48L3NjcmlwdD4KCTwhLS0gLS0+Cgk8bGluayBocmVmPSJXZWJTZXJ2aWNlU2FtcGxlLmNzcyIgcmVsPSJzdHlsZXNoZWV0Ij4KCTxzY3JpcHQgc3JjPSJXZWJTZXJ2aWNlU2FtcGxlLmpzIiB0eXBlPSJ0ZXh0L2phdmFzY3JpcHQiPjwvc2NyaXB0PgoJPCEtLSAtLT4KPC9oZWFkPgo8Ym9keSBjbGFzcz0ib255eCI+CjxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0Ij4KCW5ldyBlbnlvLnNhbXBsZS5XZWJTZXJ2aWNlU2FtcGxlKCkucmVuZGVySW50byhkb2N1bWVudC5ib2R5KTsKPC9zY3JpcHQ+CjwvYm9keT4KPC9odG1sPg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/WebServiceSample.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLnNhbXBsZS5XZWJTZXJ2aWNlU2FtcGxlIiwKCWtpbmQ6ICJGaXR0YWJsZVJvd3MiLAoJY2xhc3NlczogImVueW8tZml0IHdlYnNlcnZpY2Utc2FtcGxlIiwKCWNvbXBvbmVudHM6IFsKCQl7a2luZDogIldlYlNlcnZpY2UiLCBuYW1lOiJ5cWwiLCB1cmw6ICJodHRwOi8vcXVlcnkueWFob29hcGlzLmNvbS92MS9wdWJsaWMveXFsP2Zvcm1hdD1qc29uIiwgb25SZXNwb25zZToicHJvY2Vzc1Jlc3BvbnNlIiwgY2FsbGJhY2tOYW1lOiAiY2FsbGJhY2sifSwKCQl7a2luZDogIkZpdHRhYmxlQ29sdW1ucyIsIGNsYXNzZXM6Im9ueXgtdG9vbGJhci1pbmxpbmUiLCBjb21wb25lbnRzOiBbCgkJCXtjb250ZW50OiAiWVFMOiAifSwKCQkJe2tpbmQ6ICJvbnl4LklucHV0IiwgbmFtZToicXVlcnkiLCBmaXQ6dHJ1ZSwgdmFsdWU6J3NlbGVjdCAqIGZyb20gdXBjb21pbmcuZXZlbnRzIHdoZXJlIHdvZWlkIGluIChzZWxlY3Qgd29laWQgZnJvbSBnZW8ucGxhY2VzIHdoZXJlIHRleHQ9IlN1bm55dmFsZSwgQ0EiKSd9LAoJCQl7a2luZDogIm9ueXguUGlja2VyRGVjb3JhdG9yIiwgY29tcG9uZW50czogWwoJCQkJe2NvbnRlbnQ6IkNob29zZSBTY3JvbGxlciIsIHN0eWxlOiJ3aWR0aDoxMDBweDsifSwKCQkJCXtraW5kOiAib255eC5QaWNrZXIiLCBmbG9hdGluZzp0cnVlLCBjb21wb25lbnRzOiBbCgkJCQkJe2NvbnRlbnQ6IkFKQVgiLCBhY3RpdmU6dHJ1ZX0sCgkJCQkJe2NvbnRlbnQ6IkpTT04tUCJ9CgkJCQldfQoJCQldfSwKCQkJe2tpbmQ6ICJvbnl4LkJ1dHRvbiIsIGNvbnRlbnQ6IkZldGNoIiwgb250YXA6ImZldGNoIn0KCQldfSwKCQl7a2luZDogIm9ueXguVGV4dEFyZWEiLCBmaXQ6dHJ1ZSwgY2xhc3Nlczoid2Vic2VydmljZS1zYW1wbGUtc291cmNlIn0KCV0sCglmZXRjaDogZnVuY3Rpb24oKSB7CgkJLy8gc2VuZCBwYXJhbWV0ZXJzIHRoZSByZW1vdGUgc2VydmljZSB1c2luZyB0aGUgJ3NlbmQoKScgbWV0aG9kCgkJdGhpcy4kLnlxbC5zZW5kKHsKCQkJcTogdGhpcy4kLnF1ZXJ5LmdldFZhbHVlKCksCgkJCWpzb25wOiAodGhpcy4kLnBpY2tlci5nZXRTZWxlY3RlZCgpLmluZGV4SW5Db250YWluZXIoKT09MikKCQl9KTsKCX0sCglwcm9jZXNzUmVzcG9uc2U6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJLy8gZG8gc29tZXRoaW5nIHdpdGggaXQKCQl0aGlzLiQudGV4dEFyZWEuc2V0VmFsdWUoSlNPTi5zdHJpbmdpZnkoaW5FdmVudC5kYXRhLCBudWxsLCAyKSk7Cgl9Cn0pOw==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/assets/Sample1.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJTYW1wbGUiLAoJa2luZDogIkNvbnRyb2wiLAoJY29tcG9uZW50czogWwoJCXtuYW1lOiAiaW5wdXQiLCB0YWc6ICJpbnB1dCJ9LAoJCXt0YWc6ICJiciJ9LAoJCXtuYW1lOiAiYnV0dG9uIiwgdGFnOiAiYnV0dG9uIiwgY29udGVudDogIlNldCBDb250ZW50Iiwgb250YXA6ICJidXR0b25UYXAifSwKCQl7dGFnOiAiYnIifSwKCQl7bmFtZTogIm91dHB1dCIsIGNsYXNzZXM6ICJzYW1wbGUtb3V0cHV0In0KCV0sCglidXR0b25UYXA6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJdmFyIHZhbHVlID0gdGhpcy4kLmlucHV0Lmhhc05vZGUoKS52YWx1ZTsKCQl0aGlzLiQub3V0cHV0LnNldENvbnRlbnQodmFsdWUpOwoJfQp9KTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/assets/Sample2.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJTYW1wbGUiLAoJa2luZDogIkNvbnRyb2wiLAoJY29tcG9uZW50czogWwoJCXtuYW1lOiAiaW5wdXQiLCB0YWc6ICJpbnB1dCIsIGF0dHJpYnV0ZXM6IHt2YWx1ZTogInJlZCJ9fSwKCQl7dGFnOiAiYnIifSwKCQl7dGFnOiAiYnV0dG9uIiwgY29udGVudDogIlNldCBDb2xvciIsIG9udGFwOiAiYnV0dG9uVGFwIn0sCgkJe3RhZzogImJyIn0sCgkJe25hbWU6ICJvdXRwdXQiLCBjbGFzc2VzOiAic2FtcGxlLW91dHB1dCJ9CgldLAoJYnV0dG9uVGFwOiBmdW5jdGlvbihpblNlbmRlciwgaW5FdmVudCkgewoJCXZhciBjb2xvciA9IHRoaXMuJC5pbnB1dC5oYXNOb2RlKCkudmFsdWU7CgkJdGhpcy4kLm91dHB1dC5hcHBseVN0eWxlKCJiYWNrZ3JvdW5kIiwgY29sb3IpOwoJfQp9KTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/assets/Sample3.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJTYW1wbGUiLAoJa2luZDogIkNvbnRyb2wiLAoJY29tcG9uZW50czogWwoJCXtuYW1lOiAiaW5wdXQiLCB0YWc6ICJpbnB1dCIsIGF0dHJpYnV0ZXM6IHt2YWx1ZTogIkZvbyAifX0sCgkJe3RhZzogImJyIn0sCgkJe3RhZzogImJ1dHRvbiIsIGNvbnRlbnQ6ICJBZGQgQ29udGVudCIsIG9udGFwOiAiYWRkQ29udGVudFRhcCJ9LAoJCXt0YWc6ICJiciJ9LAoJCXt0YWc6ICJidXR0b24iLCBzdHlsZTogIm1hcmdpbi1yaWdodDogLjVlbTsiLCBjb250ZW50OiAiSGlkZSIsIG9udGFwOiAiaGlkZVRhcCJ9LAoJCXt0YWc6ICJidXR0b24iLCBzdHlsZTogIm1hcmdpbi1yaWdodDogLjVlbTsiLCBjb250ZW50OiAiU2hvdyIsIG9udGFwOiAic2hvd1RhcCJ9LAoJCXt0YWc6ICJidXR0b24iLCBjb250ZW50OiAiVG9nZ2xlIGNsYXNzIiwgb250YXA6ICJ0b2dnbGVDbGFzc1RhcCJ9LAoJCXt0YWc6ICJiciJ9LAoJCXtuYW1lOiAib3V0cHV0IiwgY2xhc3NlczogInNhbXBsZS1vdXRwdXQifQoJXSwKCWFkZENvbnRlbnRUYXA6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJdmFyIGNvbnRlbnQgPSB0aGlzLiQuaW5wdXQuaGFzTm9kZSgpLnZhbHVlOwoJCXRoaXMuJC5vdXRwdXQuYWRkQ29udGVudChjb250ZW50KTsKCX0sCgloaWRlVGFwOiBmdW5jdGlvbihpblNlbmRlciwgaW5FdmVudCkgewoJCXRoaXMuJC5vdXRwdXQuaGlkZSgpOwoJfSwKCXNob3dUYXA6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJdGhpcy4kLm91dHB1dC5zaG93KCk7Cgl9LAoJdG9nZ2xlQ2xhc3NUYXA6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJdmFyIGMgPSAic2FtcGxlLW91dHB1dCI7CgkJdmFyIGhhcyA9IHRoaXMuJC5vdXRwdXQuaGFzQ2xhc3MoYyk7CgkJdGhpcy4kLm91dHB1dC5hZGRSZW1vdmVDbGFzcyhjLCAhaGFzKTsKCX0KfSk7
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/assets/Sample4.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5raW5kKHsKCW5hbWU6ICJTYW1wbGUiLAoJa2luZDogIkNvbnRyb2wiLAoJY29tcG9uZW50czogWwoJCXtjb250ZW50OiAiQ3JlYXRlIGEgYnV0dG9uIGNvbnRyb2wgd2l0aCB0aGUgZ2l2ZW4gY29udGVudDoiLCBzdHlsZTogInBhZGRpbmc6IDEwcHggMDsifSwKCQl7bmFtZTogImlucHV0IiwgdGFnOiAiaW5wdXQiLCBhdHRyaWJ1dGVzOiB7dmFsdWU6ICJGb28ifSwgc3R5bGU6ICJkaXNwbGF5OiBibG9jazsifSwKCQl7c3R5bGU6ICJwYWRkaW5nOiAxMHB4IDA7IiwgY29tcG9uZW50czogWwoJCQl7dGFnOiAiYnV0dG9uIiwgc3R5bGU6ICJtYXJnaW4tcmlnaHQ6IDEwcHg7IiwgY29udGVudDogIkFkZCIsIG9udGFwOiAiYWRkVGFwIn0sCgkJCXt0YWc6ICJidXR0b24iLCBjb250ZW50OiAiQ2xlYXIiLCBvbnRhcDogImNsZWFyVGFwIn0sCgkJXX0sCgkJe25hbWU6ICJvdXRwdXQifQoJXSwKCWFkZFRhcDogZnVuY3Rpb24oaW5TZW5kZXIsIGluRXZlbnQpIHsKCQl2YXIgY29udGVudCA9IHRoaXMuJC5pbnB1dC5oYXNOb2RlKCkudmFsdWU7CgkJdGhpcy5jcmVhdGVDb21wb25lbnQoewoJCQl0YWc6ICJidXR0b24iLAoJCQljb250ZW50OiBjb250ZW50LAoJCQljb250YWluZXI6IHRoaXMuJC5vdXRwdXQKCQl9KS5yZW5kZXIoKTsKCX0sCgljbGVhclRhcDogZnVuY3Rpb24oaW5TZW5kZXIsIGluRXZlbnQpIHsKCQl0aGlzLiQub3V0cHV0LmRlc3Ryb3lDbGllbnRDb250cm9scygpOwoJfQp9KTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/samples/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJIlBsYXRmb3JtU2FtcGxlLmNzcyIsCgkiUGxhdGZvcm1TYW1wbGUuanMiLAoJIkFqYXhTYW1wbGUuY3NzIiwKCSJBamF4U2FtcGxlLmpzIiwKCSJKc29ucFNhbXBsZS5jc3MiLAoJIkpzb25wU2FtcGxlLmpzIiwKCSJXZWJTZXJ2aWNlU2FtcGxlLmNzcyIsCgkiV2ViU2VydmljZVNhbXBsZS5qcyIsCgkiUmVwZWF0ZXJTYW1wbGUuY3NzIiwKCSJSZXBlYXRlclNhbXBsZS5qcyIsCgkiU2Nyb2xsZXJTYW1wbGUuY3NzIiwKCSJTY3JvbGxlclNhbXBsZS5qcyIsCgkiR2VzdHVyZVNhbXBsZS5jc3MiLAoJIkdlc3R1cmVTYW1wbGUuanMiLAoJIkRyYXdlclNhbXBsZS5jc3MiLAoJIkRyYXdlclNhbXBsZS5qcyIsCgkiRnVsbHNjcmVlblNhbXBsZS5jc3MiLAoJIkZ1bGxzY3JlZW5TYW1wbGUuanMiLAoJIkF1ZGlvU2FtcGxlLmNzcyIsCgkiQXVkaW9TYW1wbGUuanMiCik7
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ajax/Ajax.js"
Content-Type: application/octet-stream; x-encoding=base64
﻿/**
	_enyo.Ajax_ is a wrapper for _XmlHttpRequest_ that uses
	the <a href="#enyo.Async">enyo.Async</a> API.

	_enyo.Ajax_ publishes all the properties of the
	<a href="#enyo.AjaxProperties">enyo.AjaxProperties</a>
	object.

	Like _enyo.Async_, _enyo.Ajax_ is an **Object**, not a **Component**.
	Do not try to make _enyo.Ajax_ objects inside a _components_ block.
	If you want to use _enyo.Ajax_ as a component, you should probably
	be using <a href="#enyo.WebService">enyo.WebService</a> instead.

	If you make changes to _enyo.Ajax_, be sure to add or update the appropriate
	[unit tests](https://github.com/enyojs/enyo/tree/master/tools/test/ajax/tests).

	For more information, see the documentation on
	[Consuming Web Services](https://github.com/enyojs/enyo/wiki/Consuming-Web-Services)
	in the Enyo Developer Guide.
*/
enyo.kind({
	name: "enyo.Ajax",
	kind: "enyo.Async",
	//* See <a href="#enyo.AjaxProperties">enyo.AjaxProperties</a> for the list of properties
	//* published by _enyo.Ajax_.
	published: enyo.AjaxProperties,
	//* @protected
	constructor: function(inParams) {
		enyo.mixin(this, inParams);
		this.inherited(arguments);
	},
	//* @public
	/**
	Sends the Ajax request with parameters _inParams_. _inParams_ values may be
	either Strings or Objects.

	_inParams_ as an Object is converted into the url query string. For
	instance, passing <code>{q: "searchTerm"}</code> will result in the addition
	of the string `q="searchTerm"` to the current url query string.

	_inParams_ as a String is used as the query part of the URL directly.

	_inParams_ will not be converted into a POST body, it will always be used as
	part of the URL query string if provided.  Use the `postBody` property for
	specifying a body.

	When the request is completed, the code will set a `xhrResponse` property
	in the `enyo.Ajax` object with the subproperties `status`, `headers`, and
	`body`.  These cache the results from the XHR for later use.  The keys for
	the `headers` object have been converted to all lower case as HTTP headers
	are case-insensitive.
	*/
	go: function(inParams) {
		this.startTimer();
		this.request(inParams);
		return this;
	},
	//* @protected
	request: function(inParams) {
		var parts = this.url.split("?");
		var uri = parts.shift() || "";
		var args = parts.length ? (parts.join("?").split("&")) : [];
		//
		var query = null;
		//
		if(enyo.isString(inParams)){
			//If inParams parameter is a string, use it as request body
			query = inParams;
		}
		else{
			//If inParams parameter is not a string, build a query from it
			if(inParams){
				query = enyo.Ajax.objectToQuery(inParams);
			}
		}
		//
		if (query) {
			args.push(query);
			query = null;
		}
		if (this.cacheBust) {
			args.push(Math.random());
		}
		//
		var url = args.length ? [uri, args.join("&")].join("?") : uri;
		//
		var xhr_headers = {};
		var body;
		if (this.method != "GET") {
			body = this.postBody;
			if (this.method === "POST" && body instanceof enyo.FormData) {
				if (body.fake) {
					xhr_headers["Content-Type"] = body.getContentType();
					body = body.toString();
				} else {
					// Nothing to do as the
					// content-type will be
					// automagically set according
					// to the FormData
				}
			} else {
				xhr_headers["Content-Type"] = this.contentType;
				if (body instanceof Object) {
					if (this.contentType.match(/^application\/json(;.*)?$/) !== null) {
						body = JSON.stringify(body);
					} else if (this.contentType === "application/x-www-form-urlencoded") {
						body = enyo.Ajax.objectToQuery(body);
					}
					else {
						body = body.toString();
					}
				}
			}
		}
		enyo.mixin(xhr_headers, this.headers);
		// don't pass in headers structure if there are no headers defined as this messes
		// up CORS code for IE8-9
		if (enyo.keys(xhr_headers).length === 0) {
			xhr_headers = undefined;
		}
		//
		try {
			this.xhr = enyo.xhr.request({
				url: url,
				method: this.method,
				callback: this.bindSafely("receive"),
				body: body,
				headers: xhr_headers,
				sync: this.sync,
				username: this.username,
				password: this.password,
				xhrFields: enyo.mixin({onprogress: this.bindSafely(this.updateProgress)}, this.xhrFields),
				mimeType: this.mimeType
			});
		}
		catch (e) {
			// IE can throw errors here if the XHR would fail CORS checks,
			// so catch and turn into a failure.
			this.fail(e);
		}
	},
	receive: function(inText, inXhr) {
		if (!this.failed && !this.destroyed) {
			var body;
			if (typeof inXhr.responseText === "string") {
				body = inXhr.responseText;
			} else {
				// IE carrying a binary
				body = inXhr.responseBody;
			}
			this.xhrResponse = {
				status: inXhr.status,
				headers: enyo.Ajax.parseResponseHeaders(inXhr),
				body: body
			};
			if (this.isFailure(inXhr)) {
				this.fail(inXhr.status);
			} else {
				this.respond(this.xhrToResponse(inXhr));
			}
		}
	},
	fail: function(inError) {
		// on failure, explicitly cancel the XHR to prevent
		// further responses.  cancellation also resets the
		// response headers & body,
		if (this.xhr) {
			enyo.xhr.cancel(this.xhr);
			this.xhr = null;
		}
		this.inherited(arguments);
	},
	xhrToResponse: function(inXhr) {
		if (inXhr) {
			return this[(this.handleAs || "text") + "Handler"](inXhr);
		}
	},
	isFailure: function(inXhr) {
		// if any exceptions are thrown while checking fields in the xhr,
		// assume a failure.
		try {
			var text = "";
			// work around IE8-9 bug where accessing responseText will thrown error
			// for binary requests.
			if (typeof inXhr.responseText === "string") {
				text = inXhr.responseText;
			}
			// Follow same failure policy as jQuery's Ajax code
			// CORS failures on FireFox will have status 0 and no responseText,
			// so treat that as failure.
			if (inXhr.status === 0 && text === "") {
				return true;
			}
			// Otherwise, status 0 may be good for local file access.  We treat the range
			// 1-199 and 300+ as failure (only 200-series code are OK).
			return (inXhr.status !== 0) && (inXhr.status < 200 || inXhr.status >= 300);
		}
		catch (e) {
			return true;
		}
	},
	xmlHandler: function(inXhr) {
		return inXhr.responseXML;
	},
	textHandler: function(inXhr) {
		return inXhr.responseText;
	},
	jsonHandler: function(inXhr) {
		var r = inXhr.responseText;
		try {
			return r && enyo.json.parse(r);
		} catch (x) {
			enyo.warn("Ajax request set to handleAs JSON but data was not in JSON format");
			return r;
		}
	},
	//* @protected
	//* Handler for ajax progress events.
	updateProgress: function(event) {
		// filter out "input" as it causes exceptions on some Firefox versions
		// due to unimplemented internal APIs
		var ev = {};
		for (var k in event) {
			if (k !== 'input') {
				ev[k] = event[k];
			}
		}
		this.sendProgress(event.loaded, 0, event.total, ev);
	},
	statics: {
		objectToQuery: function(/*Object*/ map) {
			var enc = encodeURIComponent;
			var pairs = [];
			var backstop = {};
			for (var name in map){
				var value = map[name];
				if (value != backstop[name]) {
					var assign = enc(name) + "=";
					if (enyo.isArray(value)) {
						for (var i=0; i < value.length; i++) {
							pairs.push(assign + enc(value[i]));
						}
					} else {
						pairs.push(assign + enc(value));
					}
				}
			}
			return pairs.join("&");
		}
	},
	protectedStatics: {
		parseResponseHeaders: function(xhr) {
			var headers = {};
			var headersStr = [];
			if (xhr.getAllResponseHeaders) {
				headersStr = xhr.getAllResponseHeaders().split(/\r?\n/);
			}
			for (var i = 0; i < headersStr.length; i++) {
				var headerStr = headersStr[i];
				var index = headerStr.indexOf(': ');
				if (index > 0) {
					var key = headerStr.substring(0, index).toLowerCase();
					var val = headerStr.substring(index + 2);
					headers[key] = val;
				}
			}
			return headers;
		}
	}
});

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ajax/AjaxProperties.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglDb21tb24gc2V0IG9mIHB1Ymxpc2hlZCBwcm9wZXJ0aWVzIHVzZWQgaW4gYm90aAoJPGEgaHJlZj0iI2VueW8uQWpheCI+ZW55by5BamF4PC9hPiBhbmQKCTxhIGhyZWY9IiNlbnlvLldlYlNlcnZpY2UiPmVueW8uV2ViU2VydmljZTwvYT4uCiovCmVueW8uQWpheFByb3BlcnRpZXMgPSB7CgkvKioKCQlXaGVuIHRydWUsIGFwcGVuZHMgYSByYW5kb20gbnVtYmVyIGFzIGEgcGFyYW1ldGVyIGZvciBHRVQgcmVxdWVzdHMKCQl0byB0cnkgdG8gZm9yY2UgYSBuZXcgZmV0Y2ggb2YgdGhlIHJlc291cmNlIGluc3RlYWQgb2YgcmV1c2luZyBhIGxvY2FsIGNhY2hlLgoJKi8KCWNhY2hlQnVzdDogdHJ1ZSwKCS8qKgoJCVRoZSBVUkwgZm9yIHRoZSBzZXJ2aWNlLiAgVGhpcyBjYW4gYmUgYSByZWxhdGl2ZSBVUkwgaWYgdXNlZCB0byBmZXRjaCByZXNvdXJjZXMgYnVuZGxlZCB3aXRoIHRoZSBhcHBsaWNhdGlvbi4KCSovCgl1cmw6ICIiLAoJLyoqCgkJVGhlIEhUVFAgbWV0aG9kIHRvIHVzZSBmb3IgdGhlIHJlcXVlc3QsIGRlZmF1bHRzIHRvIEdFVC4gIFN1cHBvcnRlZCB2YWx1ZXMgaW5jbHVkZQoJCSJHRVQiLCAiUE9TVCIsICJQVVQiLCBhbmQgIkRFTEVURSIuCgkqLwoJbWV0aG9kOiAiR0VUIiwgLy8ge3ZhbHVlOiAiR0VUIiwgb3B0aW9uczogWyJHRVQiLCAiUE9TVCIsICJQVVQiLCAiREVMRVRFIl19LAoJLyoqCgkJSG93IHRoZSByZXNwb25zZSB3aWxsIGJlIGhhbmRsZWQuCgkJU3VwcG9ydGVkIHZhbHVlcyBhcmU6IDxjb2RlPiJqc29uIiwgInRleHQiLCAieG1sIjwvY29kZT4KCSovCgloYW5kbGVBczogImpzb24iLCAvLyB7dmFsdWU6ICJqc29uIiwgb3B0aW9uczogWyJ0ZXh0IiwgImpzb24iLCAieG1sIl19LAoJLyoqCgkJVGhlIENvbnRlbnQtVHlwZSBoZWFkZXIgZm9yIHRoZSByZXF1ZXN0IGFzIGEgU3RyaW5nLgoJKi8KCWNvbnRlbnRUeXBlOiAiYXBwbGljYXRpb24veC13d3ctZm9ybS11cmxlbmNvZGVkIiwKCS8qKgoJCUlmIHRydWUsIG1ha2VzIGEgc3luY2hyb25vdXMgKGJsb2NraW5nKSBjYWxsLCBpZiBzdXBwb3J0ZWQuCgkqLwoJc3luYzogZmFsc2UsCgkvKioKCQlPcHRpb25hbCBhZGRpdGlvbmFsIHJlcXVlc3QgaGVhZGVycyBhcyBhIEpTIG9iamVjdCwgZS5nLgoJCTxjb2RlPnsgIlgtTXktSGVhZGVyIjogIk15IFZhbHVlIiwgIk1vb2QiOiAiSGFwcHkiIH08L2NvZGU+IG9yIG51bGwuCgkqLwoJaGVhZGVyczogbnVsbCwKCS8qKgoJCVRoZSBjb250ZW50IGZvciB0aGUgcmVxdWVzdCBib2R5IGZvciBQT1NUL1BVVCBtZXRob2RzLgoKCQlXaGVuIHBvc3RCb2R5IGlzIGEgQnVmZmVyIG9yIGEgU3RyaW5nLCBpdCBpcyBwYXNzZWQgdmVyYmF0aW0gaW4gdGhlIHJlcXVlc3QgYm9keS4KCQlXaGVuIHBvc3RCb2R5IGlzIGFuIE9iamVjdCwgdGhlIHdheSBpdCBpcyBlbmNvZGVkIGRlcGVuZHMgb24gdGhlIGNvbnRlbnRUeXBlOgoKICAgICAgICAqIGFwcGxpY2F0aW9uL2pzb24gPT4gSlNPTi5zdHJpbmdpZnkKICAgICAgICAqIGFwcGxpY2F0aW9uL3gtd3d3LXVybGVuY29lZCA9PiB1cmwtZW5jb2RlZCBwYXJhbWV0ZXJzCiAgICAgICAgKiBtdWx0aXBhcnQvZm9ybS1kYXRhID0+IHBhc3NlZCBhcyBmaWVsZHMgaW4gZW55by5Gb3JtRGF0YSAoWEhSMiBlbXVsYXRpb24pCgkqLwoJcG9zdEJvZHk6ICIiLAoJLyoqCgkJVGhlIG9wdGlvbmFsIHVzZXIgbmFtZSB0byB1c2UgZm9yIGF1dGhlbnRpY2F0aW9uIHB1cnBvc2VzLgoJKi8KCXVzZXJuYW1lOiAiIiwKCS8qKgoJCVRoZSBvcHRpb25hbCBwYXNzd29yZCB0byB1c2UgZm9yIGF1dGhlbnRpY2F0aW9uIHB1cnBvc2VzLgoJKi8KCXBhc3N3b3JkOiAiIiwKCS8qKgoJCU9wdGlvbmFsIG9iamVjdCB3aXRoIGZpZWxkcyB0byBwYXNzIGRpcmVjdGx5IHRvIHRoZSB1bmRlcmx5aW5nIFhIUiBvYmplY3QuCgkJT25lIGV4YW1wbGUgaXMgdGhlIF93aXRoQ3JlZGVudGlhbHNfIGZsYWcgdXNlZCBmb3IgY3Jvc3Mtb3JpZ2luIHJlcXVlc3RzLgoJKi8KCXhockZpZWxkczogbnVsbCwKCS8qKgoJCU9wdGlvbmFsIHN0cmluZyB0byBvdmVycmlkZSB0aGUgTUlNRS1UeXBlIGhlYWRlci4KCSovCgltaW1lVHlwZTogbnVsbAp9Owo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ajax/Async.js"
Content-Type: application/octet-stream; x-encoding=base64
77u/LyoqCglfZW55by5Bc3luY18gaXMgdGhlIGJhc2Uga2luZCBmb3IgaGFuZGxpbmcgYXN5bmNocm9ub3VzIG9wZXJhdGlvbnMuCgoJX2VueW8uQXN5bmNfIGlzIGFuICoqT2JqZWN0KiosIG5vdCBhICoqQ29tcG9uZW50Kio7IHRodXMsIHlvdSBtYXkgbm90IGRlY2xhcmUKCWFuIF9Bc3luY18gaW4gYSBfY29tcG9uZW50c18gYmxvY2suIElmIHlvdSB3YW50IHRvIHVzZSBfQXN5bmNfIGFzIGEKCWNvbXBvbmVudCwgeW91IHNob3VsZCBwcm9iYWJseSBiZSB1c2luZwoJPGEgaHJlZj0iI2VueW8uV2ViU2VydmljZSI+ZW55by5XZWJTZXJ2aWNlPC9hPiBpbnN0ZWFkLgoKCUFuIEFzeW5jIG9iamVjdCByZXByZXNlbnRzIGEgdGFzayB0aGF0IGhhcyBub3QgeWV0IGNvbXBsZXRlZC4gWW91IG1heSBhdHRhY2gKCWNhbGxiYWNrIGZ1bmN0aW9ucyB0byBhbiBBc3luYywgdG8gYmUgY2FsbGVkIHdoZW4gdGhlIHRhc2sgY29tcGxldGVzIG9yCgllbmNvdW50ZXJzIGFuIGVycm9yLgoKCU1vcmUgaW5mb3JtYXRpb24gb24gX0FzeW5jXyBhbmQgaXRzIHVzYWdlIGlzIGF2YWlsYWJsZSBpbiB0aGUgZG9jdW1lbnRhdGlvbglvbgoJPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2VueW9qcy9lbnlvL3dpa2kvQ29uc3VtaW5nLVdlYi1TZXJ2aWNlcyI+Q29uc3VtaW5nIFdlYiBTZXJ2aWNlczwvYT4KCWluIHRoZSBFbnlvIERldmVsb3BlciBHdWlkZS4KKi8KZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLkFzeW5jIiwKCWtpbmQ6ICJlbnlvLk9iamVjdCIsCglwdWJsaXNoZWQ6IHsKCQkvKioKCQkJSWYgc2V0IHRvIGEgbm9uLXplcm8gdmFsdWUsIHRoZSBudW1iZXIgb2YgbWlsbGlzZWNvbmRzIHRvCgkJCXdhaXQgYWZ0ZXIgdGhlIF9nb18gY2FsbCBiZWZvcmUgZmFpbGluZyB3aXRoIHRoZSAidGltZW91dCIgZXJyb3IKCQkqLwoJCXRpbWVvdXQ6IDAKCX0sCgkvLyogQHByb3RlY3RlZAoJZmFpbGVkOiBmYWxzZSwKCWNvbnRleHQ6IG51bGwsCgljb25zdHJ1Y3RvcjogZnVuY3Rpb24oKSB7CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCQl0aGlzLnJlc3BvbmRlcnMgPSBbXTsKCQl0aGlzLmVycm9ySGFuZGxlcnMgPSBbXTsKCQl0aGlzLnByb2dyZXNzSGFuZGxlcnMgPSBbXTsKCX0sCglhY2N1bXVsYXRlOiBmdW5jdGlvbihpbkFycmF5LCBpbk1ldGhvZEFyZ3MpIHsKCQl2YXIgZm4gPSAoaW5NZXRob2RBcmdzLmxlbmd0aCA8IDIpID8gaW5NZXRob2RBcmdzWzBdIDogZW55by5iaW5kKGluTWV0aG9kQXJnc1swXSwgaW5NZXRob2RBcmdzWzFdKTsKCQlpbkFycmF5LnB1c2goZm4pOwoJfSwKCS8vKiBAcHVibGljCgkvKioKCQlSZWdpc3RlcnMgYSByZXNwb25zZSBmdW5jdGlvbi4KCQlGaXJzdCBwYXJhbWV0ZXIgaXMgYW4gb3B0aW9uYWwgX3RoaXNfIGNvbnRleHQgZm9yIHRoZSByZXNwb25zZSBtZXRob2QuCgkJU2Vjb25kIChvciBvbmx5KSBwYXJhbWV0ZXIgaXMgdGhlIGZ1bmN0aW9uIG9iamVjdC4KCSovCglyZXNwb25zZTogZnVuY3Rpb24oLyogW2luQ29udGV4dF0sIGluUmVzcG9uZGVyICovKSB7CgkJdGhpcy5hY2N1bXVsYXRlKHRoaXMucmVzcG9uZGVycywgYXJndW1lbnRzKTsKCQlyZXR1cm4gdGhpczsKCX0sCgkvKioKCQlSZWdpc3RlcnMgYW4gZXJyb3IgaGFuZGxlci4KCQlGaXJzdCBwYXJhbWV0ZXIgaXMgYW4gb3B0aW9uYWwgX3RoaXNfIGNvbnRleHQgZm9yIHRoZSByZXNwb25zZSBtZXRob2QuCgkJU2Vjb25kIChvciBvbmx5KSBwYXJhbWV0ZXIgaXMgdGhlIGZ1bmN0aW9uIG9iamVjdC4KCSovCgllcnJvcjogZnVuY3Rpb24oLyogW2luQ29udGV4dF0sIGluUmVzcG9uZGVyICovKSB7CgkJdGhpcy5hY2N1bXVsYXRlKHRoaXMuZXJyb3JIYW5kbGVycywgYXJndW1lbnRzKTsKCQlyZXR1cm4gdGhpczsKCX0sCgkvLyogQHByb3RlY3RlZAoJcm91dGU6IGZ1bmN0aW9uKGluQXN5bmMsIGluVmFsdWUpIHsKCQl2YXIgciA9IHRoaXMuYmluZFNhZmVseSgicmVzcG9uZCIpOwoJCWluQXN5bmMucmVzcG9uc2UoZnVuY3Rpb24oaW5TZW5kZXIsIGluVmFsdWUpIHsKCQkJcihpblZhbHVlKTsKCQl9KTsKCQl2YXIgZiA9IHRoaXMuYmluZFNhZmVseSgiZmFpbCIpOwoJCWluQXN5bmMuZXJyb3IoZnVuY3Rpb24oaW5TZW5kZXIsIGluVmFsdWUpIHsKCQkJZihpblZhbHVlKTsKCQl9KTsKCQlpbkFzeW5jLmdvKGluVmFsdWUpOwoJfSwKCWhhbmRsZTogZnVuY3Rpb24oaW5WYWx1ZSwgaW5IYW5kbGVycykgewoJCXZhciByID0gaW5IYW5kbGVycy5zaGlmdCgpOwoJCWlmIChyKSB7CgkJCWlmIChyIGluc3RhbmNlb2YgZW55by5Bc3luYykgewoJCQkJdGhpcy5yb3V0ZShyLCBpblZhbHVlKTsKCQkJfSBlbHNlIHsKCQkJCS8vIGhhbmRsZXIgY2FuIHJldHVybiBhIG5ldyAndmFsdWUnCgkJCQl2YXIgdiA9IGVueW8uY2FsbCh0aGlzLmNvbnRleHQgfHwgdGhpcywgciwgW3RoaXMsIGluVmFsdWVdKTsKCQkJCS8vIC4uLiBidXQgb25seSBpZiBpdCByZXR1cm5zIHNvbWV0aGluZyBvdGhlciB0aGFuIHVuZGVmaW5lZAoJCQkJdiA9ICh2ICE9PSB1bmRlZmluZWQpID8gdiA6IGluVmFsdWU7CgkJCQkvLyBuZXh0IGhhbmRsZXIKCQkJCSh0aGlzLmZhaWxlZCA/IHRoaXMuZmFpbCA6IHRoaXMucmVzcG9uZCkuY2FsbCh0aGlzLCB2KTsKCQkJfQoJCX0KCX0sCglzdGFydFRpbWVyOiBmdW5jdGlvbigpIHsKCQl0aGlzLnN0YXJ0VGltZSA9IGVueW8ubm93KCk7CgkJaWYgKHRoaXMudGltZW91dCkgewoJCQl0aGlzLnRpbWVvdXRKb2IgPSBzZXRUaW1lb3V0KHRoaXMuYmluZFNhZmVseSgidGltZW91dENvbXBsZXRlIiksIHRoaXMudGltZW91dCk7CgkJfQoJfSwKCWVuZFRpbWVyOiBmdW5jdGlvbigpIHsKCQlpZiAodGhpcy50aW1lb3V0Sm9iKSB7CgkJCXRoaXMuZW5kVGltZSA9IGVueW8ubm93KCk7CgkJCWNsZWFyVGltZW91dCh0aGlzLnRpbWVvdXRKb2IpOwoJCQl0aGlzLnRpbWVvdXRKb2IgPSBudWxsOwoJCQl0aGlzLmxhdGVuY3kgPSB0aGlzLmVuZFRpbWUgLSB0aGlzLnN0YXJ0VGltZTsKCQl9Cgl9LAoJdGltZW91dENvbXBsZXRlOiBmdW5jdGlvbigpIHsKCQl0aGlzLnRpbWVkb3V0ID0gdHJ1ZTsKCQl0aGlzLmZhaWwoInRpbWVvdXQiKTsKCX0sCgkvLyogQHByb3RlY3RlZAoJLy8qIENhbGxlZCBhcyBwYXJ0IG9mIHRoZSBhc3luYyBpbXBsZW1lbnRhdGlvbjsgdHJpZ2dlcnMgdGhlIGhhbmRsZXIgY2hhaW4uCglyZXNwb25kOiBmdW5jdGlvbihpblZhbHVlKSB7CgkJdGhpcy5mYWlsZWQgPSBmYWxzZTsKCQl0aGlzLmVuZFRpbWVyKCk7CgkJdGhpcy5oYW5kbGUoaW5WYWx1ZSwgdGhpcy5yZXNwb25kZXJzKTsKCX0sCgkvLyogQHB1YmxpYwoJLy8qIENhbiBiZSBjYWxsZWQgZnJvbSBhbnkgaGFuZGxlciB0byB0cmlnZ2VyIHRoZSBlcnJvciBjaGFpbi4KCWZhaWw6IGZ1bmN0aW9uKGluRXJyb3IpIHsKCQl0aGlzLmZhaWxlZCA9IHRydWU7CgkJdGhpcy5lbmRUaW1lcigpOwoJCXRoaXMuaGFuZGxlKGluRXJyb3IsIHRoaXMuZXJyb3JIYW5kbGVycyk7Cgl9LAoJLy8qIENhbGxlZCBmcm9tIGFuIGVycm9yIGhhbmRsZXI7IGNsZWFycyB0aGUgZXJyb3IgY29uZGl0aW9uIGFuZCByZXN1bWVzCgkvLyogY2FsbGluZyBoYW5kbGVyIG1ldGhvZHMuCglyZWNvdmVyOiBmdW5jdGlvbigpIHsKCQl0aGlzLmZhaWxlZCA9IGZhbHNlOwoJfSwKICAgIC8vKiBAcHVibGljCgkvKioKCQlSZWdpc3RlcnMgYSBwcm9ncmVzcyBoYW5kbGVyLgoJCUZpcnN0IHBhcmFtZXRlciBpcyBhbiBvcHRpb25hbCBfdGhpc18gY29udGV4dCBmb3IgdGhlIHJlc3BvbnNlIG1ldGhvZC4KCQlTZWNvbmQgKG9yIG9ubHkpIHBhcmFtZXRlciBpcyB0aGUgZnVuY3Rpb24gb2JqZWN0LgoJCVByb2dyZXNzIGhhbmRsZXJzIGFyZSBjYWxsZWQgd2l0aCB0aGUgc2VuZGVyIGFzIHRoZSBmaXJzdCBhcmd1bWVudCBhbmQgYSBwcm9ncmVzcyBldmVudCBhcyB0aGUgc2Vjb25kIGFyZ3VtZW50LgoJKi8KCXByb2dyZXNzOiBmdW5jdGlvbigvKiBbaW5Db250ZXh0XSwgaW5SZXNwb25kZXIgKi8pIHsKCQl0aGlzLmFjY3VtdWxhdGUodGhpcy5wcm9ncmVzc0hhbmRsZXJzLCBhcmd1bWVudHMpOwoJCXJldHVybiB0aGlzOwoJfSwKCS8vKiBAcHJvdGVjdGVkCgkvLyogTm90aWZpZXMgdGhlIHByb2dyZXNzIGhhbmRsZXJzCglzZW5kUHJvZ3Jlc3M6IGZ1bmN0aW9uKGN1cnJlbnQsIG1pbiwgbWF4LCBzb3VyY2VFdmVudCkgewoJCXZhciBldmVudCA9IGVueW8ubWl4aW4oe30sIHNvdXJjZUV2ZW50KTsKCQlldmVudC50eXBlID0gJ3Byb2dyZXNzJzsKCQlldmVudC5jdXJyZW50ID0gY3VycmVudDsKCQlldmVudC5taW4gPSBtaW47CgkJZXZlbnQubWF4ID0gbWF4OwoJCWZvciAodmFyIGkgPSAwOyBpIDwgdGhpcy5wcm9ncmVzc0hhbmRsZXJzLmxlbmd0aDsgaSsrKSB7CgkJCWVueW8uY2FsbCh0aGlzLmNvbnRleHQgfHwgdGhpcywgdGhpcy5wcm9ncmVzc0hhbmRsZXJzW2ldLCBbdGhpcywgZXZlbnRdKTsKCQl9Cgl9LAoJLy8qIFN0YXJ0cyB0aGUgYXN5bmMgYWN0aXZpdHkuIE92ZXJyaWRkZW4gaW4gc3Via2luZHMuCglnbzogZnVuY3Rpb24oaW5WYWx1ZSkgewoJCXRoaXMuc2VuZFByb2dyZXNzKDAsIDAsIDEpOwoJCWVueW8uYXN5bmNNZXRob2QodGhpcywgZnVuY3Rpb24oKSB7CgkJCXRoaXMuc2VuZFByb2dyZXNzKDEsIDAsIDEpOwoJCQl0aGlzLnJlc3BvbmQoaW5WYWx1ZSk7CgkJfSk7CgkJcmV0dXJuIHRoaXM7Cgl9Cn0pOwo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ajax/Jsonp.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5Kc29ucFJlcXVlc3RfIGlzIGEgc3BlY2lhbGl6ZWQgZm9ybSBvZgoJPGEgaHJlZj0iI2VueW8uQXN5bmMiPmVueW8uQXN5bmM8L2E+IHVzZWQgZm9yIG1ha2luZyBKU09OUCByZXF1ZXN0cyB0byBhCglyZW1vdGUgc2VydmVyLiBUaGlzIGRpZmZlcnMgZnJvbSB0aGUgbm9ybWFsIFhNTEh0dHBSZXF1ZXN0IGNhbGwgaW4gdGhhdCB0aGUKCWV4dGVybmFsIHJlc291cmNlIGlzIGxvYWRlZCB1c2luZyBhICZsdDtzY3JpcHQmZ3Q7IHRhZy4gVGhpcyBhbGxvd3MgdXMgdG8KCWJ5cGFzcyB0aGUgc2FtZS1kb21haW4gcnVsZXMgdGhhdCBub3JtYWxseSBhcHBseSB0byBYSFIsIHNpbmNlIHRoZSBicm93c2VyCgl3aWxsIGxvYWQgc2NyaXB0cyBmcm9tIGFueSBhZGRyZXNzLgoKCUF0IHRoZSBzYW1lIHRpbWUsIGluIG9yZGVyIHRvIHN1Y2Nlc3NmdWxseSBsb2FkIGRhdGEgdmlhIHRoZSAmbHQ7c2NyaXB0Jmd0OwoJdGFnLCB5b3VyIGRhdGEgc291cmNlIG11c3QgYmUgYWNjZXNzZWQgdmlhIGFuIEhUVFAgR0VUIHJlcXVlc3QgYW5kIG11c3QgYmUKCWFibGUgdG8gZHluYW1pY2FsbHkgYWRkIHRoZSByZXF1ZXN0ZWQgY2FsbGJhY2sgbmFtZSBhcyBhIGZ1bmN0aW9uIHdyYXBwZXIKCWFyb3VuZCB0aGUgSlNPTiBkYXRhLgoKCUlmIHlvdSBtYWtlIGNoYW5nZXMgdG8gX2VueW8uSnNvbnBSZXF1ZXN0XywgYmUgc3VyZSB0byBhZGQgb3IgdXBkYXRlIHRoZQoJYXBwcm9wcmlhdGUgW3VuaXQgdGVzdHNdKGh0dHBzOi8vZ2l0aHViLmNvbS9lbnlvanMvZW55by90cmVlL21hc3Rlci90b29scy90ZXN0L2FqYXgvdGVzdHMpLgoKCUZvciBtb3JlIGluZm9ybWF0aW9uLCBzZWUgdGhlIGRvY3VtZW50YXRpb24gb24KCVtDb25zdW1pbmcgV2ViIFNlcnZpY2VzXShodHRwczovL2dpdGh1Yi5jb20vZW55b2pzL2VueW8vd2lraS9Db25zdW1pbmctV2ViLVNlcnZpY2VzKQoJaW4gdGhlIEVueW8gRGV2ZWxvcGVyIEd1aWRlLgoqLwplbnlvLmtpbmQoewoJbmFtZTogImVueW8uSnNvbnBSZXF1ZXN0IiwKCWtpbmQ6ICJlbnlvLkFzeW5jIiwKCXB1Ymxpc2hlZDogewoJCS8vKiBUaGUgVVJMIGZvciB0aGUgc2VydmljZQoJCXVybDogIiIsCgkJLy8qIE9wdGlvbmFsIGNoYXJhY3RlciBzZXQgdG8gdXNlIHRvIGludGVycHJldCB0aGUgcmV0dXJuIGRhdGEKCQljaGFyc2V0OiBudWxsLAoJCS8qKgoJCQlOYW1lIG9mIHRoZSBhcmd1bWVudCB0aGF0IGhvbGRzIHRoZSBjYWxsYmFjayBuYW1lLiBGb3IgZXhhbXBsZSwgdGhlCgkJCVR3aXR0ZXIgc2VhcmNoIEFQSSB1c2VzICJjYWxsYmFjayIgYXMgdGhlIHBhcmFtZXRlciB0byBob2xkIHRoZQoJCQluYW1lIG9mIHRoZSBjYWxsZWQgZnVuY3Rpb24uICBXZSB3aWxsIGF1dG9tYXRpY2FsbHkgYWRkIHRoaXMgdG8KCQkJdGhlIGVuY29kZWQgYXJndW1lbnRzLgoKCQkJSWYgdGhpcyBpcyBudWxsLCB3ZSB3b24ndCBwYXNzIGEgY2FsbGJhY2sgbmFtZSB0byB0aGUgSlNPTlAgc2VydmVyLgoJCQlUaGF0IG1vZGUgaXMgdXN1YWxseSBvbmx5IHVzZWQgaWYgeW91IGFsc28gc2V0IHRoZSBfb3ZlcnJpZGVDYWxsYmFja18KCQkJcGFyYW1ldGVyLCBzaW5jZSB3aXRob3V0IHRoaXMsIHRoZXJlJ3Mgbm8gd2F5IGZvciB0aGUgc2VydmVyIHRvIGtub3cKCQkJd2hhdCBmdW5jdGlvbiB3cmFwcGVyIHRvIHVzZS4KCQkqLwoJCWNhbGxiYWNrTmFtZTogImNhbGxiYWNrIiwKCQkvKioKCQkJV2hlbiB0cnVlLCBhcHBlbmRzIGEgcmFuZG9tIG51bWJlciBhcyBhIHBhcmFtZXRlciBmb3IgR0VUIHJlcXVlc3RzCgkJCXRvIHRyeSB0byBmb3JjZSBhIG5ldyBmZXRjaCBvZiB0aGUgcmVzb3VyY2UgaW5zdGVhZCBvZiByZXVzaW5nIGEKCQkJbG9jYWwgY2FjaGUKCQkqLwoJCWNhY2hlQnVzdDogdHJ1ZSwKCQkvKioKCQkJV2hlbiBzZXQsIHVzZSB0aGlzIGFzIHRoZSBuYW1lIG9mIHRoZSBjYWxsYmFjayBtZXRob2QgdG8gcGFzcwoJCQl0byB0aGUgcmVtb3RlIHNlcnZlci4gVGhpcyBpcyBtYWlubHkgdXNlZnVsIHdoZW4gZGVhbGluZyB3aXRoCgkJCXNlcnZlcnMgdGhhdCBhcmVuJ3QgZmxleGlibGUgaW4gaG93IHRoZXkgc3BlY2lmeSBjYWxsYmFjayBuYW1lcy4KCgkJCUlmIHlvdSBzcGVjaWZ5IHRoaXMsIHdlIHdpbGwgYWRkIGEgbWV0aG9kIHRvIHRoZSBnbG9iYWwgbmFtZXNwYWNlCgkJCXVzaW5nIHRoZSBzcGVjaWZpZWQgbmFtZSwgc28gdGhpcyBjYW4gZWFzaWx5IHN0b21wIG9uIGEgZ2xvYmFsCgkJCXZhcmlhYmxlLiBZb3UgY2FuJ3QgaGF2ZSBtdWx0aXBsZSBjYWxscyB0byBhIEpTT05QIEFQSSBhbGl2ZSBhdAoJCQl0aGUgc2FtZSB0aW1lIHVzaW5nIHRoZSBzYW1lIGNhbGxiYWNrIG1ldGhvZC4KCQkqLwoJCW92ZXJyaWRlQ2FsbGJhY2s6IG51bGwKCX0sCglwcm90ZWN0ZWRTdGF0aWNzOiB7CgkJLy8gQ291bnRlciB0byBhbGxvdyBjcmVhdGlvbiBvZiB1bmlxdWUgbmFtZSBmb3IgZWFjaCBKU09OUCByZXF1ZXN0CgkJbmV4dENhbGxiYWNrSUQ6IDAKCX0sCgkvLyogQHByb3RlY3RlZAoJYWRkU2NyaXB0RWxlbWVudDogZnVuY3Rpb24oKSB7CgkJdmFyIHNjcmlwdCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoJ3NjcmlwdCcpOwoJCXNjcmlwdC5zcmMgPSB0aGlzLnNyYzsKCQlzY3JpcHQuYXN5bmMgPSAiYXN5bmMiOwoJCWlmICh0aGlzLmNoYXJzZXQpIHsKCQkJc2NyaXB0LmNoYXJzZXQgPSB0aGlzLmNoYXJzZXQ7CgkJfQoJCS8vIG1vc3QgbW9kZXJuIGJyb3dzZXJzIGFsc28gaGF2ZSBhbiBvbmVycm9yIGhhbmRsZXIKCQlzY3JpcHQub25lcnJvciA9IHRoaXMuYmluZFNhZmVseShmdW5jdGlvbigpIHsKCQkJLy8gd2UgZG9uJ3QgZ2V0IGFuIGVycm9yIGNvZGUsIHNvIHdlJ2xsIGp1c3QgdXNlIHRoZSBnZW5lcmljIDQwMCBlcnJvciBzdGF0dXMKCQkJdGhpcy5mYWlsKDQwMCk7CgkJfSk7CgkJLy8gYWRkIHNjcmlwdCBiZWZvcmUgZXhpc3Rpbmcgc2NyaXB0IHRvIG1ha2Ugc3VyZSBpdCdzIGluIGEgdmFsaWQgcGFydCBvZiBkb2N1bWVudAoJCS8vIGh0dHA6Ly93d3cuanNwYXR0ZXJucy5jb20vdGhlLXJpZGljdWxvdXMtY2FzZS1vZi1hZGRpbmctYS1zY3JpcHQtZWxlbWVudC8KCQl2YXIgZmlyc3QgPSBkb2N1bWVudC5nZXRFbGVtZW50c0J5VGFnTmFtZSgnc2NyaXB0JylbMF07CgkJZmlyc3QucGFyZW50Tm9kZS5pbnNlcnRCZWZvcmUoc2NyaXB0LCBmaXJzdCk7CgkJdGhpcy5zY3JpcHRUYWcgPSBzY3JpcHQ7Cgl9LAoJcmVtb3ZlU2NyaXB0RWxlbWVudDogZnVuY3Rpb24oKSB7CgkJdmFyIHNjcmlwdCA9IHRoaXMuc2NyaXB0VGFnOwoJCXRoaXMuc2NyaXB0VGFnID0gbnVsbDsKCQlzY3JpcHQub25lcnJvciA9IG51bGw7CgkJaWYgKHNjcmlwdC5wYXJlbnROb2RlKSB7CgkJCXNjcmlwdC5wYXJlbnROb2RlLnJlbW92ZUNoaWxkKHNjcmlwdCk7CgkJfQoJfSwKCWNvbnN0cnVjdG9yOiBmdW5jdGlvbihpblBhcmFtcykgewoJCWVueW8ubWl4aW4odGhpcywgaW5QYXJhbXMpOwoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7Cgl9LAoJLy8qIEBwdWJsaWMKCS8vKiBTdGFydHMgdGhlIEpTT05QIHJlcXVlc3QuCglnbzogZnVuY3Rpb24oaW5QYXJhbXMpIHsKCQl0aGlzLnN0YXJ0VGltZXIoKTsKCQl0aGlzLmpzb25wKGluUGFyYW1zKTsKCQlyZXR1cm4gdGhpczsKCX0sCgkvLyogQHByb3RlY3RlZAoJanNvbnA6IGZ1bmN0aW9uKGluUGFyYW1zKSB7CgkJdmFyIGNhbGxiYWNrRnVuY3Rpb25OYW1lID0gdGhpcy5vdmVycmlkZUNhbGxiYWNrIHx8CgkJCSJlbnlvX2pzb25wX2NhbGxiYWNrXyIgKyAoZW55by5Kc29ucFJlcXVlc3QubmV4dENhbGxiYWNrSUQrKyk7CgkJLy8KCQl0aGlzLnNyYyA9IHRoaXMuYnVpbGRVcmwoaW5QYXJhbXMsIGNhbGxiYWNrRnVuY3Rpb25OYW1lKTsKCQl0aGlzLmFkZFNjcmlwdEVsZW1lbnQoKTsKCQkvLwoJCXdpbmRvd1tjYWxsYmFja0Z1bmN0aW9uTmFtZV0gPSB0aGlzLmJpbmRTYWZlbHkodGhpcy5yZXNwb25kKTsKCQkvLwoJCS8vIHNldHVwIGNsZWFudXAgaGFuZGxlcnMgZm9yIEpTT05QIGNvbXBsZXRpb24gYW5kIGZhaWx1cmUKCQl2YXIgY2xlYW51cCA9IHRoaXMuYmluZFNhZmVseShmdW5jdGlvbigpIHsKCQkJdGhpcy5yZW1vdmVTY3JpcHRFbGVtZW50KCk7CgkJCXdpbmRvd1tjYWxsYmFja0Z1bmN0aW9uTmFtZV0gPSBudWxsOwoJCX0pOwoJCXRoaXMucmVzcG9uc2UoY2xlYW51cCk7CgkJdGhpcy5lcnJvcihjbGVhbnVwKTsKCX0sCglidWlsZFVybDogZnVuY3Rpb24oaW5QYXJhbXMsIGluQ2FsbGJhY2tGdW5jdGlvbk5hbWUpIHsKCQl2YXIgcGFydHMgPSB0aGlzLnVybC5zcGxpdCgiPyIpOwoJCXZhciB1cmkgPSBwYXJ0cy5zaGlmdCgpIHx8ICIiOwoJCXZhciBhcmdzID0gcGFydHMubGVuZ3RoPyBwYXJ0cy5qb2luKCI/Iikuc3BsaXQoIiYiKTogW107CgkJLy8KCQl2YXIgYm9keUFyZ3MgPSB0aGlzLmJvZHlBcmdzRnJvbVBhcmFtcyhpblBhcmFtcywgaW5DYWxsYmFja0Z1bmN0aW9uTmFtZSk7CgkJYXJncy5wdXNoKGJvZHlBcmdzKTsKCQlpZiAodGhpcy5jYWNoZUJ1c3QpIHsKCQkJYXJncy5wdXNoKE1hdGgucmFuZG9tKCkpOwoJCX0KCQkvLwoJCXJldHVybiBbdXJpLCBhcmdzLmpvaW4oIiYiKV0uam9pbigiPyIpOwoJfSwKCS8vIGZvciBhIHN0cmluZyB2ZXJzaW9uIG9mIGluUGFyYW1zLCB3ZSBmb2xsb3cgdGhlIGNvbnZlbnRpb24gb2YKCS8vIHJlcGxhY2luZyB0aGUgc3RyaW5nICI9PyIgd2l0aCB0aGUgY2FsbGJhY2sgbmFtZS4gRm9yIHRoZSBtb3JlCgkvLyBjb21tb24gY2FzZSBvZiBpblBhcmFtcyBiZWluZyBhbiBvYmplY3QsIHdlJ2xsIGFkZCBhIGFyZ3VtZW50IG5hbWVkCgkvLyB1c2luZyB0aGUgY2FsbGJhY2tOYW1lIHB1Ymxpc2hlZCBwcm9wZXJ0eS4KCWJvZHlBcmdzRnJvbVBhcmFtczogZnVuY3Rpb24oaW5QYXJhbXMsIGluQ2FsbGJhY2tGdW5jdGlvbk5hbWUpIHsKCQlpZiAoZW55by5pc1N0cmluZyhpblBhcmFtcykpIHsKCQkJcmV0dXJuIGluUGFyYW1zLnJlcGxhY2UoIj0/IiwgIj0iICsgaW5DYWxsYmFja0Z1bmN0aW9uTmFtZSk7CgkJfSBlbHNlIHsKCQkJdmFyIHBhcmFtcyA9IGVueW8ubWl4aW4oe30sIGluUGFyYW1zKTsKCQkJaWYgKHRoaXMuY2FsbGJhY2tOYW1lKSB7CgkJCQlwYXJhbXNbdGhpcy5jYWxsYmFja05hbWVdID0gaW5DYWxsYmFja0Z1bmN0aW9uTmFtZTsKCQkJfQoJCQlyZXR1cm4gZW55by5BamF4Lm9iamVjdFRvUXVlcnkocGFyYW1zKTsKCQl9Cgl9Cn0pOwo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ajax/WebService.js"
Content-Type: application/octet-stream; x-encoding=base64
77u/Ly8qIEBwcm90ZWN0ZWQKZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLl9BamF4Q29tcG9uZW50IiwKCWtpbmQ6ICJlbnlvLkNvbXBvbmVudCIsCglwdWJsaXNoZWQ6IGVueW8uQWpheFByb3BlcnRpZXMKfSk7CgovLyogQHB1YmxpYwovKioKCV9lbnlvLldlYlNlcnZpY2VfIGlzIGEgY29tcG9uZW50IHRoYXQgcGVyZm9ybXMgV2ViIHJlcXVlc3RzIChfWG1sSHR0cFJlcXVlc3RfKS4KCglJbnRlcm5hbGx5LCBfZW55by5XZWJTZXJ2aWNlXyB1c2VzIF9lbnlvLkFzeW5jXyBzdWJraW5kcyAobmFtZWx5LAoJPGEgaHJlZj0iI2VueW8uQWpheCI+ZW55by5BamF4PC9hPiBhbmQKCTxhIGhyZWY9IiNlbnlvLkpzb25wUmVxdWVzdCI+ZW55by5Kc29ucFJlcXVlc3Q8L2E+KSB0byBtYW5hZ2UgdHJhbnNhY3Rpb25zLgoJVGhlIF9zZW5kXyBtZXRob2QgcmV0dXJucyB0aGUgQXN5bmMgaW5zdGFuY2UgdXNlZCBieSB0aGUgcmVxdWVzdC4KCglfZW55by5XZWJTZXJ2aWNlXyB1c2VzIF9lbnlvLkFqYXhfIGJ5IGRlZmF1bHQgYW5kLCBsaWtlIF9lbnlvLkFqYXhfLCBpdAoJcHVibGlzaGVzIGFsbCB0aGUgcHJvcGVydGllcyBvZiB0aGUKCTxhIGhyZWY9IiNlbnlvLkFqYXhQcm9wZXJ0aWVzIj5lbnlvLkFqYXhQcm9wZXJ0aWVzPC9hPiBvYmplY3QuCgoJVG8gdXNlIGBlbnlvLkpzb25wUmVxdWVzdGAgaW5zdGVhZCBvZiBgZW55by5BamF4YCwgc2V0IGBqc29uYCB0byBgdHJ1ZWAuCgoJSWYgeW91IG1ha2UgY2hhbmdlcyB0byBfZW55by5XZWJTZXJ2aWNlXywgYmUgc3VyZSB0byBhZGQgb3IgdXBkYXRlIHRoZQoJYXBwcm9wcmlhdGUgW3VuaXQgdGVzdHNdKGh0dHBzOi8vZ2l0aHViLmNvbS9lbnlvanMvZW55by90cmVlL21hc3Rlci90b29scy90ZXN0L2FqYXgvdGVzdHMpLgoKCUZvciBtb3JlIGluZm9ybWF0aW9uLCBzZWUgdGhlIGRvY3VtZW50YXRpb24gb24KCVtDb25zdW1pbmcgV2ViIFNlcnZpY2VzXShodHRwczovL2dpdGh1Yi5jb20vZW55b2pzL2VueW8vd2lraS9Db25zdW1pbmctV2ViLVNlcnZpY2VzKQoJaW4gdGhlIEVueW8gRGV2ZWxvcGVyIEd1aWRlLgoqLwplbnlvLmtpbmQoewoJbmFtZTogImVueW8uV2ViU2VydmljZSIsCglraW5kOiAiZW55by5fQWpheENvbXBvbmVudCIsCglwdWJsaXNoZWQ6IHsKCQkvLyogU2V0IHRvIHRydWUgdG8gdXNlIEpTT05QIHByb3RvY29sCgkJanNvbnA6IGZhbHNlLAoJCS8qKgoJCQlXaGVuIHVzaW5nIEpTT05QLCB0aGUgbmFtZSBvZiB0aGUgY2FsbGJhY2sgcGFyYW1ldGVyLgoJCQlOb3RlIHRoYXQgdGhpcyBub3QgdGhlIG5hbWUgb2YgYSBjYWxsYmFjayBmdW5jdGlvbiwgYnV0IG9ubHkKCQkJdGhlIG5hbWUgb2YgdGhlIGNhbGxiYWNrIHBhcmFtZXRlci4gRW55byB3aWxsIGNyZWF0ZSBhbgoJCQlpbnRlcm5hbCBjYWxsYmFjayBmdW5jdGlvbiBhcyBuZWNlc3NhcnkuCgkJKi8KCQljYWxsYmFja05hbWU6ICJjYWxsYmFjayIsCgkJLyoqCgkJCVdoZW4gdXNpbmcgSlNPTlAsIG9wdGlvbmFsIGNoYXJhY3RlciBzZXQgdG8gdXNlIHRvIGludGVycHJldCB0aGUKCQkJcmV0dXJuIGRhdGEKCQkqLwoJCWNoYXJzZXQ6IG51bGwsCgkJLyoqCgkJCUlmIHNldCB0byBhIG5vbi16ZXJvIHZhbHVlLCB0aGUgbnVtYmVyIG9mIG1pbGxpc2Vjb25kcyB0bwoJCQl3YWl0IGFmdGVyIHRoZSBfc2VuZF8gY2FsbCBiZWZvcmUgZmFpbGluZyB3aXRoIGEgInRpbWVvdXQiIGVycm9yCgkJKi8KCQl0aW1lb3V0OiAwCgl9LAoJZXZlbnRzOiB7CgkJLyoqCgkJCUZpcmVzIHdoZW4gYSByZXNwb25zZSBpcyByZWNlaXZlZC4KCgkJCV9pbkV2ZW50LmFqYXhfIGNvbnRhaW5zIHRoZSBBc3luYyBpbnN0YW5jZSBhc3NvY2lhdGVkIHdpdGggdGhlIHJlcXVlc3QuCgoJCQlfaW5FdmVudC5kYXRhXyBjb250YWlucyB0aGUgcmVzcG9uc2UgZGF0YS4KCQkqLwoJCW9uUmVzcG9uc2U6ICIiLAoJCS8qKgoJCQlGaXJlcyB3aGVuIGFuIGVycm9yIGlzIHJlY2VpdmVkLgoKCQkJX2luRXZlbnQuYWpheF8gY29udGFpbnMgdGhlCUFzeW5jIGluc3RhbmNlIGFzc29jaWF0ZWQgd2l0aCB0aGUgcmVxdWVzdC4KCgkJCV9pbkV2ZW50LmRhdGFfIGNvbnRhaW5zIHRoZSBlcnJvciBkYXRhLgoJCSovCgkJb25FcnJvcjogIiIsCgkJLyoqCgkJCUZpcmVzIHdoZW4gdGhlIHJlcXVlc3QgcHJvZ3Jlc3Nlcy4KCQkqLwoJCW9uUHJvZ3Jlc3M6ICIiCgl9LAoJLy8qIEBwcm90ZWN0ZWQKCWNvbnN0cnVjdG9yOiBmdW5jdGlvbihpblByb3BzKSB7CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCX0sCgkvLyogQHB1YmxpYwoJLyoqCgkJU2VuZHMgYSBXZWIgcmVxdWVzdCB3aXRoIHRoZSBwYXNzZWQtaW4gcGFyYW1ldGVycywgcmV0dXJuaW5nIHRoZQoJCWFzc29jaWF0ZWQgQXN5bmMgaW5zdGFuY2UuCgoJCV9pblByb3BzXyBpcyBhbiBvcHRpb25hbCBvYmplY3QgcGFyYW1ldGVydGhhdCAgY2FuIGJlIHVzZWQgdG8gb3ZlcnJpZGUgc29tZQoJCW9mIHRoZSBBSkFYIHByb3BlcnRpZXMgZm9yIHRoaXMgcmVxdWVzdCwgc3VjaCBhcyBzZXR0aW5nIGEgX3Bvc3RCb2R5Xy4KCSovCglzZW5kOiBmdW5jdGlvbihpblBhcmFtcywgaW5Qcm9wcykgewoJCXJldHVybiB0aGlzLmpzb25wID8gdGhpcy5zZW5kSnNvbnAoaW5QYXJhbXMsIGluUHJvcHMpIDogdGhpcy5zZW5kQWpheChpblBhcmFtcywgaW5Qcm9wcyk7Cgl9LAoJLy8qIEBwcm90ZWN0ZWQKCXNlbmRKc29ucDogZnVuY3Rpb24oaW5QYXJhbXMsIGluUHJvcHMpIHsKCQl2YXIganNvbnAgPSBuZXcgZW55by5Kc29ucFJlcXVlc3QoKTsKCQlmb3IgKHZhciBuIGluIHsndXJsJzoxLCAnY2FsbGJhY2tOYW1lJzoxLCAnY2hhcnNldCc6MSwgJ3RpbWVvdXQnOjF9KSB7CgkJCWpzb25wW25dID0gdGhpc1tuXTsKCQl9CgkJZW55by5taXhpbihqc29ucCwgaW5Qcm9wcyk7CgkJcmV0dXJuIHRoaXMuc2VuZEFzeW5jKGpzb25wLCBpblBhcmFtcyk7Cgl9LAoJc2VuZEFqYXg6IGZ1bmN0aW9uKGluUGFyYW1zLCBpblByb3BzKSB7CgkJdmFyIGFqYXggPSBuZXcgZW55by5BamF4KGluUHJvcHMpOwoJCWZvciAodmFyIG4gaW4gZW55by5BamF4UHJvcGVydGllcykgewoJCQlhamF4W25dID0gdGhpc1tuXTsKCQl9CgkJYWpheC50aW1lb3V0ID0gdGhpcy50aW1lb3V0OwoJCWVueW8ubWl4aW4oYWpheCwgaW5Qcm9wcyk7CgkJcmV0dXJuIHRoaXMuc2VuZEFzeW5jKGFqYXgsIGluUGFyYW1zKTsKCX0sCglzZW5kQXN5bmM6IGZ1bmN0aW9uKGluQWpheCwgaW5QYXJhbXMpIHsKCQlyZXR1cm4gaW5BamF4LmdvKGluUGFyYW1zKS5yZXNwb25zZSh0aGlzLCAicmVzcG9uc2UiKS5lcnJvcih0aGlzLCAiZXJyb3IiKS5wcm9ncmVzcyh0aGlzLCAicHJvZ3Jlc3MiKTsKCX0sCglyZXNwb25zZTogZnVuY3Rpb24oaW5TZW5kZXIsIGluRGF0YSkgewoJCXRoaXMuZG9SZXNwb25zZSh7YWpheDogaW5TZW5kZXIsIGRhdGE6IGluRGF0YX0pOwoJfSwKCWVycm9yOiBmdW5jdGlvbihpblNlbmRlciwgaW5EYXRhKSB7CgkJdGhpcy5kb0Vycm9yKHthamF4OiBpblNlbmRlciwgZGF0YTogaW5EYXRhfSk7Cgl9LAoJcHJvZ3Jlc3M6IGZ1bmN0aW9uKGluU2VuZGVyLCBpblByb2dyZXNzRXZlbnQpIHsKCQl0aGlzLmRvUHJvZ3Jlc3MoaW5Qcm9ncmVzc0V2ZW50KTsKCX0KfSk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ajax/cookie.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qIEBwdWJsaWMKCi8qKgoJR2V0cyBhIG5hbWVkIHZhbHVlIGZyb20gdGhlIGRvY3VtZW50IGNvb2tpZS4KKi8KZW55by5nZXRDb29raWUgPSBmdW5jdGlvbihpbk5hbWUpIHsKCXZhciBtYXRjaGVzID0gZG9jdW1lbnQuY29va2llLm1hdGNoKG5ldyBSZWdFeHAoIig/Ol58OyApIiArIGluTmFtZSArICI9KFteO10qKSIpKTsKCXJldHVybiBtYXRjaGVzID8gZGVjb2RlVVJJQ29tcG9uZW50KG1hdGNoZXNbMV0pIDogdW5kZWZpbmVkOwp9OwoKLyoqCglTZXRzIGEgbmFtZWQgdmFsdWUgaW4gdGhlIGRvY3VtZW50IGNvb2tpZSwgd2l0aCBwcm9wZXJ0aWVzLgoKCVByb3BlcnRpZXMgaW4gdGhlIG9wdGlvbmFsIF9pblByb3BzXyBhcmd1bWVudCBhcmUgYXR0YWNoZWQgdG8gdGhlIGNvb2tpZS4KCV9pblByb3BzXyBtYXkgaGF2ZSBhbiBfZXhwaXJlc18gcHJvcGVydHksIHdoaWNoIGNhbiBiZSBhIG51bWJlciBvZiBkYXlzLCBhCglEYXRlIG9iamVjdCwgb3IgYSBVVEMgdGltZSBzdHJpbmcuCgoJVG8gcmVtb3ZlIGEgY29va2llLCB1c2UgYW4gX2luUHJvcHNfIHZhbHVlIG9mIDxjb2RlPnsgIk1heC1BZ2UiOiAwIH08L2NvZGU+LgoKCUlmIGRldmVsb3BpbmcgaW4gdGhlIEdvb2dsZSBDaHJvbWUgYnJvd3NlciB3aXRoIGEgbG9jYWwgZmlsZSBhcyB5b3VyCglhcHBsaWNhdGlvbiwgc3RhcnQgQ2hyb21lIHdpdGggdGhlIDxjb2RlPi0tZW5hYmxlLWZpbGUtY29va2llczwvY29kZT4gc3dpdGNoCgl0byBhbGxvdyBjb29raWVzIHRvIGJlIHNldC4KKi8KZW55by5zZXRDb29raWUgPSBmdW5jdGlvbihpbk5hbWUsIGluVmFsdWUsIGluUHJvcHMpIHsKCXZhciBjb29raWUgPSBpbk5hbWUgKyAiPSIgKyBlbmNvZGVVUklDb21wb25lbnQoaW5WYWx1ZSk7Cgl2YXIgcCA9IGluUHJvcHMgfHwge307CgkvLwoJLy8gRklYTUU6IGV4cGlyZXM9MCBzZWVtcyB0byBkaXNhcHBlYXIgcmlnaHQgYXdheSwgbm90IG9uIGNsb3NlPyAoRkYzKSAgQ2hhbmdlIGRvY3M/Cgl2YXIgZXhwID0gcC5leHBpcmVzOwoJaWYgKHR5cGVvZiBleHAgPT0gIm51bWJlciIpIHsKCQl2YXIgZCA9IG5ldyBEYXRlKCk7CgkJZC5zZXRUaW1lKGQuZ2V0VGltZSgpICsgZXhwKjI0KjYwKjYwKjEwMDApOwoJCWV4cCA9IGQ7Cgl9CglpZiAoZXhwICYmIGV4cC50b1VUQ1N0cmluZykgewoJCXAuZXhwaXJlcyA9IGV4cC50b1VUQ1N0cmluZygpOwoJfQoJLy8KCXZhciBuYW1lLCB2YWx1ZTsKCWZvciAobmFtZSBpbiBwKXsKCQljb29raWUgKz0gIjsgIiArIG5hbWU7CgkJdmFsdWUgPSBwW25hbWVdOwoJCWlmICh2YWx1ZSAhPT0gdHJ1ZSkgewoJCQljb29raWUgKz0gIj0iICsgdmFsdWU7CgkJfQoJfQoJLy8KCS8vZW55by5sb2coY29va2llKTsKCWRvY3VtZW50LmNvb2tpZSA9IGNvb2tpZTsKfTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ajax/formdata.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCgpfZW55by5Gb3JtRGF0YV8gaXMgYW4gW1hIUjJdKGh0dHA6Ly93d3cudzMub3JnL1RSL1hNTEh0dHBSZXF1ZXN0LykKYEZvcm1EYXRhYCBpbXBsZW1lbnRhdGlvbi4gIEl0IGlzIHVzZWQgdG8gc2VuZCBgbXVsdGlwYXJ0L2Zvcm0tZGF0YWAKQWpheCByZXF1ZXN0cy4gIF9lbnlvLkJsb2JfIGlzIHRoZSBhc3NvY2lhdGVkIGNvbnRlbnQgcHJvdmlkZXIgZm9yCmZpbGUtcGFydHMuCgpOb3RlIHRoYXQgaW4gSUU8MTAsIGJvdGggX2VueW8uRm9ybURhdGFfIGFuZCBfZW55by5CbG9iXyBhcmUgbGltaXRlZAp0byBgU3RyaW5nYCBjb250ZW50LS1hbiBfZW55by5CbG9iXyBtYXkgb25seSBiZSBpbnN0YW50aWF0ZWQgdXNpbmcgYW4KYEFycmF5YCBvciBgU3RyaW5nYC4KCl9lbnlvLkZvcm1EYXRhXyBpcyBpbnNwaXJlZCBieQpbaHRtbDUtZm9ybWRhdGFdKGh0dHBzOi8vZ2l0aHViLmNvbS9mcmFuY29pczJtZXR6L2h0bWw1LWZvcm1kYXRhL2Jsb2IvbWFzdGVyL2Zvcm1kYXRhLmpzKS4KCiAgICBFbXVsYXRlIEZvcm1EYXRhIGZvciBzb21lIGJyb3dzZXJzCiAgICBNSVQgTGljZW5zZQogICAgKGMpIDIwMTAgRnJhbmNvaXMgZGUgTWV0egoKICovCihmdW5jdGlvbih3KSB7CglpZiAody5Gb3JtRGF0YSkgewoJCXRyeSB7CgkJCS8qIGpzaGludCB1bnVzZWQ6IGZhbHNlICovCgkJCXZhciB0MSA9IG5ldyB3LkZvcm1EYXRhKCk7CgkJCXZhciB0MiA9IG5ldyB3LkJsb2IoKTsKCQkJLy8gQW5kcm9pZCBDaHJvbWUgMTggd2lsbCB0aHJvdyBhbiBlcnJvciB0cnlpbmcgdG8gY3JlYXRlIHRoZXNlCgkJCWVueW8uRm9ybURhdGEgPSB3LkZvcm1EYXRhOwoJCQllbnlvLkJsb2IgPSB3LkJsb2I7CgkJCXJldHVybjsKCQl9CgkJY2F0Y2ggKGUpIHsKCQkJLy8gaWdub3JlIGVycm9yIGFuZCBmYWxsIHRocm91Z2ggdG8gZmFrZSBGb3JtRGF0YSBjb2RlCgkJfQoJfQoJZnVuY3Rpb24gRm9ybURhdGEoKSB7CgkJdGhpcy5mYWtlID0gdHJ1ZTsKCQl0aGlzLl9maWVsZHMgPSBbXTsKCQkvLyBUaGlzIGdlbmVyYXRlcyBhIDUwIGNoYXJhY3RlciBib3VuZGFyeSBzaW1pbGFyIHRvCgkJLy8gdGhvc2UgdXNlZCBieSBGaXJlZm94LiAgVGhleSBhcmUgb3B0aW1pemVkIGZvcgoJCS8vIGJveWVyLW1vb3JlIHBhcnNpbmcuCgkJdGhpcy5ib3VuZGFyeSA9ICctLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSc7CgkJZm9yICh2YXIgaSA9IDA7IGkgPCAyNDsgaSsrKSB7CgkJCXRoaXMuYm91bmRhcnkgKz0gTWF0aC5mbG9vcihNYXRoLnJhbmRvbSgpICogMTApLnRvU3RyaW5nKDE2KTsKCQl9Cgl9CglGb3JtRGF0YS5wcm90b3R5cGUuZ2V0Q29udGVudFR5cGUgPSBmdW5jdGlvbigpIHsKCQlyZXR1cm4gIm11bHRpcGFydC9mb3JtLWRhdGE7IGJvdW5kYXJ5PSIgKyB0aGlzLmJvdW5kYXJ5OwoJfTsKCUZvcm1EYXRhLnByb3RvdHlwZS5hcHBlbmQgPSBmdW5jdGlvbihrZXksIHZhbHVlLCBmaWxlbmFtZSkgewoJCXRoaXMuX2ZpZWxkcy5wdXNoKFtrZXksIHZhbHVlLCBmaWxlbmFtZV0pOwoJfTsKCUZvcm1EYXRhLnByb3RvdHlwZS50b1N0cmluZyA9IGZ1bmN0aW9uKCkgewoJCXZhciBib3VuZGFyeSA9IHRoaXMuYm91bmRhcnk7CgkJdmFyIGJvZHkgPSAiIjsKCQllbnlvLmZvckVhY2godGhpcy5fZmllbGRzLCBmdW5jdGlvbihmaWVsZCkgewoJCQlib2R5ICs9ICItLSIgKyBib3VuZGFyeSArICJcclxuIjsKCQkJaWYgKGZpZWxkWzJdIHx8IGZpZWxkWzFdLm5hbWUpIHsKCQkJCS8vIGZpbGUgdXBsb2FkCgkJCQl2YXIgZmlsZSA9IGZpZWxkWzFdLCBmaWxlbmFtZSA9IGZpZWxkWzJdIHx8IGZpbGUubmFtZTsKCQkJCWJvZHkgKz0gIkNvbnRlbnQtRGlzcG9zaXRpb246IGZvcm0tZGF0YTsgbmFtZT1cIiIrIGZpZWxkWzBdICsiXCI7IGZpbGVuYW1lPVwiIisgZmlsZW5hbWUgKyJcIlxyXG4iOwoJCQkJYm9keSArPSAiQ29udGVudC1UeXBlOiAiKyBmaWxlLnR5cGUgKyJcclxuXHJcbiI7CgkJCQlib2R5ICs9IGZpbGUuZ2V0QXNCaW5hcnkoKSArICJcclxuIjsKCQkJfSBlbHNlIHsKCQkJCS8vIGtleS12YWx1ZSBmaWVsZAoJCQkJYm9keSArPSAiQ29udGVudC1EaXNwb3NpdGlvbjogZm9ybS1kYXRhOyBuYW1lPVwiIisgZmllbGRbMF0gKyJcIjtcclxuXHJcbiI7CgkJCQlib2R5ICs9IGZpZWxkWzFdICsgIlxyXG4iOwoJCQl9CgkJfSk7CgkJYm9keSArPSAiLS0iICsgYm91bmRhcnkgKyItLSI7CgkJcmV0dXJuIGJvZHk7Cgl9OwoJZW55by5Gb3JtRGF0YSA9IEZvcm1EYXRhOwoKCWZ1bmN0aW9uIEJsb2IoaW5CdWZzLCBpbk9wdHMpIHsKCQl0aGlzLm5hbWUgPSBpbk9wdHMubmFtZTsKCQl0aGlzLnR5cGUgPSBpbk9wdHMudHlwZSB8fCAnYXBwbGljYXRpb24vb2N0ZXQtc3RyZWFtJzsKCQlpZiAoIWVueW8uaXNBcnJheShpbkJ1ZnMpKSB7CgkJCXRocm93IG5ldyBFcnJvcignZW55by5CbG9iIG9ubHkgaGFuZGxlcyBBcnJheXMgb2YgU3RyaW5ncycpOwoJCX0KCQlpZiAoKGluQnVmcy5sZW5ndGggPiAwKSAmJiB0eXBlb2YgaW5CdWZzWzBdICE9PSAnc3RyaW5nJykgewoJCQl0aHJvdyBuZXcgRXJyb3IoJ2VueW8uQmxvYiBvbmx5IGhhbmRsZXMgQXJyYXlzIG9mIFN0cmluZ3MnKTsKCQl9CgkJdGhpcy5fYnVmcyA9IGluQnVmczsgLy8gbGVhdmUgYnl0ZSBhcnJheXMgdW4tdG91Y2hlZAoJfQoJQmxvYi5wcm90b3R5cGUuZ2V0QXNCaW5hcnkgPSBmdW5jdGlvbigpIHsKCQl2YXIgZW1wdHkgPSAnJywKCQkJY29udGVudCA9IGVtcHR5LmNvbmNhdC5hcHBseShlbXB0eSwgdGhpcy5fYnVmcyk7CgkJcmV0dXJuIGNvbnRlbnQ7Cgl9OwoJZW55by5CbG9iID0gQmxvYjsKCn0pKHdpbmRvdyk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ajax/json.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5qc29uID0gewoJLy8qIEBwdWJsaWMKCS8qKgoJCVJldHVybnMgYSBKU09OIHN0cmluZyBmb3IgYSBnaXZlbiBvYmplY3QsIHVzaW5nIG5hdGl2ZSBzdHJpbmdpZnkKCQlyb3V0aW5lLgoJCTxpPmluVmFsdWU8L2k+IGlzIHRoZSBPYmplY3QgdG8gYmUgY29udmVydGVkIHRvIEpTT04uCgkJPGk+aW5SZXBsYWNlcjwvaT4gaXMgdGhlIG9wdGlvbmFsIHZhbHVlIGluY2x1c2lvbiBhcnJheSBvciByZXBsYWNlbWVudCBmdW5jdGlvbi4KCQk8aT5pblNwYWNlPC9pPiBpcyB0aGUgb3B0aW9uYWwgbnVtYmVyIG9yIHN0cmluZyB0byB1c2UgZm9yIHByZXR0eS1wcmludGluZyB3aGl0ZXNwYWNlLgoJKi8KCXN0cmluZ2lmeTogZnVuY3Rpb24oaW5WYWx1ZSwgaW5SZXBsYWNlciwgaW5TcGFjZSkgewoJCXJldHVybiBKU09OLnN0cmluZ2lmeShpblZhbHVlLCBpblJlcGxhY2VyLCBpblNwYWNlKTsKCX0sCgkvKioKCQlSZXR1cm5zIGEgSmF2YVNjcmlwdCBvYmplY3QgZm9yIGEgZ2l2ZW4gSlNPTiBzdHJpbmcsIHVzaW5nIG5hdGl2ZSBzdHJpbmdpZnkKCQlyb3V0aW5lLgoJCTxpPmluSnNvbjwvaT4gaXMgdGhlIEpTT04gc3RyaW5nIHRvIGJlIGNvbnZlcnRlZCB0byBhIEphdmFTY3JpcHQgb2JqZWN0LgoJKi8KCXBhcnNlOiBmdW5jdGlvbihpbkpzb24sIGluUmV2aXZlcikgewoJCXJldHVybiBpbkpzb24gPyBKU09OLnBhcnNlKGluSnNvbiwgaW5SZXZpdmVyKSA6IG51bGw7Cgl9Cn07Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ajax/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJIkFzeW5jLmpzIiwKCSJqc29uLmpzIiwKCSJjb29raWUuanMiLAoJInhoci5qcyIsCgkiZm9ybWRhdGEuanMiLAoJIkFqYXhQcm9wZXJ0aWVzLmpzIiwKCSJBamF4LmpzIiwKCSJKc29ucC5qcyIsCgkiV2ViU2VydmljZS5qcyIKKTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ajax/xhr.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qIEBwcm90ZWN0ZWQKLyoqCglJZiB5b3UgbWFrZSBjaGFuZ2VzIHRvIF9lbnlvLnhocl8sIGJlIHN1cmUgdG8gYWRkIG9yIHVwZGF0ZSB0aGUgYXBwcm9wcmlhdGUKCVt1bml0IHRlc3RzXShodHRwczovL2dpdGh1Yi5jb20vZW55b2pzL2VueW8vdHJlZS9tYXN0ZXIvdG9vbHMvdGVzdC9hamF4L3Rlc3RzKS4KKi8KZW55by54aHIgPSB7CgkvKioKCQk8Y29kZT5pblBhcmFtczwvY29kZT4gaXMgYW4gT2JqZWN0IHRoYXQgbWF5IGNvbnRhaW4gdGhlc2UgcHJvcGVydGllczoKCgkJLSBfdXJsXzogVGhlIFVSTCB0byByZXF1ZXN0IChyZXF1aXJlZCkuCgkJLSBfbWV0aG9kXzogVGhlIEhUVFAgbWV0aG9kIHRvIHVzZSBmb3IgdGhlIHJlcXVlc3QuIERlZmF1bHRzIHRvIEdFVC4KCQktIF9jYWxsYmFja186IENhbGxlZCB3aGVuIHJlcXVlc3QgaXMgY29tcGxldGVkLiAoT3B0aW9uYWwpCgkJLSBfYm9keV86IFNwZWNpZmljIGNvbnRlbnRzIGZvciB0aGUgcmVxdWVzdCBib2R5IGZvciBQT1NUIG1ldGhvZC4gKE9wdGlvbmFsKQoJCS0gX2hlYWRlcnNfOiBBZGRpdGlvbmFsIHJlcXVlc3QgaGVhZGVycy4gKE9wdGlvbmFsKS4gIEdpdmVuIGhlYWRlcnMgb3ZlcnJpZGUgdGhlIG9uZXMgdGhhdCBFbnlvIG1heSBzZXQgYnkgZGVmYXVsdCAoYG51bGxgIGV4cGxpY3RseSByZW1vdmluZyB0aGUgaGVhZGVyIGZyb20gdGhlIEFKQVggcmVxdWVzdCkuCgkJLSBfdXNlcm5hbWVfOiBUaGUgb3B0aW9uYWwgdXNlciBuYW1lIHRvIHVzZSBmb3IgYXV0aGVudGljYXRpb24gcHVycG9zZXMuCgkJLSBfcGFzc3dvcmRfOiBUaGUgb3B0aW9uYWwgcGFzc3dvcmQgdG8gdXNlIGZvciBhdXRoZW50aWNhdGlvbiBwdXJwb3Nlcy4KCQktIF94aHJGaWVsZHNfOiBPcHRpb25hbCBvYmplY3QgY29udGFpbmluZyBuYW1lL3ZhbHVlIHBhaXJzIHRvIG1peCBkaXJlY3RseSBpbnRvIHRoZSBnZW5lcmF0ZWQgeGhyIG9iamVjdC4KCQktIF9taW1lVHlwZV86IE9wdGlvbmFsIHN0cmluZyB0byBvdmVycmlkZSB0aGUgTUlNRS1UeXBlLgoKCQlOb3RlOiBvbiBpT1MgNiwgd2Ugd2lsbCBleHBsaWNpdHkgYWRkIGEgImNhY2hlLWNvbnRyb2w6IG5vLWNhY2hlIgoJCWhlYWRlciBmb3IgYW55IG5vbi1HRVQgcmVxdWVzdHMgdG8gd29ya2Fyb3VuZCBhIHN5c3RlbSBidWcgdGhhdCBjYXVzZWQKCQlub24tY2FjaGFibGUgcmVxdWVzdHMgdG8gYmUgY2FjaGVkLiBUbyBkaXNhYmxlIHRoaXMsIHVzZSB0aGUgX2hlYWRlcl8KCQlwcm9wZXJ0eSB0byBzcGVjaWZ5IGFuIG9iamVjdCB3aGVyZSAiY2FjaGUtY29udHJvbCIgaXMgc2V0IHRvIG51bGwuCgkqLwoJcmVxdWVzdDogZnVuY3Rpb24oaW5QYXJhbXMpIHsKCQl2YXIgeGhyID0gdGhpcy5nZXRYTUxIdHRwUmVxdWVzdChpblBhcmFtcyk7CgkJdmFyIHVybCA9IGVueW8ucGF0aC5yZXdyaXRlKHRoaXMuc2ltcGxpZnlGaWxlVVJMKGluUGFyYW1zLnVybCkpOwoJCS8vCgkJdmFyIG1ldGhvZCA9IGluUGFyYW1zLm1ldGhvZCB8fCAiR0VUIjsKCQl2YXIgYXN5bmMgPSAhaW5QYXJhbXMuc3luYzsKCQkvLwoJCWlmIChpblBhcmFtcy51c2VybmFtZSkgewoJCQl4aHIub3BlbihtZXRob2QsIHVybCwgYXN5bmMsIGluUGFyYW1zLnVzZXJuYW1lLCBpblBhcmFtcy5wYXNzd29yZCk7CgkJfSBlbHNlIHsKCQkJeGhyLm9wZW4obWV0aG9kLCB1cmwsIGFzeW5jKTsKCQl9CgkJLy8KCQllbnlvLm1peGluKHhociwgaW5QYXJhbXMueGhyRmllbGRzKTsKCQkvLyBvbmx5IHNldHVwIGhhbmRsZXIgd2hlbiB3ZSBoYXZlIGEgY2FsbGJhY2sKCQlpZiAoaW5QYXJhbXMuY2FsbGJhY2spIHsKCQkJdGhpcy5tYWtlUmVhZHlTdGF0ZUhhbmRsZXIoeGhyLCBpblBhcmFtcy5jYWxsYmFjayk7CgkJfQoJCS8vCgkJaW5QYXJhbXMuaGVhZGVycyA9IGluUGFyYW1zLmhlYWRlcnMgfHwge307CgkJLy8gd29yayBhcm91bmQgaU9TIDYuMCBidWcgd2hlcmUgbm9uLUdFVCByZXF1ZXN0cyBhcmUgY2FjaGVkCgkJLy8gc2VlIGh0dHA6Ly93d3cuZWludGVybmFscy5jb20vYmxvZy93ZWItZGV2ZWxvcG1lbnQvaW9zNi0wLWNhY2hpbmctYWpheC1wb3N0LXJlcXVlc3RzCgkJaWYgKG1ldGhvZCAhPT0gIkdFVCIgJiYgZW55by5wbGF0Zm9ybS5pb3MgJiYgZW55by5wbGF0Zm9ybS5pb3MgPT0gNikgewoJCQlpZiAoaW5QYXJhbXMuaGVhZGVyc1siY2FjaGUtY29udHJvbCJdICE9PSBudWxsKSB7CgkJCQlpblBhcmFtcy5oZWFkZXJzWyJjYWNoZS1jb250cm9sIl0gPSBpblBhcmFtcy5oZWFkZXJzWydjYWNoZS1jb250cm9sJ10gfHwgIm5vLWNhY2hlIjsKCQkJfQoJCX0KCQkvLyB1c2VyLXNldCBoZWFkZXJzIG92ZXJyaWRlIGFueSBwbGF0Zm9ybS1kZWZhdWx0CgkJaWYgKHhoci5zZXRSZXF1ZXN0SGVhZGVyKSB7CgkJCWZvciAodmFyIGtleSBpbiBpblBhcmFtcy5oZWFkZXJzKSB7CgkJCQlpZiAoaW5QYXJhbXMuaGVhZGVyc1trZXldKSB7CgkJCQkJeGhyLnNldFJlcXVlc3RIZWFkZXIoa2V5LCBpblBhcmFtcy5oZWFkZXJzW2tleV0pOwoJCQkJfQoJCQl9CgkJfQoJCS8vCgkJaWYoKHR5cGVvZiB4aHIub3ZlcnJpZGVNaW1lVHlwZSA9PSAiZnVuY3Rpb24iKSAmJiBpblBhcmFtcy5taW1lVHlwZSkgewoJCQl4aHIub3ZlcnJpZGVNaW1lVHlwZShpblBhcmFtcy5taW1lVHlwZSk7CgkJfQoJCS8vCgkJeGhyLnNlbmQoaW5QYXJhbXMuYm9keSB8fCBudWxsKTsKCQlpZiAoIWFzeW5jICYmIGluUGFyYW1zLmNhbGxiYWNrKSB7CgkJCXhoci5vbnJlYWR5c3RhdGVjaGFuZ2UoeGhyKTsKCQl9CgkJcmV0dXJuIHhocjsKCX0sCgkvLyogcmVtb3ZlIGFueSBjYWxsYmFja3MgdGhhdCBtaWdodCBiZSBzZXQgZnJvbSBFbnlvIGNvZGUgZm9yIGFuIGV4aXN0aW5nIFhIUgoJLy8qIGFuZCBzdG9wIHRoZSBYSFIgZnJvbSBjb21wbGV0aW5nLgoJY2FuY2VsOiBmdW5jdGlvbihpblhocikgewoJCWlmIChpblhoci5vbmxvYWQpIHsKCQkJaW5YaHIub25sb2FkID0gbnVsbDsKCQl9CgkJaWYgKGluWGhyLm9ucmVhZHlzdGF0ZWNoYW5nZSkgewoJCQlpblhoci5vbnJlYWR5c3RhdGVjaGFuZ2UgPSBudWxsOwoJCX0KCQlpZiAoaW5YaHIuYWJvcnQpIHsKCQkJaW5YaHIuYWJvcnQoKTsKCQl9Cgl9LAoJLy8qIEBwcm90ZWN0ZWQKCW1ha2VSZWFkeVN0YXRlSGFuZGxlcjogZnVuY3Rpb24oaW5YaHIsIGluQ2FsbGJhY2spIHsKCQlpZiAod2luZG93LlhEb21haW5SZXF1ZXN0ICYmIGluWGhyIGluc3RhbmNlb2Ygd2luZG93LlhEb21haW5SZXF1ZXN0KSB7CgkJCWluWGhyLm9ubG9hZCA9IGZ1bmN0aW9uKCkgewoJCQkJdmFyIGRhdGE7CgkJCQlpZiAoaW5YaHIucmVzcG9uc2VUeXBlID09PSAiYXJyYXlidWZmZXIiKSB7CgkJCQkJZGF0YSA9IGluWGhyLnJlc3BvbnNlOwoJCQkJfSBlbHNlIGlmICh0eXBlb2YgaW5YaHIucmVzcG9uc2VUZXh0ID09PSAic3RyaW5nIikgewoJCQkJCWRhdGEgPSBpblhoci5yZXNwb25zZVRleHQ7CgkJCQl9CgkJCQlpbkNhbGxiYWNrLmFwcGx5KG51bGwsIFtkYXRhLCBpblhocl0pOwoJCQl9OwoJCX0KCQlpblhoci5vbnJlYWR5c3RhdGVjaGFuZ2UgPSBmdW5jdGlvbigpIHsKCQkJaWYgKGluWGhyLnJlYWR5U3RhdGUgPT0gNCkgewoJCQkJdmFyIGRhdGE7CgkJCQlpZiAoaW5YaHIucmVzcG9uc2VUeXBlID09PSAiYXJyYXlidWZmZXIiKSB7CgkJCQkJZGF0YSA9IGluWGhyLnJlc3BvbnNlOwoJCQkJfSBlbHNlIGlmICh0eXBlb2YgaW5YaHIucmVzcG9uc2VUZXh0ID09PSAic3RyaW5nIikgewoJCQkJCWRhdGEgPSBpblhoci5yZXNwb25zZVRleHQ7CgkJCQl9CgkJCQlpbkNhbGxiYWNrLmFwcGx5KG51bGwsIFtkYXRhLCBpblhocl0pOwoJCQl9CgkJfTsKCX0sCglpbk9yaWdpbjogZnVuY3Rpb24oaW5VcmwpIHsKCQl2YXIgYSA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoImEiKSwgcmVzdWx0ID0gZmFsc2U7CgkJYS5ocmVmID0gaW5Vcmw7CgkJLy8gcHJvdG9jb2wgaXMgIjoiIGZvciByZWxhdGl2ZSBVUkxzCgkJaWYgKGEucHJvdG9jb2wgPT09ICI6IiB8fAoJCQkJKGEucHJvdG9jb2wgPT09IHdpbmRvdy5sb2NhdGlvbi5wcm90b2NvbCAmJgoJCQkJCWEuaG9zdG5hbWUgPT09IHdpbmRvdy5sb2NhdGlvbi5ob3N0bmFtZSAmJgoJCQkJCWEucG9ydCA9PT0gKHdpbmRvdy5sb2NhdGlvbi5wb3J0IHx8CgkJCQkJCSh3aW5kb3cubG9jYXRpb24ucHJvdG9jb2wgPT09ICJodHRwczoiID8gIjQ0MyIgOiAiODAiKSkpKSB7CgkJCXJlc3VsdCA9IHRydWU7CgkJfQoJCXJldHVybiByZXN1bHQ7Cgl9LAoJc2ltcGxpZnlGaWxlVVJMOiBmdW5jdGlvbihpblVybCkgewoJCXZhciBhID0gZG9jdW1lbnQuY3JlYXRlRWxlbWVudCgiYSIpOwoJCWEuaHJlZiA9IGluVXJsOwoJCS8vIHByb3RvY29sIGlzICI6IiBmb3IgcmVsYXRpdmUgVVJMcwoJCWlmIChhLnByb3RvY29sID09PSAiZmlsZToiIHx8CgkJCWEucHJvdG9jb2wgPT09ICI6IiAmJiB3aW5kb3cubG9jYXRpb24ucHJvdG9jb2wgPT09ICJmaWxlOiIpIHsKCQkJLy8gbGVhdmUgb2ZmIHNlYXJjaCBhbmQgaGFzaCBwYXJ0cyBvZiB0aGUgVVJMCgkJCXJldHVybiBhLnByb3RvY29sICsgJy8vJyArIGEuaG9zdCArIGEucGF0aG5hbWU7CgkJfSBlbHNlIGlmIChhLnByb3RvY29sID09PSAiOiIgJiYgd2luZG93LmxvY2F0aW9uLnByb3RvY29sID09PSAieC13bWFwcDA6IikgewoJCQkvLyBleHBsaWNpdGx5IHJldHVybiBhYnNvbHV0ZSBVUkwgZm9yIFdpbmRvd3MgUGhvbmUgOCwgYXMgYW4gYWJzb2x1dGUgcGF0aCBpcyByZXF1aXJlZCBmb3IgbG9jYWwgZmlsZXMKCQkJcmV0dXJuIHdpbmRvdy5sb2NhdGlvbi5wcm90b2NvbCArICIvLyIgKyB3aW5kb3cubG9jYXRpb24ucGF0aG5hbWUuc3BsaXQoJy8nKVswXSArICIvIiArIGEuaG9zdCArIGEucGF0aG5hbWU7CgkJfSBlbHNlIHsKCQkJcmV0dXJuIGluVXJsOwoJCX0KCX0sCglnZXRYTUxIdHRwUmVxdWVzdDogZnVuY3Rpb24oaW5QYXJhbXMpIHsKCQl0cnkgewoJCQkvLyBvbmx5IHVzZSBYRG9tYWluUmVxdWVzdCB3aGVuIGl0IGV4aXN0cywgbm8gZXh0cmEgaGVhZGVycyB3ZXJlIHNldCwgYW5kIHRoZQoJCQkvLyB0YXJnZXQgVVJMIG1hcHMgdG8gYSBkb21haW4gb3RoZXIgdGhhbiB0aGUgZG9jdW1lbnQgb3JpZ2luLgoJCQlpZiAoZW55by5wbGF0Zm9ybS5pZSA8IDEwICYmIHdpbmRvdy5YRG9tYWluUmVxdWVzdCAmJiAhaW5QYXJhbXMuaGVhZGVycyAmJgoJCQkJIXRoaXMuaW5PcmlnaW4oaW5QYXJhbXMudXJsKSAmJiAhL15maWxlOlwvXC8vLnRlc3Qod2luZG93LmxvY2F0aW9uLmhyZWYpKSB7CgkJCQlyZXR1cm4gbmV3IHdpbmRvdy5YRG9tYWluUmVxdWVzdCgpOwoJCQl9CgkJfSBjYXRjaChlKSB7fQoJCXRyeSB7CgkJCXJldHVybiBuZXcgWE1MSHR0cFJlcXVlc3QoKTsKCQl9IGNhdGNoKGUpIHt9CgkJcmV0dXJuIG51bGw7Cgl9Cn07Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/boot/boot.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8gbWFjaGluZSBmb3IgYSBsb2FkZXIgaW5zdGFuY2UKZW55by5tYWNoaW5lID0gewoJc2hlZXQ6IGZ1bmN0aW9uKGluUGF0aCkgewoJCXZhciB0eXBlID0gInRleHQvY3NzIjsKCQl2YXIgcmVsID0gInN0eWxlc2hlZXQiOwoJCXZhciBpc0xlc3MgPSAoaW5QYXRoLnNsaWNlKC01KSA9PSAiLmxlc3MiKTsKCQlpZiAoaXNMZXNzKSB7CgkJCWlmICh3aW5kb3cubGVzcykgewoJCQkJLy8gSWYgY2xpZW50LXNpZGUgbGVzcyBpcyBsb2FkZWQsIGluc2VydCB0aGUgbGVzcyBzdHlsZXNoZWV0CgkJCQl0eXBlID0gInRleHQvbGVzcyI7CgkJCQlyZWwgPSAic3R5bGVzaGVldC9sZXNzIjsKCQkJfSBlbHNlIHsKCQkJCS8vIE90aGVyd2lzZSwgd2UgZXhwZWN0IGEgY3NzIGZpbGUgb2YgdGhlIHNhbWUgbmFtZSB0byBleGlzdAoJCQkJaW5QYXRoID0gaW5QYXRoLnNsaWNlKDAsIGluUGF0aC5sZW5ndGgtNCkgKyAiY3NzIjsKCQkJfQoJCX0KCQl2YXIgbGluazsKCQlpZiAoZW55by5ydW50aW1lTG9hZGluZyB8fCBpc0xlc3MpIHsKCQkJbGluayA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoJ2xpbmsnKTsKCQkJbGluay5ocmVmID0gaW5QYXRoOwoJCQlsaW5rLm1lZGlhID0gInNjcmVlbiI7CgkJCWxpbmsucmVsID0gcmVsOwoJCQlsaW5rLnR5cGUgPSB0eXBlOwoJCQlkb2N1bWVudC5nZXRFbGVtZW50c0J5VGFnTmFtZSgnaGVhZCcpWzBdLmFwcGVuZENoaWxkKGxpbmspOwoJCX0gZWxzZSB7CgkJCS8qIGpzaGludCBldmlsOiB0cnVlICovCgkJCWRvY3VtZW50LndyaXRlKAoJCQkJJzxsaW5rIGhyZWY9IicgKyBpblBhdGggKyAnIiBtZWRpYT0ic2NyZWVuIiByZWw9IicgKwoJCQkJcmVsICsgJyIgdHlwZT0iJyArIHR5cGUgKyAnIiAvPicpOwoJCQkvKiBqc2hpbnQgZXZpbDogZmFsc2UgKi8KCQl9CgkJaWYgKGlzTGVzcyAmJiB3aW5kb3cubGVzcykgewoJCQl3aW5kb3cubGVzcy5zaGVldHMucHVzaChsaW5rKTsKCQkJaWYgKCFlbnlvLmxvYWRlci5maW5pc2hDYWxsYmFja3MubGVzc1JlZnJlc2gpIHsKCQkJCWVueW8ubG9hZGVyLmZpbmlzaENhbGxiYWNrcy5sZXNzUmVmcmVzaCA9IGZ1bmN0aW9uKCkgewoJCQkJCXdpbmRvdy5sZXNzLnJlZnJlc2godHJ1ZSk7CgkJCQl9OwoJCQl9CgkJfQoJfSwKCXNjcmlwdDogZnVuY3Rpb24oaW5TcmMsIG9uTG9hZCwgb25FcnJvcikgewoJCWlmIChlbnlvLnJ1bnRpbWVMb2FkaW5nKSB7CgkJCXZhciBzY3JpcHQgPSBkb2N1bWVudC5jcmVhdGVFbGVtZW50KCdzY3JpcHQnKTsKCQkJc2NyaXB0LnNyYyA9IGluU3JjOwoJCQlzY3JpcHQub25sb2FkID0gb25Mb2FkOwoJCQlzY3JpcHQub25lcnJvciA9IG9uRXJyb3I7CgkJCXNjcmlwdC5jaGFyc2V0ID0gInV0Zi04IjsKCQkJZG9jdW1lbnQuZ2V0RWxlbWVudHNCeVRhZ05hbWUoJ2hlYWQnKVswXS5hcHBlbmRDaGlsZChzY3JpcHQpOwoJCX0gZWxzZSB7CgkJCS8qIGpzaGludCBldmlsOiB0cnVlICovCgkJCWRvY3VtZW50LndyaXRlKAoJCQkJJzxzY3JpJyArICdwdCBzcmM9IicgKyBpblNyYyArICciJyArCgkJCQkob25Mb2FkID8gJyBvbmxvYWQ9IicgKyBvbkxvYWQgKyAnIicgOiAnJykgKwoJCQkJKG9uRXJyb3IgPyAnIG9uZXJyb3I9IicgKyBvbkVycm9yICsgJyInIDogJycpICsKCQkJCSc+PC9zY3JpJyArICdwdD4nKTsKCQkJLyoganNoaW50IGV2aWw6IGZhbHNlICovCgkJfQoJfSwKCWluamVjdDogZnVuY3Rpb24oaW5Db2RlKSB7CgkJLyoganNoaW50IGV2aWw6IHRydWUgKi8KCQlkb2N1bWVudC53cml0ZSgnPHNjcmknICsgJ3B0IHR5cGU9InRleHQvamF2YXNjcmlwdCI+JyArIGluQ29kZSArICI8L3NjcmkiICsgInB0PiIpOwoJCS8qIGpzaGludCBldmlsOiBmYWxzZSAqLwoJfQp9OwoKLy8gY3JlYXRlIGEgZGVwZW5kZW5jeSBwcm9jZXNzb3IgdXNpbmcgb3VyIHNjcmlwdCBtYWNoaW5lCmVueW8ubG9hZGVyID0gbmV3IGVueW8ubG9hZGVyRmFjdG9yeShlbnlvLm1hY2hpbmUpOwoKLy8gZGVwZW5kZW5jeSBBUEkgdXNlcyBlbnlvIGxvYWRlcgplbnlvLmRlcGVuZHMgPSBmdW5jdGlvbigpIHsKCXZhciBsZHIgPSBlbnlvLmxvYWRlcjsKCWlmICghbGRyLnBhY2thZ2VGb2xkZXIpIHsKCQl2YXIgdGFnID0gZW55by5sb2NhdGVTY3JpcHQoInBhY2thZ2UuanMiKTsKCQlpZiAodGFnICYmIHRhZy5wYXRoKSB7CgkJCWxkci5hbGlhc1BhY2thZ2UodGFnLnBhdGgpOwoJCQlsZHIucGFja2FnZUZvbGRlciA9IHRhZy5wYXRoICsgIi8iOwoJCQkvL2NvbnNvbGUubG9nKCJkZXRlY3RlZCBQQUNLQUdFRk9MREVSIFsiICsgbGRyLnBhY2thZ2VGb2xkZXIgKyAiXSIpOwoJCX0KCX0KCWxkci5sb2FkLmFwcGx5KGxkciwgYXJndW1lbnRzKTsKfTsKCi8vIFJ1bnRpbWUgbG9hZGVyCi8vIFVzYWdlOiBlbnlvLmxvYWQoZGVwZW5kcywgW29uTG9hZENhbGxiYWNrXSkKLy8gIHdoZXJlIC0gZGVwZW5kcyBpcyBzdHJpbmcgb3IgYXJyYXkgb2Ygc3RyaW5nIHBhdGhzIHRvIHBhY2thZ2UuanMsIHNjcmlwdCwgb3IgY3NzIHRvIGxvYWQKLy8gICAgICAgIC0gZG9uZUNhbGxiYWNrIGlzIGZpcmVkIGFmdGVyIGZpbGUgb3IgcGFja2FnZSBsb2FkaW5nIGhhcyBjb21wbGV0ZWQKLy8gT25seSBvbmUgZmlsZS9wYWNrYWdlIGlzIGxvYWRlZCBhdCBhIHRpbWU7IGFkZGl0aW9uYWwgY2FsbHMgYXJlIHF1ZXVlZCBhbmQgbG9hZGluZyBkZWZlcnJlZAooZnVuY3Rpb24oKSB7Cgl2YXIgZW55byA9IHdpbmRvdy5lbnlvOwoJdmFyIHJ1bnRpbWVMb2FkUXVldWUgPSBbXTsKCWVueW8ubG9hZCA9IGZ1bmN0aW9uKGRlcGVuZHMsIG9uTG9hZENhbGxiYWNrKSB7CgkJcnVudGltZUxvYWRRdWV1ZS5wdXNoKGFyZ3VtZW50cyk7CgkJaWYgKCFlbnlvLnJ1bnRpbWVMb2FkaW5nKSB7CgkJCWVueW8ucnVudGltZUxvYWRpbmcgPSB0cnVlOwoJCQlydW50aW1lTG9hZCgpOwoJCX0KCX07CglmdW5jdGlvbiBydW50aW1lTG9hZChvbkxvYWQpIHsKCQlpZiAob25Mb2FkKSB7CgkJCW9uTG9hZCgpOyAvLyBSdW4gdXNlciBjYWxsYmFjayBmdW5jdGlvbgoJCX0KCQlpZiAocnVudGltZUxvYWRRdWV1ZS5sZW5ndGgpIHsKCQkJdmFyIGFyZ3MgPSBydW50aW1lTG9hZFF1ZXVlLnNoaWZ0KCk7CgkJCXZhciBkZXBlbmRzID0gYXJnc1swXTsKCQkJdmFyIGRlcGVuZHNBcmcgPSBlbnlvLmlzQXJyYXkoZGVwZW5kcykgPyBkZXBlbmRzIDogW2RlcGVuZHNdOwoJCQl2YXIgb25Mb2FkQ2FsbGJhY2sgPSBhcmdzWzFdOwoJCQllbnlvLmxvYWRlci5maW5pc2hDYWxsYmFja3MucnVudGltZUxvYWRlciA9IGZ1bmN0aW9uKGluQmxvY2spIHsKCQkJCS8vIE9uY2UgbG9hZGVyIGlzIGRvbmUgbG9hZGluZyBhIHBhY2thZ2UsIHdlIGNoYWluIGEgY2FsbCB0byBydW50aW1lTG9hZCgpLAoJCQkJLy8gd2hpY2ggd2lsbCBjYWxsIHRoZSBvbkxvYWRDYWxsYmFjayBmcm9tIHRoZSBvcmlnaW5hbCBsb2FkIGNhbGwsIHBhc3NpbmcKCQkJCS8vIGEgcmVmZXJlbmNlIHRvIHRoZSBkZXBlbmRzIGFyZ3VtZW50IGZyb20gdGhlIG9yaWdpbmFsIGNhbGwgZm9yIHRyYWNraW5nLAoJCQkJLy8gZm9sbG93ZWQgYnkga2lja2luZyBvZmYgYW55IGFkZGl0aW9uYWxseSBxdWV1ZWQgbG9hZCgpIGNhbGxzCgkJCQlydW50aW1lTG9hZChmdW5jdGlvbigpIHsKCQkJCQlpZiAob25Mb2FkQ2FsbGJhY2spIHsKCQkJCQkJb25Mb2FkQ2FsbGJhY2soaW5CbG9jayk7CgkJCQkJfQoJCQkJfSk7CgkJCX07CgkJCWVueW8ubG9hZGVyLnBhY2thZ2VGb2xkZXIgPSAiLi8iOwoJCQkvLyBLaWNrIG9mZiBuZXh0IHF1ZXVlZCBjYWxsIHRvIGxvYWRlcgoJCQllbnlvLmRlcGVuZHMuYXBwbHkodGhpcywgZGVwZW5kc0FyZyk7CgkJfSBlbHNlIHsKCQkJZW55by5ydW50aW1lTG9hZGluZyA9IGZhbHNlOwoJCQllbnlvLmxvYWRlci5wYWNrYWdlRm9sZGVyID0gIiI7CgkJfQoJfQp9KSgpOwoKLy8gcHJlZGVmaW5lZCBwYXRoIGFsaWFzZXMKZW55by5wYXRoLmFkZFBhdGhzKHsKCWVueW86IGVueW8uYXJncy5yb290LAoJbGliOiAiJGVueW8vLi4vbGliIgp9KTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/boot/enyo.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uKCkgewoJLy8gZW55byBjYW4gdXNlIGluZm9ybWF0aW9uIGZyb20gdGhlIHNjcmlwdCB0YWcgdGhhdCBsb2FkcyB0aGlzIGJvb3RzdHJhcCBmaWxlCgl2YXIgdGhpc1NjcmlwdCA9ICJlbnlvLmpzIjsKCgkvKiBnbG9iYWwgZW55bzp0cnVlICovCgllbnlvID0gd2luZG93LmVueW8gfHwge307CgoJZW55by5sb2NhdGVTY3JpcHQgPSBmdW5jdGlvbihpbk5hbWUpIHsKCQl2YXIgc2NyaXB0cyA9IGRvY3VtZW50LmdldEVsZW1lbnRzQnlUYWdOYW1lKCJzY3JpcHQiKTsKCQlmb3IgKHZhciBpPXNjcmlwdHMubGVuZ3RoLTEsIHMsIHNyYywgbD1pbk5hbWUubGVuZ3RoOyAoaT49MCkgJiYgKHM9c2NyaXB0c1tpXSk7IGktLSkgewoJCQlpZiAoIXMubG9jYXRlZCkgewoJCQkJc3JjID0gcy5nZXRBdHRyaWJ1dGUoInNyYyIpIHx8ICIiOwoJCQkJaWYgKHNyYy5zbGljZSgtbCkgPT0gaW5OYW1lKSB7CgkJCQkJcy5sb2NhdGVkID0gdHJ1ZTsKCQkJCQlyZXR1cm4ge3BhdGg6IHNyYy5zbGljZSgwLCBNYXRoLm1heCgwLCBzcmMubGFzdEluZGV4T2YoIi8iKSkpLCBub2RlOiBzfTsKCQkJCX0KCQkJfQoJCX0KCX07CgoJZW55by5hcmdzID0gZW55by5hcmdzIHx8IHt9OwoKCXZhciB0YWcgPSBlbnlvLmxvY2F0ZVNjcmlwdCh0aGlzU2NyaXB0KTsKCWlmICh0YWcpIHsKCQkvLyBpbmZlciB0aGUgZnJhbWV3b3JrIHBhdGggZnJvbSB0aGUgZG9jdW1lbnQsIHVubGVzcyB0aGUgdXNlciBoYXMgc3BlY2lmaWVkIG9uZSBleHBsaWNpdGx5CgkJZW55by5hcmdzLnJvb3QgPSAoZW55by5hcmdzLnJvb3QgfHwgdGFnLnBhdGgpLnJlcGxhY2UoIi9zb3VyY2UiLCAiIik7CgkJLy8gYWxsIGF0dHJpYnV0ZXMgb2YgdGhlIGJvb3RzdHJhcCBzY3JpcHQgdGFnIGJlY29tZSBlbnlvLmFyZ3MKCQlmb3IgKHZhciBpPTAsIGFsID0gdGFnLm5vZGUuYXR0cmlidXRlcy5sZW5ndGgsIGl0OyAoaSA8IGFsKSAmJiAoaXQgPSB0YWcubm9kZS5hdHRyaWJ1dGVzLml0ZW0oaSkpOyBpKyspIHsKCQkJZW55by5hcmdzW2l0Lm5vZGVOYW1lXSA9IGl0LnZhbHVlOwoJCX0KCX0KfSkoKTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/boot/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJImVueW8uanMiLAoJInJlYWR5LmpzIiwKCSIuLi8uLi9sb2FkZXIuanMiLAoJImJvb3QuanMiCik7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/boot/ready.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uIChzY29wZSkgewoKCS8vIHdlIG5lZWQgdG8gcmVnaXN0ZXIgYXBwcm9wcmlhdGVseSB0byBrbm93IHdoZW4KCS8vIHRoZSBkb2N1bWVudCBpcyBvZmZpY2lhbGx5IHJlYWR5LCB0byBlbnN1cmUgdGhhdAoJLy8gY2xpZW50IGNvZGUgaXMgb25seSBnb2luZyB0byBleGVjdXRlIGF0IHRoZQoJLy8gYXBwcm9wcmlhdGUgdGltZQoKCXZhciBkb2MgPSBzY29wZS5kb2N1bWVudDsKCXZhciBxdWV1ZSA9IFtdOwoJdmFyIHJlYWR5ID0gKCJjb21wbGV0ZSIgPT09IGRvYy5yZWFkeVN0YXRlKTsKCXZhciBydW47Cgl2YXIgaW5pdDsKCXZhciByZW1vdmU7Cgl2YXIgYWRkOwoJdmFyIGZsdXNoOwoKCWVueW8ucmVhZHkgPSBmdW5jdGlvbiAoZm4sIGNvbnRleHQpIHsKCQlpZiAocmVhZHkpIHsKCQkJcnVuKGZuLCBjb250ZXh0KTsKCQl9CgkJZWxzZSB7CgkJCXF1ZXVlLnB1c2goW2ZuLCBjb250ZXh0XSk7CgkJfQoJfTsKCglydW4gPSBmdW5jdGlvbiAoZm4sIGNvbnRleHQpIHsKCQlmbi5jYWxsKGNvbnRleHQgfHwgZW55by5nbG9iYWwpOwoJfTsKCglpbml0ID0gZnVuY3Rpb24gKGV2ZW50KSB7CgkJLy8gaWYgd2UncmUgaW50ZXJhY3RpdmUsIGl0IHNob3VsZCBiZSBzYWZlIHRvIG1vdmUKCQkvLyBmb3J3YXJkIGJlY2F1c2UgdGhlIGNvbnRlbnQgaGFzIGJlZW4gcGFyc2VkCgkJaWYgKChyZWFkeSA9ICgiaW50ZXJhY3RpdmUiID09PSBkb2MucmVhZHlTdGF0ZSkpKSB7CgkJCWlmICghfmVueW8uaW5kZXhPZihldmVudC50eXBlLCBbIkRPTUNvbnRlbnRMb2FkZWQiLCAicmVhZHlzdGF0ZWNoYW5nZSJdKSkgewoJCQkJcmVtb3ZlKGV2ZW50LnR5cGUsIGluaXQpOwoJCQkJZmx1c2goKTsKCQkJfQoJCX0KCQkvLyBmb3IgYW4gSUU4IGZhbGxiYWNrIGFuZCBsZWdhY3kgV2ViS2l0IChpbmNsdWRpbmcgd2ViT1MgMy54IGFuZCBsZXNzKSBhbmQgYXNzdXJhbmNlCgkJaWYgKChyZWFkeSA9ICgiY29tcGxldGUiID09PSBkb2MucmVhZHlTdGF0ZSB8fCAibG9hZGVkIiA9PT0gZG9jLnJlYWR5U3RhdGUpKSkgewoJCQlyZW1vdmUoZXZlbnQudHlwZSwgaW5pdCk7CgkJCWZsdXNoKCk7CgkJfQoJfTsKCglhZGQgPSBmdW5jdGlvbiAoZXZlbnQsIGZuKSB7CgkJdmFyIG5hbWUgPSBkb2MuYWRkRXZlbnRMaXN0ZW5lcj8gImFkZEV2ZW50TGlzdGVuZXIiOiAiYXR0YWNoRXZlbnQiOwoJCXZhciBvbiA9IG5hbWUgPT09ICJhdHRhY2hFdmVudCI/ICJvbiI6ICIiOwoJCWRvY1tuYW1lXShvbiArIGV2ZW50LCBmbiwgZmFsc2UpOwoJfTsKCglyZW1vdmUgPSBmdW5jdGlvbiAoZXZlbnQsIGZuKSB7CgkJdmFyIG5hbWUgPSBkb2MuYWRkRXZlbnRMaXN0ZW5lcj8gInJlbW92ZUV2ZW50TGlzdGVuZXIiOiAiZGV0YWNoRXZlbnQiOwoJCXZhciBvbiA9IG5hbWUgPT09ICJkZXRhY2hFdmVudCI/ICJvbiI6ICIiOwoJCWRvY1tuYW1lXShvbiArIGV2ZW50LCBmbiwgZmFsc2UpOwoJfTsKCglmbHVzaCA9IGZ1bmN0aW9uICgpIHsKCQlpZiAocmVhZHkgJiYgcXVldWUubGVuZ3RoKSB7CgkJCXdoaWxlIChxdWV1ZS5sZW5ndGgpIHsKCQkJCXJ1bi5hcHBseShzY29wZSwgcXVldWUuc2hpZnQoKSk7CgkJCX0KCQl9Cgl9OwoKCS8vIG9rLCBsZXQncyBob29rIHRoaXMgdXAKCWFkZCgiRE9NQ29udGVudExvYWRlZCIsIGluaXQpOwoJYWRkKCJyZWFkeXN0YXRlY2hhbmdlIiwgaW5pdCk7Cgp9KSh3aW5kb3csIGVueW8pOwo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/Control.js"
Content-Type: application/octet-stream; x-encoding=base64
/**
	_enyo.Control_ is a component that controls a DOM node (i.e., an element in
	the user interface). Controls are generally visible and the user often
	interacts with them directly. While things like buttons and input boxes are
	obviously controls, in Enyo, a control may become as complex as an entire
	application.

	If you make changes to _enyo.Control_, be sure to add or update the appropriate
	[unit tests](https://github.com/enyojs/enyo/tree/master/tools/test/core/tests).

	For more information, see the documentation on
	<a href="https://github.com/enyojs/enyo/wiki/Creating-Controls">Controls</a>
	in the Enyo Developer Guide.
*/
enyo.kind({
	name: "enyo.Control",
	kind: "enyo.UiComponent",
	published: {
		/**
			HTML tag name to use for the control. If null, no tag is generated;
			only the contents are used.
		*/
		tag: "div",
		//* Hash of DOM attributes to apply to the generated HTML tag
		attributes: null,
		//* Space-delimited set of CSS classes to apply to the generated HTML tag
		classes: "",
		//* Style attribute to apply to the generated HTML tag
		style: "",
		/**
			Content that will be generated inside the HTML tag; defaults to
			plain text unless _allowHtml_ is true
		*/
		content: "",
		//* Boolean indicating whether the tag will be visible in the document
		showing: true,
		//* If false, HTML codes in _content_ are escaped before rendering
		allowHtml: false,
		//
		// convenience properties for common attributes
		//
		//* Shortcut for setting _src_ attribute in _attributes_ hash. Overrides that value.
		src: "",
		//
		// esoteric
		//
		/**
			Set to false if the control should not generate any HTML. Used to
			inhibit generation of popups until they're shown at runtime.
		*/
		canGenerate: true,
		//
		// ad hoc properties:
		//
		/**
			Flag used by control layouts to determine which control will expand
			to fill the available space. This only has meaning when the control
			is being used as a child of a control with a version of FittableLayout
			as its layoutKind.

			TODO: We would like to be able to test for the existence of the fit
			property without setting a default (of null) since it is a boolean
			flag. This is a temporary fix.
		*/
		fit: null,
		//* Used by Ares design editor for design objects
		isContainer: false,
		model: null
	},
	handlers: {
		//* Controls will call a user-provided _tap_ method when tapped upon.
		ontap: "tap"
	},
	mixins: ["enyo.ControllerSupport"],
	//*@protected
	_isView: true,
	noDefer: true,
	//* The default kind for controls created inside this control that don't
	//* specify their own kind
	defaultKind: "Control",
	//* A set of CSS classes that are applied to controls created inside this control
	controlClasses: "",
	//* @protected
	node: null,
	generated: false,
	create: function() {
		// initialize style databases
		this.initStyles();
		// superkind initialization
		this.inherited(arguments);
		// 'showing' is tertiary method for modifying display style
		// setting 'display: none;' style at initialization time will
		// not work if 'showing' is true.
		this.showingChanged();
		// Notes:
		// - 'classes' does not reflect the complete set of classes on an object; the complete set is in
		//   this.attributes.class. The '*Class' apis affect this.attributes.class.
		// - use addClass instead of setClasses here, by convention 'classes' is reserved for instance objects
		// - inheritors should 'addClass' to add classes
		// - setClasses removes the old classes and adds the new one, setClassAttribute replaces all classes
		this.addClass(this.kindClasses);
		this.addClass(this.classes);
		this.initProps(["id", "content", "src", "model"]);
	},
	destroy: function() {
		this.removeNodeFromDom();
		enyo.Control.unregisterDomEvents(this.id);
		this.inherited(arguments);
	},
	importProps: function(inProps) {
		this.inherited(arguments);
		// each instance has its own attributes array, the union of the prototype attributes and user-specified attributes
		this.attributes = enyo.mixin(enyo.clone(this.kindAttributes), this.attributes);
	},
	initProps: function(inPropNames) {
		// for each named property, trigger the *Changed handler if the property value is truthy
		for (var i=0, n, cf; (n=inPropNames[i]); i++) {
			if (this[n]) {
				// FIXME: awkward
				cf = n + "Changed";
				if (this[cf]) {
					this[cf]();
				}
			}
		}
	},
	//*@protected
	dispatchEvent: function (inEventName, inEvent, inSender) {
		// prevent dispatch and bubble of events that are strictly internal (e.g. enter/leave)
		if (this.strictlyInternalEvents[inEventName] && this.isInternalEvent(inEvent)) {
			return true;
		}

		return this.inherited(arguments);
	},
	classesChanged: function(inOld) {
		this.removeClass(inOld);
		this.addClass(this.classes);
	},
	modelChanged: function () {
		if (this.model && enyo.isModel(this.model)) {
			// if bindings haven't been initialized yet then this
			// would be unnecessary
			if (this._didSetupBindings) {
				this.rebuildBindings();
			}
		}
	},
	// modify components we create ourselves
	/*
	adjustComponentProps: function(inProps) {
		if (this.controlClasses) {
			inProps.classes = (inProps.classes ? inProps.classes + " " : "") + this.controlClasses;
		}
		this.inherited(arguments);
	},
	*/
	addChild: function(inControl) {
		inControl.addClass(this.controlClasses);
		this.inherited(arguments);
	},
	removeChild: function(inControl) {
		this.inherited(arguments);
		inControl.removeClass(this.controlClasses);
	},
	// event filter
	strictlyInternalEvents: {onenter: 1, onleave: 1},
	isInternalEvent: function(inEvent) {
		var rdt = enyo.dispatcher.findDispatchTarget(inEvent.relatedTarget);
		return rdt && rdt.isDescendantOf(this);
	},
	//
	//* @public
	/**
		Returns the DOM node representing the control.
		If the control is not currently rendered, returns null.

		If hasNode() returns a value, the _node_ property will be valid and
		can be checked directly.

		Once hasNode() is called, the returned value is made available in
		the _node_ property of this control.

		A control will only return a node if it has been rendered.

			if (this.hasNode()) {
				enyo.log(this.node.nodeType);
			}
	*/
	hasNode: function() {
		// 'generated' is used to gate access to expensive findNodeById call
		return this.generated && (this.node || this.findNodeById());
	},
	/**
		Appends the string value of _inAddendum_ to the _content_ of this
		control.
	*/
	addContent: function(inAddendum) {
		this.setContent(this.get("content") + inAddendum);
	},
	/**
		Gets the value of an attribute on this object.

		If this control has a node, the attribute value is retrieved from the
		node; otherwise, it's read from the _attributes_ property of the control
		itself.

		Caveat: If the control is rendered, the _attributes_ property is used to
		construct the rendering, and values that have changed on the node itself
		are lost.

			// Get the value attribute for this DomNode
			var value = this.getAttribute("tabIndex");
	*/
	getAttribute: function(inName) {
		return this.hasNode() ? this.node.getAttribute(inName) : this.attributes[inName];
	},
	/**
		Sets the value of an attribute on this object. Pass null _inValue_ to
		remove an attribute.

			// set the tabIndex attribute for this DomNode
			this.setAttribute("tabIndex", 3);
			...
			// remove the index attribute
			this.setAttribute("index", null);
	*/
	setAttribute: function(inName, inValue) {
		this.attributes[inName] = inValue;
		if (this.hasNode()) {
			this.attributeToNode(inName, inValue);
		}
		this.invalidateTags();
	},
	/**
		Gets the value of a property named _inName_ directly from the DOM node.
		A caller-specified default value, _inDefault_, is returned when the DOM
		node has not yet been created.
	*/
	getNodeProperty: function(inName, inDefault) {
		if (this.hasNode()) {
			return this.node[inName];
		} else {
			return inDefault;
		}
	},
	/**
		Sets the value of the _inName_ property on the control's DOM node to
		_inValue_, if and only if the DOM node has been rendered.  This method
		does not alter any values cached in local properties of the
		_enyo.Control_ instance.
	*/
	setNodeProperty: function(inName, inValue) {
		if (this.hasNode()) {
			this.node[inName] = inValue;
		}
	},
	/**
		Convenience function for setting the _class_ attribute.
		The _class_ attribute represents the CSS classes assigned to this object;
		it is a string that can contain multiple CSS classes separated by spaces.

			this.$.control.setClassAttribute("box blue-border highlighted");
	*/
	setClassAttribute: function(inClass) {
		this.setAttribute("class", inClass);
	},
	/**
		Convenience function for getting the _class_ attribute.
		The _class_ attribute represents the CSS classes assigned to this object;
		it is a string that can contain multiple CSS classes separated by spaces.

			var cssClasses = this.$.control.getClassAttribute();
	*/
	getClassAttribute: function() {
		return this.attributes["class"] || "";
	},
	/**
		Returns true if the _class_ attribute contains a substring matching
		_inClass_.

		The _class_ attribute is a string that can contain multiple CSS classes.
		This method tests whether a particular class is part of the set of
		classes on this control.

			// returns true if _class_ is "bar foo baz", but false for "barfoobaz"
			var hasFooClass = this.$.control.hasClass("foo");
	*/
	hasClass: function(inClass) {
		return inClass && ((" " + this.getClassAttribute() + " ").indexOf(" " + inClass + " ") >= 0);
	},
	/**
		Adds CSS class name _inClass_ to the _class_ attribute of this object.

			// add the highlight class to this object
			this.addClass("highlight");
	*/
	addClass: function(inClass) {
		if (inClass && !this.hasClass(inClass)) {
			var c = this.getClassAttribute();
			this.setClassAttribute(c + (c ? " " : "") + inClass);
		}
	},
	/**
		Removes substring _inClass_ from the _class_ attribute of this object.

		_inClass_ must have no leading or trailing spaces.

		Using a compound class name is supported, but the name is treated
		atomically. For example, given _"a b c"_, _removeClass("a b")_ will
		produce _"c"_, but _removeClass("a c")_ will produce _"a b c"_.

			// remove the highlight class from this object
			this.removeClass("highlight");
	*/
	removeClass: function(inClass) {
		if (inClass && this.hasClass(inClass)) {
			var c = this.getClassAttribute();
			c = (" " + c + " ").replace(" " + inClass + " ", " ").slice(1, -1);
			this.setClassAttribute(c);
		}
	},
	/**
		Adds or removes substring _inClass_ from the _class_ attribute of this
		object based on the value of _inTrueToAdd_.

		If _inTrueToAdd_ is truthy, then _inClass_ is added; otherwise,
		_inClass_ is removed.

			// add or remove the highlight class, depending on the "highlighted" property
			this.addRemoveClass("highlight", this.highlighted);
	*/
	addRemoveClass: function(inClass, inTrueToAdd) {
		this[inTrueToAdd ? "addClass" : "removeClass"](inClass);
	},
	//
	// styles
	//
	//* @protected
	initStyles: function() {
		this.domStyles = this.domStyles || {};
		enyo.Control.cssTextToDomStyles(this.kindStyle, this.domStyles);
		this.domCssText = enyo.Control.domStylesToCssText(this.domStyles);
	},
	styleChanged: function() {
		// FIXME: stomping on domStyles is problematic, there may be styles on this object
		// applied by layouts or other objects.
		// We may need a 'runtimeStyles' concept separate from a 'userStyles' concept, although
		// it's not clear what API calls like 'applyStyle' would affect, and which concept would take
		// precedence when there is a conflict.
		// Perhaps we can separate 'style' completely from 'domStyles'. API methods like applyStyle
		// would affect domStyles, and the two style databases would be combined at render-time.
		// Alternatively, we can disallow changing "style" string at runtime and allow it to be set
		// at init-time only (as it was in pre-ares enyo).
		//this.domStyles = {};
		//this.addStyles(this.kindStyle);
		//this.addStyles(this.style);
		this.invalidateTags();
		this.renderStyles();
	},
	//* @public
	/**
		Applies a single style value to this object.

			this.$.box.applyStyle("z-index", 4);

		You may remove a style by setting its value to null.

			this.$.box.applyStyle("z-index", null);
	*/
	applyStyle: function(inStyle, inValue) {
		this.domStyles[inStyle] = inValue;
		this.domStylesChanged();
	},
	/**
		Adds CSS styles to the set of styles assigned to this object.

		_inCssText_ is a string containing CSS styles in text format.

			this.$.box.addStyles("background-color: red; padding: 4px;");
	*/
	addStyles: function(inCssText) {
		enyo.Control.cssTextToDomStyles(inCssText, this.domStyles);
		this.domStylesChanged();
	},
	/**
		Returns the computed value of a CSS style named from _inStyle_
		for the DOM node of the control. If the node hasn't been generated,
		returns _inDefault_ as a default value. This uses CSS-style property
		names, not JavaScript-style names, so use "font-family" instead of
		"fontFamily".
	*/
	getComputedStyleValue: function(inStyle, inDefault) {
		if (this.hasNode()) {
			return enyo.dom.getComputedStyleValue(this.node, inStyle);
		}
		return inDefault;
	},
	//* @protected
	domStylesChanged: function() {
		this.domCssText = enyo.Control.domStylesToCssText(this.domStyles);
		this.invalidateTags();
		this.renderStyles();
	},
	stylesToNode: function() {
		this.node.style.cssText = this.style + (this.style[this.style.length-1] == ';' ? ' ' : '; ') + this.domCssText;
	},
	setupBodyFitting: function() {
		enyo.dom.applyBodyFit();
		this.addClass("enyo-fit enyo-clip");
	},
	/*
		If the platform is Android or Android-Chrome, don't include
		the css rule -webkit-overflow-scrolling: touch, as it is
		not supported in Android and leads to overflow issues
		(ENYO-900 and ENYO-901)
		Similarly, BB10 has issues repainting out-of-viewport content
		when -webkit-overflow-scrolling is used (ENYO-1396)
	*/
	setupOverflowScrolling: function() {
		if(enyo.platform.android || enyo.platform.androidChrome || enyo.platform.blackberry) {
			return;
		}
		document.getElementsByTagName("body")[0].className += " webkitOverflowScrolling";
	},
	//
	//
	//* @public
	/**
		Renders this object into DOM, generating a DOM node if needed.
	*/
	render: function() {
		if (this.parent) {
			// allow the parent to perform setup tasks
			// note: ('parent.generated' may change here)
			this.parent.beforeChildRender(this);
			// don't render if the parent has not generated
			if (!this.parent.generated) {
				return this;
			}
			if (this.tag === null) {
				// can't render a null element, have to render parent instead
				this.parent.render();
				return this;
			}
		}
		if (!this.hasNode()) {
			this.renderNode();
		}
		if (this.hasNode()) {
			this.renderDom();
			if (this.generated) {
				this.rendered();
			}
		}
		// return 'this' to support method chaining
		return this;
	},
	/**
		Renders this object into the DOM node referenced by _inParentNode_.
		If rendering into the document body element, appropriate styles will
		be used to have it expand to fill the whole window.
	*/
	renderInto: function(inParentNode) {
		// clean up render flags and memoizations
		this.teardownRender();
		// inParentNode can be a string id or a node reference
		var pn = enyo.dom.byId(inParentNode);
		var noFit = enyo.exists(this.fit) && this.fit === false;
		//console.log(noFit);
		if (pn == document.body && !noFit) {
			this.setupBodyFitting();
		} else if (this.fit) {
			this.addClass("enyo-fit enyo-clip");
		}
		// for IE10 support, we want full support over touch actions in Enyo-rendered areas
		this.addClass("enyo-no-touch-action");
		// add css to enable hw-accelerated scrolling on non-Android platforms (ENYO-900, ENYO-901)
		this.setupOverflowScrolling();
		// generate our HTML
		enyo.dom.setInnerHtml(pn, this.generateHtml());
		// post-rendering tasks
		if (this.generated) {
			this.rendered();
		}
		// support method chaining
		return this;
	},
	/**
		Uses _document.write_ to output the control into the document.
		If the control has _fit: true_ defined, appropriate styles will be set
		to have it expand to fill its container.

		Note that this has all the limitations that _document.write_ has.
		It only works while the page is loading, so you can't call this from an
		event handler. Also, it will not work in certain environments, such as
		Chrome Packaged Apps or Windows 8.
	*/
	write: function() {
		/* jshint evil:true */
		if (this.fit) {
			this.setupBodyFitting();
		}
		// for IE10 support, we want full support over touch actions in Enyo-rendered areas
		this.addClass("enyo-no-touch-action");
		// add css to enable hw-accelerated scrolling on non-Android platforms (ENYO-900, ENYO-901)
		this.setupOverflowScrolling();
		document.write(this.generateHtml());
		// post-rendering tasks
		if (this.generated) {
			this.rendered();
		}
		// support method chaining
		return this;
	},
	/**
		Override this method to perform tasks that require access to the DOM node.

			rendered: function() {
				this.inherited(arguments);
				// do some task
			}
	*/
	rendered: function() {
		// CAVEAT: Currently we use one entry point ('reflow') for
		// post-render layout work *and* post-resize layout work.
		this.reflow();
		for (var i=0, c; (c=this.children[i]); i++) {
			if (c.generated) {
				c.rendered();
			}
		}
	},
	/**
		Shows this node (alias for _setShowing(true)_).
	*/
	show: function() {
		this.setShowing(true);
	},
	/**
		Hides this node (alias for _setShowing(false)_).
	*/
	hide: function() {
		this.setShowing(false);
	},
	//* Sets focus to this control.
	focus: function() {
		if (this.hasNode()) {
			this.node.focus();
		}
	},
	//* Blurs this control.
	blur: function() {
		if (this.hasNode()) {
			this.node.blur();
		}
	},
	//* Returns true if the control is focused.
	hasFocus: function() {
		if (this.hasNode()) {
			return document.activeElement === this.node;
		}
	},
	/**
		Returns an object describing the geometry of this object, like so:

			{left: _offsetLeft_, top: _offsetTop_, width: _offsetWidth_, height: _offsetHeight_}

		Values returned are only valid if _hasNode()_ is truthy.
		If there's no DOM node for the object, this returns a bounds structure with
		_undefined_ as the value of all fields.

			var bounds = this.getBounds();
			enyo.log(bounds.width);
	*/
	getBounds: function() {
		var n = this.node || this.hasNode();
		var b = enyo.dom.getBounds(n);
		return b || {left: undefined, top: undefined, width: undefined, height: undefined};
	},
	/**
		Sets any or all of the geometry style properties _width_, _height_,
		_left_, _top_, _right_ and _bottom_.

		Values may be specified as strings (with units included), or as numbers
		when a unit is provided in _inUnit_.

			this.setBounds({width: 100, height: 100}, "px"); // adds style properties like "width: 100px; height: 100px;"
			//
			this.setBounds({width: "10em", right: "30pt"}); // adds style properties like "width: 10em; right: 30pt;"
	*/
	setBounds: function(inBounds, inUnit) {
		var s = this.domStyles, unit = inUnit || "px";
		var extents = ["width", "height", "left", "top", "right", "bottom"];
		for (var i=0, b, e; (e=extents[i]); i++) {
			b = inBounds[e];
			if (b || b === 0) {
				s[e] = b + (!enyo.isString(b) ? unit : '');
			}
		}
		this.domStylesChanged();
	},
	getAbsoluteBounds: function() {
		var l = 0,
			t = 0,
			n = this.hasNode(),
			w = n ? n.offsetWidth : 0,
			h = n ? n.offsetHeight : 0;

		while(n) {
			l += n.offsetLeft - (n.offsetParent ? n.offsetParent.scrollLeft : 0);
			t += n.offsetTop  - (n.offsetParent ? n.offsetParent.scrollTop	: 0);
			n = n.offsetParent;
		}

		return {
			top		: t,
			left	: l,
			bottom	: document.body.offsetHeight - t - h,
			right   : document.body.offsetWidth  - l - w,
			height	: h,
			width	: w
		};
	},
	//* @protected
	// expensive, other methods do work to avoid calling here
	findNodeById: function() {
		return this.id && (this.node = enyo.dom.byId(this.id));
	},
	idChanged: function(inOld) {
		if (inOld) {
			enyo.Control.unregisterDomEvents(inOld);
		}
		this.setAttribute("id", this.id);
		if (this.id) {
			enyo.Control.registerDomEvents(this.id, this);
		}
	},
	contentChanged: function() {
		if (this.hasNode()) {
			this.renderContent();
		}
	},
	getSrc: function() {
		return this.getAttribute("src");
	},
	srcChanged: function() {
		if (!this.src) {
			return;
		}
		this.setAttribute("src", enyo.path.rewrite(this.src));
	},
	attributesChanged: function() {
		this.invalidateTags();
		this.renderAttributes();
	},
	//
	// HTML rendering
	//
	generateHtml: function() {
		if (this.canGenerate === false) {
			return '';
		}
		// do this first in case content generation affects outer html (styles or attributes)
		var c = this.generateInnerHtml();
		// generate tag, styles, attributes
		var h = this.generateOuterHtml(c);
		// NOTE: 'generated' is used to gate access to findNodeById in
		// hasNode, because findNodeById is expensive.
		// NOTE: we typically use 'generated' to mean 'created in DOM'
		// but that has not actually happened at this point.
		// We set 'generated = true' here anyway to avoid having to walk the
		// control tree a second time (to set it later).
		// The contract is that insertion in DOM will happen synchronously
		// to generateHtml() and before anybody should be calling hasNode().
		this.generated = true;
		return h;
	},
	generateInnerHtml: function() {
		// Flow can alter the way that html content is rendered inside
		// the container regardless of whether there are children.
		this.flow();
		if (this.children.length) {
			return this.generateChildHtml();
		} else {
			return this.allowHtml ? this.get("content") : enyo.Control.escapeHtml(this.get("content"));
		}
	},
	generateChildHtml: function() {
		var results = '';
		for (var i=0, c; (c=this.children[i]); i++) {
			var h = c.generateHtml();
			results += h;
		}
		return results;
	},
	generateOuterHtml: function(inContent) {
		if (!this.tag) {
			return inContent;
		}
		if (!this.tagsValid) {
			this.prepareTags();
		}
		return this._openTag + inContent + this._closeTag;
	},
	invalidateTags: function() {
		this.tagsValid = false;
	},
	prepareTags: function() {
		var htmlStyle = this.domCssText + this.style;
		this._openTag = '<' +
			this.tag +
			(htmlStyle ? ' style="' + htmlStyle + '"' : "") +
			enyo.Control.attributesToHtml(this.attributes);
		if (enyo.Control.selfClosing[this.tag]) {
			this._openTag += '/>';
			this._closeTag =  '';
		} else {
			this._openTag += '>';
			this._closeTag =  '</' + this.tag + '>';
		}
		this.tagsValid = true;
	},
	// DOM, aka direct-to-node, rendering
	attributeToNode: function(inName, inValue) {
		if (inValue === null || inValue === false || inValue === "") {
			this.node.removeAttribute(inName);
		} else {
			this.node.setAttribute(inName, inValue);
		}
	},
	attributesToNode: function() {
		for (var n in this.attributes) {
			this.attributeToNode(n, this.attributes[n]);
		}
	},
	getParentNode: function() {
		return this.parentNode || (this.parent && (this.parent.hasNode() || this.parent.getParentNode()));
	},
	addNodeToParent: function() {
		if (this.node) {
			var pn = this.getParentNode();
			if (pn) {
				if (this.addBefore !== undefined) {
					this.insertNodeInParent(pn, this.addBefore && this.addBefore.hasNode());
				} else {
					this.appendNodeToParent(pn);
				}
			}
		}
	},
	appendNodeToParent: function(inParentNode) {
		inParentNode.appendChild(this.node);
	},
	insertNodeInParent: function(inParentNode, inBeforeNode) {
		inParentNode.insertBefore(this.node, inBeforeNode || inParentNode.firstChild);
	},
	removeNodeFromDom: function() {
		if (this.hasNode() && this.node.parentNode) {
			this.node.parentNode.removeChild(this.node);
		}
	},
	disconnectDom: function () {
		if (this.generated) {
			this.disconnectChildrenDom();
		}
		this._node = this.node;
		this.node = null;
		this.generated = false;
		this._domDisconnected = true;
	},
	disconnectChildrenDom: function () {
		for (var i=0, c; (c=this.children[i]); ++i) {
			c.disconnectDom();
		}
	},
	connectChildrenDom: function () {
		for (var i=0, c; (c=this.children[i]); ++i) {
			c.connectDom();
		}
	},
	connectDom: function () {
		if (this._domDisconnected) {
			this.connectChildrenDom();
			this.node = this._node;
			this._node = null;
			this.generated = true;
			this._domDisconnected = false;
		}
	},
	teardownRender: function() {
		if (this.generated) {
			this.teardownChildren();
		}
		this.node = null;
		this.generated = false;
	},
	teardownChildren: function() {
		for (var i=0, c; (c=this.children[i]); i++) {
			c.teardownRender();
		}
	},
	renderNode: function() {
		this.teardownRender();
		this.node = document.createElement(this.tag);
		this.addNodeToParent();
		this.generated = true;
	},
	renderDom: function() {
		this.renderAttributes();
		this.renderStyles();
		this.renderContent();
	},
	renderContent: function() {
		if (this.generated) {
			this.teardownChildren();
		}
		if (this.node) {
			enyo.dom.setInnerHtml(this.node, this.generateInnerHtml());
		}
	},
	renderReusingNode: function () {
		if (this.children.length) {
			for (var i=0, c; (c=this.children[i]); ++i) {
				c.renderReusingNode();
			}
		} else {
			if (this.generated && (this.node = this.hasNode())) {
				enyo.dom.setInnerHtml(this.node, this.generateInnerHtml());
			}
		}
	},
	renderStyles: function() {
		if (this.hasNode()) {
			this.stylesToNode();
		}
	},
	renderAttributes: function() {
		if (this.hasNode()) {
			this.attributesToNode();
		}
	},
	beforeChildRender: function() {
		// if we are generated, we should flow before rendering a child
		// if not, the render context isn't ready anyway
		if (this.generated) {
			this.flow();
		}
	},
	syncDisplayToShowing: function() {
		var ds = this.domStyles;
		if (this.showing) {
			// note: only show a node if it's actually hidden
			// this way we prevent overriding the value of domStyles.display
			if (ds.display == "none") {
				this.applyStyle("display", this._displayStyle || "");
			}
		} else {
			// cache the previous showing value of display
			// note: we could use a class to hide a node, but then
			// hide would not override a setting of display: none in style,
			// which seems bad.
			this._displayStyle = (ds.display == "none" ? "" : ds.display);
			this.applyStyle("display", "none");
		}
	},
	showingChanged: function() {
		this.syncDisplayToShowing();
	},
	getShowing: function() {
		// 'showing' specifically means domStyles.display !== 'none'.
		// 'showing' does not imply the node is actually visible or even rendered in DOM,
		// it simply reflects this state of this specific property as a convenience.
		this.showing = (this.domStyles.display != "none");
		return this.showing;
	},
	//* Return true if this and all parents are showing
	getAbsoluteShowing: function() {
		var b = this.getBounds();

		if(this.getShowing() === false || (b.height === 0 && b.width === 0)) {
			return false;
		}

		if(this.parent && this.parent.getAbsoluteShowing) {
			return this.parent.getAbsoluteShowing();
		} else {
			return true;
		}
	},
	//
	//
	fitChanged: function(inOld) {
		this.parent.reflow();
	},
	//* Return true if this control is the current fullscreen control
	isFullscreen: function() {
		return (this.hasNode() && this.hasNode() === enyo.fullscreen.getFullscreenElement());
	},
	//* Send request to make this control fullscreen
	requestFullscreen: function() {
		if (!this.hasNode()) {
			return false;
		}

		if (enyo.fullscreen.requestFullscreen(this)) {
			return true;
		}

		return false;
	},
	//* Send request to take this control out of fullscreen
	cancelFullscreen: function() {
		if (this.isFullscreen()) {
			enyo.fullscreen.cancelFullscreen();
			return true;
		}

		return false;
	},
	//
	//
	statics: {
		/**
			Returns passed-in string with ampersand, less-than, and greater-than
			characters replaced by HTML entities, e.g.,
			'&lt;code&gt;"This &amp; That"&lt;/code&gt;' becomes
			'&amp;lt;code&amp;gt;"This &amp;amp; That"&amp;lt;/code&amp;gt;'
		*/
		escapeHtml: function(inText) {
			return inText != null ? String(inText).replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;') : '';
		},
		//* @protected
		registerDomEvents: function(inId, inControl) {
			enyo.$[inId] = inControl;
		},
		unregisterDomEvents: function(inId) {
			enyo.$[inId] = null;
		},
		selfClosing: {img: 1, hr: 1, br: 1, area: 1, base: 1, basefont: 1, input: 1, link: 1, meta: 1,
			command: 1, embed: 1, keygen: 1, wbr: 1, param: 1, source: 1, track: 1, col: 1},
		cssTextToDomStyles: function(inText, inStyleHash) {
			if (inText) {
				// remove spaces between rules, then split rules on delimiter (;)
				var rules = inText.replace(/; /g, ";").split(";");
				// parse string styles into name/value pairs
				for (var i=0, s, n, v, rule; (rule=rules[i]); i++) {
					// "background-image: url(http://foo.com/foo.png)" => ["background-image", "url(http", "//foo.com/foo.png)"]
					s = rule.split(":");
					// n = "background-image", s = ["url(http", "//foo.com/foo.png)"]
					n = s.shift();
					// v = ["url(http", "//foo.com/foo.png)"].join(':') = "url(http://foo.com/foo.png)"
					v = s.join(':');
					// store name/value pair
					inStyleHash[n] = v;
				}
			}
		},
		domStylesToCssText: function(inStyleHash) {
			var n, v, text = '';
			for (n in inStyleHash) {
				v = inStyleHash[n];
				if ((v !== null) && (v !== undefined) && (v !== "")) {
					text += n + ':' + v + ';';
				}
			}
			return text;
		},
		stylesToHtml: function(inStyleHash) {
			var cssText = enyo.Control.domStylesToCssText(inStyleHash);
			return (cssText ? ' style="' + cssText + '"' : "");
		},
		/**
			Returns passed-in string with ampersand and double quote characters
			replaced by HTML entities, e.g., 'hello from "Me & She"' becomes
			'hello from &amp;quot;Me &amp;amp; She&amp;quot;'
		*/
		escapeAttribute: function(inText) {
			return !enyo.isString(inText) ? inText : String(inText).replace(/&/g,'&amp;').replace(/\"/g,'&quot;');
		},
		attributesToHtml: function(inAttributeHash) {
			var n, v, h = '';
			for (n in inAttributeHash) {
				v = inAttributeHash[n];
				if (v !== null && v !== false && v !== "") {
					h += ' ' + n + '="' + enyo.Control.escapeAttribute(v) + '"';
				}
			}
			return h;
		}
	}
});

enyo.defaultCtor = enyo.Control;

enyo.Control.subclass = function(ctor, props) {
	// Control classes may declare properties that are intended
	// to stack with superclass properties.
	//
	// We resort to prototype magic to assemble these properties
	// at kind declaration time, in the interest of efficiency
	// and ease of use.
	//
	// However, the properties are no longer 'live' in prototypes
	// because of this magic--i.e., changes to the prototype of
	// a Control subclass will not necessarily be reflected in
	// instances of that control (e.g., chained prototypes).
	//
	// These properties are also renamed to kind* to allow
	// combining with instance properties.
	//
	var proto = ctor.prototype;
	//
	// 'kindClasses' comes either from our inheritance chain (e.g., proto's prototype chain)
	// or has been forced by a kind declaration.
	//
	if (proto.classes) {
		var kc = proto.kindClasses;
		proto.kindClasses = (kc ? kc + " " : "") + proto.classes;
		proto.classes = "";
	}
	if (proto.style) {
		var ks = proto.kindStyle;
		proto.kindStyle = (ks ? ks + ";" : "") + proto.style;
		proto.style = "";
	}
	if (props.attributes) {
		var ka = proto.kindAttributes;
		proto.kindAttributes = enyo.mixin(enyo.clone(ka), proto.attributes);
		proto.attributes = null;
	}
};

//*@public
/**
	Also usable as _enyo.View_.
*/
enyo.View = enyo.Control;

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/animation.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uKCkgewoJdmFyIG1zID0gTWF0aC5yb3VuZCgxMDAwLzYwKTsKCXZhciBwcmVmaXggPSBbIndlYmtpdCIsICJtb3oiLCAibXMiLCAibyIsICIiXTsKCXZhciByID0gInJlcXVlc3RBbmltYXRpb25GcmFtZSI7Cgl2YXIgYyA9ICJjYW5jZWwiICsgZW55by5jYXAocik7CgkvLyBmYWxsYmFjayBvbiBzZXRUaW1lb3V0IGFuZCBjbGVhclRpbWVvdXQKCXZhciBfcmVxdWVzdEZyYW1lID0gZnVuY3Rpb24oaW5DYWxsYmFjaykgewoJCXJldHVybiB3aW5kb3cuc2V0VGltZW91dChpbkNhbGxiYWNrLCBtcyk7Cgl9OwoJdmFyIF9jYW5jZWxGcmFtZSA9IGZ1bmN0aW9uKGluSWQpIHsKCQlyZXR1cm4gd2luZG93LmNsZWFyVGltZW91dChpbklkKTsKCX07Cglmb3IgKHZhciBpID0gMCwgcGwgPSBwcmVmaXgubGVuZ3RoLCBwLCB3Yywgd3I7IChwID0gcHJlZml4W2ldKSB8fCBpIDwgcGw7IGkrKykgewoJCS8vIGlmIHdlJ3JlIG9uIGlvcyA2IGp1c3QgdXNlIHNldFRpbWVvdXQsIHJlcXVlc3RBbmltYXRpb25GcmFtZSBoYXMgc29tZSBraW5rcyBjdXJyZW50bHkKCQlpZiAoZW55by5wbGF0Zm9ybS5pb3MgPj0gNikgewoJCQlicmVhazsKCQl9CgoJCS8vIGlmIHByZWZpeGVkLCBiZWNvbWVzIFJlcXVlc3QgYW5kIENhbmNlbAoJCXdjID0gcCA/IChwICsgZW55by5jYXAoYykpIDogYzsKCQl3ciA9IHAgPyAocCArIGVueW8uY2FwKHIpKSA6IHI7CgkJLy8gVGVzdCBmb3IgY2FuY2VsUmVxdWVzdEFuaW1hdGlvbkZyYW1lLCBiZWNhdXNlIHNvbWUgYnJvd3NlcnMgKEZpcmVmaXggNC0xMCkgaGF2ZSBhIHJlcXVlc3Qgd2l0aG91dCBhIGNhbmNlbAoJCWlmICh3aW5kb3dbd2NdKSB7CgkJCV9jYW5jZWxGcmFtZSA9IHdpbmRvd1t3Y107CgkJCV9yZXF1ZXN0RnJhbWUgPSB3aW5kb3dbd3JdOwoJCQlpZiAocCA9PSAid2Via2l0IikgewoJCQkJLyoKCQkJCQlOb3RlOiBJbiBDaHJvbWUsIHRoZSBmaXJzdCByZXR1cm4gdmFsdWUgb2Ygd2Via2l0UmVxdWVzdEFuaW1hdGlvbkZyYW1lIGlzIDAuCgkJCQkJV2UgbWFrZSAxIGJvZ3VzIGNhbGwgc28gdGhlIGZpcnN0IHVzZWQgcmV0dXJuIHZhbHVlIG9mIHdlYmtpdFJlcXVlc3RBbmltYXRpb25GcmFtZSBpcyA+IDAsIGFzIHRoZSBzcGVjIHJlcXVpcmVzLgoJCQkJCVRoaXMgbWFrZXMgaXQgc28gdGhhdCB0aGUgcmVxdWVzdElkIGlzIGFsd2F5cyB0cnV0aHkuCgkJCQkJKHdlIGNob29zZSB0byBkbyB0aGlzIHJhdGhlciB0aGFuIHdyYXBwaW5nIHRoZSBuYXRpdmUgZnVuY3Rpb24gdG8gYXZvaWQgdGhlIG92ZXJoZWFkKQoJCQkJKi8KCQkJCV9jYW5jZWxGcmFtZShfcmVxdWVzdEZyYW1lKGVueW8ubm9wKSk7CgkJCX0KCQkJYnJlYWs7CgkJfQoJfQoJLyoqCgkJUmVxdWVzdHMgYW4gYW5pbWF0aW9uIGNhbGxiYWNrLgoKCQlPbiBjb21wYXRpYmxlIGJyb3dzZXJzLCBpZiBfaW5Ob2RlXyBpcyBkZWZpbmVkLCB0aGUgY2FsbGJhY2sgd2lsbCBmaXJlIG9ubHkgaWYgX2luTm9kZV8gaXMgdmlzaWJsZS4KCgkJUmV0dXJucyBhIHJlcXVlc3QgaWQgdG8gYmUgdXNlZCB3aXRoIFtlbnlvLmNhbmNlbFJlcXVlc3RBbmltYXRpb25GcmFtZV0oI2VueW8uY2FuY2VsUmVxdWVzdEFuaW1hdGlvbkZyYW1lKS4KCSovCgllbnlvLnJlcXVlc3RBbmltYXRpb25GcmFtZSA9IGZ1bmN0aW9uKGluQ2FsbGJhY2ssIGluTm9kZSkgewoJCXJldHVybiBfcmVxdWVzdEZyYW1lKGluQ2FsbGJhY2ssIGluTm9kZSk7Cgl9OwoJLyoqCgkJQ2FuY2VscyBhIHJlcXVlc3RlZCBhbmltYXRpb24gY2FsbGJhY2sgd2l0aCB0aGUgc3BlY2lmaWVkIGlkLgoJKi8KCWVueW8uY2FuY2VsUmVxdWVzdEFuaW1hdGlvbkZyYW1lID0gZnVuY3Rpb24oaW5JZCkgewoJCXJldHVybiBfY2FuY2VsRnJhbWUoaW5JZCk7Cgl9Owp9KSgpOwoKLyoqCglBbiBhc3NvcnRtZW50IG9mIGludGVycG9sYXRpb24gZnVuY3Rpb25zIGZvciBhbmltYXRpb25zLgoKCVNpbWlsYXIgaW4gZnVuY3Rpb24gdG8gQ1NTMyB0cmFuc2l0aW9ucy4KCglJbnRlbmRlZCBmb3IgdXNlIHdpdGggW2VueW8uZWFzZWRMZXJwXSgjZW55by5lYXNlZExlcnApCiovCmVueW8uZWFzaW5nID0gewoJY3ViaWNJbjogZnVuY3Rpb24obikgewoJCXJldHVybiBNYXRoLnBvdyhuLCAzKTsKCX0sCgljdWJpY091dDogZnVuY3Rpb24obikgewoJCXJldHVybiBNYXRoLnBvdyhuIC0gMSwgMykgKyAxOwoJfSwKCWV4cG9PdXQ6IGZ1bmN0aW9uKG4pIHsKCQlyZXR1cm4gKG4gPT0gMSkgPyAxIDogKC0xICogTWF0aC5wb3coMiwgLTEwICogbikgKyAxKTsKCX0sCglxdWFkSW5PdXQ6IGZ1bmN0aW9uKG4pewoJCW4gPSBuICogMjsKCQlpZiAobiA8IDEpIHsKCQkJcmV0dXJuIE1hdGgucG93KG4sIDIpIC8gMjsKCQl9CgkJcmV0dXJuIC0xICogKCgtLW4pICogKG4gLSAyKSAtIDEpIC8gMjsKCX0sCglsaW5lYXI6IGZ1bmN0aW9uKG4pIHsKCQlyZXR1cm4gbjsKCX0KfTsKCi8qKgoJR2l2ZXMgYW4gaW50ZXJwb2xhdGlvbiBvZiBhbiBhbmltYXRlZCB0cmFuc2l0aW9uJ3MgZGlzdGFuY2UgZnJvbSAwIHRvIDEuCgoJR2l2ZW4gYSBzdGFydCB0aW1lIChfaW5UMF8pIGFuZCBhbiBhbmltYXRpb24gZHVyYXRpb24gKF9pbkR1cmF0aW9uXyksIGFwcGxpZXMKCXRoZSBfaW5FYXNpbmdfIGZ1bmN0aW9uIHRvIHRoZSBwZXJjZW50YWdlIG9mIHRpbWUgZWxhcHNlZCAvIGR1cmF0aW9uLCBjYXBwZWQKCWF0IDEwMCUuCiovCmVueW8uZWFzZWRMZXJwID0gZnVuY3Rpb24oaW5UMCwgaW5EdXJhdGlvbiwgaW5FYXNpbmcsIHJldmVyc2UpIHsKCXZhciBsZXJwID0gKGVueW8ubm93KCkgLSBpblQwKSAvIGluRHVyYXRpb247CglpZiAocmV2ZXJzZSkgewoJCXJldHVybiBsZXJwID49IDEgPyAwIDogKDEgLSBpbkVhc2luZygxIC0gbGVycCkpOwoJfSBlbHNlIHsKCQlyZXR1cm4gbGVycCA+PSAxID8gMSA6IGluRWFzaW5nKGxlcnApOwoJfQp9OwoKLyoqCglHaXZlcyBhbiBpbnRlcnBvbGF0aW9uIG9mIGFuIGFuaW1hdGVkIHRyYW5zaXRpb24ncyBkaXN0YW5jZSBmcm9tIHN0YXJ0VmFsdWUgdG8gaW5WYWx1ZUNoYW5nZS4KCglBcHBsaWVzIHRoZSBfaW5FYXNpbmdfIGZ1bmN0aW9uIHdpdGggYSB3aWRlciByYW5nZSBvZiB2YXJpYWJsZXMgdG8gYWxsb3cgZm9yIG1vcmUgY29tcGxleCBhbmltYXRpb25zCiovCmVueW8uZWFzZWRDb21wbGV4TGVycCA9IGZ1bmN0aW9uKGluVDAsIGluRHVyYXRpb24sIGluRWFzaW5nLCByZXZlcnNlLCBpblRpbWUsIGluU3RhcnRWYWx1ZSwgaW5WYWx1ZUNoYW5nZSkgewoJdmFyIGxlcnAgPSAoZW55by5ub3coKSAtIGluVDApIC8gaW5EdXJhdGlvbjsKCWlmIChyZXZlcnNlKSB7CgkJcmV0dXJuIGluRWFzaW5nKDEgLSBsZXJwLCBpblRpbWUsIGluU3RhcnRWYWx1ZSwgaW5WYWx1ZUNoYW5nZSwgaW5EdXJhdGlvbik7Cgl9IGVsc2UgewoJCXJldHVybiBpbkVhc2luZyhsZXJwLCBpblRpbWUsIGluU3RhcnRWYWx1ZSwgaW5WYWx1ZUNoYW5nZSwgaW5EdXJhdGlvbik7Cgl9Cn07Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/cordova.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCkxpc3RlbnMgZm9yIENvcmRvdmEgc3BlY2lmaWMgZXZlbnRzCgpFdmVudHMgYXJlIGV4cG9zZWQgdGhyb3VnaCB0aGUgW1NpZ25hbHNdKCNlbnlvLlNpZ25hbHMpIGtpbmQgYnkgYWRkaW5nIGNhbGxiYWNrIGhhbmRsZXJzLgoKRXhhbXBsZToKCmVueW8ua2luZCh7CgluYW1lOiAiQXBwIiwKCWNvbXBvbmVudHM6IFsKCQl7a2luZDogIlNpZ25hbHMiLCBvbmRldmljZXJlYWR5OiAiZGV2aWNlcmVhZHkifSwKCQkuLi4KCQldLAoJZGV2aWNlcmVhZHk6IGZ1bmN0aW9uKCkgewoJLy8gQ29yZG92YSBBUEkgZXhpc3RzIGF0IHRoaXMgcG9pbnQgZm9yd2FyZAoJfQp9KTsKCkxpc3Qgb2YgQ29yZG92YSBldmVudHMgZGV0YWlsZWQgb24gdGhlIFtQaG9uZUdhcCBEb2NzXShodHRwOi8vZG9jcy5waG9uZWdhcC5jb20vZW4vMS42LjAvcGhvbmVnYXBfZXZlbnRzX2V2ZW50cy5tZC5odG1sI0V2ZW50cykKKi8KLy8qIEBwcm90ZWN0ZWQKZW55by5yZWFkeShmdW5jdGlvbigpewoJaWYgKHdpbmRvdy5jb3Jkb3ZhIHx8IHdpbmRvdy5QaG9uZUdhcCkgewoJCXZhciBwZ2UgPSBbCgkJCSJkZXZpY2VyZWFkeSIsCgkJCSJwYXVzZSIsCgkJCSJyZXN1bWUiLAoJCQkib25saW5lIiwKCQkJIm9mZmxpbmUiLAoJCQkiYmFja2J1dHRvbiIsCgkJCSJsb2NhbGVjaGFuZ2UiLAoJCQkiYmF0dGVyeWNyaXRpY2FsIiwKCQkJImJhdHRlcnlsb3ciLAoJCQkiYmF0dGVyeXN0YXR1cyIsCgkJCSJtZW51YnV0dG9uIiwKCQkJInNlYXJjaGJ1dHRvbiIsCgkJCSJzdGFydGNhbGxidXR0b24iLAoJCQkiZW5kY2FsbGJ1dHRvbiIsCgkJCSJ2b2x1bWVkb3duYnV0dG9uIiwKCQkJInZvbHVtZXVwYnV0dG9uIgoJCV07CgkJZm9yICh2YXIgaT0wLCBlOyAoZT1wZ2VbaV0pOyBpKyspIHsKCQkJLy8gc29tZSBjb3Jkb3ZhIGV2ZW50cyBoYXZlIG5vIHR5cGUsIHNvIGVueW8uZGlzcGF0Y2ggZmFpbHMKCQkJZG9jdW1lbnQuYWRkRXZlbnRMaXN0ZW5lcihlLCBlbnlvLmJpbmQoZW55by5TaWduYWxzLCAic2VuZCIsICJvbiIgKyBlKSwgZmFsc2UpOwoJCX0KCX0KfSk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/dispatcher.js"
Content-Type: application/octet-stream; x-encoding=base64
(function (enyo) {

	//* @protected
	enyo.$ = {};

	enyo.dispatcher = {
		// these events come from document
		events: ["mousedown", "mouseup", "mouseover", "mouseout", "mousemove", "mousewheel",
			"click", "dblclick", "change", "keydown", "keyup", "keypress", "input",
			"paste", "copy", "cut"],
		// these events come from window
		windowEvents: ["resize", "load", "unload", "message", "hashchange"],
		// these events come from css
		cssEvents: ["webkitTransitionEnd", "transitionend"],
		// feature plugins (aka filters)
		features: [],
		connect: function() {
			var d = enyo.dispatcher, i, n;
			for (i=0; (n=d.events[i]); i++) {
				d.listen(document, n);
			}
			for (i=0; (n=d.cssEvents[i]); i++) {
				d.listen(document, n);
			}
			for (i=0; (n=d.windowEvents[i]); i++) {
				// Chrome Packaged Apps don't like "unload"
				if(n === "unload" &&
					(typeof window.chrome === "object") &&
					window.chrome.app) {
					continue;
				}

				d.listen(window, n);
			}
			for (i=0; (n=d.cssEvents[i]); i++) {
				d.listen(document, n);
			}
		},
		listen: function(inListener, inEventName, inHandler) {
			var d = enyo.dispatch;
			if (inListener.addEventListener) {
				this.listen = function(inListener, inEventName, inHandler) {
					inListener.addEventListener(inEventName, inHandler || d, false);
				};
			} else {
				//enyo.log("IE8 COMPAT: using 'attachEvent'");
				this.listen = function(inListener, inEvent, inHandler) {
					inListener.attachEvent("on" + inEvent, function(e) {
						e.target = e.srcElement;
						if (!e.preventDefault) {
							e.preventDefault = enyo.iePreventDefault;
						}
						return (inHandler || d)(e);
					});
				};
			}
			this.listen(inListener, inEventName, inHandler);
		},
		stopListening: function(inListener, inEventName, inHandler) {
			var d = enyo.dispatch;
			if (inListener.addEventListener) {
				this.stopListening = function(inListener, inEventName, inHandler) {
					inListener.removeEventListener(inEventName, inHandler || d, false);
				};
			} else {
				//enyo.log("IE8 COMPAT: using 'detachEvent'");
				this.stopListening = function(inListener, inEvent, inHandler) {
					inListener.detachEvent("on" + inEvent, inHandler || d);
				};
			}
			this.stopListening(inListener, inEventName, inHandler);
		},
		//* Fires an event for Enyo to listen for.
		dispatch: function(e) {
			// Find the control who maps to e.target, or the first control that maps to an ancestor of e.target.
			var c = this.findDispatchTarget(e.target) || this.findDefaultTarget(e);
			// Cache the original target
			e.dispatchTarget = c;
			// support pluggable features return true to abort immediately or set e.preventDispatch to avoid processing.
			for (var i=0, fn; (fn=this.features[i]); i++) {
				if (fn.call(this, e) === true) {
					return;
				}
			}
			if (c && !e.preventDispatch) {
				this.dispatchBubble(e, c);
			}
		},
		//* Takes an event target and finds the corresponding Enyo control.
		findDispatchTarget: function(inNode) {
			var t, n = inNode;
			// FIXME: Mozilla: try/catch is here to squelch "Permission denied to access property xxx from a non-chrome context"
			// which appears to happen for scrollbar nodes in particular. It's unclear why those nodes are valid targets if
			// it is illegal to interrogate them. Would like to trap the bad nodes explicitly rather than using an exception block.
			try {
				while (n) {
					if ((t = enyo.$[n.id])) {
						// there could be multiple nodes with this id, the relevant node for this event is n
						// we don't push this directly to t.node because sometimes we are just asking what
						// the target 'would be' (aka, calling findDispatchTarget from handleMouseOverOut)
						t.eventNode = n;
						break;
					}
					n = n.parentNode;
				}
			} catch(x) {
				enyo.log(x, n);
			}
			return t;
		},
		//* Returns the default Enyo control for events.
		findDefaultTarget: function(e) {
			return enyo.master;
		},
		dispatchBubble: function(e, c) {
			return c.bubble("on" + e.type, e, c);
		}
	};

	// Called in the context of an event
	enyo.iePreventDefault = function() {
		try {
			this.returnValue = false;
		}
		catch(e) {
			// do nothing
		}
	};

	enyo.dispatch = function(inEvent) {
		return enyo.dispatcher.dispatch(inEvent);
	};

	enyo.bubble = function(inEvent) {
		// '|| window.event' clause needed for IE8
		var e = inEvent || window.event;
		if (e) {
			// We depend on e.target existing for event tracking and dispatching.
			if (!e.target) {
				e.target = e.srcElement;
			}
			enyo.dispatch(e);
		}
	};

	// This string is set on event handlers attributes for DOM elements that
	// don't normally bubble (like onscroll) so that they can participate in the
	// Enyo event system.
	enyo.bubbler = "enyo.bubble(arguments[0])";

	// The code below helps make Enyo compatible with Google Packaged Apps
	// Content Security Policy(http://developer.chrome.com/extensions/contentSecurityPolicy.html),
	// which, among other things, forbids the use of inline scripts.
	// We replace online scripting with equivalent means, leaving enyo.bubbler
	// for backward compatibility.
	(function() {
		var bubbleUp = function() {
			enyo.bubble(arguments[0]);
		};

		/**
			Makes given events bubble on a specified Enyo control.
		*/
		enyo.makeBubble = function() {
			var args = Array.prototype.slice.call(arguments, 0),
				control = args.shift();

			if((typeof control === "object") && (typeof control.hasNode === "function")) {
				enyo.forEach(args, function(event) {
					if(this.hasNode()) {
						enyo.dispatcher.listen(this.node, event, bubbleUp);
					}
				}, control);
			}
		};
		/**
			Removes the event listening and bubbling initiated by _enyo.makeBubble()_
			on a specific control.
		 */
		enyo.unmakeBubble = function() {
			var args = Array.prototype.slice.call(arguments, 0),
				control = args.shift();

			if((typeof control === "object") && (typeof control.hasNode === "function")) {
				enyo.forEach(args, function(event) {
					if(this.hasNode()) {
						enyo.dispatcher.stopListening(this.node, event, bubbleUp);
					}
				}, control);
			}
		};
	})();

	// FIXME: we need to create and initialize dispatcher someplace else to allow overrides
	enyo.requiresWindow(enyo.dispatcher.connect);

	// generate a tapped event for a raw-click event
	enyo.dispatcher.features.push(
		function (e) {
			if ("click" === e.type) {
				if (e.clientX === 0 && e.clientY === 0) {
					// this allows the click to dispatch as well
					// but note the tap event will fire first
					var cp = enyo.clone(e);
					cp.type = "tap";
					enyo.dispatch(cp);
				}
			}
		}
	);
	
})(enyo);

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/dom.css"
Content-Type: application/octet-stream; x-encoding=base64
LyogdGhpbmdzIHdlIGFsd2F5cyB3YW50ICovCmJvZHkgewoJZm9udC1mYW1pbHk6ICdIZWx2ZXRpY2EgTmV1ZScsICdOaW1idXMgU2FucyBMJywgQXJpYWwsIHNhbnMtc2VyaWY7Cn0KCi8qIGFsbG93IGh3LWFjY2VsZXJhdGVkIHNjcm9sbGluZyBvbiBwbGF0Zm9ybXMgdGhhdCBzdXBwb3J0IGl0ICovCmJvZHkud2Via2l0T3ZlcmZsb3dTY3JvbGxpbmcgewoJLXdlYmtpdC1vdmVyZmxvdy1zY3JvbGxpbmc6IHRvdWNoOwp9CgovKiBmb3IgYXBwcyAqLwouZW55by1kb2N1bWVudC1maXQgewoJbWFyZ2luOiAwOwoJaGVpZ2h0OiAxMDAlOwoJLyogbm90ZTogZ2l2aW5nIGh0bWwgb3ZlcmZsb3c6IGF1dG8gaXMgb2RkIGFuZCB3YXMgb25seSBldmVyIGRvbmUgdG8gYXZvaWQgZHVwbGljYXRpb24KCQlob3dldmVyLCBhbmRyb2lkIDQuMDQgc29tZXRpbWVzIGRvZXMgbm90IGhpZGUgbm9kZXMgd2hlbiB0aGVpciBkaXNwbGF5IGlzIHNldCB0byBub25lCgkJaWYgZG9jdW1lbnQgaXMgb3ZlcmZsb3cgYXV0by4KCSovCglwb3NpdGlvbjogcmVsYXRpdmU7Cn0KCi5lbnlvLWJvZHktZml0IHsKCW1hcmdpbjogMDsKCWhlaWdodDogMTAwJTsKCS8qIGhlbHBzIHByZXZlbnQgaW9zIHBhZ2Ugc2Nyb2xsICovCglvdmVyZmxvdzogYXV0bzsKCXBvc2l0aW9uOiByZWxhdGl2ZTsKfQoKLmVueW8tbm8tdG91Y2gtYWN0aW9uIHsKCS1tcy10b3VjaC1hY3Rpb246IG5vbmU7Cn0KCi8qIHJlc2V0ICovCgpidXR0b24gewoJZm9udC1zaXplOiBpbmhlcml0OwoJZm9udC1mYW1pbHk6IGluaGVyaXQ7Cn0KYnV0dG9uOjotbW96LWZvY3VzLWlubmVyIHsKICAgIGJvcmRlcjogMDsKICAgIHBhZGRpbmc6IDA7Cn0KCi8qIHVzZXIgc2VsZWN0aW9uICovCgouZW55by11bnNlbGVjdGFibGUgewoJY3Vyc29yOiBkZWZhdWx0OwoJLW1zLXVzZXItc2VsZWN0OiBub25lOwoJLXdlYmtpdC11c2VyLXNlbGVjdDogbm9uZTsKCS1tb3otdXNlci1zZWxlY3Q6IC1tb3otbm9uZTsKCXVzZXItc2VsZWN0OiBub25lOwp9CgouZW55by11bnNlbGVjdGFibGU6OnNlbGVjdGlvbiwgLmVueW8tdW5zZWxlY3RhYmxlIDo6c2VsZWN0aW9uIHsKCWNvbG9yOiB0cmFuc3BhcmVudDsKfQoKLmVueW8tc2VsZWN0YWJsZSB7CgljdXJzb3I6IGF1dG87CgktbXMtdXNlci1zZWxlY3Q6IGVsZW1lbnQ7Cgktd2Via2l0LXVzZXItc2VsZWN0OiB0ZXh0OwoJLW1vei11c2VyLXNlbGVjdDogdGV4dDsKCXVzZXItc2VsZWN0OiB0ZXh0Owp9CgouZW55by1zZWxlY3RhYmxlOjpzZWxlY3Rpb24sIC5lbnlvLXNlbGVjdGFibGUgOjpzZWxlY3Rpb24gewoJYmFja2dyb3VuZDogIzMyOTdGRDsKCWNvbG9yOiAjRkZGOwp9CgovKiBsYXlvdXQgKi8KCmJvZHkgLmVueW8tZml0IHsKCXBvc2l0aW9uOiBhYnNvbHV0ZTsKCWxlZnQ6IDA7Cgl0b3A6IDA7CglyaWdodDogMDsKCWJvdHRvbTogMDsKfQoKLmVueW8tY2xpcCB7CglvdmVyZmxvdzogaGlkZGVuOwp9CgouZW55by1ib3JkZXItYm94IHsKCS13ZWJraXQtYm94LXNpemluZzogYm9yZGVyLWJveDsKCS1tb3otYm94LXNpemluZzogYm9yZGVyLWJveDsKCWJveC1zaXppbmc6IGJvcmRlci1ib3g7Cn0KCi8qIGNvbXBvc2l0aW5nICovCgouZW55by1jb21wb3NpdGUgewoJLXdlYmtpdC10cmFuc2Zvcm06IHRyYW5zbGF0ZVooMCk7CgktbW96LXRyYW5zZm9ybTogdHJhbnNsYXRlWigwKTsKCS1tcy10cmFuc2Zvcm06IHRyYW5zbGF0ZVooMCk7Cgktby10cmFuc2Zvcm06IHRyYW5zbGF0ZVooMCk7Cgl0cmFuc2Zvcm06IHRyYW5zbGF0ZVooMCk7Cn0KCi8qIEZ1bGxzY3JlZW4gQ1NTICovCjotd2Via2l0LWZ1bGwtc2NyZWVuIHsKCXdpZHRoOiAxMDAlICFpbXBvcnRhbnQ7CgloZWlnaHQ6IDEwMCUgIWltcG9ydGFudDsKfQo6LW1vei1mdWxsLXNjcmVlbiB7Cgl3aWR0aDogMTAwJSAhaW1wb3J0YW50OwoJaGVpZ2h0OiAxMDAlICFpbXBvcnRhbnQ7Cn0KOi1tcy1mdWxsLXNjcmVlbiB7Cgl3aWR0aDogMTAwJSAhaW1wb3J0YW50OwoJaGVpZ2h0OiAxMDAlICFpbXBvcnRhbnQ7Cn0KOi1vLWZ1bGwtc2NyZWVuIHsKCXdpZHRoOiAxMDAlICFpbXBvcnRhbnQ7CgloZWlnaHQ6IDEwMCUgIWltcG9ydGFudDsKfQo6ZnVsbC1zY3JlZW4gewoJd2lkdGg6IDEwMCUgIWltcG9ydGFudDsKCWhlaWdodDogMTAwJSAhaW1wb3J0YW50Owp9Ci8qIEZhbGxiYWNrIEZ1bGxzY3JlZW4gQ1NTICovCmJvZHkgLmVueW8tZnVsbHNjcmVlbiB7Cglwb3NpdGlvbjogYWJzb2x1dGUgIWltcG9ydGFudDsKCWxlZnQ6IDAgIWltcG9ydGFudDsKCXRvcDogMCAhaW1wb3J0YW50OwoJcmlnaHQ6IDAgIWltcG9ydGFudDsKCWJvdHRvbTogMCAhaW1wb3J0YW50OwoJd2lkdGg6IDEwMCUgIWltcG9ydGFudDsKCWhlaWdodDogMTAwJSAhaW1wb3J0YW50OwoJYm94LXNpemluZzogYm9yZGVyLWJveCAhaW1wb3J0YW50Owp9Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/dom.js"
Content-Type: application/octet-stream; x-encoding=base64
//* @public

/**
	Allow bootstrapping in environments that do not have a window object right away.
*/
enyo.requiresWindow = function(inFunction) {
	inFunction();
};

enyo.dom = {
	/**
		Shortcut for _document.getElementById_ if _id_ is a string, otherwise returns _id_.
		Uses _window.document_ unless a document is specified in the (optional) _doc_
		parameter.

			// find 'node' if it's a string id, or return it unchanged if it's already a node reference
			var domNode = enyo.dom.byId(node);
	*/
	byId: function(id, doc){
		return (typeof id == "string") ? (doc || document).getElementById(id) : id;
	},
	/**
		return string with ampersand, less-than, and greater-than characters
		replaced with HTML entities, e.g.

			'&lt;code&gt;"This &amp; That"&lt;/code&gt;'

		becomes

			'&amp;lt;code&amp;gt;"This &amp;amp; That"&amp;lt;/code&amp;gt;'
	*/
	escape: function(inText) {
		return inText !== null ? String(inText).replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;') : '';
	},
	/**
		Returns an object describing the geometry of this node, like so:

			{left: _offsetLeft_, top: _offsetTop_, width: _offsetWidth_, height: _offsetHeight_}
	*/
	getBounds: function(n) {
		if (n) {
			return {left: n.offsetLeft, top: n.offsetTop, width: n.offsetWidth, height: n.offsetHeight};
		}
		else {
			return null;
		}
	},
	//* @protected
	// this is designed to be copied into the computedStyle object
	_ie8GetComputedStyle: function(prop) {
		var re = /(\-([a-z]){1})/g;
		if (prop === 'float') {
			prop = 'styleFloat';
		} else if (re.test(prop)) {
			prop = prop.replace(re, function () {
				return arguments[2].toUpperCase();
			});
		}
		return this[prop] !== undefined ? this[prop] : null;
	},
	getComputedStyle: function(inNode) {
		if(enyo.platform.ie < 9 && inNode && inNode.currentStyle) {
			//simple window.getComputedStyle polyfill for IE8
			var computedStyle = enyo.clone(inNode.currentStyle);
			computedStyle.getPropertyValue = this._ie8GetComputedStyle;
			computedStyle.setProperty = function() {
				return inNode.currentStyle.setExpression.apply(inNode.currentStyle, arguments);
			};
			computedStyle.removeProperty = function() {
				return inNode.currentStyle.removeAttribute.apply(inNode.currentStyle, arguments);
			};
			return computedStyle;
		} else {
			return window.getComputedStyle && inNode && window.getComputedStyle(inNode, null);
		}
	},
	getComputedStyleValue: function(inNode, inProperty, inComputedStyle) {
		var s   = inComputedStyle || this.getComputedStyle(inNode),
			nIE = enyo.platform.ie;

		s = s ? s.getPropertyValue(inProperty) : null;

		if (nIE) {
			var oConversion = {
				'thin'   : (nIE > 8 ? 2 : 1) + 'px',
				'medium' : (nIE > 8 ? 4 : 3) + 'px',
				'thick'  : (nIE > 8 ? 6 : 5) + 'px',
				'none'   : '0'
			};
			if (typeof oConversion[s] != 'undefined') {
				s = oConversion[s];
			}

			if (s == 'auto') {
				switch (inProperty) {
				case 'width':
					s = inNode.offsetWidth;
					break;
				case 'height':
					s = inNode.offsetHeight;
					break;
				}
			}
		}

		return s;
	},
	getFirstElementByTagName: function(inTagName) {
		var e = document.getElementsByTagName(inTagName);
		return e && e[0];
	},
	applyBodyFit: function() {
		var h = this.getFirstElementByTagName("html");
		if (h) {
			h.className += " enyo-document-fit";
		}
		var b = this.getFirstElementByTagName("body");
		if (b) {
			b.className += " enyo-body-fit";
		}
		enyo.bodyIsFitting = true;
	},
	getWindowWidth: function() {
		if (window.innerWidth) {
			return window.innerWidth;
		}
		if (document.body && document.body.offsetWidth) {
			return document.body.offsetWidth;
		}
		if (document.compatMode=='CSS1Compat' &&
			document.documentElement &&
			document.documentElement.offsetWidth ) {
			return document.documentElement.offsetWidth;
		}
		return 320;
	},
	getWindowHeight: function() {
		if (window.innerHeight) {
			return window.innerHeight;
		}
		if (document.body && document.body.offsetHeight) {
			return document.body.offsetHeight;
		}
		if (document.compatMode=='CSS1Compat' &&
			document.documentElement &&
			document.documentElement.offsetHeight ) {
			return document.documentElement.offsetHeight;
		}
		return 480;
	},
	// Workaround for lack of compareDocumentPosition support in IE8
	// Code MIT Licensed, John Resig; source: http://ejohn.org/blog/comparing-document-position/
	compareDocumentPosition: function(a, b) {
		return a.compareDocumentPosition ?
		a.compareDocumentPosition(b) :
		a.contains ?
			(a != b && a.contains(b) && 16) +
			(a != b && b.contains(a) && 8) +
			(a.sourceIndex >= 0 && b.sourceIndex >= 0 ?
				(a.sourceIndex < b.sourceIndex && 4) +
				(a.sourceIndex > b.sourceIndex && 2) :
				1) +
			0 :
			0;
	},
	// moved from FittableLayout.js into common protected code
	_ieCssToPixelValue: function(inNode, inValue) {
		var v = inValue;
		// From the awesome hack by Dean Edwards
		// http://erik.eae.net/archives/2007/07/27/18.54.15/#comment-102291
		var s = inNode.style;
		// store style and runtime style values
		var l = s.left;
		var rl = inNode.runtimeStyle && inNode.runtimeStyle.left;
		// then put current style in runtime style.
		if (rl) {
			inNode.runtimeStyle.left = inNode.currentStyle.left;
		}
		// apply given value and measure its pixel value
		s.left = v;
		v = s.pixelLeft;
		// finally restore previous state
		s.left = l;
		if (rl) {
			s.runtimeStyle.left = rl;
		}
		return v;
	},
	_pxMatch: /px/i,
	getComputedBoxValue: function(inNode, inProp, inBoundary, inComputedStyle) {
		var s = inComputedStyle || this.getComputedStyle(inNode);
		if (s) {
			var p = s.getPropertyValue(inProp + "-" + inBoundary);
			return p === "auto" ? 0 : parseInt(p, 10);
		} else if (inNode && inNode.currentStyle) {
			var v = inNode.currentStyle[inProp + enyo.cap(inBoundary)];
			if (!v.match(this._pxMatch)) {
				v = this._ieCssToPixelValue(inNode, v);
			}
			return parseInt(v, 0);
		}
		return 0;
	},
	//* @public
	//* Gets the boundaries of a node's margin or padding box.
	calcBoxExtents: function(inNode, inBox) {
		var s = this.getComputedStyle(inNode);
		return {
			top: this.getComputedBoxValue(inNode, inBox, "top", s),
			right: this.getComputedBoxValue(inNode, inBox, "right", s),
			bottom: this.getComputedBoxValue(inNode, inBox, "bottom", s),
			left: this.getComputedBoxValue(inNode, inBox, "left", s)
		};
	},
	//* Gets the calculated padding of a node.
	calcPaddingExtents: function(inNode) {
		return this.calcBoxExtents(inNode, "padding");
	},
	//* Gets the calculated margin of a node.
	calcMarginExtents: function(inNode) {
		return this.calcBoxExtents(inNode, "margin");
	},
	/**
		Returns an object like `{top: 0, left: 0, bottom: 100, right: 100, height: 10, width: 10}`
		that represents the object's position relative to `relativeToNode` (suitable for absolute
		positioning within that parent node). Negative values mean part of the object is not visible.
		If you leave `relativeToNode` undefined (or it is not a parent element), then the position
		will be relative to the viewport and suitable for absolute positioning in a floating layer.
	*/
	calcNodePosition: function(inNode, relativeToNode) {
		// Parse upward and grab our positioning relative to the viewport
		var top = 0,
			left = 0,
			node = inNode,
			width = node.offsetWidth,
			height = node.offsetHeight,
			transformProp = enyo.dom.getStyleTransformProp(),
			xregex = /translateX\((-?\d+)px\)/i,
			yregex = /translateY\((-?\d+)px\)/i,
			borderLeft = 0, borderTop = 0,
			totalHeight = 0, totalWidth = 0,
			offsetAdjustLeft = 0, offsetAdjustTop = 0;

		if (relativeToNode) {
			totalHeight = relativeToNode.offsetHeight;
			totalWidth = relativeToNode.offsetWidth;
		} else {
			totalHeight = (document.body.parentNode.offsetHeight > this.getWindowHeight() ? this.getWindowHeight() - document.body.parentNode.scrollTop : document.body.parentNode.offsetHeight);
			totalWidth = (document.body.parentNode.offsetWidth > this.getWindowWidth() ? this.getWindowWidth() - document.body.parentNode.scrollLeft : document.body.parentNode.offsetWidth);
		}

		if (node.offsetParent) {
			do {
				// Adjust the offset if relativeToNode is a child of the offsetParent
				// For IE 8 compatibility, have to use integer 8 instead of Node.DOCUMENT_POSITION_CONTAINS
				if (relativeToNode && this.compareDocumentPosition(relativeToNode, node.offsetParent) & 8) {
					offsetAdjustLeft = relativeToNode.offsetLeft;
					offsetAdjustTop = relativeToNode.offsetTop;
				}
				// Ajust our top and left properties based on the position relative to the parent
				left += node.offsetLeft - (node.offsetParent ? node.offsetParent.scrollLeft : 0) - offsetAdjustLeft;
				if (transformProp && xregex.test(node.style[transformProp])) {
					left += parseInt(node.style[transformProp].replace(xregex, '$1'), 10);
				}
				top += node.offsetTop - (node.offsetParent ? node.offsetParent.scrollTop : 0) - offsetAdjustTop;
				if (transformProp && yregex.test(node.style[transformProp])) {
					top += parseInt(node.style[transformProp].replace(yregex, '$1'), 10);
				}
				// Need to correct for borders if any exist on parent elements
				if (node !== inNode) {
					if (node.currentStyle) {
						// Oh IE, we do so love working around your incompatibilities
						borderLeft = parseInt(node.currentStyle.borderLeftWidth, 10);
						borderTop = parseInt(node.currentStyle.borderTopWidth, 10);
					} else if (window.getComputedStyle) {
						borderLeft = parseInt(window.getComputedStyle(node, '').getPropertyValue('border-left-width'), 10);
						borderTop = parseInt(window.getComputedStyle(node, '').getPropertyValue('border-top-width'), 10);
					} else {
						// No computed style options, so try the normal style object (much less robust)
						borderLeft = parseInt(node.style.borderLeftWidth, 10);
						borderTop = parseInt(node.style.borderTopWidth, 10);
					}
					if (borderLeft) {
						left += borderLeft;
					}
					if (borderTop) {
						top += borderTop;
					}
				}
				// Continue if we have an additional offsetParent, and either don't have a relativeToNode or the offsetParent is contained by the relativeToNode (if offsetParent contains relativeToNode, then we have already calculated up to the node, and can safely exit)
				// For IE 8 compatibility, have to use integer 16 instead of Node.DOCUMENT_POSITION_CONTAINED_BY
			} while ((node = node.offsetParent) && (!relativeToNode || this.compareDocumentPosition(relativeToNode, node) & 16));
		}
		return {
			'top': top,
			'left': left,
			'bottom': totalHeight - top - height,
			'right': totalWidth - left - width,
			'height': height,
			'width': width
		};
	},
	setInnerHtml: function(node, html) {
		node.innerHTML = html;
	}
};

// override setInnerHtml for Windows 8 HTML applications
if (typeof window.MSApp !== "undefined") {
	enyo.dom.setInnerHtml = function(node, html) {
		window.MSApp.execUnsafeLocalFunction(function() {
			node.innerHTML = html;
		});
	};
}

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/drag.js"
Content-Type: application/octet-stream; x-encoding=base64
/**
	Enyo supports a cross-platform set of drag events.  These events allow users
	to write a single set of event handlers for applications that run on both
	mobile and desktop platforms.

	The following events are provided:

	* "dragstart"
	* "dragfinish"
	* "drag"
	* "drop"
	* "dragover"
	* "dragout"
	* "hold"
	* "release"
	* "holdpulse"
	* "flick"

	For more information on these events, see the documentation on
	[User Input](https://github.com/enyojs/enyo/wiki/User-Input) in the Enyo
	Developer Guide.
*/

//* @protected
enyo.dispatcher.features.push(
	function(e) {
		// NOTE: beware of properties in enyo.gesture inadvertently mapped to event types
		if (enyo.gesture.drag[e.type]) {
			return enyo.gesture.drag[e.type](e);
		}
	}
);

//* @public
enyo.gesture.drag = {
	//* @protected
	hysteresisSquared: 16,
	holdPulseDelay: 200,
	trackCount: 5,
	minFlick: 0.1,
	minTrack: 8,
	down: function(e) {
		// tracking if the mouse is down
		//enyo.log("tracking ON");
		// Note: 'tracking' flag indicates interest in mousemove, it's turned off
		// on mouseup
		// make sure to stop dragging in case the up event was not received.
		this.stopDragging(e);
		this.cancelHold();
		this.target = e.target;
		this.startTracking(e);
		this.beginHold(e);
	},
	move: function(e) {
		if (this.tracking) {
			this.track(e);
			// If the mouse is not down and we're tracking a drag, abort.
			// this error condition can occur on IE/Webkit after interaction with a scrollbar.
			if (!e.which) {
				this.stopDragging(e);
				this.cancelHold();
				this.tracking = false;
				//enyo.log("enyo.gesture.drag: mouse must be down to drag.");
				return;
			}
			if (this.dragEvent) {
				this.sendDrag(e);
			} else if (this.dy*this.dy + this.dx*this.dx >= this.hysteresisSquared) {
				this.sendDragStart(e);
				this.cancelHold();
			}
		}
	},
	up: function(e) {
		this.endTracking(e);
		this.stopDragging(e);
		this.cancelHold();
	},
	leave: function(e) {
		if (this.dragEvent) {
			this.sendDragOut(e);
		}
	},
	stopDragging: function(e) {
		if (this.dragEvent) {
			this.sendDrop(e);
			var handled = this.sendDragFinish(e);
			enyo.pool.releaseObject(this.dragEvent);
			this.dragEvent = null;
			return handled;
		}
	},
	makeDragEvent: function(inType, inTarget, inEvent, inInfo) {
		var adx = Math.abs(this.dx), ady = Math.abs(this.dy);
		var h = adx > ady;
		// suggest locking if off-axis < 22.5 degrees
		var l = (h ? ady/adx : adx/ady) < 0.414;
		var e = enyo.pool.claimObject();
		// var e = {
		e.type = inType;
		e.dx = this.dx;
		e.dy = this.dy;
		e.ddx = this.dx - this.lastDx;
		e.ddy = this.dy - this.lastDy;
		e.xDirection = this.xDirection;
		e.yDirection = this.yDirection;
		e.pageX = inEvent.pageX;
		e.pageY = inEvent.pageY;
		e.clientX = inEvent.clientX;
		e.clientY = inEvent.clientY;
		e.horizontal = h;
		e.vertical = !h;
		e.lockable = l;
		e.target = inTarget;
		e.dragInfo = inInfo;
		e.ctrlKey = inEvent.ctrlKey;
		e.altKey = inEvent.altKey;
		e.metaKey = inEvent.metaKey;
		e.shiftKey = inEvent.shiftKey;
		e.srcEvent = inEvent.srcEvent;
		// };
		//Fix for IE8, which doesn't include pageX and pageY properties
		if(enyo.platform.ie==8 && e.target) {
			e.pageX = e.clientX + e.target.scrollLeft;
			e.pageY = e.clientY + e.target.scrollTop;
		}
		e.preventDefault = enyo.gesture.preventDefault;
		e.disablePrevention = enyo.gesture.disablePrevention;
		return e;
	},
	sendDragStart: function(e) {
		//enyo.log("dragstart");
		this.dragEvent = this.makeDragEvent("dragstart", this.target, e);
		enyo.dispatch(this.dragEvent);
	},
	sendDrag: function(e) {
		//enyo.log("sendDrag to " + this.dragEvent.target.id + ", over to " + e.target.id);
		// send dragOver event to the standard event target
		var synth = this.makeDragEvent("dragover", e.target, e, this.dragEvent.dragInfo);
		enyo.dispatch(synth);
		// send drag event to the drag source
		synth.type = "drag";
		synth.target = this.dragEvent.target;
		enyo.dispatch(synth);
		enyo.pool.releaseObject(synth);
	},
	sendDragFinish: function(e) {
		//enyo.log("dragfinish");
		var synth = this.makeDragEvent("dragfinish", this.dragEvent.target, e, this.dragEvent.dragInfo);
		synth.preventTap = function() {
			if (e.preventTap) {
				e.preventTap();
			}
		};
		enyo.dispatch(synth);
		enyo.pool.releaseObject(synth);
	},
	sendDragOut: function(e) {
		var synth = this.makeDragEvent("dragout", e.target, e, this.dragEvent.dragInfo);
		enyo.dispatch(synth);
		enyo.pool.releaseObject(synth);
	},
	sendDrop: function(e) {
		var synth = this.makeDragEvent("drop", e.target, e, this.dragEvent.dragInfo);
		synth.preventTap = function() {
			if (e.preventTap) {
				e.preventTap();
			}
		};
		enyo.dispatch(synth);
		enyo.pool.releaseObject(synth);
	},
	startTracking: function(e) {
		this.tracking = true;
		// note: use clientX/Y to be compatible with ie8
		this.px0 = e.clientX;
		this.py0 = e.clientY;
		// this.flickInfo = {startEvent: e, moves: []};
		this.flickInfo = enyo.pool.claimObject();
		this.flickInfo.startEvent = e;
		// FIXME: so we're trying to reuse objects where possible, should
		// do the same in scenarios like this for arrays
		this.flickInfo.moves = [];
		this.track(e);
	},
	track: function(e) {
		this.lastDx = this.dx;
		this.lastDy = this.dy;
		this.dx = e.clientX - this.px0;
		this.dy = e.clientY - this.py0;
		this.xDirection = this.calcDirection(this.dx - this.lastDx, 0);
		this.yDirection = this.calcDirection(this.dy - this.lastDy, 0);
		//
		var ti = this.flickInfo;
		ti.moves.push({
			x: e.clientX,
			y: e.clientY,
			// t: enyo.now()
			t: enyo.bench()
		});
		// track specified # of points
		if (ti.moves.length > this.trackCount) {
			ti.moves.shift();
		}
	},
	endTracking: function(e) {
		this.tracking = false;
		var ti = this.flickInfo;
		var moves = ti && ti.moves;
		if (moves && moves.length > 1) {
			// note: important to use up time to reduce flick
			// velocity based on time between move and up.
			var l = moves[moves.length-1];
			// var n = enyo.now();
			var n = enyo.bench();
			// take the greatest of flick between each tracked move and last move
			for (var i=moves.length-2, dt=0, x1=0, y1=0, x=0, y=0, sx=0, sy=0, m; (m=moves[i]); i--) {
				// this flick (this move - last move) / (this time - last time)
				dt = n - m.t;
				x1 = (l.x - m.x) / dt;
				y1 = (l.y - m.y) / dt;
				// establish flick direction
				sx = sx || (x1 < 0 ? -1 : (x1 > 0 ? 1 : 0));
				sy = sy || (y1 < 0 ? -1 : (y1 > 0 ? 1 : 0));
				// if either axis is a greater flick than previously recorded use this one
				if ((x1 * sx > x * sx) || (y1 * sy > y * sy)) {
					x = x1;
					y = y1;
				}
			}
			var v = Math.sqrt(x*x + y*y);
			if (v > this.minFlick) {
				// generate the flick using the start event so it has those coordinates
				this.sendFlick(ti.startEvent, x, y, v);
			}
		}
		this.flickInfo = null;
		enyo.pool.releaseObject(ti);
	},
	calcDirection: function(inNum, inDefault) {
		return inNum > 0 ? 1 : (inNum < 0 ? -1 : inDefault);
	},
	beginHold: function(e) {
		// this.holdStart = enyo.now();
		this.holdStart = enyo.bench();
		// clone the event to ensure it stays alive on IE upon returning to event loop
		// var clonedEvent = enyo.clone(e);
		var $ce = enyo.mixin(enyo.pool.claimObject(), e);
		// $ce.srcEvent = enyo.clone(e.srcEvent);
		$ce.srcEvent = enyo.mixin(enyo.pool.claimObject(), e.srcEvent);
		this._holdJobFunction = enyo.bind(this, "sendHoldPulse", $ce);
		this._holdJobFunction.ce = $ce;
		this.holdJob = setInterval(this._holdJobFunction, this.holdPulseDelay);
	},
	cancelHold: function() {
		clearInterval(this.holdJob);
		this.holdJob = null;
		if (this._holdJobFunction) {
			var $ce = this._holdJobFunction.ce;
			this._holdJobFunction = null;
			enyo.pool.releaseObject($ce.srcEvent);
			enyo.pool.releaseObject($ce);
		}
		if (this.sentHold) {
			this.sentHold = false;
			this.sendRelease(this.holdEvent);
		}
	},
	sendHoldPulse: function(inEvent) {
		if (!this.sentHold) {
			this.sentHold = true;
			this.sendHold(inEvent);
		}
		var e = enyo.gesture.makeEvent("holdpulse", inEvent);
		// e.holdTime = enyo.now() - this.holdStart;
		e.holdTime = enyo.bench() - this.holdStart;
		enyo.dispatch(e);
		enyo.pool.releaseObject(e);
	},
	sendHold: function(inEvent) {
		this.holdEvent = inEvent;
		var e = enyo.gesture.makeEvent("hold", inEvent);
		enyo.dispatch(e);
		enyo.pool.releaseObject(e);
	},
	sendRelease: function(inEvent) {
		var e = enyo.gesture.makeEvent("release", inEvent);
		enyo.dispatch(e);
		enyo.pool.releaseObject(e);
	},
	sendFlick: function(inEvent, inX, inY, inV) {
		var e = enyo.gesture.makeEvent("flick", inEvent);
		e.xVelocity = inX;
		e.yVelocity = inY;
		e.velocity = inV;
		enyo.dispatch(e);
		enyo.pool.releaseObject(e);
	}
};
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/gesture.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qIEBwdWJsaWMKLyoqCglFbnlvIHN1cHBvcnRzIGEgc2V0IG9mIG5vcm1hbGl6ZWQgZXZlbnRzIHRoYXQgd29yayBzaW1pbGFybHkgYWNyb3NzCWFsbAoJc3VwcG9ydGVkIHBsYXRmb3Jtcy4gVGhlc2UgZXZlbnRzIGFyZSBwcm92aWRlZCBzbyB0aGF0IHVzZXJzIGNhbiB3cml0ZSBhCglzaW5nbGUgc2V0IG9mIGV2ZW50IGhhbmRsZXJzIGZvciBhcHBsaWNhdGlvbnMgdGhhdCBydW4gb24gYm90aCBtb2JpbGUgYW5kCglkZXNrdG9wIHBsYXRmb3Jtcy4gIFRoZXkgYXJlIG5lZWRlZCBiZWNhdXNlIGRlc2t0b3AgYW5kIG1vYmlsZSBwbGF0Zm9ybXMKCWhhbmRsZSBiYXNpYyBpbnB1dCBkaWZmZXJlbnRseS4KCglGb3IgbW9yZSBpbmZvcm1hdGlvbiBvbiBub3JtYWxpemVkIGlucHV0IGV2ZW50cyBhbmQgdGhlaXIgYXNzb2NpYXRlZAoJcHJvcGVydGllcywJc2VlCXRoZSBkb2N1bWVudGF0aW9uIG9uCglbVXNlciBJbnB1dF0oaHR0cHM6Ly9naXRodWIuY29tL2VueW9qcy9lbnlvL3dpa2kvVXNlci1JbnB1dCkgaW4gdGhlIEVueW8KCURldmVsb3BlciBHdWlkZS4KKi8KZW55by5nZXN0dXJlID0gewoJLy8qIEBwcm90ZWN0ZWQKCWV2ZW50UHJvcHM6IFsidGFyZ2V0IiwgInJlbGF0ZWRUYXJnZXQiLCAiY2xpZW50WCIsICJjbGllbnRZIiwgInBhZ2VYIiwgInBhZ2VZIiwKCQkic2NyZWVuWCIsICJzY3JlZW5ZIiwgImFsdEtleSIsICJjdHJsS2V5IiwgIm1ldGFLZXkiLCAic2hpZnRLZXkiLAoJCSJkZXRhaWwiLCAiaWRlbnRpZmllciIsICJkaXNwYXRjaFRhcmdldCIsICJ3aGljaCIsICJzcmNFdmVudCJdLAoJbWFrZUV2ZW50OiBmdW5jdGlvbihpblR5cGUsIGluRXZlbnQpIHsKCQkvLyB2YXIgZSA9IHt0eXBlOiBpblR5cGV9OwoJCXZhciBlID0gZW55by5wb29sLmNsYWltT2JqZWN0KCk7CgkJZS50eXBlID0gaW5UeXBlOwoJCWZvciAodmFyIGk9MCwgcDsgKHA9dGhpcy5ldmVudFByb3BzW2ldKTsgaSsrKSB7CgkJCWVbcF0gPSBpbkV2ZW50W3BdOwoJCX0KCQllLnNyY0V2ZW50ID0gZS5zcmNFdmVudCB8fCBpbkV2ZW50OwoJCWUucHJldmVudERlZmF1bHQgPSB0aGlzLnByZXZlbnREZWZhdWx0OwoJCWUuZGlzYWJsZVByZXZlbnRpb24gPSB0aGlzLmRpc2FibGVQcmV2ZW50aW9uOwoJCS8vCgkJLy8gbm9ybWFsaXplIGV2ZW50LndoaWNoIGFuZCBldmVudC5wYWdlWC9ldmVudC5wYWdlWQoJCS8vIE5vdGUgdGhhdCB3aGlsZSAid2hpY2giIHdvcmtzIGluIElFOSwgaXQgaXMgYnJva2VuIGZvciBtb3VzZW1vdmUuIFRoZXJlZm9yZSwKCQkvLyBpbiBJRSwgdXNlIHdpbmRvdy5ldmVudC5idXR0b24KCQlpZiAoZW55by5wbGF0Zm9ybS5pZSA8IDEwKSB7CgkJCS8vRml4IGZvciBJRTgsIHdoaWNoIGRvZXNuJ3QgaW5jbHVkZSBwYWdlWCBhbmQgcGFnZVkgcHJvcGVydGllcwoJCQlpZihlbnlvLnBsYXRmb3JtLmllPT04ICYmIGUudGFyZ2V0KSB7CgkJCQllLnBhZ2VYID0gZS5jbGllbnRYICsgZS50YXJnZXQuc2Nyb2xsTGVmdDsKCQkJCWUucGFnZVkgPSBlLmNsaWVudFkgKyBlLnRhcmdldC5zY3JvbGxUb3A7CgkJCX0KCQkJdmFyIGIgPSB3aW5kb3cuZXZlbnQgJiYgd2luZG93LmV2ZW50LmJ1dHRvbjsKCQkJaWYgKGIpIHsKCQkJCS8vIG11bHRpLWJ1dHRvbiBub3Qgc3VwcG9ydGVkLCBwcmlvcml0eTogbGVmdCwgcmlnaHQsIG1pZGRsZQoJCQkJLy8gKG5vdGU6IElFIGJpdG1hc2sgaXMgMT1sZWZ0LCAyPXJpZ2h0LCA0PWNlbnRlcik7CgkJCQllLndoaWNoID0gYiAmIDEgPyAxIDogKGIgJiAyID8gMiA6IChiICYgNCA/IDMgOiAwKSk7CgkJCX0KCQl9IGVsc2UgaWYgKGVueW8ucGxhdGZvcm0ud2Vib3MgfHwgd2luZG93LlBhbG1TeXN0ZW0pIHsKCQkJLy8gVGVtcG9yYXJ5IGZpeCBmb3Igb3dvczogaXQgZG9lcyBub3QgY3VycmVudGx5IHN1cHBseSAnd2hpY2gnIG9uIG1vdmUgZXZlbnRzCgkJCS8vIGFuZCB0aGUgdXNlciBhZ2VudCBzdHJpbmcgZG9lc24ndCBpZGVudGlmeSBpdHNlbGYgc28gd2UgdGVzdCBmb3IgUGFsbVN5c3RlbQoJCQlpZiAoZS53aGljaCA9PT0gMCkgewoJCQkJZS53aGljaCA9IDE7CgkJCX0KCQl9CgkJcmV0dXJuIGU7Cgl9LAoJZG93bjogZnVuY3Rpb24oaW5FdmVudCkgewoJCS8vIGNhbmNlbCBhbnkgaG9sZCBzaW5jZSBpdCdzIHBvc3NpYmxlIGluIGNvcm5lciBjYXNlcyB0byBnZXQgYSBkb3duIHdpdGhvdXQgYW4gdXAKCQl2YXIgZSA9IHRoaXMubWFrZUV2ZW50KCJkb3duIiwgaW5FdmVudCk7CgkJZW55by5kaXNwYXRjaChlKTsKCQl0aGlzLmRvd25FdmVudCA9IGU7Cgl9LAoJbW92ZTogZnVuY3Rpb24oaW5FdmVudCkgewoJCXZhciBlID0gdGhpcy5tYWtlRXZlbnQoIm1vdmUiLCBpbkV2ZW50KTsKCQkvLyBpbmNsdWRlIGRlbHRhIGFuZCBkaXJlY3Rpb24gdi4gZG93biBpbmZvIGluIG1vdmUgZXZlbnQKCQllLmR4ID0gZS5keSA9IGUuaG9yaXpvbnRhbCA9IGUudmVydGljYWwgPSAwOwoJCWlmIChlLndoaWNoICYmIHRoaXMuZG93bkV2ZW50KSB7CgkJCWUuZHggPSBpbkV2ZW50LmNsaWVudFggLSB0aGlzLmRvd25FdmVudC5jbGllbnRYOwoJCQllLmR5ID0gaW5FdmVudC5jbGllbnRZIC0gdGhpcy5kb3duRXZlbnQuY2xpZW50WTsKCQkJZS5ob3Jpem9udGFsID0gTWF0aC5hYnMoZS5keCkgPiBNYXRoLmFicyhlLmR5KTsKCQkJZS52ZXJ0aWNhbCA9ICFlLmhvcml6b250YWw7CgkJfQoJCWVueW8uZGlzcGF0Y2goZSk7CgkJZW55by5wb29sLnJlbGVhc2VPYmplY3QoZSk7Cgl9LAoJdXA6IGZ1bmN0aW9uKGluRXZlbnQpIHsKCQl2YXIgZSA9IHRoaXMubWFrZUV2ZW50KCJ1cCIsIGluRXZlbnQpOwoJCXZhciB0YXBQcmV2ZW50ZWQgPSBmYWxzZTsKCQllLnByZXZlbnRUYXAgPSBmdW5jdGlvbigpIHsKCQkJdGFwUHJldmVudGVkID0gdHJ1ZTsKCQl9OwoJCWVueW8uZGlzcGF0Y2goZSk7CgkJaWYgKCF0YXBQcmV2ZW50ZWQgJiYgdGhpcy5kb3duRXZlbnQgJiYgdGhpcy5kb3duRXZlbnQud2hpY2ggPT0gMSkgewoJCQl0aGlzLnNlbmRUYXAoZSk7CgkJfQoJCWVueW8ucG9vbC5yZWxlYXNlT2JqZWN0KHRoaXMuZG93bkV2ZW50KTsKCQl0aGlzLmRvd25FdmVudCA9IG51bGw7CgkJZW55by5wb29sLnJlbGVhc2VPYmplY3QoZSk7Cgl9LAoJb3ZlcjogZnVuY3Rpb24oaW5FdmVudCkgewoJCXZhciBlID0gdGhpcy5tYWtlRXZlbnQoImVudGVyIiwgaW5FdmVudCk7CgkJZW55by5kaXNwYXRjaChlKTsKCQllbnlvLnBvb2wucmVsZWFzZU9iamVjdChlKTsKCX0sCglvdXQ6IGZ1bmN0aW9uKGluRXZlbnQpIHsKCQl2YXIgZSA9IHRoaXMubWFrZUV2ZW50KCJsZWF2ZSIsIGluRXZlbnQpOwoJCWVueW8uZGlzcGF0Y2goZSk7CgkJZW55by5wb29sLnJlbGVhc2VPYmplY3QoZSk7Cgl9LAoJc2VuZFRhcDogZnVuY3Rpb24oaW5FdmVudCkgewoJCS8vIFRoZSBjb21tb24gYW5jZXN0b3IgZm9yIHRoZSBkb3duL3VwIHBhaXIgaXMgdGhlIG9yaWdpbiBmb3IgdGhlIHRhcCBldmVudAoJCXZhciB0ID0gdGhpcy5maW5kQ29tbW9uQW5jZXN0b3IodGhpcy5kb3duRXZlbnQudGFyZ2V0LCBpbkV2ZW50LnRhcmdldCk7CgkJaWYgKHQpIHsKCQkJdmFyIGUgPSB0aGlzLm1ha2VFdmVudCgidGFwIiwgaW5FdmVudCk7CgkJCWUudGFyZ2V0ID0gdDsKCQkJZW55by5kaXNwYXRjaChlKTsKCQkJZW55by5wb29sLnJlbGVhc2VPYmplY3QoZSk7CgkJfQoJfSwKCWZpbmRDb21tb25BbmNlc3RvcjogZnVuY3Rpb24oaW5BLCBpbkIpIHsKCQl2YXIgcCA9IGluQjsKCQl3aGlsZSAocCkgewoJCQlpZiAodGhpcy5pc1RhcmdldERlc2NlbmRhbnRPZihpbkEsIHApKSB7CgkJCQlyZXR1cm4gcDsKCQkJfQoJCQlwID0gcC5wYXJlbnROb2RlOwoJCX0KCX0sCglpc1RhcmdldERlc2NlbmRhbnRPZjogZnVuY3Rpb24oaW5DaGlsZCwgaW5QYXJlbnQpIHsKCQl2YXIgYyA9IGluQ2hpbGQ7CgkJd2hpbGUoYykgewoJCQlpZiAoYyA9PSBpblBhcmVudCkgewoJCQkJcmV0dXJuIHRydWU7CgkJCX0KCQkJYyA9IGMucGFyZW50Tm9kZTsKCQl9Cgl9Cn07CgovLyogQHByb3RlY3RlZAoKLy8gaW5zdGFsbGVkIG9uIGV2ZW50cyBhbmQgY2FsbGVkIGluIGV2ZW50IGNvbnRleHQKZW55by5nZXN0dXJlLnByZXZlbnREZWZhdWx0ID0gZnVuY3Rpb24oKSB7CglpZiAodGhpcy5zcmNFdmVudCkgewoJCXRoaXMuc3JjRXZlbnQucHJldmVudERlZmF1bHQoKTsKCX0KfTsKCmVueW8uZ2VzdHVyZS5kaXNhYmxlUHJldmVudGlvbiA9IGZ1bmN0aW9uKCkgewoJdGhpcy5wcmV2ZW50RGVmYXVsdCA9IGVueW8ubm9wOwoJaWYgKHRoaXMuc3JjRXZlbnQpIHsKCQl0aGlzLnNyY0V2ZW50LnByZXZlbnREZWZhdWx0ID0gZW55by5ub3A7Cgl9Cn07CgplbnlvLmRpc3BhdGNoZXIuZmVhdHVyZXMucHVzaCgKCWZ1bmN0aW9uKGUpIHsKCQkvLyBOT1RFOiBiZXdhcmUgb2YgcHJvcGVydGllcyBpbiBlbnlvLmdlc3R1cmUgaW5hZHZlcnRlbnRseSBtYXBwZWQgdG8gZXZlbnQgdHlwZXMKCQlpZiAoZW55by5nZXN0dXJlLmV2ZW50c1tlLnR5cGVdKSB7CgkJCXJldHVybiBlbnlvLmdlc3R1cmUuZXZlbnRzW2UudHlwZV0oZSk7CgkJfQoJfQopOwoKZW55by5nZXN0dXJlLmV2ZW50cyA9IHsKCW1vdXNlZG93bjogZnVuY3Rpb24oZSkgewoJCWVueW8uZ2VzdHVyZS5kb3duKGUpOwoJfSwKCW1vdXNldXA6IGZ1bmN0aW9uKGUpIHsKCQllbnlvLmdlc3R1cmUudXAoZSk7Cgl9LAoJbW91c2Vtb3ZlOiAgZnVuY3Rpb24oZSkgewoJCWVueW8uZ2VzdHVyZS5tb3ZlKGUpOwoJfSwKCW1vdXNlb3ZlcjogIGZ1bmN0aW9uKGUpIHsKCQllbnlvLmdlc3R1cmUub3ZlcihlKTsKCX0sCgltb3VzZW91dDogIGZ1bmN0aW9uKGUpIHsKCQllbnlvLmdlc3R1cmUub3V0KGUpOwoJfQp9OwoKLy8gRmlyZWZveCBtb3VzZXdoZWVsIGhhbmRsaW5nCmVueW8ucmVxdWlyZXNXaW5kb3coZnVuY3Rpb24oKSB7CglpZiAoZG9jdW1lbnQuYWRkRXZlbnRMaXN0ZW5lcikgewoJCWRvY3VtZW50LmFkZEV2ZW50TGlzdGVuZXIoIkRPTU1vdXNlU2Nyb2xsIiwgZnVuY3Rpb24oaW5FdmVudCkgewoJCQkvLyB2YXIgZSA9IGVueW8uY2xvbmUoaW5FdmVudCk7CgkJCXZhciBlID0gZW55by5taXhpbihlbnlvLnBvb2wuY2xhaW1PYmplY3QoKSwgaW5FdmVudCk7CgkJCWUucHJldmVudERlZmF1bHQgPSBmdW5jdGlvbigpIHsKCQkJCWluRXZlbnQucHJldmVudERlZmF1bHQoKTsKCQkJfTsKCQkJZS50eXBlID0gIm1vdXNld2hlZWwiOwoJCQl2YXIgcCA9IGUuVkVSVElDQUxfQVhJUyA9PSBlLmF4aXMgPyAid2hlZWxEZWx0YVkiIDogIndoZWVsRGVsdGFYIjsKCQkJZVtwXSA9ICBlLmRldGFpbCAqIC00MDsKCQkJZW55by5kaXNwYXRjaChlKTsKCQkJZW55by5wb29sLnJlbGVhc2VPYmplY3QoZSk7CgkJfSwgZmFsc2UpOwoJfQp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/modal.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qIEBwcm90ZWN0ZWQKCi8qKgoJRXZlbnQgbW9kYWwgY2FwdHVyZSBmZWF0dXJlOiBjYXB0dXJlIGV2ZW50cyB0byBhIHNwZWNpZmljCgljb250cm9sIHZpYSBlbnlvLmRpc3BhdGNoZXIuY2FwdHVyZShpbkNvbnRyb2wsIGluU2hvdWxkRm9yd2FyZCkKCXJlbGVhc2UgZXZlbnRzIHZpYSBlbnlvLmRpc3BhdGNoZXIucmVsZWFzZSgpCiovCmVueW8uZGlzcGF0Y2hlci5mZWF0dXJlcy5wdXNoKGZ1bmN0aW9uKGUpIHsKCXZhciBjID0gZS5kaXNwYXRjaFRhcmdldDsKCXZhciB3YW50cyA9IHRoaXMuY2FwdHVyZVRhcmdldCAmJiAhdGhpcy5ub0NhcHR1cmVFdmVudHNbZS50eXBlXTsKCXZhciBuZWVkcyA9IHdhbnRzICYmICEoYyAmJiBjLmlzRGVzY2VuZGFudE9mICYmIGMuaXNEZXNjZW5kYW50T2YodGhpcy5jYXB0dXJlVGFyZ2V0KSk7CglpZiAobmVlZHMpIHsKCQl2YXIgYzEgPSBlLmNhcHR1cmVUYXJnZXQgPSB0aGlzLmNhcHR1cmVUYXJnZXQ7CgkJLy8gTk9URTogV2UgZG8gbm90IHdhbnQgcmVsZWFzaW5nIGNhcHR1cmUgd2hpbGUgYW4gZXZlbnQgaXMgYmVpbmcgcHJvY2Vzc2VkIHRvIGFsdGVyCgkJLy8gdGhlIHdheSB0aGUgZXZlbnQgcHJvcGFnYXRlcy4gVGhlcmVmb3JlIGRlY2lkZSBpZiB0aGUgZXZlbnQgc2hvdWxkIGZvcndhcmQKCQkvLyBiZWZvcmUgdGhlIGNhcHR1cmUgdGFyZ2V0IHJlY2VpdmVzIHRoZSBldmVudCAoc2luY2UgaXQgbWF5IHJlbGVhc2UgY2FwdHVyZSkuCgkJdmFyIHNob3VsZEZvcndhcmQgPSAodGhpcy5hdXRvRm9yd2FyZEV2ZW50c1tlLnR5cGVdIHx8IHRoaXMuZm9yd2FyZEV2ZW50cyk7CgkJdGhpcy5kaXNwYXRjaEJ1YmJsZShlLCBjMSk7CgkJaWYgKCFzaG91bGRGb3J3YXJkKSB7CgkJCWUucHJldmVudERpc3BhdGNoID0gdHJ1ZTsKCQl9Cgl9Cn0pOwoKLy8KLy8JTk9URTogVGhpcyBvYmplY3QgaXMgYSBwbHVnLWluOyB0aGVzZSBtZXRob2RzIHNob3VsZAovLwliZSBjYWxsZWQgb24gX2VueW8uZGlzcGF0Y2hlcl8sIGFuZCBub3Qgb24gdGhlIHBsdWctaW4gaXRzZWxmLgovLwplbnlvLm1peGluKGVueW8uZGlzcGF0Y2hlciwgewoJbm9DYXB0dXJlRXZlbnRzOiB7bG9hZDogMSwgdW5sb2FkOjEsIGVycm9yOiAxfSwKCWF1dG9Gb3J3YXJkRXZlbnRzOiB7bGVhdmU6IDEsIHJlc2l6ZTogMX0sCgljYXB0dXJlczogW10sCgkvLyogQ2FwdHVyZSBldmVudHMgZm9yIGBpblRhcmdldGAgYW5kIG9wdGlvbmFsbHkgZm9yd2FyZCB0aGVtCgljYXB0dXJlOiBmdW5jdGlvbihpblRhcmdldCwgaW5TaG91bGRGb3J3YXJkKSB7CgkJdmFyIGluZm8gPSB7dGFyZ2V0OiBpblRhcmdldCwgZm9yd2FyZDogaW5TaG91bGRGb3J3YXJkfTsKCQl0aGlzLmNhcHR1cmVzLnB1c2goaW5mbyk7CgkJdGhpcy5zZXRDYXB0dXJlSW5mbyhpbmZvKTsKCX0sCgkvLyogUmVtb3ZlIHRoZSBzcGVjaWZpZWQgdGFyZ2V0IGZyb20gdGhlIGNhcHR1cmUgbGlzdAoJcmVsZWFzZTogZnVuY3Rpb24oaW5UYXJnZXQpIHsKCQlmb3IgKHZhciBpID0gdGhpcy5jYXB0dXJlcy5sZW5ndGggLSAxOyBpID49IDA7IGktLSkgewoJCQlpZiAodGhpcy5jYXB0dXJlc1tpXS50YXJnZXQgPT09IGluVGFyZ2V0KSB7CgkJCQl0aGlzLmNhcHR1cmVzLnNwbGljZShpLDEpOwoJCQkJdGhpcy5zZXRDYXB0dXJlSW5mbyh0aGlzLmNhcHR1cmVzW3RoaXMuY2FwdHVyZXMubGVuZ3RoLTFdKTsKCQkJCWJyZWFrOwoJCQl9CgkJfQoJfSwKCS8vKiBTZXQgdGhlIGluZm9ybWF0aW9uIGZvciBhIGNhcHR1cmVkIGV2ZW50CglzZXRDYXB0dXJlSW5mbzogZnVuY3Rpb24oaW5JbmZvKSB7CgkJdGhpcy5jYXB0dXJlVGFyZ2V0ID0gaW5JbmZvICYmIGluSW5mby50YXJnZXQ7CgkJdGhpcy5mb3J3YXJkRXZlbnRzID0gaW5JbmZvICYmIGluSW5mby5mb3J3YXJkOwoJfQp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJImRvbS5jc3MiLAoJImRvbS5qcyIsCgkidHJhbnNmb3JtLmpzIiwKCSJDb250cm9sLmpzIiwKCSJwbGF0Zm9ybS5qcyIsCgkiYW5pbWF0aW9uLmpzIiwKCSJjb3Jkb3ZhLmpzIiwKCSJkaXNwYXRjaGVyLmpzIiwKCSJwcmV2aWV3LmpzIiwKCSJtb2RhbC5qcyIsCgkiZ2VzdHVyZS5qcyIsCgkiZHJhZy5qcyIsCgkidHJhbnNpdGlvbi5qcyIKKTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/platform.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglEZXRlcm1pbmVzIE9TIHZlcnNpb25zIG9mIHBsYXRmb3JtcyB0aGF0IG5lZWQgc3BlY2lhbCB0cmVhdG1lbnQuCglDYW4gaGF2ZSBvbmUgb2YgdGhlIGZvbGxvd2luZyBwcm9wZXJ0aWVzOgoKCSogYW5kcm9pZAoJKiBhbmRyb2lkQ2hyb21lIChDaHJvbWUgb24gQW5kcm9pZCwgc3RhbmRhcmQgc3RhcnRpbmcgaW4gNC4xKQoJKiBhbmRyb2lkRmlyZWZveAoJKiBpZQoJKiBpb3MKCSogd2Vib3MKCSogYmxhY2tiZXJyeQoJKiBzYWZhcmkgKGRlc2t0b3AgdmVyc2lvbikKCSogY2hyb21lIChkZXNrdG9wIHZlcnNpb24pCgkqIGZpcmVmb3ggKGRlc2t0b3AgdmVyc2lvbikKCglJZiB0aGUgcHJvcGVydHkgaXMgZGVmaW5lZCwgaXRzIHZhbHVlIHdpbGwgYmUgdGhlIG1ham9yIHZlcnNpb24JbnVtYmVyCglvZiB0aGUgcGxhdGZvcm0uCgoJRXhhbXBsZToKCgkJLy8gYW5kcm9pZCAyIGRvZXMgbm90IGhhdmUgM2QgY3NzCgkJaWYgKGVueW8ucGxhdGZvcm0uYW5kcm9pZCA8IDMpIHsKCQkJdCA9ICJ0cmFuc2xhdGUoMzBweCwgNTBweCkiOwoJCX0gZWxzZSB7CgkJCXQgPSAidHJhbnNsYXRlM2QoMzBweCwgNTBweCwgMCkiOwoJCX0KCQl0aGlzLmFwcGx5U3R5bGUoIi13ZWJraXQtdHJhbnNmb3JtIiwgdCk7CiovCmVueW8ucGxhdGZvcm0gPSB7CgkvLyogVHJ1ZSBpZiB0aGUgcGxhdGZvcm0gaGFzIG5hdGl2ZSBzaW5nbGUgZmluZ2VyIGV2ZW50cwoJdG91Y2g6IEJvb2xlYW4oKCJvbnRvdWNoc3RhcnQiIGluIHdpbmRvdykgfHwgd2luZG93Lm5hdmlnYXRvci5tc01heFRvdWNoUG9pbnRzKSwKCS8vKiBUcnVlIGlmIHRoZSBwbGF0Zm9ybSBoYXMgbmF0aXZlIGRvdWJsZSBmaW5nZXIgZXZlbnRzCglnZXN0dXJlOiBCb29sZWFuKCgib25nZXN0dXJlc3RhcnQiIGluIHdpbmRvdykgfHwgd2luZG93Lm5hdmlnYXRvci5tc01heFRvdWNoUG9pbnRzKQp9OwoKLy8qIEBwcm90ZWN0ZWQKKGZ1bmN0aW9uKCkgewoJdmFyIHVhID0gbmF2aWdhdG9yLnVzZXJBZ2VudDsKCXZhciBlcCA9IGVueW8ucGxhdGZvcm07Cgl2YXIgcGxhdGZvcm1zID0gWwoJCS8vIEFuZHJvaWQgNCsgdXNpbmcgQ2hyb21lCgkJe3BsYXRmb3JtOiAiYW5kcm9pZENocm9tZSIsIHJlZ2V4OiAvQW5kcm9pZCAuKiBDaHJvbWVcLyhcZCspWy5cZF0rL30sCgkJLy8gQW5kcm9pZCAyIC0gNAoJCXtwbGF0Zm9ybTogImFuZHJvaWQiLCByZWdleDogL0FuZHJvaWQgKFxkKykvfSwKCQkvLyBLaW5kbGUgRmlyZQoJCS8vIEZvcmNlIHZlcnNpb24gdG8gMiwgKGRlc2t0b3AgbW9kZSBkb2VzIG5vdCBsaXN0IGFuZHJvaWQgdmVyc2lvbikKCQl7cGxhdGZvcm06ICJhbmRyb2lkIiwgcmVnZXg6IC9TaWxrXC8xLi8sIGZvcmNlVmVyc2lvbjogMiwgZXh0cmE6IHtzaWxrOiAxfX0sCgkJLy8gS2luZGxlIEZpcmUgSEQKCQkvLyBGb3JjZSB2ZXJzaW9uIHRvIDQKCQl7cGxhdGZvcm06ICJhbmRyb2lkIiwgcmVnZXg6IC9TaWxrXC8yLi8sIGZvcmNlVmVyc2lvbjogNCwgZXh0cmE6IHtzaWxrOiAyfX0sCgkJLy8gV2luZG93cyBQaG9uZSA3IC0gOAoJCXtwbGF0Zm9ybTogIndpbmRvd3NQaG9uZSIsIHJlZ2V4OiAvV2luZG93cyBQaG9uZSAoPzpPUyApPyhcZCspWy5cZF0rL30sCgkJLy8gSUUgOCAtIDEwCgkJe3BsYXRmb3JtOiAiaWUiLCByZWdleDogL01TSUUgKFxkKykvfSwKCQkvLyBpT1MgMyAtIDUKCQkvLyBBcHBsZSBsaWtlcyB0byBtYWtlIHRoaXMgY29tcGxpY2F0ZWQKCQl7cGxhdGZvcm06ICJpb3MiLCByZWdleDogL2lQKD86aG9uZXxhZDsoPzogVTspPyBDUFUpIE9TIChcZCspL30sCgkJLy8gd2ViT1MgMSAtIDMKCQl7cGxhdGZvcm06ICJ3ZWJvcyIsIHJlZ2V4OiAvKD86d2VifGhwdylPU1wvKFxkKykvfSwKCQkvLyBkZXNrdG9wIFNhZmFyaQoJCXtwbGF0Zm9ybTogInNhZmFyaSIsIHJlZ2V4OiAvVmVyc2lvblwvKFxkKylbLlxkXStccytTYWZhcmkvfSwKCQkvLyBkZXNrdG9wIENocm9tZQoJCXtwbGF0Zm9ybTogImNocm9tZSIsIHJlZ2V4OiAvQ2hyb21lXC8oXGQrKVsuXGRdKy99LAoJCS8vIEZpcmVmb3ggb24gQW5kcm9pZAoJCXtwbGF0Zm9ybTogImFuZHJvaWRGaXJlZm94IiwgcmVnZXg6IC9BbmRyb2lkOy4qRmlyZWZveFwvKFxkKykvfSwKCQkvLyBGaXJlZm94T1MKCQl7cGxhdGZvcm06ICJmaXJlZm94T1MiLCByZWdleDogL01vYmlsZTsuKkZpcmVmb3hcLyhcZCspL30sCgkJLy8gZGVza3RvcCBGaXJlZm94CgkJe3BsYXRmb3JtOiAiZmlyZWZveCIsIHJlZ2V4OiAvRmlyZWZveFwvKFxkKykvfSwKCQkvLyBCbGFja2JlcnJ5IDEwKwoJCXtwbGF0Zm9ybTogImJsYWNrYmVycnkiLCByZWdleDogL0JCMVxkOy4qVmVyc2lvblwvKFxkK1wuXGQrKS99LAoJCS8vIFRpemVuCgkJe3BsYXRmb3JtOiAidGl6ZW4iLCByZWdleDogL1RpemVuIChcZCspL30KCV07Cglmb3IgKHZhciBpID0gMCwgcCwgbSwgdjsgKHAgPSBwbGF0Zm9ybXNbaV0pOyBpKyspIHsKCQltID0gcC5yZWdleC5leGVjKHVhKTsKCQlpZiAobSkgewoJCQlpZiAocC5mb3JjZVZlcnNpb24pIHsKCQkJCXYgPSBwLmZvcmNlVmVyc2lvbjsKCQkJfSBlbHNlIHsKCQkJCXYgPSBOdW1iZXIobVsxXSk7CgkJCX0KCQkJZXBbcC5wbGF0Zm9ybV0gPSB2OwoJCQlpZiAocC5leHRyYSkgewoJCQkJZW55by5taXhpbihlcCwgcC5leHRyYSk7CgkJCX0KCQkJYnJlYWs7CgkJfQoJfQoJLy8gdGhlc2UgcGxhdGZvcm1zIG9ubHkgYWxsb3cgb25lIGFyZ3VtZW50IGZvciBjb25zb2xlLmxvZwoJZW55by5kdW1iQ29uc29sZSA9IEJvb2xlYW4oZXAuYW5kcm9pZCB8fCBlcC5pb3MgfHwgZXAud2Vib3MpOwp9KSgpOw==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/preview.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglEaXNwYXRjaGVyIHByZXZpZXcgZmVhdHVyZTogYWxsb3cgY29udHJvbCBhbmNlc3RvcnMgb2YgdGhlIGV2ZW50Cgl0YXJnZXQgYSBjaGFuY2UgKGVsZGVzdCBmaXJzdCkgdG8gcmVhY3QgdmlhIGltcGxlbWVudGluZyAicHJldmlld0RvbUV2ZW50LiIKKi8KLy8qIEBwcm90ZWN0ZWQKKGZ1bmN0aW9uKCkgewoJdmFyIGZuID0gInByZXZpZXdEb21FdmVudCI7CgkvLwoJdmFyIHByZXZpZXcgPSB7CgkJZmVhdHVyZTogZnVuY3Rpb24oZSkgewoJCQlwcmV2aWV3LmRpc3BhdGNoKGUsIGUuZGlzcGF0Y2hUYXJnZXQpOwoJCX0sCgkJZGlzcGF0Y2g6IGZ1bmN0aW9uKGUsIGMpIHsKCQkJdmFyIGwkID0gdGhpcy5idWlsZExpbmVhZ2UoYyk7CgkJCS8vIGhhbmRsZXJzIHJldHVybiB0cnVlIHRvIGFib3J0IHByZXZpZXcgYW5kIHByZXZlbnQgZGVmYXVsdCBldmVudCBwcm9jZXNzaW5nLgoJCQlmb3IgKHZhciBpPTAsIGw7IChsPWwkW2ldKTsgaSsrKSB7CgkJCQlpZiAobFtmbl0gJiYgbFtmbl0oZSkgPT09IHRydWUpIHsKCQkJCQllLnByZXZlbnREaXNwYXRjaCA9IHRydWU7CgkJCQkJcmV0dXJuOwoJCQkJfQoJCQl9CgkJfSwKCQkvLyB3ZSBhc2NlbmQgbWFraW5nIGEgbGlzdCBvZiBlbnlvIGNvbnRyb2xzCgkJLy8gTk9URTogdGhlIGNvbnRyb2wgaXMgY29uc2lkZXJlZCBpdHMgb3duIGFuY2VzdG9yCgkJYnVpbGRMaW5lYWdlOiBmdW5jdGlvbihpbkNvbnRyb2wpIHsKCQkJdmFyIGwgPSBbXSwgYyA9IGluQ29udHJvbDsKCQkJd2hpbGUgKGMpIHsKCQkJCWwudW5zaGlmdChjKTsKCQkJCWMgPSBjLnBhcmVudDsKCQkJfQoJCQlyZXR1cm4gbDsKCQl9Cgl9OwoJLy8KCWVueW8uZGlzcGF0Y2hlci5mZWF0dXJlcy5wdXNoKHByZXZpZXcuZmVhdHVyZSk7Cn0pKCk7
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/transform.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qIEBwcm90ZWN0ZWQKKGZ1bmN0aW9uKCkgewoJZW55by5kb20uY2FsY0NhbkFjY2VsZXJhdGUgPSBmdW5jdGlvbigpIHsKCQkvKiBBbmRyb2lkIDIgaXMgYSBsaWFyOiBpdCBkb2VzIE5PVCBzdXBwb3J0IDNEIHRyYW5zZm9ybXMsIGV2ZW4gdGhvdWdoIFBlcnNwZWN0aXZlIGlzIHRoZSBiZXN0IGNoZWNrICovCgkJaWYgKGVueW8ucGxhdGZvcm0uYW5kcm9pZCA8PSAyKSB7CgkJCXJldHVybiBmYWxzZTsKCQl9CgkJdmFyIHAkID0gWyJwZXJzcGVjdGl2ZSIsICJXZWJraXRQZXJzcGVjdGl2ZSIsICJNb3pQZXJzcGVjdGl2ZSIsICJtc1BlcnNwZWN0aXZlIiwgIk9QZXJzcGVjdGl2ZSJdOwoJCWZvciAodmFyIGk9MCwgcDsgKHA9cCRbaV0pOyBpKyspIHsKCQkJaWYgKHR5cGVvZiBkb2N1bWVudC5ib2R5LnN0eWxlW3BdICE9ICJ1bmRlZmluZWQiKSB7CgkJCQlyZXR1cm4gdHJ1ZTsKCQkJfQoJCX0KCQlyZXR1cm4gZmFsc2U7Cgl9OwoJdmFyIGNzc1RyYW5zZm9ybVByb3BzID0gWyJ0cmFuc2Zvcm0iLCAiLXdlYmtpdC10cmFuc2Zvcm0iLCAiLW1vei10cmFuc2Zvcm0iLCAiLW1zLXRyYW5zZm9ybSIsICItby10cmFuc2Zvcm0iXTsKCXZhciBzdHlsZVRyYW5zZm9ybVByb3BzID0gWyJ0cmFuc2Zvcm0iLCAid2Via2l0VHJhbnNmb3JtIiwgIk1velRyYW5zZm9ybSIsICJtc1RyYW5zZm9ybSIsICJPVHJhbnNmb3JtIl07CgllbnlvLmRvbS5nZXRDc3NUcmFuc2Zvcm1Qcm9wID0gZnVuY3Rpb24oKSB7CgkJaWYgKHRoaXMuX2Nzc1RyYW5zZm9ybVByb3ApIHsKCQkJcmV0dXJuIHRoaXMuX2Nzc1RyYW5zZm9ybVByb3A7CgkJfQoJCXZhciBpID0gZW55by5pbmRleE9mKHRoaXMuZ2V0U3R5bGVUcmFuc2Zvcm1Qcm9wKCksIHN0eWxlVHJhbnNmb3JtUHJvcHMpOwoJCXRoaXMuX2Nzc1RyYW5zZm9ybVByb3AgPSBjc3NUcmFuc2Zvcm1Qcm9wc1tpXTsKCQlyZXR1cm4gdGhpcy5fY3NzVHJhbnNmb3JtUHJvcDsKCX07CgllbnlvLmRvbS5nZXRTdHlsZVRyYW5zZm9ybVByb3AgPSBmdW5jdGlvbigpIHsKCQlpZiAodGhpcy5fc3R5bGVUcmFuc2Zvcm1Qcm9wIHx8ICFkb2N1bWVudC5ib2R5KSB7CgkJCXJldHVybiB0aGlzLl9zdHlsZVRyYW5zZm9ybVByb3A7CgkJfQoJCWZvciAodmFyIGkgPSAwLCBwOyAocCA9IHN0eWxlVHJhbnNmb3JtUHJvcHNbaV0pOyBpKyspIHsKCQkJaWYgKHR5cGVvZiBkb2N1bWVudC5ib2R5LnN0eWxlW3BdICE9ICJ1bmRlZmluZWQiKSB7CgkJCQl0aGlzLl9zdHlsZVRyYW5zZm9ybVByb3AgPSBwOwoJCQkJcmV0dXJuIHRoaXMuX3N0eWxlVHJhbnNmb3JtUHJvcDsKCQkJfQoJCX0KCX07CgllbnlvLmRvbS5kb21UcmFuc2Zvcm1zVG9Dc3MgPSBmdW5jdGlvbihpblRyYW5zZm9ybXMpIHsKCQl2YXIgbiwgdiwgdGV4dCA9ICcnOwoJCWZvciAobiBpbiBpblRyYW5zZm9ybXMpIHsKCQkJdiA9IGluVHJhbnNmb3Jtc1tuXTsKCQkJaWYgKCh2ICE9PSBudWxsKSAmJiAodiAhPT0gdW5kZWZpbmVkKSAmJiAodiAhPT0gIiIpKSB7CgkJCQl0ZXh0ICs9ICBuICsgJygnICsgdiArICcpICc7CgkJCX0KCQl9CgkJcmV0dXJuIHRleHQ7Cgl9OwoJZW55by5kb20udHJhbnNmb3Jtc1RvRG9tID0gZnVuY3Rpb24oaW5Db250cm9sKSB7CgkJdmFyIHQgPSB0aGlzLmRvbVRyYW5zZm9ybXNUb0NzcyhpbkNvbnRyb2wuZG9tVHJhbnNmb3Jtcyk7CgkJdmFyIHN0ID0gaW5Db250cm9sLmhhc05vZGUoKSA/IGluQ29udHJvbC5ub2RlLnN0eWxlIDogbnVsbDsKCQl2YXIgZHMgPSBpbkNvbnRyb2wuZG9tU3R5bGVzOwoJCXZhciBzcCA9IHRoaXMuZ2V0U3R5bGVUcmFuc2Zvcm1Qcm9wKCk7CgkJdmFyIGNwID0gdGhpcy5nZXRDc3NUcmFuc2Zvcm1Qcm9wKCk7CgkJaWYgKHNwICYmIGNwKSB7CgkJCWRzW2NwXSA9IHQ7CgkJCWlmIChzdCkgewoJCQkJc3Rbc3BdID0gdDsKCQkJfSBlbHNlIHsKCQkJCWluQ29udHJvbC5kb21TdHlsZXNDaGFuZ2VkKCk7CgkJCX0KCQl9Cgl9OwoJLy8qIEBwdWJsaWMKCS8qKgoJCVJldHVybnMgdHJ1ZSBpZiB0aGUgcGxhdGZvcm0gc3VwcG9ydHMgQ1NTMyBUcmFuc2Zvcm1zCgkqLwoJZW55by5kb20uY2FuVHJhbnNmb3JtID0gZnVuY3Rpb24oKSB7CgkJcmV0dXJuIEJvb2xlYW4odGhpcy5nZXRTdHlsZVRyYW5zZm9ybVByb3AoKSk7Cgl9OwoJLyoqCgkJUmV0dXJucyB0cnVlIGlmIHBsYXRmb3JtIHN1cHBvcnRzIENTUzMgM0QgVHJhbnNmb3Jtcy4KCgkJVHlwaWNhbGx5IHVzZWQgbGlrZSB0aGlzOgoKCQkJaWYgKGVueW8uZG9tLmNhbkFjY2VsZXJhdGUoKSkgewoJCQkJZW55by5kb20udHJhbnNmb3JtVmFsdWUodGhpcy4kLnNsaWRpbmdUaGluZywgInRyYW5zbGF0ZTNkIiwgeCArICIsIiArIHkgKyAiLCIgKyAiMCIpCgkJCX0gZWxzZSB7CgkJCQllbnlvLmRvbS50cmFuc2Zvcm1WYWx1ZSh0aGlzLiQuc2xpZGluZ1RoaW5nLCAidHJhbnNsYXRlIiwgeCArICIsIiArIHkpOwoJCQl9CgoJKi8KCWVueW8uZG9tLmNhbkFjY2VsZXJhdGUgPSBmdW5jdGlvbigpIHsKCQlyZXR1cm4gdGhpcy5hY2NlbGVyYW5kbyAhPT0gdW5kZWZpbmVkID8gdGhpcy5hY2NlbGVyYW5kbzogZG9jdW1lbnQuYm9keSAmJiAodGhpcy5hY2NlbGVyYW5kbyA9IHRoaXMuY2FsY0NhbkFjY2VsZXJhdGUoKSk7Cgl9OwoJLyoqCgkJQXBwbGllcyBhIHNlcmllcyBvZiB0cmFuc2Zvcm1zIHRvIF9pbkNvbnRyb2xfLCB1c2luZyB0aGUgcGxhdGZvcm0ncyBwcmVmaXhlZCB0cmFuc2Zvcm0gcHJvcGVydHkuCgoJCSoqTm90ZToqKiBUcmFuc2Zvcm1zIGFyZSBub3QgY29tbXV0YXRpdmUsIHNvIG9yZGVyIGlzIGltcG9ydGFudAoKCQlUcmFuc2Zvcm0gdmFsdWVzIGFyZSB1cGRhdGVkIGJ5IHN1Y2Nlc3NpdmUgY2FsbHM6CgoJCQllbnlvLmRvbS50cmFuc2Zvcm0oY29udHJvbCwge3RyYW5zbGF0ZTogIjMwcHgsIDQwcHgiLCBzY2FsZTogMiwgcm90YXRlOiAiMjBkZWcifSk7CgkJCWVueW8uZG9tLnRyYW5zZm9ybShjb250cm9sLCB7c2NhbGU6IDMsIHNrZXdYOiAiLTMwZGVnIn0pOwoKCQlpcyBlcXVpdmFsZW50IHRvOgoKCQkJZW55by5kb20udHJhbnNmb3JtKGNvbnRyb2wsIHt0cmFuc2xhdGU6ICIzMHB4LCA0MHB4Iiwgc2NhbGU6IDMsIHJvdGF0ZTogIjIwZGVnIiwgc2tld1g6ICItMzBkZWcifSk7CgoJCVdoZW4gYXBwbHlpbmcgdGhlc2UgdHJhbnNmb3JtcyBpbiB3ZWJraXQgYnJvd3NlciwgdGhpcyBpcyBlcXVpdmFsZW50IHRvOgoKCQkJY29udHJvbC5hcHBseVN0eWxlKCItd2Via2l0LXRyYW5zZm9ybSIsICJ0cmFuc2xhdGUoMzBweCwgNDBweCkgc2NhbGUoMykgcm90YXRlKDIwZGVnKSBza2V3WCgtMzBkZWcpIik7CgoJCUFuZCBpbiBmaXJlZm94LCB0aGlzIGlzIGVxdWl2YWxlbnQgdG86CgoJCQljb250cm9sLmFwcGx5U3R5bGUoIi1tb3otdHJhbnNmb3JtIiwgInRyYW5zbGF0ZSgzMHB4LCA0MHB4KSBzY2FsZSgzKSByb3RhdGUoMjBkZWcpIHNrZXdYKC0zMGRlZykiKTsKCgkqLwoJZW55by5kb20udHJhbnNmb3JtID0gZnVuY3Rpb24oaW5Db250cm9sLCBpblRyYW5zZm9ybXMpIHsKCQl2YXIgZCA9IGluQ29udHJvbC5kb21UcmFuc2Zvcm1zID0gaW5Db250cm9sLmRvbVRyYW5zZm9ybXMgfHwge307CgkJZW55by5taXhpbihkLCBpblRyYW5zZm9ybXMpOwoJCXRoaXMudHJhbnNmb3Jtc1RvRG9tKGluQ29udHJvbCk7Cgl9OwoKCS8qKgoJCUFwcGx5IGEgc2luZ2xlIHRyYW5zZm9ybSB0byBfaW5Db250cm9sXy4KCgkJRXhhbXBsZToKCgkJCXRhcDogZnVuY3Rpb24oaW5TZW5kZXIsIGluRXZlbnQpIHsKCQkJCXZhciBjID0gaW5FdmVudC5vcmlnaW5hdG9yOwoJCQkJdmFyIHIgPSBjLnJvdGF0aW9uIHx8IDA7CgkJCQlyID0gKHIgKyA0NSkgJSAzNjA7CgkJCQljLnJvdGF0aW9uID0gcjsKCQkJCWVueW8uZG9tLnRyYW5zZm9ybVZhbHVlKGMsICJyb3RhdGUiLCByKTsKCQkJfQoKCQlUaGlzIHdpbGwgcm90YXRlIHRoZSB0YXBwZWQgY29udHJvbCBieSA0NSBkZWdyZWVzIGNsb2Nrd2lzZS4KCSovCgllbnlvLmRvbS50cmFuc2Zvcm1WYWx1ZSA9IGZ1bmN0aW9uKGluQ29udHJvbCwgaW5UcmFuc2Zvcm0sIGluVmFsdWUpIHsKCQl2YXIgZCA9IGluQ29udHJvbC5kb21UcmFuc2Zvcm1zID0gaW5Db250cm9sLmRvbVRyYW5zZm9ybXMgfHwge307CgkJZFtpblRyYW5zZm9ybV0gPSBpblZhbHVlOwoJCXRoaXMudHJhbnNmb3Jtc1RvRG9tKGluQ29udHJvbCk7Cgl9OwoJLy8qIEBwcm90ZWN0ZWQKCS8qKgoJCUFwcGxpZXMgYSB0cmFuc2Zvcm0gdGhhdCBzaG91bGQgdHJpZ2dlciBHUFUgY29tcG9zaXRpbmcgZm9yIF9pbkNvbnRyb2xfCgkqLwoJZW55by5kb20uYWNjZWxlcmF0ZSA9IGZ1bmN0aW9uKGluQ29udHJvbCwgaW5WYWx1ZSkgewoJCXZhciB2ID0gaW5WYWx1ZSA9PSAiYXV0byIgPyB0aGlzLmNhbkFjY2VsZXJhdGUoKSA6IGluVmFsdWU7CgkJdGhpcy50cmFuc2Zvcm1WYWx1ZShpbkNvbnRyb2wsICJ0cmFuc2xhdGVaIiwgdiA/IDAgOiBudWxsKTsKCX07Cn0pKCk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/dom/transition.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kb20udHJhbnNpdGlvbiA9IChlbnlvLnBsYXRmb3JtLmlvcyB8fCBlbnlvLnBsYXRmb3JtLmFuZHJvaWQgfHwgZW55by5wbGF0Zm9ybS5jaHJvbWUgfHwgZW55by5wbGF0Zm9ybS5hbmRyb2lkQ2hyb21lIHx8IGVueW8ucGxhdGZvcm0uc2FmYXJpKQoJPyAiLXdlYmtpdC10cmFuc2l0aW9uIgoJOiAoZW55by5wbGF0Zm9ybS5maXJlZm94IHx8IGVueW8ucGxhdGZvcm0uZmlyZWZveE9TIHx8IGVueW8ucGxhdGZvcm0uYW5kcm9pZEZpcmVmb3gpCgkJPyAiLW1vei10cmFuc2l0aW9uIgoJCTogInRyYW5zaXRpb24iOw==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/enyo.design"
Content-Type: application/octet-stream; x-encoding=base64
ewogICAgInBhbGV0dGUiOiBbCiAgICAgICB7CiAgICAgICAgICAgICJuYW1lIjogIkVueW8iLAogICAgICAgICAgICAiaXRlbXMiOiBbCiAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgIm5hbWUiOiAiZW55by5TY3JvbGxlciIsCiAgICAgICAgICAgICAgICAgICAgImtpbmQiOiAiZW55by5TY3JvbGxlciIsCiAgICAgICAgICAgICAgICAgICAgImRlc2NyaXB0aW9uIjogIkEgc2Nyb2xsZXIiLAogICAgICAgICAgICAgICAgICAgICJjb25maWciOiB7CiAgICAgICAgICAgICAgICAgICAgICAgICJraW5kIjogImVueW8uU2Nyb2xsZXIiLAogICAgICAgICAgICAgICAgICAgICAgICAiZml0IjogdHJ1ZQogICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIH0sCiAgICAgICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAgICAgIm5hbWUiOiAiZW55by5SZXBlYXRlciIsCiAgICAgICAgICAgICAgICAgICAgImtpbmQiOiAiZW55by5SZXBlYXRlciIsCiAgICAgICAgICAgICAgICAgICAgImRlc2NyaXB0aW9uIjogIkEgcmVwZWF0ZXIiLAogICAgICAgICAgICAgICAgICAgICJjb25maWciOiB7CiAgICAgICAgICAgICAgICAgICAgICAgICJraW5kIjogImVueW8uUmVwZWF0ZXIiCiAgICAgICAgICAgICAgICAgICAgfSwKICAgICAgICAgICAgICAgICAgICAib3B0aW9ucyI6IHsKICAgICAgICAgICAgICAgICAgICAgICAgImlzUmVwZWF0ZXIiOiB0cnVlCiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgfQogICAgICAgICAgICBdCiAgICAgICAgfQogICAgXSwKICAgICJpbnNwZWN0b3IiOiBbCiAgICAgICAgewogICAgICAgICAgICAidHlwZSI6ICJraW5kIiwKICAgICAgICAgICAgIm5hbWUiOiAiZW55by5SZXBlYXRlciIsCiAgICAgICAgICAgICJldmVudHMiOiB7CiAgICAgICAgICAgICAgICAib25TZXR1cEl0ZW0iOiB7CiAgICAgICAgICAgICAgICAgICAgImZpbHRlckxldmVsIjogInVzZWZ1bCIKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgIH0KICAgIF0KfQo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ext/BooleanBinding.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uIChlbnlvKSB7CgoJLy8qQHB1YmxpYwoJLyoqCgkJX2VueW8uQm9vbGVhbkJpbmRpbmdfIGlzIGEgYmluZGluZyB0aGF0IHdpbGwgdHlwZS1jYXN0IGFueSB0cnV0aHkgb3IgZmFsc2V5CgkJdmFsdWUgdG8gYW4gZXhwbGljaXQgYm9vbGVhbiB2YWx1ZS4KCSovCgllbnlvLmtpbmQoewoKCQkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCQkvLyBQVUJMSUMgUFJPUEVSVElFUwoKCQkvLypAcHVibGljCgkJbmFtZTogImVueW8uQm9vbGVhbkJpbmRpbmciLAoKCQkvLypAcHVibGljCgkJa2luZDogImVueW8uQmluZGluZyIsCgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIFBST1RFQ1RFRCBNRVRIT0RTCgoJCS8vKkBwcm90ZWN0ZWQKCQl0cmFuc2Zvcm06IGZ1bmN0aW9uICh2YWx1ZSwgZGlyZWN0aW9uLCBiaW5kaW5nKSB7CgkJCXJldHVybiAhISB2YWx1ZTsKCQl9CgoJfSk7Cgp9KShlbnlvKTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ext/BooleanOnlyBinding.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uIChlbnlvKSB7CgoJLy8qQHB1YmxpYwoJLyoqCgkJX2VueW8uQm9vbGVhbk9ubHlCaW5kaW5nXyBpcyBhIGJpbmRpbmcgdGhhdCB3aWxsIG9ubHkgY29udGludWUgcHJvcGFnYXRpb24KCQlpZiB0aGUgdmFsdWUgYmVpbmcgcGFzc2VkIGlzIGFuIGV4cGxpY2l0IGJvb2xlYW4gdmFsdWUuCgkqLwoJZW55by5raW5kKHsKCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gUFVCTElDIFBST1BFUlRJRVMKCgkJLy8qQHB1YmxpYwoJCW5hbWU6ICJlbnlvLkJvb2xlYW5Pbmx5QmluZGluZyIsCgoJCS8vKkBwdWJsaWMKCQlraW5kOiAiZW55by5CaW5kaW5nIiwKCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gUFJPVEVDVEVEIE1FVEhPRFMKCgkJLy8qQHByb3RlY3RlZAoJCXRyYW5zZm9ybTogZnVuY3Rpb24gKHZhbHVlLCBkaXJlY3Rpb24sIGJpbmRpbmcpIHsKCQkJaWYgKHZhbHVlICE9PSB0cnVlICYmIHZhbHVlICE9PSBmYWxzZSkgewoJCQkJYmluZGluZy5zdG9wKCk7CgkJCX0KCQkJcmV0dXJuIHZhbHVlOwoJCX0KCgl9KTsKCn0pKGVueW8pOwo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ext/InputBinding.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uIChlbnlvKSB7CgoJLy8qQHB1YmxpYwoJLyoqCgkJVGhpcyBiaW5kaW5nIGlzIGRlc2lnbmVkIHRvIHdvcmsgd2l0aCBpdHMgc291cmNlIG9yIHRhcmdldAoJCWJlaW5nIGFuIF9pbnB1dF8gd2l0aCBhbiBvcHRpb25hbCBfcGxhY2Vob2xkZXJfIHZhbHVlLiBUaGlzCgkJd2lsbCBrZWVwIGFuIGlucHV0IGZyb20gc2hvd2luZyBhbiB1Z2x5IHVuZGVmaW5lZCB3aGVuIHRoZXJlCgkJaXMgbm8gY29udGVudCBhbmQgaW5zdGVhZCBwcm9wYWdhdGUgdGhlIF9wbGFjZWhvbGRlcl8gdmFsdWUKCQl0byB0aGUgb3Bwb3NpdGUgZW5kLgoJKi8KCWVueW8ua2luZCh7CgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIFBVQkxJQyBQUk9QRVJUSUVTCgoJCS8vKkBwdWJsaWMKCQluYW1lOiAiZW55by5JbnB1dEJpbmRpbmciLAoKCQkvLypAcHVibGljCgkJa2luZDogImVueW8uQmluZGluZyIsCgoJCS8vKkBwdWJsaWMKCQl0d29XYXk6IHRydWUsCgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIFBST1RFQ1RFRCBNRVRIT0RTCgoJCS8vKkBwcm90ZWN0ZWQKCQl0cmFuc2Zvcm06IGZ1bmN0aW9uICh2YWx1ZSwgZGlyZWN0aW9uLCBiaW5kaW5nKSB7CgkJCXZhciBzb3VyY2UgPSBiaW5kaW5nLnNvdXJjZSB8fCB7fTsKCQkJdmFyIHRhcmdldCA9IGJpbmRpbmcudGFyZ2V0IHx8IHt9OwoJCQl2YXIgcGggPSBzb3VyY2UucGxhY2hvbGRlciB8fCB0YXJnZXQucGxhY2Vob2xkZXIgfHwgIiI7CgkJCWlmICghZW55by5leGlzdHModmFsdWUpIHx8IG51bGwgPT09IHZhbHVlIHx8IHZhbHVlLmxlbmd0aCA9PT0gMCkgewoJCQkJcmV0dXJuIHBoOwoJCQl9IGVsc2UgewoJCQkJcmV0dXJuIHZhbHVlOwoJCQl9CgkJfQoKCX0pOwoKfSkoZW55byk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ext/macroize.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qIEBwdWJsaWMKCi8qKgoJUG9wdWxhdGVzIGEgc3RyaW5nIHRlbXBsYXRlIHdpdGggZGF0YSB2YWx1ZXMuCgoJUmV0dXJucyBhIGNvcHkgb2YgX2luVGV4dF8sIHdpdGggbWFjcm9zIGRlZmluZWQgYnkgX2luUGF0dGVybl8gcmVwbGFjZWQgYnkKCW5hbWVkIHZhbHVlcyBpbiBfaW5NYXBfLgoKCV9pblBhdHRlcm5fIG1heSBiZSBvbWl0dGVkLCBpbiB3aGljaCBjYXNlIHRoZSBkZWZhdWx0IG1hY3JvIHBhdHRlcm4gaXMgdXNlZC4KCVRoZSBkZWZhdWx0IHBhdHRlcm4gbWF0Y2hlcyBtYWNyb3Mgb2YgdGhlIGZvcm0KCgkJeyRuYW1lfQoKCUV4YW1wbGU6CgoJCS8vIFJldHVybnMgIk15IG5hbWUgaXMgQmFybmV5LiIKCQllbnlvLm1hY3JvaXplKCJNeSBuYW1lIGlzIHskbmFtZX0uIiwge25hbWU6ICJCYXJuZXkifSk7CgoJRG90IG5vdGF0aW9uIGlzIHN1cHBvcnRlZCwgbGlrZSBzbzoKCgkJdmFyIGluZm8gPSB7CgkJCXByb2R1Y3RfMDogewoJCQkJbmFtZTogIkdpem1vIgoJCQkJd2VpZ2h0OiAzCgkJCX0KCQl9CgkJLy8gUmV0dXJucyAiRWFjaCBHaXptbyB3ZWlnaHMgMyBwb3VuZHMuIgoJCWVueW8ubWFjcm9pemUoIkVhY2ggeyRwcm9kdWN0XzAubmFtZX0gd2VpZ2hzIHskcHJvZHVjdF8wLndlaWdodH0gcG91bmRzLiIsIGluZm8pOwoqLwplbnlvLm1hY3JvaXplID0gZnVuY3Rpb24oaW5UZXh0LCBpbk1hcCwgaW5QYXR0ZXJuKSB7Cgl2YXIgdiwgd29ya2luZywgcmVzdWx0ID0gaW5UZXh0LCBwYXR0ZXJuID0gaW5QYXR0ZXJuIHx8IGVueW8ubWFjcm9pemUucGF0dGVybjsKCXZhciBmbiA9IGZ1bmN0aW9uKG1hY3JvLCBuYW1lKSB7CgkJdiA9IGVueW8uZ2V0UGF0aC5jYWxsKGluTWFwLCBuYW1lKTsKCQlpZiAodiA9PT0gdW5kZWZpbmVkIHx8IHYgPT09IG51bGwpIHsKCQkJcmV0dXJuICJ7JCIgKyBuYW1lICsgIn0iOwoJCX0KCQl3b3JraW5nID0gdHJ1ZTsKCQlyZXR1cm4gdjsKCX07Cgl2YXIgcHJldmVudCA9IDA7CglkbyB7CgkJd29ya2luZyA9IGZhbHNlOwoJCXJlc3VsdCA9IHJlc3VsdC5yZXBsYWNlKHBhdHRlcm4sIGZuKTsKCQkvLyBpZiBpdGVyYXRpbmcgbW9yZSB0aGFuIDIwIHRpbWVzLCB3ZSBhc3N1bWUgYSByZWN1cnNpb24gKHdlIHNob3VsZCBwcm9iYWJseSB0aHJvdykKCQlpZiAoKytwcmV2ZW50ID49IDIwKSB7CgkJCXRocm93KCJlbnlvLm1hY3JvaXplOiByZWN1cnNpb24gdG9vIGRlZXAiKTsKCQl9Cgl9IHdoaWxlICh3b3JraW5nKTsKCXJldHVybiByZXN1bHQ7Cn07CgovKioKCVNpbWlsYXIgdG8gX2VueW8ubWFjcm9pemVfLCBidXQgcGVyZm9ybXMgb25seSBvbmUgaXRlcmF0aW9uIG9mIHRoZSBfcmVwbGFjZV8KCWNhbGwuIFRoaXMgbWVhbnMgdGhhdCByZWN1cnNpdmUgZXhwYW5zaW9uIG9mIG1hY3JvcyBpc24ndCBwb3NzaWJsZSwgYnV0IGl0Cglhdm9pZHMgdGhlIGV4dHJhIHByb2Nlc3NpbmcgbmVlZGVkIHRvIGZpbmQgcmVjdXJzaXZlIHVzZS4KKi8KZW55by5xdWlja01hY3JvaXplID0gZnVuY3Rpb24oaW5UZXh0LCBpbk1hcCwgaW5QYXR0ZXJuKSB7Cgl2YXIgdiwgcmVzdWx0ID0gaW5UZXh0LCBwYXR0ZXJuID0gaW5QYXR0ZXJuIHx8IGVueW8ubWFjcm9pemUucGF0dGVybjsKCXZhciBmbiA9IGZ1bmN0aW9uKG1hY3JvLCBuYW1lKSB7CgkJaWYgKG5hbWUgaW4gaW5NYXApIHsKCQkJdiA9IGluTWFwW25hbWVdOwoJCX0gZWxzZSB7CgkJCXYgPSBlbnlvLmdldFBhdGguY2FsbChpbk1hcCwgbmFtZSk7CgkJfQoJCXJldHVybiAodiA9PT0gdW5kZWZpbmVkIHx8IHYgPT09IG51bGwpID8gInskIiArIG5hbWUgKyAifSIgOiB2OwoJfTsKCXJlc3VsdCA9IHJlc3VsdC5yZXBsYWNlKHBhdHRlcm4sIGZuKTsKCXJldHVybiByZXN1bHQ7Cn07CgovLyogQHByb3RlY3RlZAoKLy8gTWF0Y2hlcyBtYWNyb3Mgb2YgdGhlIGZvcm0geyRuYW1lfS4KZW55by5tYWNyb2l6ZS5wYXR0ZXJuID0gL1x7XCQoW157fV0qKVx9L2c7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ext/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJIm1hY3JvaXplLmpzIiwKCSJJbnB1dEJpbmRpbmcuanMiLAoJIkJvb2xlYW5CaW5kaW5nLmpzIiwKCSJCb29sZWFuT25seUJpbmRpbmcuanMiCik7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/Application.js"
Content-Type: application/octet-stream; x-encoding=base64
(function (enyo) {

	//*@public
	/**
		Tracks the applications running at any given time. For the sake of
		convenience, and in order to provide some debugging tools, we collect
		the running apps here for later reference. When _enyo.Application_
		instances are destroyed, they know to remove themselves from this table.
	*/
	var applications = enyo.applications = {};

	//*@protected
	/**
		Used internally; maintains registration of applications with the
		framework.
	*/
	var register = function (app) {
		applications[app.id] = app;
	};

	//*@protected
	/**
		Used internally; unregisters applications that have been destroyed.
	*/
	var unregister = function (app) {
		var kind = app.kindName;
		var kinds = applications[kind] || [];
		var idx = enyo.indexOf(app, kinds);
		if (!~idx) {
			kinds.splice(idx, 1);
		}
	};

	//*@protected
	var _destroy_controllers = function () {
		var name;
		// we iterate over the controllers that are stored in
		// the application hash
		for (name in this.controllers) {
			// we can't know that all of those controllers were placed
			// there by this application or that its owner hasn't been
			// set to something else (owner implying this application
			// instanced the controller and thus is responsible for its
			// life-cycle) so we test for that and destroy it if it is
			if (this === this.controllers[name].owner) {
				if ("function" === typeof this.controllers[name].destroy) {
					this.controllers[name].destroy();
				}
			}
			// regardless of ownership we release the controller reference
			// and hope whoever was responsible for the controller will
			// clean it up
			delete this.controllers[name];
		}
		// remove the reference to the object entirely
		delete this.controllers;
	};

	//*@protected
	var _setup_controllers = function () {
		var kinds = this.controllers || [];
		var controllers = this.controllers = {};
		var len = kinds.length;
		var idx = 0;
		var kind;
		for (; idx < len; ++idx) {
			kind = kinds[idx];
			// we need the name of the instance whether the controller is global
			// or app-specific
			var name = kind.name;
			// there is the optional global flag that indicates if the controller
			// is to be instanced outside the scope of the application
			var global = Boolean(kind.global);
			var Ctor;
			var inst;
			// cleanup
			delete kind.global;
			delete kind.name;
			// if the definition does not supply a controller kind, we add one
			if (!("kind" in kind)) {
				kind.kind = "enyo.Controller";
			}
			// create a kind constructor for the controller with all of the given
			// properties
			Ctor = enyo.kind(kind);
			inst = new Ctor({owner: this, app: this});
			// if the controller is not a global controller, we create it as part
			// of our applications controller store
			if (false === global) {
				controllers[name] = inst;
			} else {
				enyo.setPath(name, inst);
			}
		}
	};

	//*@protected
	enyo.kind.postConstructors.push(function () {
		if (!this._is_application) {
			return;
		}

		// now that any controllers for the application have been
		// initialized, we test to see if we're supposed to
		// automatically start
		if (true === this.autoStart) {
			this.start();
		}
	});

	//*@public
	/**
		_enyo.Application_ is a kind used to coordinate execution of a given
		collection of _enyo_ objects. There may be one or more instances
		running--with certain limitations, such as which one is rendered into
		the _document.body_. (There is no limitation if each instance is
		rendered into a separate DOM node, or if the instances are nested.)

		This kind also provides the ability to namespace and automatically
		initialize any controllers of the application.
	*/
	enyo.kind({

		// ...........................
		// PUBLIC PROPERTIES

		//*@public
		name: "enyo.Application",

		//*@public
		kind: "enyo.ViewController",

		//*@public
		autoStart: true,

		//*@public
		renderOnStart: true,

		//*@public
		controllers: null,

		//*@public
		concat: ["controllers"],

		// ...........................
		// PROTECTED PROPERTIES

		//*@protected
		_is_application: true,

		// ...........................
		// PUBLIC METHODS

		//*@public
		/**
			If the _autoStart_ flag is set to true, this method is
			automatically executed when the constructor is called. Otherwise,
			it may be executed whenever the application should begin execution.
		*/
		start: function () {
			if (true === this.renderOnStart) {
				this.render();
			}
		},

		// ...........................
		// PROTECTED METHODS

		//*@protected
		constructor: function (props) {
			if (props && enyo.exists(props.name)) {
				enyo.setPath(props.name, this);
				this.id = props.name;
				delete props.name;
			} else {
				this.id = enyo.uid("_application_");
			}
			this.inherited(arguments);
			// we register kind of early in the process in case any controllers
			// or other initialization assumes it will be there...
			register(this);
		},

		//*@protected
		constructed: function () {
			// we need to make sure that the controllers are already initialized
			// before we create our view according to the view controller's API
			_setup_controllers.call(this);
			// now we let it continue as usual
			this.inherited(arguments);
		},

		//*@protected
		/**
			The overloaded \_create_view method ensures that the appropriate
			values are supplied to the new view instance.
		*/
		_create_view: function () {
			// this is the constructor for the view kind
			var Ctor = this.get("_view_kind");
			// the properties we want to supply to the view are the
			// app (the reference to this application instance) and the
			// \_bubble_target so events are bubbled to us
			this.set("view", new Ctor({app: this, _bubble_target: this}));
		},

		//*@protected
		_make_view_name: function () {
			return enyo.uid("_application_view_");
		},

		//*@protected
		destroy: function () {
			// release/destroy all controllers associated with
			// this instance of the application
			_destroy_controllers.call(this);
			// do the normal breakdown
			this.inherited(arguments);
			// unregister this as an active application
			unregister(this);
		}

	});

}(enyo));

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/Binding.js"
Content-Type: application/octet-stream; x-encoding=base64
(function () {

	// add bindings to the automatically concatenatable properties in enyo
	enyo.concat.push("bindings");

	//*@protected
	/**
		Used internally to track bindings.
	*/
	var map = {};

	//*@protected
	/**
		Used internally to track references to bindings to allow for
		several different events to occur that will still cleanup
		bindings.
	*/
	var register = function (binding) {
		map[binding.id] = binding;
		binding._registered = true;
	};

	//*@protected
	/**
		Used internally to remove references to bindings.
	*/
	var unregister = function (id) {
		id = id && id.id? id.id: id;
		if (map[id]) {
			delete map[id];
		}
	};

	//*@protected
	var count = 0;

	//*@protected
	/**
		Used internally as part of the getParts method.
	*/
	var fromRoot = function (root, parts) {
		// check to see if the part of the path is relative to the root
		var piece = root[(parts || [])[0]];
		// if the root here is enyo.global then we need to ensure that
		// the piece is actually an object
		// this is very, very important
		if (enyo.exists(piece)) {
			if (enyo.global === root) {
				return "object" === typeof piece? root: undefined;
			} else {
				return root;
			}
		}
	};

	//*@protected
	/**
		Used by the binding's setter for both targets and sources
		when determing whether or not to force a notification to fire.
		We cannot easily determine this for types that are passed
		by reference (e.g. arrays and native JavaScript objects). But
		for these types it much clearer in nearly all cases.
	*/
	var _forceRegex = /(string|number|boolean)/;

	//*@protected
	/**
		Used internally to determine from the given information what the
		source and target paths and properties are. There is one exception case
		between determining parts for the source and the target in bindings so
		the optional third parameter helps it to use the correct algorithm.
	*/
	var getParts = function (path, context) {
		/* jshint debug: true */
		if (this.debug) {
			debugger;
		}
		var parts;
		var idx = 0;
		var ret = {};
		var root;
		var cur;
		var prop;
		var base;
		var part;
		var owner = this.owner;
		var local = path[0] === "."? true: false;
		path = path[0] === "."? path.slice(1): path;
		parts = path.split(".");
		root = local? context || owner: context || fromRoot(enyo.global, parts) || owner;
		base = root;
		ret.property = prop = parts.length > 1? parts.pop(): path;
		if (prop === path || (!local && context)) {
			ret.base = base;
		} else {
			cur = base;
			for (; idx < parts.length; ++idx) {
				part = parts[idx];
				if (!part) {
					continue;
				}
				cur = cur[part];
				if (!cur || "string" === typeof cur) {
					if (part !== prop) {
						ret.base = null;
					}
					return ret;
				}
			}
			if (part !== path) {
				base = cur;
			}
			ret.base = base;
		}
		return ret;
	};

	//*@protected
	/**
		Initially called during construction to setup the properties
		of the binding appropriately exiting on specific conditions
		silently if the target or source could not be properly
		determined or found.
	*/
	var setup = function () {
		/* jshint debug: true */
		var debug = this.debug;
		// for browsers that support this kind of debugging
		if (true === debug) {
			debugger;
		}
		// register the binding globally for cleanup purposes
		var connect = this.autoConnect;
		var sync = this.autoSync;
		var source = this.setupSource();
		var target = this.setupTarget();
		var refreshing = this._refreshing;
		if (!this._registered) {
			register(this);
		}
		// setup the transform if we can
		if (true !== refreshing) {
			this.setupTransform();
		}
		// if we are refreshing and cannot find
		// one of these parts we need to reset the targets
		// value if possible (happens frequently in proxy/model-
		// controllers who's model has been set to null)
		if (!(source && target)) {
			if (refreshing) {
				if (target) {
					// set the target's value to null to let
					// it know we can't sync the real value from
					// the source
					this.setTargetValue(null);
				}
			}
			return;
		}
		// this will fail silently if setup went aury for
		// either the target or source
		// we allow the process of connecting the ends to be
		// interrupted if either end has been destroyed, we
		// self-destruct
		try {
			if (connect || refreshing) {
				this.connect();
			}
		} catch (err) {
			if ("binding-destroyed" === err) {
				return;
			}
			else {
				throw err;
			}
		}
		if (sync || refreshing) {
			this.sync();
		}
	};

	//*@protected
	function Transform (fn, binding) {
		this.transformer = fn;
		this.binding = binding;
	}

	//*@protected
	Transform.prototype = {
		transform: function (value, direction) {
			var fn = this.transformer;
			var binding = this.binding;
			var context = binding.owner || enyo.global;
			return fn.call(context, value, direction, binding);
		},
		destroy: function () {
			this.transformer = null;
			this.binding = null;
		}
	};

	//*@public
	enyo.kind({

		// ...........................
		// PUBLIC PROPERTIES

		//*@public
		name: "enyo.Binding",

		//*@public
		kind: null,

		//*@public
		source: null,

		//*@public
		target: null,

		//*@public
		to: null,

		//*@public
		from: null,

		//*@public
		autoConnect: true,

		//*@public
		autoSync: true,

		//*@public
		owner: null,

		//*@public
		transform: null,

		//*@public
		oneWay: true,

		//*@public
		twoWay: false,

		//*@public
		destroyed: false,

		//*@public
		debug: false,

		//*@public
		statics: {
			find: function (id) {
				return map[id];
			}
		},

		// ...........................
		// PROTECTED PROPERTIES

		//*@protected
		_sourceProperty: null,

		//*@protected
		_targetProperty: null,

		//*@protected
		_sourceResponder: null,

		//*@protected
		_targetResponder: null,

		//*@protected
		_isConnected: false,

		//*@protected
		_synchronizing: false,

		//*@protected
		_refreshing: false,

		//*@protected
		_registered: false,

		// ...........................
		// COMPUTED PROPERTIES

		// ...........................
		// PUBLIC METHODS

		// ...........................
		// PROTECTED METHODS

		//*@protected
		sync: function () {
			if (true === this._isConnected) {
				this.syncFromSource();
			}
		},

		//*@protected
		refresh: function () {
			this._refreshing = true;
			setup.call(this);
			this._refreshing = false;
		},

		//*@protected
		rebuild: function() {
			this.disconnect();
			this.source = null;
			this._sourceProperty = null;
			this.target = null;
			this._targetProperty = null;
			this.refresh();
		},

		//*@public
		/**
			Call this method to connect this binding to its
			source (and target). This only registers the responders
			but does not automatically synchronize the values.
		*/
		connect: function () {
			if (true === this._isConnected) {
				return;
			}
			if (true === this.destroyed) {
				return;
			}
			this.connectSource();
			this.connectTarget();
			if (this.sourceConnected && this.targetConnected) {
				this._isConnected = true;
			} else {
				this._isConnected = false;
			}
		},

		//*@public
		/**
			Call this method to disconnect this binding from
			its source (and target).
		*/
		disconnect: function () {
			if (false === this._isConnected) {
				return;
			}
			this.disconnectSource();
			this.disconnectTarget();
			this._isConnected = false;
		},

		//*@protected
		setupSource: function () {
			var parts;
			var base;
			var property = this._sourceProperty;
			var source = this.source;
			var from = this.from;
			if (source && property) {
				return true;
			}
			if (!from) {
				return false;
			}
			parts = getParts.call(this, from, source);
			base = parts.base;
			property = parts.property;
			if (!base || "object" !== typeof base) {
				return false;
			}
			this.source = base;
			this._sourceProperty = property;
			return true;
		},

		//*@protected
		setupTarget: function () {
			var parts;
			var base;
			var property = this._targetProperty;
			var target = this.target;
			var to = this.to;
			if (target && property) {
				return true;
			}
			if (!to) {
				return false;
			}
			parts = getParts.call(this, to, target);
			base = parts.base;
			property = parts.property;
			if (!base || "object" !== typeof base) {
				return false;
			}
			this.target = base;
			this._targetProperty = property;
			return true;
		},

		//*@protected
		stop: function () {
			throw "stop-binding";
		},

		//*@protected
		connectSource: function () {
			var source = this.source;
			var property = this._sourceProperty;
			var fn = this._sourceResponder;
			if (!(source instanceof enyo.Object)) {
				this.sourceConnected = false;
				return false;
			}
			// only create the responder if it doesn't already exist
			if (!enyo.exists(fn) || "function" !== typeof fn) {
				fn = enyo.bind(this, this.syncFromSource);
				this._sourceResponder = fn;
			}
			// in the event that the source actually exists but has been destroyed
			if (true === source.destroyed) {
				// we need to be destroyed so we can also be cleaned up
				this.destroy();
				throw "binding-destroyed";
			}
			// if it is already connected don't do anything
			if (true === this.sourceConnected) {
				return true;
			}
			if (!enyo.exists(source)) {
				this.sourceConnected = false;
				return false;
			}
			// assign the binding's id to the responder for debugging
			fn.bindingId = this.id;
			// add the observer for the property on the source object
			source.addObserver(property, fn);
			this.sourceConnected = true;
			return true;
		},

		//*@protected
		connectTarget: function () {
			var target = this.target;
			var property = this._targetProperty;
			var fn = this._targetResponder;
			var oneWay = this.oneWay;
			if (!(target instanceof enyo.Object)) {
				this.targetConnected = false;
				return false;
			}
			// in the event that the target actually exists but has been destroyed
			if (true === target.destroyed) {
				// we need to be destroyed so we can also be cleaned up
				this.destroy();
				throw "binding-destroyed";
			}
			// if this is a one way binding there is nothing to do
			if (true === oneWay) {
				this.targetConnected = true;
				return true;
			}
			// only create the responder if it doesn't already exist
			if (!enyo.exists(fn) || "function" !== typeof fn) {
				fn = enyo.bind(this, this.syncFromTarget);
				this._targetResponder = fn;
			}
			// if it is already connected don't do anything else
			if (true === this.targetConnected) {
				return true;
			}
			if (!enyo.exists(target)) {
				this.targetConnected = false;
				return false;
			}
			fn.bindingId = this.id;
			target.addObserver(property, fn);
			this.targetConnected = true;
			return true;
		},

		//*@protected
		syncFromSource: function () {
			var twoWay = this.twoWay;
			var value = this.getSourceValue();
			var transformer = this.transform;
			// if this is a two way binding we need to
			// disconnect from the target first to ensure
			// we don't catch the update response
			// TODO: rethink this approach as try/catch are
			// costly in general...
			try {
				value = transformer.transform(value, "source");
			} catch (err) {
				// the transform was interrupted, do not complete
				if ("stop-binding" === err) {
					return;
				} else {
					throw err;
				}
			}
			if (twoWay) {
				this._synchronizing = true;
				this.disconnectTarget();
			}
			this.setTargetValue(value);
			if (twoWay) {
				this.connectTarget();
				this._synchronizing = false;
			}
		},

		//*@protected
		syncFromTarget: function () {
			var value = this.getTargetValue();
			var transformer = this.transform;
			// TODO: same as for syncFromSource
			try {
				value = transformer.transform(value, "target");
			} catch (err) {
				// the transform was interrupted, do not complete
				if ("stop-binding" === err) {
					return;
				} else {
					throw err;
				}
			}
			this.disconnectSource();
			this.setSourceValue(value);
			this.connectSource();
		},

		//*@protected
		disconnectSource: function () {
			var source = this.source;
			var property = this._sourceProperty;
			var fn = this._sourceResponder;
			if (!enyo.exists(source)) {
				return;
			}
			source.removeObserver(property, fn);
			this.sourceConnected = false;
		},

		//*@protected
		disconnectTarget: function () {
			var target = this.target;
			var fn = this._targetResponder;
			var property = this._targetProperty;
			if (!enyo.exists(target)) {
				return;
			}
			if ("function" === typeof fn) {
				target.removeObserver(property, fn);
			}
			this.targetConnected = false;
		},

		//*@protected
		setSourceValue: function (value) {
			var source = this.source;
			var property = this._sourceProperty;
			var force = !_forceRegex.test(typeof value);
			source.set(property, value, force);
		},

		//*@protected
		setTargetValue: function (value) {
			var target = this.target;
			var property = this._targetProperty;
			var force = !_forceRegex.test(typeof value);
			target.set(property, value, force);
		},

		//*@protected
		getSourceValue: function () {
			var source = this.source;
			var property = this._sourceProperty;
			return source.get(property);
		},

		//*@protected
		getTargetValue: function () {
			var target = this.target;
			var property = this._targetProperty;
			return target.get(property);
		},

		//*@protected
		setupTransform: function () {
			var transform = this.transform;
			var owner = this.owner || {};
			// if it is a string we try and locate it on the owner
			// or as a global method
			if ("string" === typeof transform) {
				transform = owner[transform] || enyo.getPath.call(owner, transform)
					|| enyo.getPath.call(enyo.global, transform);
			}
			// if we couldn't find anything go ahead and setup a default
			// to simply return the value
			if ("function" !== typeof transform) {
				transform = this.transform = function(value) {
					return value;
				};
			}
			if (!(transform instanceof Transform)) {
				this.transform = new Transform(transform, this);
			}
		},

		//*@public
		/**
			Call this method to prepare this object to be
			cleaned up by the garbage collector.
		*/
		destroy: function () {
			if (true === this.destroyed) {
				return;
			}
			// we set this right away so that we don't wind up
			// in an infinite loop
			this.destroyed = true;
			this.disconnect();
			this.source = null;
			this.target = null;
			this._sourceResponder = null;
			this._targetResponder = null;
			enyo.Binding.bindingCount--;
			if (this.transform) {
				this.transform.destroy();
				this.transform = null;
			}
			if (this.owner) {
				this.owner.removeBinding(this);
			}
			// make sure to unregister the binding reference
			unregister(this);
		},

		//*@protected
		constructor: function () {
			var idx = 0;
			var len = arguments.length;
			var oneWay;
			var twoWay;
			// increment our binding counter for debugging purposes
			count++;
			// take any properties that were passed in and apply them
			// to this binding instance
			for (; idx < len; ++idx) {
				enyo.mixin(this, arguments[idx]);
			}
			// generate a new id for this binding
			this.id = enyo.uid("binding");
			// we need to make sure the binding's setup understands if
			// this is a one-way or two-way binding
			oneWay = this.oneWay;
			twoWay = this.twoWay;
			// regardless of what the oneWay flag is set to, priority
			// resolution defers to the twoWay flag initially
			if (true === twoWay) {
				this.oneWay = false;
			}
			// now we check our deferred flag to see if this is a two-way
			// binding having been set via the oneWay flag
			else if (false === oneWay) {
				this.twoWay = true;
			}
			// run our initialization routines
			setup.call(this);
		}

		// ...........................
		// OBSERVERS

	});

}());

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/Component.js"
Content-Type: application/octet-stream; x-encoding=base64
/**
	_enyo.Component_ is the fundamental building block for Enyo applications.
	Components are designed to fit together, allowing complex behaviors to be
	fashioned from smaller bits of functionality.

	Component constructors take a single argument (sometimes called a
	_Component configuration_), a JavaScript object that defines various
	properties to be initialized on the Component. For example:

		// create a new component, initialize its name property to 'me'.
		var c = new enyo.Component({
			name: "me"
		});

	When a Component is instantiated, items configured in its _components_
	property are instantiated, too:

		// create a new component, which itself has a component
		var c = new enyo.Component({
			name: "me",
			components: [
				{kind: "Component", name: "other"}
			]
		});

	In this case, when _me_ is created, _other_ is also created, and we say that
	_me owns other_. In other words, the _owner_ property of _other_ equals
	_me_. Notice that you can specify the _kind_ of _other_ explicitly in its
	configuration block, to tell _me_ what constructor to use to create _other_.

	Note that _kind_ values may be references to actual kinds or string-names of
	kinds. Kind names that	do not resolve directly to kinds are looked up in
	default namespaces. In this case, _kind: "Component"_ resolves to
	_enyo.Component_.

	To move a component, use the _setOwner_ method to change the component's owner.
	If you want to make a component unowned, use _setOwner(null)_.

	If you make changes to _enyo.Component_, be sure to add or update the
	appropriate [unit tests](https://github.com/enyojs/enyo/tree/master/tools/test/core/tests).

	For more information, see the documentation on
	[Components](https://github.com/enyojs/enyo/wiki/Creating-Components)
	in the Enyo Developer Guide.
*/
enyo.kind({
	name: "enyo.Component",
	kind: "enyo.Object",
	published: {
		/**
			A unique name for the component within its owner. This is used to
			set the access name in the owner's _$_ hash.  If not specified, a
			default name will be provided based on the name of the object's
			kind, optionally with a number suffix if more than one instance
			exists in the owner.
		*/
		name: "",
		/**
			A unique id for the component, usually automatically generated based
			on its position within the component hierarchy, although it may also
			be directly specified. _enyo.Control_ uses this id value for the DOM
			id attribute.
		*/
		id: "",
		/**
			The component that owns this component. It is usually implicitly
			defined during creation based on the _createComponent_ call or
			_components_ hash.
		*/
		owner: null
	},
	//* @protected
	protectedStatics: {
		// for memoizing kind-prefix names in nameComponent
		_kindPrefixi: {},
		// for naming the unnamed
		_unnamedKindNumber: 0
	},
	defaultKind: "Component",
	noDefer: true,
	handlers: {},
	mixins: ["enyo.MixinComponentSupport", "enyo.ApplicationSupport"],
	__jobs: {},
	toString: function() {
		return this.kindName;
	},
	constructor: function(props) {
		// initialize instance objects
		this._componentNameMap = {};
		this.$ = {};
		this.inherited(arguments);
	},
	constructed: function(inProps) {
		this.handlers = enyo.mixin(enyo.clone(this.kindHandlers), this.handlers);
		// perform initialization
		this.create(inProps);
	},
	create: function() {
		this.ownerChanged();
		this.initComponents();
	},
	initComponents: function() {
		// The _components_ property in kind declarations is renamed to
		// _kindComponents_ by the Component subclass mechanism.  This makes it
		// easy for the developer to distinguish kindComponents from the components
		// in _this.components_, without having to worry about the actual difference.
		//
		// Specifically, the difference is that kindComponents are constructed as
		// owned by this control (whereas components in _this.components_ are not).
		// In addition, kindComponents are marked with the _isChrome: true_ flag.
		this.createChrome(this.kindComponents);
		this.createClientComponents(this.components);
	},
	createChrome: function(inComponents) {
		this.createComponents(inComponents, {isChrome: true});
	},
	createClientComponents: function(inComponents) {
		this.createComponents(inComponents, {owner: this.getInstanceOwner()});
	},
	getInstanceOwner: function() {
		return (!this.owner || this.owner.notInstanceOwner) ? this : this.owner;
	},
	//* @public
	/**
		Removes this component from its owner (sets _owner_ to null) and does
		any necessary cleanup. The component is flagged with a _destroyed: true_
		property. Usually, the component will be suitable for garbage collection
		after being destroyed, unless user code keeps a reference to it.
	*/
	destroy: function() {
		this.destroyComponents();
		this.setOwner(null);
		this.inherited(arguments);
		this.stopAllJobs();
	},
	/**
		Destroys all owned components.
	*/
	destroyComponents: function() {
		enyo.forEach(this.getComponents(), function(c) {
			// This local components list may be stale as components
			// we owned when the loop started could have been destroyed
			// by containers. Avoid redestroying components by testing
			// destroyed flag.
			if (!c.destroyed) {
				c.destroy();
			}
		});
	},
	//* @protected
	makeId: function() {
		var delim = "_", pre = this.owner && this.owner.getId();
		var baseName = this.name || ("@@" + (++enyo.Component._unnamedKindNumber));
		return (pre ? pre + delim : "") + baseName;
	},
	ownerChanged: function(inOldOwner) {
		if (inOldOwner && inOldOwner.removeComponent) {
			inOldOwner.removeComponent(this);
		}
		if (this.owner && this.owner.addComponent) {
			this.owner.addComponent(this);
		}
		if (!this.id) {
			this.id = this.makeId();
		}
		//this.id = this.makeId();
	},
	nameComponent: function(inComponent) {
		var prefix = enyo.Component.prefixFromKindName(inComponent.kindName);
		// get last memoized name index
		var n, i = this._componentNameMap[prefix] || 0;
		// find an available name
		do {
			n = prefix + (++i > 1 ? String(i) : "");
		} while (this.$[n]);
		// memoize next likely-unique id tag for this prefix
		this._componentNameMap[prefix] = Number(i);
		// set and return
		inComponent.name = n;
		return inComponent.name;
	},
	/**
		Adds _inComponent_ to the list of components owned by the current
		component (i.e., _this.$_).
	*/
	addComponent: function(inComponent) {
		var n = inComponent.getName();
		if (!n) {
			n = this.nameComponent(inComponent);
		}
		if (this.$[n]) {
			this.warn('Duplicate component name "' + n + '" in owner "' + this.id + '" violates ' +
				'unique-name-under-owner rule, replacing existing component in the hash and continuing, ' +
				'but this is an error condition and should be fixed.');
		}
		this.$[n] = inComponent;
	},
	//* Removes _inComponent_ from the list of components owned by the current
	//* component (i.e., _this.$_).
	removeComponent: function(inComponent) {
		delete this.$[inComponent.getName()];
	},
	//* @public
	/**
		Returns an array of owned components; in other words, converts the _$_
		hash into an array and returns the array.
	*/
	getComponents: function() {
		var results = [];
		for (var n in this.$) {
			results.push(this.$[n]);
		}
		return results;
	},
	//* @protected
	adjustComponentProps: function(inProps) {
		if (this.defaultProps) {
			enyo.mixin(inProps, this.defaultProps);
		}
		inProps.kind = inProps.kind || inProps.isa || this.defaultKind;
		inProps.owner = inProps.owner || this;
	},
	_createComponent: function(inInfo, inMoreInfo) {
		if (!inInfo.kind && ("kind" in inInfo)) {
			throw "enyo.create: Attempt to create a null kind. Check dependencies for [" + inInfo.name + "].";
		}
		// CAVEAT: inInfo and inMoreInfo are copied before mutation, but it's only a shallow copy
		var props = enyo.mixin(enyo.clone(inMoreInfo), inInfo);
		this.adjustComponentProps(props);
		return enyo.Component.create(props);
	},
	//* @public
	/**
		Creates and returns a component as defined by the combination of
		_inInfo_ and _inMoreInfo_. Properties in _inInfo_ override properties in
		_inMoreInfo_.

		The created component passes through initialization machinery provided
		by the creating component, which may supply special handling.
		Unless the owner is explicitly specified, the new component will be
		owned by the instance on which _createComponent_ is called.

			// Create a new component named _dynamic_ owned by _this_
			// (will be available as this.$.dynamic).
			this.createComponent({name: "dynamic"});

			// Create a new component named _another_ owned by _other_
			// (will be available as other.$.another).
			this.createComponent({name: "another"}, {owner: other});
	*/
	createComponent: function(inInfo, inMoreInfo) {
		// createComponent and createComponents both delegate to the protected method (_createComponent),
		// allowing overrides to customize createComponent and createComponents separately.
		return this._createComponent(inInfo, inMoreInfo);
	},
	/**
		Creates Component objects as defined by the array of configurations
		_inInfos_. Each configuration in _inInfos_ is combined with _inCommonInfo_,
		as described in _createComponent_.

		Returns an array of references to the created components.

			// ask foo to create components _bar_ and _zot_, but set the owner of
			// both components to _this_.
			this.$.foo.createComponents([
				{name: "bar"},
				{name: "zot"}
			], {owner: this});
	*/
	createComponents: function(inInfos, inCommonInfo) {
		if (inInfos) {
			var cs = [];
			for (var i=0, ci; (ci=inInfos[i]); i++) {
				cs.push(this._createComponent(ci, inCommonInfo));
			}
			return cs;
		}
	},
	//* @protected
	getBubbleTarget: function() {
		return this._bubble_target || this.owner;
	},
	//* @public
	/**
		Bubbles an event up an object chain, starting with _this_.

		If a handler for this event returns true (aka _handled_),
		bubbling is stopped.

		Handlers always have this signature:

			function(inSender, inEvent)

		where _inSender_ refers to the component that most recently
		propagated the event and _inEvent_ is an object containing
		event information.

		_inEvent_ will have at least one property, _originator_, which
		references the component that triggered the event in the first place.
	*/
	bubble: function(inEventName, inEvent, inSender) {
		if (this._silenced) {
			return;
		}
		var e = inEvent || {};
		// FIXME: is this the right place?
		// if (!("originator" in e)) {
		if (!enyo.exists(e.originator)) {
			e.originator = inSender || this;
			// FIXME: use indirection here?
			//e.delegate = e.originator.delegate || e.originator.owner;
		}
		return this.dispatchBubble(inEventName, e, inSender || this);
	},
	/**
		Bubbles an event up an object chain, starting <b>above</b> _this_.

		If a handler for this event returns true (i.e., _handled_),
		bubbling is stopped.

		Handlers always have this signature:

			function(inSender, inEvent)

		where _inSender_ refers to the component that most recently
		propagated the event and _inEvent_ is an object containing
		event information.

		_inEvent_ will have at least one property, _originator_, which
		references the component that triggered the event in the first place.
	*/
	bubbleUp: function(inEventName, inEvent, inSender) {
		if (this._silenced) {
			return;
		}
		// Bubble to next target
		var next = this.getBubbleTarget();
		var delegate = inEvent.delegate;
		if (next) {
			return next.dispatchBubble(inEventName, inEvent, delegate || this);
		}
		return false;
	},
	//* @protected
	/**
		Sends an event to a named delegate. This object may dispatch an event
		to itself via a handler, or to its owner via an event property, e.g.:

			handlers {
				// 'tap' events dispatched to this.tapHandler
				ontap: "tapHandler"
			}

			// 'tap' events dispatched to 'tapHandler' delegate in this.owner
			ontap: "tapHandler"
	*/
	dispatchEvent: function(name, event, sender) {
		if (this._silenced) {
			return;
		}
		// if the event has a delegate associated with it we grab that
		// for reference
		var delegate = (event || (event = enyo.pool.claimObject())).delegate;
		var ret;
		// bottleneck event decoration
		this.decorateEvent(name, event, sender);
		// dispatch via the handlers block if possible
		if (this.handlers && this.handlers[name] &&
			this.dispatch(this.handlers[name], event, sender)) {
			return true;
		}

		if (this[name]) {
			if ("function" === typeof this[name]) {
				if (this._isController || (delegate && this === delegate.owner)) {
					return this.dispatch(name, event, sender);
				}
			} else {
				// otherwise we dispatch it up because it is a remap of another event
				if (!delegate) {
					event.delegate = this;
				}
				ret = this.bubbleUp(this[name], event, sender);
				delete event.delegate;
				return ret;
			}
		}
	},
	// internal - try dispatching event to self, if that fails bubble it up the tree
	dispatchBubble: function(inEventName, inEvent, inSender) {
		if (this._silenced) {
			return;
		}
		// Try to dispatch from here, stop bubbling on truthy return value
		if (this.dispatchEvent(inEventName, inEvent, inSender)) {
			return true;
		}
		// Bubble to next target
		return this.bubbleUp(inEventName, inEvent, inSender);
	},
	decorateEvent: function(inEventName, inEvent, inSender) {
		// an event may float by us as part of a dispatchEvent chain or delegateEvent
		// both call this method so intermediaries can decorate inEvent
	},
	bubbleDelegation: function(inDelegate, inName, inEventName, inEvent, inSender) {
		if (this._silenced) {
			return;
		}
		// next target in bubble sequence
		var next = this.getBubbleTarget();
		if (next) {
			return next.delegateEvent(inDelegate, inName, inEventName, inEvent, inSender);
		}
	},
	delegateEvent: function(inDelegate, inName, inEventName, inEvent, inSender) {
		if (this._silenced) {
			return;
		}
		// override this method to play tricks with delegation
		// bottleneck event decoration
		this.decorateEvent(inEventName, inEvent, inSender);
		// by default, dispatch this event if we are in fact the delegate
		if (inDelegate == this) {
			return this.dispatch(inName, inEvent, inSender);
		}
		return this.bubbleDelegation(inDelegate, inName, inEventName, inEvent, inSender);
	},
	stopAllJobs: function() {
		for (var jobName in this.__jobs) {
			this.stopJob(jobName);
		}
	},
	//* @public
	/**
		Dispatches the event to named delegate _inMethodName_, if it exists.
		Subkinds may re-route dispatches.
		Note that both 'handlers' events and events delegated from owned controls
		arrive here. If you need to handle these differently, you may also need
		to override _dispatchEvent_.
	*/
	dispatch: function(inMethodName, inEvent, inSender) {
		if (this._silenced) {
			return;
		}
		var fn = inMethodName && this[inMethodName];
		if (fn && "function" === typeof fn) {
			// TODO: we use inSender || this but the inSender argument
			// to keep unit tests working will be deprecated in the future
			return fn.call(this, inSender || this, inEvent);
		}
	},
	/**
		Sends a message to myself and all of my components.
		You can stop a waterfall into components owned by a
		receiving object by returning a truthy value from
		the event handler.
	*/
	waterfall: function(name, event, sender) {
		if (this._silenced) {
			return;
		}
		event = event || {};
		//this.log(name, (sender || this).name, "=>", this.name);
		if (this.dispatchEvent(name, event, sender)) {
			return true;
		}
		this.waterfallDown(name, event, sender || this);
	},
	/**
		Sends a message to all of my components, but not myself.
		You can stop a waterfall into components owned by a
		receiving object by returning a truthy value from
		the event handler.
	*/
	waterfallDown: function(name, event, sender) {
		if (this._silenced) {
			return;
		}
		for (var n in this.$) {
			this.$[n].waterfall(name, event, sender);
		}
	},
	//*@protected
	_silenced: false,
	//*@protected
	_silence_count: 0,
	//*@public
	/**
		Sets a flag that disables event propagation for this component. Also
		increments an internal counter that tracks the number of times the
		_unsilence_ method must be called before event propagation will continue.
	*/
	silence: function () {
		this._silenced = true;
		this._silence_count += 1;
	},

	//*@public
	/**
		Allows event propagation for this component if the internal silence counter
		is 0; otherwise, decrements the counter by one.  For event propagation to
		resume, this method must be called one time for each call to _silence_.
	*/
	unsilence: function () {
		if (0 !== this._silence_count) {
			--this._silence_count;
		}
		if (0 === this._silence_count) {
			this._silenced = false;
		}
	},
	/**
		Creates a new job tied to this instance of the component. If the component
		is destroyed, any jobs associated with it will be stopped.

		If you start a job with the same name as a pending job, the original job
		will be stopped; this can be useful for resetting timeouts.
	*/
	startJob: function(inJobName, inJob, inWait) {
		// allow strings as job names, they map to local method names
		if (enyo.isString(inJob)) {
			inJob = this[inJob];
		}
		// stop any existing jobs with same name
		this.stopJob(inJobName);
		this.__jobs[inJobName] = setTimeout(this.bindSafely(function() {
			this.stopJob(inJobName);
			// call "inJob" with this bound to the component.
			inJob.call(this);
		}), inWait);
	},
	/**
		Stops a component-specific job before it has been activated.
	*/
	stopJob: function(inJobName) {
		if (this.__jobs[inJobName]) {
			clearTimeout(this.__jobs[inJobName]);
			delete this.__jobs[inJobName];
		}
	},
	/**
		Execute the method _inJob_ immediately, then prevent
		any other calls to throttleJob with the same _inJobName_ from running
		for the	next _inWait_ milliseconds.
	*/
	throttleJob: function(inJobName, inJob, inWait) {
		// if we still have a job with this name pending, return immediately
		if (this.__jobs[inJobName]) {
			return;
		}
		// allow strings as job names, they map to local method names
		if (enyo.isString(inJob)) {
			inJob = this[inJob];
		}
		inJob.call(this);
		this.__jobs[inJobName] = setTimeout(this.bindSafely(function() {
			this.stopJob(inJobName);
		}), inWait);
	}
});

//* @protected

enyo.defaultCtor = enyo.Component;

// Creates new instances from config objects. This method looks up the
// proper constructor based on the provided _kind_ attribute.
enyo.create = enyo.Component.create = function(inConfig) {
	if (!inConfig.kind && ("kind" in inConfig)) {
		throw "enyo.create: Attempt to create a null kind. Check dependencies for [" + (inConfig.name || "") + "].";
	}
	var kind = inConfig.kind || inConfig.isa || enyo.defaultCtor;
	var Ctor = enyo.constructorForKind(kind);
	if (!Ctor) {
		enyo.error('no constructor found for kind "' + kind + '"');
		Ctor = enyo.Component;
	}
	return new Ctor(inConfig);
};

enyo.Component.subclass = function(ctor, props) {
	// Note: To reduce API surface area, sub-components are declared only as
	// 'components' in both kind and instance declarations.
	//
	// However, 'components' from kind declarations must be handled separately
	// at creation time.
	//
	// We rename the property here to avoid having
	// to interrogate the prototype at creation time.
	//
	var proto = ctor.prototype;
	//
	if (props.components) {
		proto.kindComponents = props.components;
		delete proto.components;
	}
	//
	// handlers are merged with supertype handlers
	// and kind time.
	//
	if (props.handlers) {
		var kh = proto.kindHandlers;
		proto.kindHandlers = enyo.mixin(enyo.clone(kh), proto.handlers);
		proto.handlers = null;
	}
	// events property defines published events for Component kinds
	if (props.events) {
		this.publishEvents(ctor, props);
	}
};

enyo.Component.publishEvents = function(ctor, props) {
	var es = props.events;
	if (es) {
		var cp = ctor.prototype;
		for (var n in es) {
			this.addEvent(n, es[n], cp);
		}
	}
};

enyo.Component.addEvent = function(inName, inValue, inProto) {
	var v, fn;
	if (!enyo.isString(inValue)) {
		v = inValue.value;
		fn = inValue.caller;
	} else {
		if (inName.slice(0, 2) != 'on') {
			enyo.warn("enyo.Component.addEvent: event names must start with 'on'. " + inProto.kindName +
				" event '" + inName + "' was auto-corrected to 'on" + inName + "'.");
			inName = "on" + inName;
		}
		v = inValue;
		fn = "do" + enyo.cap(inName.slice(2));
	}
	inProto[inName] = v;
	if (!inProto[fn]) {
		inProto[fn] = function(payload) {
			// bubble this event
			var $e = payload || enyo.pool.claimObject();
			var $d = $e.delegate;
			// delete payload.delegate;
			$e.delegate = undefined;
			this.bubble(inName, $e);
			if ($d) {
				$e.delegate = $d;
			}
			if (!payload || !payload._pooled) {
				enyo.pool.releaseObject($e);
			}
		};
		// NOTE: Mark this function as a generated event handler to allow us to
		// do event chaining. Is this too complicated?
		//inProto[fn]._dispatcher = true;
	}
};

enyo.Component.prefixFromKindName = function(inKindName) {
	var prefix = enyo.Component._kindPrefixi[inKindName];
	if (!prefix) {
		// memoize naming information for this kind
		var l = inKindName.lastIndexOf(".");
		prefix = (l >= 0) ? inKindName.slice(l+1) : inKindName;
		// lower-case the leading char
		prefix = prefix.charAt(0).toLowerCase() + prefix.slice(1);
		// memoize result
		enyo.Component._kindPrefixi[inKindName] = prefix;
	}
	return prefix;
};

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/Controller.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qQHB1YmxpYwovKioKCV9lbnlvLkNvbnRyb2xsZXJfIGlzIHRoZSBiYXNlIGtpbmQgZm9yIGFsbCBjb250cm9sbGVycyBpbiBFbnlvLiBBbgoJYWJzdHJhY3Qga2luZCwgaXQgaXMgYSBkZWxlZ2F0ZS9jb21wb25lbnQgdGhhdCBpcyBkZXNpZ25lZCB0byBiZSBhCglwcm94eSBmb3IgaW5mb3JtYXRpb24uCiovCmVueW8ua2luZCh7CgoJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkvLyBQVUJMSUMgUFJPUEVSVElFUwoKCS8vKkBwdWJsaWMKCW5hbWU6ICJlbnlvLkNvbnRyb2xsZXIiLAoKCS8vKkBwdWJsaWMKCWtpbmQ6ICJlbnlvLk11bHRpcGxlRGlzcGF0Y2hDb21wb25lbnQiLAoKCS8vKkBwdWJsaWMKCS8qKgoJCVRoZSBkZWZhdWx0IHNvdXJjZSBvZiBpbmZvcm1hdGlvbiBmb3IgYWxsIGluc3RhbmNlcyBvZiBfZW55by5Db250cm9sbGVyXwoJCWFuZCBpdHMgc3Via2luZHMuIEluIHNvbWUgY2FzZXMsIHRoaXMgd2lsbCBiZSBhIGNvbXB1dGVkIHByb3BlcnR5IHRvCgkJZmFjaWxpdGF0ZSBvdmVybG9hZGluZy4gSXQgbWF5IGNvbnRhaW4gYW55IHR5cGUgb2YgZGF0YS4KCSovCglkYXRhOiBudWxsLAoKCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJLy8gUFJPVEVDVEVEIFBST1BFUlRJRVMKCgkvLypAcHJvdGVjdGVkCglfaXNDb250cm9sbGVyOiB0cnVlCgoJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkvLyBDT01QVVRFRCBQUk9QRVJUSUVTCgoJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkvLyBQVUJMSUMgTUVUSE9EUwoKCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJLy8gUFJPVEVDVEVEIE1FVEhPRFMKCgkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCS8vIE9CU0VSVkVSUwoKfSk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/Layout.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5MYXlvdXRfIGlzIHRoZSBiYXNlIGtpbmQgZm9yIGxheW91dCBraW5kcy4gIFRoZXNlIGFyZSB1c2VkIGJ5Cgk8YSBocmVmPSIjZW55by5VaUNvbXBvbmVudCI+ZW55by5VaUNvbXBvbmVudDwvYT4tYmFzZWQgY29udHJvbHMgdG8gYWxsb3cgZm9yCglhcnJhbmdpbmcgb2YgdGhlIGNoaWxkcmVuIGJ5IHNldHRpbmcgdGhlIF9sYXlvdXRLaW5kXyBwcm9wZXJ0eS4KCglEZXJpdmVkIGtpbmRzIHdpbGwgdXN1YWxseSBwcm92aWRlIHRoZWlyIG93biBfbGF5b3V0Q2xhc3NfIHByb3BlcnR5IHRvCglhZmZlY3QgdGhlIENTUyBydWxlcyB1c2VkLCBhbmQgbWF5IGFsc28gaW1wbGVtZW50IHRoZSBfZmxvd18gYW5kIF9yZWZsb3dfCgltZXRob2RzLiBfZmxvd18gaXMgY2FsbGVkIGR1cmluZyBjb250cm9sIHJlbmRlcmluZywgd2hpbGUgX3JlZmxvd18gaXMgY2FsbGVkCgl3aGVuIHRoZSBhc3NvY2lhdGVkIGNvbnRyb2wgaXMgcmVzaXplZC4KKi8KZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLkxheW91dCIsCglraW5kOiBudWxsLAoJLy8qIENTUyBjbGFzcyB0aGF0J3MgYWRkZWQgdG8gdGhlIGNvbnRyb2wgdXNpbmcgdGhpcyBsYXlvdXQga2luZAoJbGF5b3V0Q2xhc3M6ICIiLAoJLy8qIEBwcm90ZWN0ZWQKCWNvbnN0cnVjdG9yOiBmdW5jdGlvbihpbkNvbnRhaW5lcikgewoJCXRoaXMuY29udGFpbmVyID0gaW5Db250YWluZXI7CgkJaWYgKGluQ29udGFpbmVyKSB7CgkJCWluQ29udGFpbmVyLmFkZENsYXNzKHRoaXMubGF5b3V0Q2xhc3MpOwoJCX0KCX0sCglkZXN0cm95OiBmdW5jdGlvbigpIHsKCQlpZiAodGhpcy5jb250YWluZXIpIHsKCQkJdGhpcy5jb250YWluZXIucmVtb3ZlQ2xhc3ModGhpcy5sYXlvdXRDbGFzcyk7CgkJfQoJfSwKCS8vIHN0YXRpYyBwcm9wZXJ0eSBsYXlvdXQKCWZsb3c6IGZ1bmN0aW9uKCkgewoJfSwKCS8vIGR5bmFtaWMgbWVhc3VyaW5nIGxheW91dAoJcmVmbG93OiBmdW5jdGlvbigpIHsKCX0KfSk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/MultipleDispatchComponent.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qQHB1YmxpYwovKioKCVRoZSBgZW55by5NdWx0aXBsZURpc3BhdGNoQ29tcG9uZW50YCBpcyBhIHB1cmVseSBhYnN0cmFjdCBraW5kCglhbmQgc2ltcGx5IGFkZHMgYSBjb21tb24gYW5jZXN0b3IgZm9yIGBlbnlvLkNvbXBvbmVudHNgIHRoYXQKCW5lZWQgdGhlIGBlbnlvLk11bHRpcGxlRGlzcGF0Y2hTdXBwb3J0YCBtaXhpbi4KKi8KZW55by5raW5kKHsKCQoJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkvLyBQVUJMSUMgUFJPUEVSVElFUwoKCS8vKkBwdWJsaWMKCW5hbWU6ICJlbnlvLk11bHRpcGxlRGlzcGF0Y2hDb21wb25lbnQiLAoJCgkvLypAcHVibGljCglraW5kOiAiZW55by5Db21wb25lbnQiLAoJCgkvLypAcHVibGljCgltaXhpbnM6IFsKCQkiZW55by5NdWx0aXBsZURpc3BhdGNoU3VwcG9ydCIKCV0KCQp9KTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/Object.js"
Content-Type: application/octet-stream; x-encoding=base64
/**
_enyo.Object_ lies at the heart of the Enyo framework's implementations of
property publishing, computed properties (via the _ComputedPropertySupport_
mixin), and data binding (via the _BindingSupport_ mixin). It also provides
several utility functions for its subkinds.

Published properties are declared in a hash called _published_ within a call
to _enyo.kind_. To get and set values for these properties, call
_get(&lt;propertyName&gt;)_ and _set(&lt;propertyName&gt;, &lt;value&gt;)_,
respectively.

By convention, the setter for a published property will trigger an optional
_&lt;propertyName&gt;Changed_ method when called.

For more information, see the [documentation on Published
Properties](https://github.com/enyojs/enyo/wiki/Published-Properties) in the
Enyo Developer Guide.
*/
enyo.kind({
	name: "enyo.Object",
	//* @protected
	// has no base kind
	kind: null,
	noDefer: true,
	//*@public
	// concatenated properties (default)
	concat: enyo.concat,
	//*@public
	/**
		An array of strings representing mixins to be applied
		to this kind at the end of the constructor routine.
	*/
	mixins: [
		"enyo.MixinSupport",
		"enyo.ObserverSupport",
		"enyo.ComputedSupport",
		"enyo.BindingSupport"
	],

	constructor: function(props) {
		enyo._objectCount++;
		this.importProps(props);
	},

	importProps: function (props) {
		if (props) {
			for (var key in props) {
				if (!props.hasOwnProperty(key)) {
					continue;
				}
				this[key] = props[key];
			}
		}
	},

	//* @public
	//* Destroys object with passed-in name.
	destroyObject: function(inName) {
		if (this[inName] && this[inName].destroy) {
			this[inName].destroy();
		}
		this[inName] = null;
	},
	/**
		Sends a log message to the console, prepended with the name of the kind
		and method from which _log_ was invoked.  Multiple arguments are coerced
		to String and joined with spaces.

			enyo.kind({
				name: "MyObject",
				kind: "enyo.Object",
				hello: function() {
					this.log("says", "hi");
					// shows in the console: MyObject.hello: says hi
				}
			});
	*/
	log: function() {
		var acc = arguments.callee.caller;
		var nom = ((acc ? acc.nom : "") || "(instance method)") + ":";
		enyo.logging.log("log", [nom].concat(enyo.cloneArray(arguments)));
	},
	//* Same as _log_, except uses the console's warn method (if it exists).
	warn: function() {
		this._log("warn", arguments);
	},
	//* Same as _log_, except uses the console's error method (if it exists).
	error: function() {
		this._log("error", arguments);
	},
	//* @protected
	_log: function(inMethod, inArgs) {
		if (enyo.logging.shouldLog(inMethod)) {
			try {
				throw new Error();
			} catch(x) {
				enyo.logging._log(inMethod, [inArgs.callee.caller.nom + ": "].concat(enyo.cloneArray(inArgs)));
				enyo.log(x.stack);
			}
		}
	},
	//*@protected
	/**
		Accepts a string property as its only parameter. Evaluates the
		property and, if the value is itself a string, attempts to resolve
		an object from the string. The goal is to determine whether the
		property is a constructor, an instance, or neither. See
		_lang.js#enyo.findAndInstance_ for more information.

		If a method exists of the form `{property}FindAndInstance`, it will
		be used as the callback, with two parameters accepted--the constructor
		(if it was found) and the instance (if it was found or created). This
		allows for those methods to be overloaded by subkinds.
	*/
	findAndInstance: function (property) {
		// if there isn't a property, do nothing
		if (!enyo.exists(property)) {
			return;
		}
		var fn = this[property + "FindAndInstance"];
		// go ahead and call the enyo-scoped version of this method
		return enyo.findAndInstance.call(this, property, fn, this);
	},

	//*@public
	/**
		Retrieves the value of a property or computed property.  Pass in the
		name of (or path to) the desired property or computed property. For
		computed properties, the value of the property is returned, not the
		function. Returns undefined if the requested path relative to the
		object cannot be found or resolved.

		This method is backwards-compatible and will automatically call any
		existing _getter_ method that uses the getProperty naming convention.
		(Moving forward, however, Enyo code should use computed properties
		instead of relying on the getter naming convention.)
	*/
	get: function (path) {
		return enyo.getPath.apply(this, arguments);
	},
	//*@public
	/**
		Sets the value of a property (or path). Pass in the property (or path)
		and the value to be set. If the value is different from the previous
		value, any observers of the property will be automatically notified of
		the change.

		The _force_ parameter is optional; if true, the property's value will be
		updated even if the passed-in value is the same as the current value,
		and observers will be notified of the update.

		The force parameter is optional; if true, the value will be updated
		even if it's the same as the current value, and observers will be
		notified of the update.

		If the property is a computed property, the intended value will be
		passed to the computed property (but will not be returned).

		This method is backwards-compatible and will call any setter that uses
		the setProperty naming convention. (Moving forward, however, Enyo code
		should use computed properties or observers instead of relying on the
		setter naming convention.)
	*/
	set: function (path, value, force) {
		return enyo.setPath.apply(this, arguments);
	},

	//*@public
	/**
		Binds a callback to this object. If the object has been destroyed, the
		bound method will be aborted cleanly with no value returned.

		This method should generally be used instead of `enyo.bind` for running
		code in the context of an instance of _enyo.Object_ or one of its
		subkinds.
	*/
	bindSafely: function(method/*, bound arguments*/) {
		var scope = this;
		if (enyo.isString(method)) {
			if (this[method]) {
				method = this[method];
			} else {
				throw(['enyo.Object.bindSafely: this["', method, '"] is null (this="', this, '")'].join(''));
			}
		}
		if (enyo.isFunction(method)) {
			var args = enyo.cloneArray(arguments, 1);
			return function() {
				if (scope.destroyed) {
					return;
				}
				var nargs = enyo.cloneArray(arguments);
				return method.apply(scope, args.concat(nargs));
			};
		} else {
			throw(['enyo.Object.bindSafely: this["', method, '"] is not a function (this="', this, '")'].join(''));
		}
	},
	//*@protected
	destroy: function () {
		// Since JS objects are never truly destroyed (GC'd) until all references are
		// gone, we might have some delayed action on this object that needs access
		// to this flag.
		this.destroyed = true;
	},

	_isObject: true
});

//* @protected

enyo._objectCount = 0;

enyo.Object.subclass = function(ctor, props) {
	this.publish(ctor, props);
	this.overload(ctor, props);
};

enyo.Object.publish = function(ctor, props) {
	var pp = props.published;
	if (pp) {
		var cp = ctor.prototype;
		for (var n in pp) {
			// need to make sure that even though a property is "published"
			// it does not overwrite any computed properties
			if (props[n] && enyo.isFunction(props[n]) && props[n].isProperty) {
				continue;
			}
			enyo.Object.addGetterSetter(n, pp[n], cp);
		}
	}
};

//*@protected
/**
	We need to find special cases and ensure that the overloaded
	getter of a published property of a parent kind is flagged for
	the global getter and setter.
*/
enyo.Object.overload = function (ctor, props) {
	var proto = ctor.prototype.base? ctor.prototype.base.prototype: {};
	var regex = /^(get|set).*/;
	var name;
	var prop;
	for (name in props) {
		if (!regex.test(name)) {
			continue;
		}
		prop = props[name];
		if ("function" === typeof prop) {
			if (proto[name]) {
				prop.overloaded = true;
			}
		}
	}
};

//*@protected
/**
	This method creates a getter/setter for a published property of
	an _enyo.Object_, but is deprecated. It is maintained for purposes
	of backwards compatability. The preferred method is to mark public
	and protected (private) methods and properties using documentation or
	other means and rely on the _get_ and _set_ methods of _enyo.Object_
	instances.
*/
enyo.Object.addGetterSetter = function (property, value, proto) {
	var getter = "get" + enyo.cap(property);
	var setter = "set" + enyo.cap(property);
	var fn;
	// set the initial value for the prototype
	proto[property] = value;
	fn = proto[getter];
	// if there isn't already a getter provided create one
	if ("function" !== typeof fn) {
		fn = proto[getter] = function () {return this.get(property);};
		fn.overloaded = false;
	} else if (false !== fn.overloaded) {
		// otherwise we need to mark it as having been overloaded
		// so the global getter knows not to ignore it
		fn.overloaded = true;
	}
	// if there isn't already a setter provided, create one
	fn = proto[setter];
	if ("function" !== typeof fn) {
		fn = proto[setter] = function () {return this.set(property, arguments[0]);};
		fn.overloaded = false;
	} else if (false !== fn.overloaded) {
		// otherwise we need to mark it as having been overloaded
		// so the global setter knows not to ignore it
		fn.overloaded = true;
	}
};

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/Oop.js"
Content-Type: application/octet-stream; x-encoding=base64
﻿//*@protected
/**
	Default properties of enyo kinds to concatenate as opposed to
	overwriting. These are automatically used unless explicitly
	removed.
*/
enyo.concat = ["concat"];

//*@protected
/**
	Is called during kind-initialization to make sure that any property
	noted to be concatenated will be (must be an array) so that those values
	will not be lost by subclasses overriding that property.
*/
enyo.handleConcatenatedProperties = function (ctor, proto) {
	var properties = enyo.merge(ctor.concat || [], proto.concat || []);
	var prop;
	var right;
	var left;
	while (properties.length) {
		prop = properties.shift();
		left = ctor[prop];
		right = proto[prop];
		if ((left instanceof Array) && (right instanceof Array)) {
			ctor[prop] = enyo.merge(left, right);
			// remove the reference to the property so it will not
			// conflict later
			delete proto[prop];
		}
	}
};

//* @public
/**
	Creates a JavaScript constructor function with a prototype defined by
	_inProps_. __All constructors must have a unique name__.

	_enyo.kind_ makes it easy to build a constructor-with-prototype (like a
	class) that has advanced features like prototype-chaining (inheritance).

	A plug-in system is included for extending the abilities of the kind
	generator, and constructors are allowed to perform custom operations when
	subclassed.

	If you make changes to _enyo.kind_, be sure to add or update the appropriate
	[unit tests](https://github.com/enyojs/enyo/tree/master/tools/test/core/tests).

	For more information, see the documentation on
	[Creating Kinds](https://github.com/enyojs/enyo/wiki/Creating-Kinds)
	in the Enyo Developer Guide.
*/
enyo.kind = function(inProps) {
	var name = inProps.name || "";
	// cannot defer unnamed kinds, kinds with static sections, or ones with
	// noDefer flag set
	if (name && !inProps.noDefer && !inProps.statics) {
		// make a deferred constructor to avoid a lot of kind
		// processing if we're never used
		var ctor = function DeferredCtor() {
			var FinalCtor;
			// check for cached final constructor first, used mainly when
			// developers directly use kind names in their components instead of
			// strings that are resolved at runtime.
			if (DeferredCtor._FinalCtor) {
				FinalCtor = DeferredCtor._FinalCtor;
			} else {
				if (!(this instanceof DeferredCtor)) {
					throw "enyo.kind: constructor called directly, not using 'new'";
				}
				FinalCtor = DeferredCtor._finishKindCreation();
			}
			var obj = enyo.delegate(FinalCtor.prototype);
			FinalCtor.apply(obj, arguments);
			return obj;
		};
		ctor._finishKindCreation = function() {
			ctor._finishKindCreation = undefined;
			enyo.setPath(name, undefined);
			var FinalCtor = enyo.kind.finish(inProps);
			ctor._FinalCtor = FinalCtor;
			inProps = null;
			return FinalCtor;
		};
		if ((name && !enyo.getPath(name)) || enyo.kind.allowOverride) {
			enyo.setPath(name, ctor);
		}
		else if (name) {
			enyo.error("enyo.kind: " + name + " is already in use by another " +
				"kind, all kind definitions must have unique names.");
		}
		return ctor;
	} else {
		// create anonymous kinds immediately
		return enyo.kind.finish(inProps);
	}
};
//* @protected
enyo.kind.finish = function(inProps) {
	// kind-name to constructor map could be faulty now that a new kind exists, so we simply destroy the memoizations
	enyo._kindCtors = {};
	// extract 'name' property
	var name = inProps.name || "";
	delete inProps.name;
	// extract 'kind' property
	var hasKind = ("kind" in inProps);
	var kind = inProps.kind;
	delete inProps.kind;
	// establish base class reference
	var base = enyo.constructorForKind(kind);
	var isa = base && base.prototype || null;
	// if we have an explicit kind property with value undefined, we probably
	// tried to reference  a kind that is not yet in scope
	if (hasKind && kind === undefined || base === undefined) {
		var problem = kind === undefined ? 'undefined kind' : 'unknown kind (' + kind + ')';
		throw "enyo.kind: Attempt to subclass an " + problem + ". Check dependencies for [" + (name || "<unnamed>") + "].";
	}
	// make a boilerplate constructor
	var ctor = enyo.kind.makeCtor();
	// semi-reserved word 'constructor' causes problems with Prototype and IE, so we rename it here
	if (inProps.hasOwnProperty("constructor")) {
		inProps._constructor = inProps.constructor;
		delete inProps.constructor;
	}
	// create our prototype
	//ctor.prototype = isa ? enyo.delegate(isa) : {};
	enyo.setPrototype(ctor, isa ? enyo.delegate(isa) : {});

	// there are special cases where a base class has a property
	// that may need to be concatenated with a subclasses implementation
	// as opposed to completely overwriting it...
	enyo.handleConcatenatedProperties(ctor.prototype, inProps);

	// put in our props
	enyo.mixin(ctor.prototype, inProps);
	// alias class name as 'kind' in the prototype
	// but we actually only need to set this if a new name was used
	// not if it is inheriting from a kind anonymously
	if (name) {
		ctor.prototype.kindName = name;
	}
	// this is for anonymous constructors
	else {
		ctor.prototype.kindName = base && base.prototype? base.prototype.kindName: "";
	}
	// cache superclass constructor
	ctor.prototype.base = base;
	// reference our real constructor
	ctor.prototype.ctor = ctor;
	// support pluggable 'features'
	enyo.forEach(enyo.kind.features, function(fn){ fn(ctor, inProps); });
	// put reference into namespace
	if ((name && !enyo.getPath(name)) || enyo.kind.allowOverride) {
		enyo.setPath(name, ctor);
	}
	else if (name) {
		enyo.error("enyo.kind: " + name + " is already in use by another " +
			"kind, all kind definitions must have unique names.");
	}
	return ctor;
};

//* @public
/**
	Creates a Singleton of a given kind with a given definition.
	__The name property will be the instance name of the singleton
	and must be unique__.

		enyo.singleton({
			kind: "enyo.Control",
			name: "app.MySingleton",
			published: {
				value: "foo"
			},
			makeSomething: function() {
				//...
			}
		});

		app.MySingleton.makeSomething();
		app.MySingleton.setValue("bar");
*/
enyo.singleton = function(conf, context) {
	// extract 'name' property (the name of our singleton)
	var name = conf.name;
	delete(conf.name);
	// create an unnamed kind and save its constructor's function
	var Kind = enyo.kind(conf);
	var inst;
	// create the singleton with the previous name and constructor
	enyo.setPath.call(context || enyo.global, name, (inst = new Kind()));
	return inst;
};

//* @protected
enyo.kind.makeCtor = function() {
	return function ctor() {
		if (!(this instanceof ctor)) {
			throw "enyo.kind: constructor called directly, not using 'new'";
		}

		// two-pass instantiation
		var result;
		var cargs = arguments;
		if (this._constructor) {
			// pure construction
			result = this._constructor.apply(this, arguments);
		}
		// defer initialization until entire constructor chain has finished
		if (this.constructed) {
			// post-constructor initialization
			this.constructed.apply(this, arguments);
		}

		for (var idx = 0; idx < enyo.kind.postConstructors.length; ++idx) {
			enyo.kind.postConstructors[idx].apply(this, cargs);
		}

		if (result) {
			return result;
		}
	};
};

// classes referenced by name can omit this namespace (e.g. "Button" instead of "enyo.Button")
enyo.kind.defaultNamespace = "enyo";

//
// feature hooks for the oop system
//
enyo.kind.features = [];


//*@protected
/**
	Post-initialize functions (after constructor has completed).
*/
enyo.kind.postConstructors = [];

//
// 'inherited' feature
//
enyo.kind.features.push(function(ctor, props) {
	var proto = ctor.prototype;
	if (!proto.inherited) {
		proto.inherited = enyo.kind.inherited;
	}
	if (proto.base) {
		// decorate function properties to support inherited (do this ex post facto so that
		// ctor.prototype is known, relies on elements in props being copied by reference)
		for (var n in props) {
			var p = props[n];
			if (enyo.isFunction(p)) {
				p._inherited = proto.base.prototype[n];
				// FIXME: we used to need some extra values for inherited, then inherited got cleaner
				// but in the meantime we used these values to support logging in Object.
				// For now we support this legacy situation, by suppling logging information here.
				p.nom = proto.kindName + '.' + n + '()';
			}
		}
	}
});

//*@protected
/**
	This method is called by enyo.Object's attempting to
	access super-methods of a parent class (kind) by calling
	_this.inherited(arguments)_ from within a kind method. This
	can only be done safely when there is known to be a super
	class with the same method.
*/
enyo.kind.inherited = function (originals, replacements) {
	// one-off methods are the fast track
	var target = originals.callee;
	var fn = target._inherited;

	// regardless of how we got here, just ensure we actually
	// have a function to call or else we throw a console
	// warning to notify developers they are calling a
	// super method that doesn't exist
	if ("function" === typeof fn) {
		return fn.apply(this, replacements? enyo.mixin(originals, replacements): originals);
	} else {
		enyo.warn("enyo.kind.inherited: unable to find requested " +
			"super-method from -> " + originals.callee.nom + " in " + this.kindName);
	}
};

//
// 'statics' feature
//
enyo.kind.features.push(function(ctor, props) {
	// install common statics
	if (!ctor.subclass) {
		ctor.subclass = enyo.kind.statics.subclass;
	}
	if (!ctor.extend) {
		ctor.extend = enyo.kind.statics.extend;
	}
	// move props statics to constructor
	if (props.statics) {
		enyo.mixin(ctor, props.statics);
		delete ctor.prototype.statics;
	}
	// also support protectedStatics which won't interfere with defer
	if (props.protectedStatics) {
		enyo.mixin(ctor, props.protectedStatics);
		delete ctor.prototype.protectedStatics;
	}
	// allow superclass customization
	var base = ctor.prototype.base;
	while (base) {
		base.subclass(ctor, props);
		base = base.prototype.base;
	}
});

enyo.kind.statics = {
	subclass: function(ctor, props) {
		//enyo.log("subclassing [" + ctor.prototype.kind + "] from [", this.prototype.kind + "]");
	},
	extend: function(props) {
		// make sure to allow concatenated properties to function
		// as expected
		enyo.handleConcatenatedProperties(this.prototype, props);
		enyo.mixin(this.prototype, props);
		// support pluggable 'features'
		var ctor = this;
		enyo.forEach(enyo.kind.features, function(fn){ fn(ctor, props); });
	}
};

enyo.checkConstructor = function(inKind) {
	if (enyo.isFunction(inKind)) {
		// if a deferred enyo kind, finish that work first
		if (inKind._FinalCtor) {
			return inKind._FinalCtor;
		}
		if (inKind._finishKindCreation) {
			return inKind._finishKindCreation();
		}
	}
	return inKind;
};

//
// factory for kinds identified by strings
//
enyo._kindCtors = {};

enyo.constructorForKind = function(inKind) {
	if (inKind === null) {
		return inKind;
	} else if (inKind === undefined) {
		return enyo.defaultCtor;
	}
	else if (enyo.isFunction(inKind)) {
		return enyo.checkConstructor(inKind);
	}

	// use memoized constructor if available...
	var ctor = enyo._kindCtors[inKind];
	if (ctor) {
		return ctor;
	}
	// otherwise look it up and memoize what we find
	//
	// if inKind is an object in enyo, say "Control", then ctor = enyo["Control"]
	// if inKind is a path under enyo, say "Heritage.Button", then ctor = enyo["Heritage.Button"] || enyo.Heritage.Button
	// if inKind is a fully qualified path, say "enyo.Heritage.Button", then ctor = enyo["enyo.Heritage.Button"] || enyo.enyo.Heritage.Button || enyo.Heritage.Button
	//
	// Note that kind "Foo" will resolve to enyo.Foo before resolving to global "Foo".
	// This is important so "Image" will map to built-in Image object, instead of enyo.Image control.
	ctor = enyo.Theme[inKind] || enyo[inKind] || enyo.getPath.call(enyo, inKind, true) || window[inKind] || enyo.getPath(inKind);
	// if this is a deferred kind, run the follow-up code then refetch the kind's constructor
	if (ctor && ctor._finishKindCreation) {
		ctor = ctor._finishKindCreation();
	}
	enyo._kindCtors[inKind] = ctor;
	return ctor;
};

//
// namespace for current theme ("enyo.Theme.Button" references the Button specialization for the current theme)
//
enyo.Theme = {};

enyo.registerTheme = function(inNamespace) {
	enyo.mixin(enyo.Theme, inNamespace);
};

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/Router.js"
Content-Type: application/octet-stream; x-encoding=base64
(function (enyo) {

	//*@protected
	/**
		Any instance of a router will be referenced here for the global
		hash change handler.
	*/
	var listeners = [];

	//*@protected
	/**
		This is the intended global window.hashchange event handler. If
		another handler is arbitrarily registered for this event then it
		will not fire.
	*/
	var hashDidChange = function (hash) {
		var list = listeners;
		var len = list.length;
		var idx = 0;
		for (; idx < len; ++idx) {
			list[idx]._hashChanged(hash);
		}
	};

	//*@protected
	var token = /\:[a-zA-Z0-9]*/g;

	//*@protected
	var prepare = function (str) {
		return str[0] === "#"? str.slice(1): str;
	};

	//*@protected
	/**
		All of our actively-supported browsers support this method of
		registering for hashchange events.
	*/
	enyo.ready(function () {
		enyo.dispatcher.listen(window, "hashchange", hashDidChange);
	});

	//*@public
	/**
		_enyo.Router_ is a controller with the ability to interpret changes
		in the url as well as set changes to the url in a manner that is
		compatible across browsers. With defined route handling, the
		application state can be managed more closely with respect to the
		location state of the browser. There may be multiple router instances
		active at any one time. Routers only interact with the hash portion
		of the browser location and will not force a reload of the current
		page.

		Routes may be defined in several ways and may be added at startup or
		added programmatically at a later time.

		A route is a declarative hash with the following structure:

			{path: "some/path", handler: "function", context: "context"}

		The path is a string that may be static (explicitly matched) or
		dynamic (matched based on dynamic placeholders). Dynamic paths
		may name elements to be matched, e.g.:

			{path: ":user/:id"}

		In this case, the handler would be called with two parameters
		filled with the values matched by that structure. It is important
		to note that a dynamic route will apply the first match it finds,
		so care must be used to ensure that the correct route is matched
		(e.g., `/users/:user/:id` is more exact because of the static
		`users` portion of the path).

		The handler may be a function reference or a string that will be
		mapped to a function. A check for the function's existence is
		conducted first on the router, then on any provided context, and
		finally in the global scope. If a context is provided, the
		function will be executed under that context regardless of where
		the function was found.

		The context property may be an object, an instance or a string that
		will be mapped to an object if possible.

		Note that, currently, only letters and numbers are supported in
		dynamic routes.
	*/
	enyo.kind({

		// ...........................
		// PUBLIC PROPERTIES

		//*@public
		name: "enyo.Router",

		//*@public
		/**
			If true, the router will respond to hash changes or internal events.
			If this flag is set to false, it will stop responding. This may be
			changed at any time.
		*/
		listening: true,

		//*@public
		/**
			If true, the router will neither respond to hash changes in the
			browser nor be able to trigger them. Instead, it may be used
			internally to maintain or trigger state changes in an application
			without changing location.
		*/
		internalOnly: false,

		//*@public
		/**
			If the desire is to force the current browser location to a particular
			path on startup set this value to true. Will be ignored if _triggerOnStart_
			is false.
		*/
		defaultPathOnStart: false,

		//*@public
		/**
			The _defaultRoute_ should have the same structure as a normal
			route (hash). It may be arbitrarily assigned to this property
			or mixed into the routes array with a special _default: true_
			flag set. For any unmatched hash changes, this route will be
			executed and passed the path that was not matched.
		*/
		defaultRoute: null,

		//*@public
		/**
			By default, when a router is created, it will attempt to trigger
			the correct route for the current browser location. Set this
			to false to prevent this from happening.
		*/
		triggerOnStart: true,

		//*@public
		/**
			The router will attempt to track history based on the events
			that have been generated through it.
		*/
		useHistory: false,

		//*@public
		/**
			The _routes_ array constitutes the handlers for this router.
			Routes are string paths, static or dynamic, that route particular
			hash change events. They are defined in an array of hashes with
			a _handler_ (function), _path_ (for static and dynamic paths), an
			optional _context_ (for the handler), or a _default_ boolean true|false
			where that handler will be used when no other route can handle the
			hashchange event.
		*/
		routes: null,

		// ...........................
		// PROTECTED PROPERTIES

		//*@protected
		kind: "enyo.Controller",

		//*@protected
		_staticRoutes: null,

		//*@protected
		_dynamicRoutes: null,

		//*@protected
		_current: "",

		//*@protected
		_history: null,

		// ...........................
		// COMPUTED PROPERTIES

		//*@public
		/**
			A computed property that will return the location as
			known by this router. This property will be synchronized
			with _window.location.hash_ unless it's _internalOnly_
			flag was set to true. Passing a string to this property
			via _set("location", "mylocation")_ will update the inner
			location known by this router.
		*/
		location: enyo.computed(function (loc) {
			if (loc) {
				loc = prepare(loc);
				if (!this.internalOnly) {
					enyo.asyncMethod(this, "trigger", {location: loc, change: true});
				} else {
					this.set("_current", loc);
				}
			} else {
				return prepare(this.get("_current"));
			}
		}, "_current", {cached: true}),

		//*@public
		/**
			Returns the string for the default path (if any otherwise empty
			string).
		*/
		defaultPath: enyo.computed(function () {
			return this.defaultRoute? this.defaultRoute.path: "";
		}),

		// ...........................
		// PUBLIC METHODS

		//*@public
		/**
			Triggers a change without necessarily requiring a change to	occur.
			If called without a parameter, it will force the route that matches
			the current browser location to fire. If a string is passed in, this
			method will trigger an internal-only event (i.e., the	browser
			location will not be changed). If it is passed a hash, the method
			will try to use a _location_ property while looking for optional
			_change_ and _global_ properties.

			If the _change_ property is present and true, it will force a
			_location.hash_ change in the browser (this is always global).
			If the _global_ property is present and true and _change_ is
			not present or false, it will trigger an internal event that
			all routers will respond to (not just this instance).
		*/
		trigger: function (params) {
			if (!params) {
				params = {location: this.get("_current")};
			} else if ("string" === typeof params) {
				params = {location: params};
			}
			var loc = params.location;
			var global = params.global;
			var change = params.change;
			var current = this.get("location");
			if (change) {
				if (current !== loc) {
					window.location.hash = loc;
				} else {
					this._hashChanged(loc);
				}
			} else {
				if (global) {
					hashDidChange(loc);
				} else {
					this._hashChanged(loc);
				}
			}
		},

		//*@public
		/**
			In very rare circumstances it may be useful to pass a path
			to the routes without using trigger or global hash changes
			with _path_ being a string that will be evaluated against
			the routes owned by this router.
		*/
		handle: function (path) {
			// fast track is to check against static routes first
			if (this._handleStatic(path)) {
				return;
			}
			// then we check against dynamic paths in this simple scheme
			else if (this._handleDynamic(path)) {
				/* do nothing */
			}
			else {
				this._handleDefault(path);
			}
		},

		//*@public
		/**
			If history is enabled and some history exists will attempt
			to revert the current known location to the previous one in
			the stack.
		*/
		back: function () {
			if (this.useHistory) {
				if (this._history.length >= 2) {
					// we shift the current location off the stack
					this._history.shift();
					// we shift the requested location off the stack
					// but reapply it
					this.set("location", this._history.shift());
				}
			}
		},

		//*@public
		/**
			To arbitrarily add history. Optional second parameter can be
			a boolean true to place the location at the lowest (first) position
			of the stack or an integer indicating its exact location in the
			stack. If the index is out of bounds it will be added at the lowest
			position (same as boolean true for second parameter). Returns callee
			for chaining.
		*/
		addHistory: function (location, idx) {
			if (this.useHistory) {
				switch (typeof idx) {
				case "undefined":
					this._history.unshift(location);
					break;
				case "number":
					if (idx >= 0 && idx < this._history.length) {
						this._history.splice(idx, 0, location);
					}
					break;
				case "boolean":
					this._history.push(location);
					break;
				}
			}
			return this;
		},

		//*@public
		/**
			Clears any history the router has currently stored. Returns
			callee for chaining.
		*/
		clearHistory: function () {
			this._history = [];
			return this;
		},

		//*@public
		/**
			Can be used to programmatically add routes to the router
			where _route_ is a hash as described by the _routes_ array.
			Returns callee for chaining.
		*/
		addRoute: function (route) {
			var statics = this._staticRoutes;
			var dynamic = this._dynamicRoutes;
			var regex;
			if (true === route['default']) {
				this.defaultRoute = route;
			}
			else if (token.test(route.path)) {
				regex = new RegExp(route.path.replace(token, "([a-zA-Z0-9-]*)"));
				route.regex = regex;
				dynamic.push(route);
			} else {
				statics[route.path] = route;
			}
			return this;
		},

		// ...........................
		// PROTECTED METHODS

		//*@protected
		constructor: function () {
			this._staticRoutes = {};
			this._dynamicRoutes = [];
			this.routes = this.routes || [];
			this._history = this._history || [];
			this.inherited(arguments);
		},

		//*@protected
		create: function () {
			this.inherited(arguments);
			// make sure to initialize our routes prior
			// to registering for events
			this._setupRoutes();
			// make sure we're up to date
			this.set("_current", prepare(window.location.hash));
			// ok, register for events
			listeners.push(this);
			// ok, if we need to go ahead and route our current
			// location, lets do it
			if (this.triggerOnStart) {
				if (this.defaultPathOnStart) {
					this.trigger({change: true, location: this.get("defaultPath")});
				} else {
					this.trigger();
				}
			}
		},

		//*@protected
		destroy: function () {
			var idx = enyo.indexOf(this, listeners);
			if (!~idx) {
				listeners.splice(idx, 1);
			}
			this.inherited(arguments);
		},

		//*@protected
		_hashChanged: function (hash) {
			var $hash = (function (prop) {
				if (!enyo.isString(prop)) {
					// some browsers do not support the newUrl property
					// so we're forced to look at the current hash
					prop = prop.newUrl || window.location.hash;
				}
				return prepare(prop);
			})(hash);
			if (this.listening) {
				this.set("_current", $hash);
				this.handle($hash);
			}
		},

		//*@protected
		_execHandler: function (context, handler, args, route) {
			var $fn = handler;
			var $ctx = "string" === typeof context? enyo.getPath.call(this, context): context || this;
			// if the handler is defined as a string, we need to determine if
			// it is relative to the router, relative to the context, or a named
			// function in the global scope
			if ("string" === typeof handler) {
				// first check to see if the handler is a named property
				// on the router; otherwise, try the context itself
				$fn = this[handler] || $ctx[handler];
				if ("function" === typeof $fn) {
					// in case we actually found it, let's not go hunting
					// next time
					route.handler = $fn;
					route.context = $ctx;
				}
			}
			// if we have an actual handler, let's execute it now
			if ($fn && "function" === typeof $fn) {
				$fn.apply($ctx, args);
				return true;
			}
			// otherwise we couldn't determine what we were supposed to
			// do here
			return false;
		},

		//*@protected
		_handleStatic: function (path) {
			var statics = this._staticRoutes;
			var route;
			var handler;
			var context;
			if ((route = statics[path])) {
				handler = route.handler;
				context = route.context;
				return this._execHandler(context, handler, [path], route);
			}
			return false;
		},

		//*@protected
		_handleDynamic: function (path) {
			var dynamic = this._dynamicRoutes;
			var regex;
			var route;
			var handler;
			var context;
			var matches;
			var idx = 0;
			var len = dynamic.length;
			for (; idx < len; ++idx) {
				route = dynamic[idx];
				regex = route.regex;
				if ((matches = regex.exec(path))) {
					// we need to strip off the full match so we can
					// use the other matches as parameters
					matches = matches.slice(1);
					handler = route.handler;
					context = route.context;
					return this._execHandler(context, handler, matches, route);
				}
			}
			return false;
		},

		//*@protected
		_handleDefault: function (path) {
			var route = this.defaultRoute || {};
			var context = route.context;
			var handler = route.handler;
			return this._execHandler(context, handler, [path], route);
		},

		//*@protected
		_setupRoutes: function () {
			var routes = this.routes;
			var idx = 0;
			var len = routes.length;
			var route;
			for (; idx < len; ++idx) {
				route = routes[idx];
				if (!route) {
					continue;
				}
				this.addRoute(route);
			}
		},

		//*@protected
		_currentChanged: function () {
			if (this.useHistory) {
				this._history.unshift(this.get("location"));
			}
		}

	});

}(enyo));

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/Signals.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5TaWduYWxzXyBjb21wb25lbnRzIGFyZSB1c2VkIHRvIGxpc3RlbiB0byBnbG9iYWwgbWVzc2FnZXMuCgoJQW4gb2JqZWN0IHdpdGggYSBTaWduYWxzIGNvbXBvbmVudCBjYW4gbGlzdGVuIHRvIG1lc3NhZ2VzIHNlbnQgZnJvbSBhbnl3aGVyZQoJYnkgZGVjbGFyaW5nIGhhbmRsZXJzIGZvciB0aGVtLgoKCURPTSBldmVudHMgdGhhdCBoYXZlIG5vIG5vZGUgdGFyZ2V0cyBhcmUgYnJvYWRjYXN0IGFzIHNpZ25hbHMuIFRoZXNlIGV2ZW50cwoJaW5jbHVkZSBXaW5kb3cgZXZlbnRzLCBsaWtlIF9vbmxvYWRfIGFuZCBfb25iZWZvcmV1bmxvYWRfLCBhbmQgZXZlbnRzIHRoYXQKCW9jY3VyIGRpcmVjdGx5IG9uIF9kb2N1bWVudF8sIGxpa2UgX29ua2V5cHJlc3NfIGlmIF9kb2N1bWVudF8gaGFzIHRoZSBmb2N1cy4KCglGb3IgbW9yZSBpbmZvcm1hdGlvbiwgc2VlIHRoZQoJPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2VueW9qcy9lbnlvL3dpa2kvU2lnbmFscyI+U2lnbmFscyBkb2N1bWVudGF0aW9uPC9hPgoJaW4gdGhlIEVueW8gRGV2ZWxvcGVyIEd1aWRlLgoqLwplbnlvLmtpbmQoewoJbmFtZTogImVueW8uU2lnbmFscyIsCglraW5kOiAiZW55by5Db21wb25lbnQiLAoJLy8qIEBwcm90ZWN0ZWQKCWNyZWF0ZTogZnVuY3Rpb24oKSB7CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCQllbnlvLlNpZ25hbHMuYWRkTGlzdGVuZXIodGhpcyk7Cgl9LAoJZGVzdHJveTogZnVuY3Rpb24oKSB7CgkJZW55by5TaWduYWxzLnJlbW92ZUxpc3RlbmVyKHRoaXMpOwoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7Cgl9LAoJbm90aWZ5OiBmdW5jdGlvbihpbk1zZywgaW5QYXlsb2FkKSB7CgkJdGhpcy5kaXNwYXRjaEV2ZW50KGluTXNnLCBpblBheWxvYWQpOwoJfSwKCXByb3RlY3RlZFN0YXRpY3M6IHsKCQlsaXN0ZW5lcnM6IFtdLAoJCWFkZExpc3RlbmVyOiBmdW5jdGlvbihpbkxpc3RlbmVyKSB7CgkJCXRoaXMubGlzdGVuZXJzLnB1c2goaW5MaXN0ZW5lcik7CgkJfSwKCQlyZW1vdmVMaXN0ZW5lcjogZnVuY3Rpb24oaW5MaXN0ZW5lcikgewoJCQllbnlvLnJlbW92ZShpbkxpc3RlbmVyLCB0aGlzLmxpc3RlbmVycyk7CgkJfSwKCX0sCglzdGF0aWNzOiB7CgkJc2VuZDogZnVuY3Rpb24oaW5Nc2csIGluUGF5bG9hZCkgewoJCQllbnlvLmZvckVhY2godGhpcy5saXN0ZW5lcnMsIGZ1bmN0aW9uKGwpIHsKCQkJCWwubm90aWZ5KGluTXNnLCBpblBheWxvYWQpOwoJCQl9KTsKCQl9Cgl9Cn0pOwo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/UiComponent.js"
Content-Type: application/octet-stream; x-encoding=base64
/**
	_enyo.UiComponent_ implements a container strategy suitable for presentation
	layers.

	UiComponent itself is abstract. Concrete subkinds include
	<a href="#enyo.Control">enyo.Control</a> (for HTML/DOM) and
	<a href="#enyo.canvas.Control">enyo.canvas.Control</a>
	(for Canvas contexts).
*/
enyo.kind({
	name: "enyo.UiComponent",
	kind: "enyo.Component",
	published: {
		//* The UiComponent that physically contains this component in the DOM
		container: null,
		/**
			The UiComponent that owns this component for purposes of event
			propagation
		*/
		parent: null,
		/**
			The UiComponent that will physically contain new items added by
			calls to _createComponent_
		*/
		controlParentName: "client",
		//* A kind used to manage the size and placement of child components
		layoutKind: ""
	},
	handlers: {
		onresize: "resizeHandler"
	},
	/**
		When set, provides a control reference used to indicate where a
		newly-created component should be added in the UiComponent's array of
		children. This is typically used when dynamically creating children
		(rather than at design time). If set to null, the new control will be
		added at the beginning of the array; if set to a specific existing
		control, the new control will be added before the specified control. If
		left undefined, the default behavior is to add the new control at the
		end of the array.
	*/
	addBefore: undefined,
	//* @protected
	protectedStatics: {
		_resizeFlags: {showingOnly: true} // don't waterfall these events into hidden controls
	},

	create: function() {
		this.controls = [];
		this.children = [];
		this.containerChanged();
		this.inherited(arguments);
		this.layoutKindChanged();
	},
	destroy: function() {
		// Destroys all non-chrome controls (regardless of owner).
		this.destroyClientControls();
		// Removes us from our container.
		this.setContainer(null);
		// Destroys chrome controls owned by this.
		this.inherited(arguments);
	},
	importProps: function(inProps) {
		this.inherited(arguments);
		if (!this.owner) {
			//this.log("registering ownerless control [" + this.kindName + "] with enyo.master");
			this.owner = enyo.master;
		}
	},
	// As implemented, _controlParentName_ only works to identify an owned
	// control created via _createComponents_ (i.e., usually in our _components_
	// block).	To attach a _controlParent_ via other means, one must call
	// _discoverControlParent_ or set _controlParent_ directly.
	//
	// We could call _discoverControlParent_ in _addComponent_, but it would
	// cause a lot of useless checking.
	createComponents: function() {
		var results = this.inherited(arguments);
		this.discoverControlParent();
		return results;
	},
	discoverControlParent: function() {
		this.controlParent = this.$[this.controlParentName] || this.controlParent;
	},
	adjustComponentProps: function(inProps) {
		// Components we create have us as a container by default.
		inProps.container = inProps.container || this;
		this.inherited(arguments);
	},
	// containment
	containerChanged: function(inOldContainer) {
		if (inOldContainer) {
			inOldContainer.removeControl(this);
		}
		if (this.container) {
			this.container.addControl(this, this.addBefore);
		}
	},
	// parentage
	parentChanged: function(inOldParent) {
		if (inOldParent && inOldParent != this.parent) {
			inOldParent.removeChild(this);
		}
	},
	//* @public
	// Note: Oddly, a Control is considered a descendant of itself.
	isDescendantOf: function(inAncestor) {
		var p = this;
		while (p && p!=inAncestor) {
			p = p.parent;
		}
		return inAncestor && (p == inAncestor);
	},
	/**
		Returns all controls.
	*/
	getControls: function() {
		return this.controls;
	},
	/**
		Returns all non-chrome controls.
	*/
	getClientControls: function() {
		var results = [];
		for (var i=0, cs=this.controls, c; (c=cs[i]); i++) {
			if (!c.isChrome) {
				results.push(c);
			}
		}
		return results;
	},
	/**
		Destroys "client controls", the same set of controls returned by
		_getClientControls_.
	*/
	destroyClientControls: function() {
		var c$ = this.getClientControls();
		for (var i=0, c; (c=c$[i]); i++) {
			c.destroy();
		}
	},
	//* @protected
	addControl: function(inControl, inBefore) {
		// Called to add an already created control to the object's control list. It is
		// not used to create controls and should likely not be called directly.
		// It can be overridden to detect when controls are added.
		if (inBefore !== undefined) {
			var idx = (inBefore === null) ? 0 : this.indexOfChild(inBefore);
			this.controls.splice(idx, 0, inControl);
		} else {
			this.controls.push(inControl);
		}
		// When we add a Control, we also establish a parent.
		this.addChild(inControl, inBefore);
	},
	removeControl: function(inControl) {
		// Called to remove a control from the object's control list. As with addControl it
		// can be overridden to detect when controls are removed.
		// When we remove a Control, we also remove it from its parent.
		inControl.setParent(null);
		return enyo.remove(inControl, this.controls);
	},
	indexOfControl: function(inControl) {
		return enyo.indexOf(inControl, this.controls);
	},
	indexOfClientControl: function(inControl) {
		return enyo.indexOf(inControl, this.getClientControls());
	},
	indexInContainer: function() {
		return this.container.indexOfControl(this);
	},
	clientIndexInContainer: function() {
		return this.container.indexOfClientControl(this);
	},
	controlAtIndex: function(inIndex) {
		return this.controls[inIndex];
	},
	// children
	addChild: function(inChild, inBefore) {
		// if inBefore is undefined, add to the end of the child list.
		// If it's null, add to front of list, otherwise add before the
		// specified control.
		//
		// allow delegating the child to a different container
		if (this.controlParent /*&& !inChild.isChrome*/) {
			// this.controlParent might have a controlParent, and so on; seek the ultimate parent
			// inBefore is not passed because that control won't be in the controlParent's scope
			this.controlParent.addChild(inChild);
		} else {
			// NOTE: addChild drives setParent.
			// It's the opposite for setContainer, where containerChanged (in Containable)
			// drives addControl.
			// Because of the way 'parent' is derived from 'container', this difference is
			// helpful for implementing controlParent.
			// By the same token, since 'parent' is derived from 'container', setParent is
			// not intended to be called by client code. Therefore, the lack of parallelism
			// should be private to this implementation.
			// Set the child's parent property to this
			inChild.setParent(this);
			// track in children array
			if (inBefore !== undefined) {
				var idx = (inBefore === null) ? 0 : this.indexOfChild(inBefore);
				this.children.splice(idx, 0, inChild);
			} else {
				this.children.push(inChild);
			}
		}
	},
	removeChild: function(inChild) {
		return enyo.remove(inChild, this.children);
	},
	indexOfChild: function(inChild) {
		return enyo.indexOf(inChild, this.children);
	},
	layoutKindChanged: function() {
		if (this.layout) {
			this.layout.destroy();
		}
		this.layout = enyo.createFromKind(this.layoutKind, this);
		if (this.generated) {
			this.render();
		}
	},
	flow: function() {
		if (this.layout) {
			this.layout.flow();
		}
	},
	// CAVEAT: currently we use the entry point for both
	// post-render layout work *and* post-resize layout work.
	reflow: function() {
		if (this.layout) {
			this.layout.reflow();
		}
	},
	/**
		Call after this control has been resized to allow it to process the size change.
		To respond to a resize, override _resizeHandler_ instead.
	*/
	// syntactic sugar for 'waterfall("onresize")'
	resized: function() {
		this.waterfall("onresize", enyo.UiComponent._resizeFlags);
		this.waterfall("onpostresize", enyo.UiComponent._resizeFlags);
	},
	//* @protected
	resizeHandler: function() {
		// FIXME: once we are in the business of reflowing layouts on resize, then we have an
		// inside/outside problem: some scenarios will need to reflow before child
		// controls reflow, and some will need to reflow after. Even more complex scenarios
		// have circular dependencies, and can require multiple passes or other resolution.
		// When we can rely on CSS to manage reflows we do not have these problems.
		this.reflow();
	},
	/**
		Sends a message to all my descendents.
	*/
	waterfallDown: function(name, event, sender) {
		event = event || {};
		// Note: Controls will generally be both in a $ hash and a child list somewhere.
		// Attempt to avoid duplicated messages by sending only to components that are not
		// UiComponent, as those components are guaranteed not to be in a child list.
		// May cause a problem if there is a scenario where a UiComponent owns a pure
		// Component that in turn owns Controls.
		//
		// waterfall to all pure components
		for (var n in this.$) {
			if (!(this.$[n] instanceof enyo.UiComponent)) {
				this.$[n].waterfall(name, event, sender);
			}
		}
		// waterfall to my children
		for (var i=0, cs=this.children, c; (c=cs[i]); i++) {
			// Do not send {showingOnly: true} events to hidden controls. This flag is set for resize events
			// which are broadcast from within the framework. This saves a *lot* of unnecessary layout.
			// TODO: Maybe remember that we did this, and re-send those messages on setShowing(true)?
			// No obvious problems with it as-is, though
			if (c.showing || !(event && event.showingOnly)) {
				c.waterfall(name, event, sender);
			}
		}
	},
	getBubbleTarget: function() {
		return this._bubble_target || this.parent || this.owner;
	}
});

enyo.createFromKind = function(inKind, inParam) {
	var Ctor = inKind && enyo.constructorForKind(inKind);
	if (Ctor) {
		return new Ctor(inParam);
	}
};

//
// Default owner for ownerless UiComponents to allow notifying such UiComponents
// of important system events like window resize.
//
// NOTE: Ownerless UiComponents will not be garbage collected unless explicitly
// destroyed, as they will be referenced by _enyo.master_.
//
enyo.master = new enyo.Component({
	name: "master",
	notInstanceOwner: true,
	eventFlags: {showingOnly: true}, // don't waterfall these events into hidden controls
	getId: function() {
		return '';
	},
	isDescendantOf: enyo.nop,
	bubble: function(inEventName, inEvent, inSender) {
		//enyo.log("master event: " + inEventName);
		if (inEventName == "onresize") {
			// Resize is special; waterfall this message.
			// This works because master is a Component, so it waterfalls
			// to its owned Components (i.e., master has no children).
			enyo.master.waterfallDown("onresize", this.eventFlags);
			enyo.master.waterfallDown("onpostresize", this.eventFlags);
		} else {
			// All other top-level events are sent only to interested Signal
			// receivers.
			enyo.Signals.send(inEventName, inEvent);
		}
	}
});

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/ViewController.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uIChlbnlvKSB7CgoJLy8qQHB1YmxpYwoJLyoqCgkJX2VueW8uVmlld0NvbnRyb2xsZXJfIGlzIGFuIGFic3RyYWN0IGtpbmQgZGVzaWduZWQgZm9yIHVzZSBpbiBhCgkJdGlnaHRseS1jb3VwbGVkIGNvbnRyb2xsZXIgYW5kIHZpZXcsIGluIHdoaWNoIHRoZSBjb250cm9sbGVyIG93bnMKCQl0aGUgdmlldyBhbmQgbWFpbnRhaW5zIGl0cyBzdGF0ZSBhbmQgbGlmZWN5Y2xlLgoJKi8KCWVueW8ua2luZCh7CgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIFBVQkxJQyBQUk9QRVJUSUVTCgoJCS8vKkBwdWJsaWMKCQluYW1lOiAiZW55by5WaWV3Q29udHJvbGxlciIsCgoJCS8vKkBwdWJsaWMKCQlraW5kOiAiZW55by5Db250cm9sbGVyIiwKCgkJLy8qQHB1YmxpYwoJCS8qKgoJCQlUaGlzIG1heSBiZSBhIHN0cmluZyByZXByZXNlbnRpbmcgYSBraW5kLCBhIGNvbnN0cnVjdG9yIGZvciBhIGtpbmQsCgkJCW9yIGFuIG9iamVjdCBsaXRlcmFsIGRlZmluaW5nIHRoZSB2aWV3IHN0cnVjdHVyZSB0byBiZSBpbnN0YW5jZWQuCgkJCU9uY2UgdGhlIHZpZXcgY29udHJvbGxlciBoYXMgYmVlbiBpbml0aWFsaXplZCwgdGhpcyBwcm9wZXJ0eSB3aWxsCgkJCWJlIGEgcmVmZXJlbmNlIHRvIHRoZSBpbnN0YW5jZSBvZiB0aGUgdmlldyBvd25lZCBieSB0aGlzIGNvbnRyb2xsZXIuCgkJKi8KCQl2aWV3OiBudWxsLAoKCQkvLypAcHVibGljCgkJLyoqCgkJCUEgc3RyaW5nIHRoYXQgcmVwcmVzZW50cyB0aGUgdGFyZ2V0IERPTSBlbGVtZW50IGluIHdoaWNoIHRvIHJlbmRlcgoJCQl0aGUgY29udHJvbGxlcidzIHZpZXcuIEJ5IGRlZmF1bHQsIHRoaXMgaXMgc2V0IHRvIHRoZSBzcGVjaWFsCgkJCWlkZW50aWZpZXI6IGAiZG9jdW1lbnQuYm9keSJgLiBJdCBtYXkgYWxzbyBiZSBzZXQgdG8gYSB1bmlxdWUgRE9NCgkJCWVsZW1lbnQgX19pZF9fIGF0dHJpYnV0ZS4KCQkqLwoJCXJlbmRlclRhcmdldDogImRvY3VtZW50LmJvZHkiLAoKCQkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCQkvLyBQUk9URUNURUQgUFJPUEVSVElFUwoKCQkvLypAcHJvdGVjdGVkCgkJX2lzX3ZpZXdfY29udHJvbGxlcjogdHJ1ZSwKCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gQ09NUFVURUQgUFJPUEVSVElFUwoKCQkvLypAcHJvdGVjdGVkCgkJLyoqCgkJCU9uIG9iamVjdCBpbml0aWFsaXphdGlvbiwgZmluZHMgb3IgY3JlYXRlcyB0aGUgYXBwcm9wcmlhdGUKCQkJa2luZCBmb3IgdGhlIHZpZXcgb2YgdGhpcyBjb250cm9sbGVyLgoJCSovCgkJX3ZpZXdfa2luZDogZW55by5jb21wdXRlZChmdW5jdGlvbiAoKSB7CgkJCS8vIHRoZSBvcmlnaW5hbCBkZWZpbml0aW9uIGFzIHN1cHBsaWVkIGJ5IHRoZSBjb250cm9sbGVyJ3MKCQkJLy8gb3duIGRlZmluaXRpb24KCQkJdmFyIHZpZXcgPSB0aGlzLnZpZXc7CgkJCS8vIGlmIGl0IGlzIGEgZnVuY3Rpb24sIHdlIGFzc3VtZSBpdCBpcyBhIGNvbnN0cnVjdG9yCgkJCWlmICgiZnVuY3Rpb24iID09PSB0eXBlb2YgdmlldykgewoJCQkJcmV0dXJuIHZpZXc7CgkJCX0KCQkJLy8gaWYgaXQgaXMgYW4gb2JqZWN0IGxpdGVyYWwsIHdlIGFzc3VtZSBpdCBpcyBhIGRlZmluaXRpb24KCQkJLy8gYW5kIG5vdGUgdGhhdCB3ZSBjcmVhdGUgYW4gYW5vbnltb3VzIGtpbmQgZm9yIHRoZSB2aWV3CgkJCS8vIHNvIGl0IGhhcyBhbGwgb2YgdGhlIG5vcm1hbCBzZXR1cCBvZiBhIGZ1bGwga2luZAoJCQlpZiAoIm9iamVjdCIgPT09IHR5cGVvZiB2aWV3KSB7CgkJCQlpZiAoIXZpZXcubmFtZSkgewoJCQkJCXZpZXcubmFtZSA9IHRoaXMuX21ha2Vfdmlld19uYW1lKCk7CgkJCQl9CgkJCQlyZXR1cm4gZW55by5raW5kKHZpZXcpOwoJCQl9CgkJCS8vIGlmIGl0IGlzIGEgc3RyaW5nLCB3ZSBhdHRlbXB0IHRvIGZpbmQgdGhlIGNvbnN0cnVjdG9yCgkJCS8vIGl0IHNob3VsZCBiZSBwb2ludGluZyB0bwoJCQlpZiAoInN0cmluZyIgPT09IHR5cGVvZiB2aWV3KSB7CgkJCQl2aWV3ID0gZW55by5nZXRQYXRoKHZpZXcpOwoJCQkJaWYgKCF2aWV3LnByb3RvdHlwZS5raW5kTmFtZSkgewoJCQkJCXZpZXcucHJvdG90eXBlLmtpbmROYW1lID0gdGhpcy5fbWFrZV92aWV3X25hbWUoKTsKCQkJCX0KCQkJCXJldHVybiB2aWV3OwoJCQl9CgkJCS8vIGlmIHdlIGdldCBoZXJlLCB3ZSBoYWQgbm90aGluZywgYW5kIGluIHRoYXQgY2FzZSB3ZQoJCQkvLyBjYW4ndCBkbyBhbnl0aGluZwoJCQl0aHJvdyB0aGlzLmtpbmROYW1lICsgIiBjYW5ub3QgaW5pdGlhbGl6ZSB3aXRob3V0IGEgdmFsaWQgdmlldyBkZWZpbmVkIjsKCQl9LCB7Y2FjaGVkOiB0cnVlfSksCgoJCS8vKkBwcm90ZWN0ZWQKCQkvKioKCQkJRmluZHMgdGhlIGFwcHJvcHJpYXRlIGZvcm0gb2YgdGhlIHJlbmRlciB0YXJnZXQuCgkJKi8KCQlfcmVuZGVyX3RhcmdldDogZW55by5jb21wdXRlZChmdW5jdGlvbiAoKSB7CgkJCS8vIHRoZSBvcmlnaW5hbCBzdXBwbGllZCB2YXJpYWJsZSBmb3IgdGhlIHJlbmRlciB0YXJnZXQKCQkJdmFyIHRhcmdldCA9IHRoaXMucmVuZGVyVGFyZ2V0OwoJCQkvLyB3ZSBhdHRlbXB0IHRvIGZpbmQgdGhlIGFjdHVhbCB0YXJnZXQgbm9kZQoJCQl0YXJnZXQgPSBlbnlvLmRvbS5ieUlkKHRhcmdldCkgfHwgZW55by5nZXRQYXRoKHRhcmdldCk7CgkJCWlmICh0YXJnZXQpIHsKCQkJCXJldHVybiB0YXJnZXQ7CgkJCX0KCQkJLy8gaWYgd2UgY2FuJ3QgZmluZCB0aGUgdGFyZ2V0LCB3ZSBjYW4ndCByZW5kZXIgaXQgaW50byBhbnl0aGluZzsKCQkJLy8gYmV0dGVyIHRvIGZpbmQgb3V0IG5vdwoJCQl0aHJvdyB0aGlzLmtpbmROYW1lICsgIiBjYW5ub3QgZmluZCB0aGUgcmVuZGVyIHRhcmdldDogIiArIHRoaXMucmVuZGVyVGFyZ2V0OwoJCX0sICJyZW5kZXJUYXJnZXQiLCB7Y2FjaGVkOiB0cnVlfSksCgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIFBVQkxJQyBNRVRIT0RTCgoJCS8vKkBwdWJsaWMKCQkvKioKCQkJUmVuZGVycyB0aGUgY29udHJvbGxlcidzIHZpZXcgaW50byB0aGUgRE9NLiBJZiB0aGUgdmlldyBpcyBhbHJlYWR5CgkJCWluIHRoZSBET00sIHRoZSBlbGVtZW50IGlzIHJlLXJlbmRlcmVkIGluIHBsYWNlLgoJCSovCgkJcmVuZGVyOiBmdW5jdGlvbiAoKSB7CgkJCS8vIGluc3RhbmNlIG9mIHRoZSB2aWV3CgkJCXZhciB2aWV3ID0gdGhpcy52aWV3OwoJCQl2YXIgdGFyZ2V0ID0gdGhpcy5nZXQoIl9yZW5kZXJfdGFyZ2V0Iik7CgkJCS8vIGlmIHRoZSB2aWV3IGFscmVhZHkgaGFzIGEgRE9NIG5vZGUsIHdlIGRvbid0IG5lZWQgdG8KCQkJLy8gYXR0ZW1wdCB0byByZS1pbnNlcnQgaXQKCQkJaWYgKHZpZXcuaGFzTm9kZSgpKSB7CgkJCQl2aWV3LnJlbmRlcigpOwoJCQl9IGVsc2UgewoJCQkJLy8gb3RoZXJ3aXNlLCB3ZSBuZWVkIHRvIHJlbmRlciBpdCBpbnRvIHRoZSBET00KCQkJCXZpZXcucmVuZGVySW50byh0YXJnZXQpOwoJCQl9CgkJfSwKCgkJLy8qQHB1YmxpYwoJCS8qKgoJCQlJbW1lZGlhdGVseSByZW5kZXJzIHRoZSBjb250cm9sbGVyJ3MgdmlldyBpbnRvIHRoZSBwYXNzZWQtaW4gdGFyZ2V0CgkJCShlaXRoZXIgYSBub2RlIHJlZmVyZW5jZSBvciBzIHN0cmluZyByZXByZXNlbnRpbmcgYSBub2RlIGlkIGF0dHJpYnV0ZQoJCQlpbiB0aGUgRE9NKS4KCQkqLwoJCXJlbmRlckludG86IGZ1bmN0aW9uICh0YXJnZXQpIHsKCQkJLy8gdXBkYXRlIHRoZSByZW5kZXIgdGFyZ2V0IGZvciB0aGUgY29udHJvbGxlcgoJCQl0aGlzLnNldCgicmVuZGVyVGFyZ2V0IiwgdGFyZ2V0KTsKCQkJLy8gbm93IHdlIHJlbmRlciBhcyB1c3VhbAoJCQl0aGlzLnJlbmRlcigpOwoJCX0sCgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIFBST1RFQ1RFRCBNRVRIT0RTCgoJCS8vKkBwcm90ZWN0ZWQKCQljb25zdHJ1Y3RlZDogZnVuY3Rpb24gKCkgewoJCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCQkvLyBlbnN1cmUgd2UgaGF2ZSBjcmVhdGVkIHRoZSB2aWV3IGluc3RhbmNlLCBub3RlIHRoYXQKCQkJLy8gdGhpcyBpcyBkb25lIGhlcmUgX3ByaW9yXyB0byBtaXhpbiBpbml0aWFsaXphdGlvbgoJCQkvLyAod2hpY2ggdGFrZXMgcGxhY2UgYWZ0ZXIgYWxsIGNvbnN0cnVjdGlvbiBpcyBkb25lKQoJCQkvLyBidXQgdGhpcyBhbGxvd3Mgc3Via2luZHMgdG8gb3ZlcmxvYWQgdGhlIGNvbnN0cnVjdGVkCgkJCS8vIG1ldGhvZCB0byBjb250cm9sIHRoZSBmbG93CgkJCXRoaXMuX2NyZWF0ZV92aWV3KCk7CgkJfSwKCgkJLy8qQHByb3RlY3RlZAoJCS8qKgoJCQlDcmVhdGVzIHRoZSBhY3R1YWwgaW5zdGFuY2Ugb2YgdGhlIGNvbnRyb2xsZXIncyB2aWV3LiBTaG91bGQKCQkJYmUgb3ZlcmxvYWRlZCBmb3Igc3BlY2lhbCBiZWhhdmlvcnMuCgkJKi8KCQlfY3JlYXRlX3ZpZXc6IGZ1bmN0aW9uICgpIHsKCQkJLy8gcmV0cmlldmUgdGhlIGNvbnN0cnVjdG9yIGZvciB0aGUgdmlldyBhbmQgaW1tZWRpYXRlbHkKCQkJLy8gaW5zdGFuY2UgaXQgd2hpbGUgYWxzbyB1cGRhdGluZyB0aGUgX3ZpZXdfIHByb3BlcnR5IHRvCgkJCS8vIHRoZSByZWZlcmVuY2UgZm9yIHRoaXMgbmV3IHZpZXcKCQkJdGhpcy5zZXQoInZpZXciLCBuZXcgdGhpcy5nZXQoIl92aWV3X2tpbmQiKSk7CgkJfSwKCgkJLy8qQHByb3RlY3RlZAoJCV9tYWtlX3ZpZXdfbmFtZTogZnVuY3Rpb24gKCkgewoJCQlyZXR1cm4gZW55by51aWQoIl92aWV3X2NvbnRyb2xsZXJfdmlld18iKTsKCQl9CgoJfSk7Cgp9KGVueW8pKTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/data/Collection.js"
Content-Type: application/octet-stream; x-encoding=base64
(function (enyo) {

	//*@public
	/**
		The `enyo.Collection` _kind_ is designed to work with _arrays_ of `enyo.Models`.
		Out of the box they work with `toMany` _relations_ between _models_ with a _defined schema_.
		They have their own default mechanisms for retrieving arrays of data as well as filtering
		its content. They have the ability of being used as an exposed _controller_ or an automatically
		generated relational container depending on your need. Like the other components of the data
		layer in _enyo_ it defaults to functioning with typical _REST_ requests.

		[see enyo.Model](#), [see enyo.Store](#), [see enyo.Source](#)
	*/
	enyo.kind({

		// ...........................
		// PUBLIC PROPERTIES

		//*@public
		name: "enyo.Collection",

		//*@public
		kind: "enyo.Controller",

		//*@public
		/**
			A collection uses the `model` property to know what constructor
			to use when _data_ is handed to it. If an array of records are
			added that are normal data hashes it will convert them using this
			_kind_. It can be a _string_ that will be resolved to a _constructor_
			or a _constructor_ directly.
		*/
		model: "enyo.Model",

		//*@public
		/**
			Like `enyo.Model` an `enyo.Collection` has several statuses it will
			set depending on its current action. The following are the available
			options for status.

			- BUSY.FETCHING
			- BUSY.DESTROYING
			- CLEAN
			- ERROR.SOURCE
			- ERROR.RESPONSE
			- ERROR.TYPE

			[see enyo.Model.status](#)
		*/
		status: enyo.Model.CLEAN,

		//*@public
		/**
			The number of elements in the _collection_.
		*/
		length: 0,

		//*@public
		/**
			Used by the `enyo.Source` in the application to generate the appropriate
			request. By default it uses a simple _REST_ scheme. The `url` is a static
			_string_ that will be post-fixed to the root domain for the `enyo.Source`.
			In more complex setups [see `query`](#) for overloading implications and
			adding dynamic `url` handling possibilities.
		*/
		url: "",

		//*@public
		/**
			A `relation` is an optional _relational_ definition object and must be of
			the type `enyo.toOne` ([see enyo.toOne](#)) as defined in the `enyo.Model`
			documentation. If a `relationKey` is defined it will know to watch for the
			_value change_ notifications from the related _model_ and update accordingly.
		*/
		relation: null,

		//*@public
		events: {
			onModelChanged: "",
			onModelAdded: "",
			onModelsAdded: "",
			onModelRemoved: "",
			onModelsRemoved: "",
			onModelDestroyed: ""
		},

		//*@public
		handlers: {
			onChange: "_modelChanged",
			onDestroy: "_modelDestroyed"
		},

		// ...........................
		// PROTECTED PROPERTIES

		//*@protected
		_store: null,

		// ...........................
		// COMPUTED PROPERTIES

		//*@public
		/**
			This computed property represents the underlying array of _models_.
			The `data` property may be set directly - but it will automatically
			replace all of the current content (if any). [See add](#) if the
			desire is not to replace but extend the current dataset. This
			_computed property_ may be overloaded in more complex scenarios
			requiring filtering and conditionally supplied datasets.
		*/
		data: enyo.computed(function (data) {
			if (data) {
				this.removeAll();
				this.add(data);
			} else {
				return this._store;
			}
		}, "length", {cached: true, defer: true}),

		//*@public
		/**
			Used by `enyo.Source` to generate the appropriate request for
			`fetching`. May be overloaded to produce dynamic _queries_.
		*/
		query: enyo.computed(function () {
			return this.url || this.model.prototype.get("query");
		}),

		// ...........................
		// PUBLIC METHODS

		//*@public
		/**
			Called by the `enyo.Source` for the application and used to
			build query parameters that will be applied as a query-string
			to the resulting request. By default this method does nothing.
			To add or modify parameters already set for the query, use the
			_options.queryParams_ hash (will always exist). These options
			are key-value pairs that will be serialized according to the
			`requestKind` of the `enyo.Source`.
		*/
		buildQueryParams: function (model, options) {
			// look at options.queryParams for a hash to add-to or modify
		},

		//*@public
		/**
			Returns an array of all of the raw datasets for any records
			in the _collection_. May be overloaded to propertly supply
			subsets of data on request. Optional parameter `local` will
			ensure that all _models_ use their local keys (default is to
			use remote keys if they are defined in the schema).
		*/
		raw: function (local) {
			return this.map(function (model) {
				return model.raw(local);
			});
		},

		//*@public
		/**
			Returns a JSON stringified version of the _collection_ array.
			Optional `local` parameter will ensure that _models_ use their
			local keys (default is to use remote keys if defined in the schema).
		*/
		toJSON: function (useLocalKeys) {
			return enyo.json.stringify(this.raw(useLocalKeys));
		},

		//*@public
		/**
			Fetch this _collection_. Accepts an options hash including a
			`success` method and/or an `error` method. A `fetch` request with
			results will _add_ those results to the _collection_. If you want
			to replace the contents use the `replace` options key or call
			`fetchAndReplace`.
		*/
		fetch: function (options) {
			var $options = options? enyo.clone(options): {};
			$options.success = this.bindSafely("didFetch", options || {});
			$options.error = this.bindSafely("didFail", "fetch", options || {});
			this.set("status", enyo.Model.BUSY.FETCHING);
			enyo.store.fetch(this, $options);
		},

		//*@public
		/**
			By default `fetch` will add to the _collection_ and keep any existing
			content. If you wish to replace all content with the results call this
			method using the same options.
		*/
		fetchAndReplace: function (options) {
			var $options = options? enyo.clone(options): {};
			$options.replace = true;
			this.fetch($options);
		},

		//*@public
		/**
			Not typically called directly but overloadable for extensibility.
		*/
		didFetch: function (options, result) {
			var data = this.filterData(result);

			if (options.replace) {
				this.removeAll();
			}
			this.add(data);
			this.startNotifications();
			if (options.success) {
				options.success(options, result);
			}
			this.set("status", enyo.Model.CLEAN);
		},

		//*@public
		/**
			Overload this method for handling fail-states. The `which`
			parameter will be `"fail"`. Sets the `status` to `ERROR.RESPONSE`.
		*/
		didFail: function (which, options) {
			this.set("status", enyo.Model.ERROR.RESPONSE);
		},

		//*@public
		/**
			TODO: Not implemented
		*/
		push: function () {
			enyo.warn("enyo.Collection.push: not currently implemented");
		},

		//*@public
		/**
			TODO: Not implemented
		*/
		pop: function () {
			enyo.warn("enyo.Collection.pop: not currently implemented");
		},

		//*@public
		/**
			TODO: Not implemented
		*/
		shift: function () {
			enyo.warn("enyo.Collection.shift: not currently implemented");
		},

		//*@public
		/**
			TODO: Not implemented
		*/
		unshift: function () {
			enyo.warn("enyo.Collection.unshift: not currently implemented");
		},

		//*@public
		/**
			Returns the index of the requested `value` in the _collection_
			starting from the optional `idx`.
		*/
		indexOf: function (value, idx) {
			return enyo.indexOf(value, this._store, idx);
		},

		//*@public
		/**
			Returns the last index of the `value` in the _collection_ starting
			from the optional `idx`.
		*/
		lastIndexOf: function (value, idx) {
			return enyo.lastIndexOf(value, this._store, idx);
		},

		//*@public
		/**
			TODO: Not implemented
		*/
		splice: function () {
			enyo.warn("enyo.Collection.splice: not currently implemented");
		},

		//*@public
		/**
			Returns a mapped array according to the return value of `fn` and
			the optional `context` to execute `fn` with (default is the _collection_).

			[see enyo.map](#)
		*/
		map: function (fn, context) {
			return enyo.map(this._store, fn, context || this);
		},

		//*@public
		/**
			Returns a filtered array of _models_ from the _collection_ according
			to the `true` or `false` return of `fn` and the optional `context`
			(default is the _collection_).

			[see enyo.filter](#)
		*/
		filter: function (fn, context) {
			return enyo.filter(this._store, fn, context || this);
		},

		//*@public
		/**
			In implementations where the result of a `fetch` may not be the array
			of records and needs to first be filtered overload this method. Return
			the array of records to be supplied during a `didFetch`.
		*/
		filterData: function (data) {
			return data;
		},

		//*@public
		/**
			Returns `true` or `false` whether the _collection_ contains `value`.
		*/
		contains: function (value) {
			return !!~enyo.indexOf(this._store, value);
		},

		//*@public
		/**
			Returns the _model_ at `idx` in the _collection_.
		*/
		at: function (idx) {
			return this._store[idx];
		},

		//*@public
		/**
			Add a single record to the _collection_. The `record` parameter
			may be an instance of `enyo.Model` or an object literal. If it is
			an object literal it will be converted to the _kind_ defined by the
			`model` property of this _collection_.
		*/
		add: function (record) {
			var idx = this._store.length;
			if (enyo.isArray(record)) {
				return this.addMany.apply(this, arguments);
			}
			if (!(record instanceof this.model)) {
				record = new this.model(record);
			}
			record._addCollection(this);
			this._store.push(record);
			this.set("length", this._store.length);
			if (!this._silenced) {
				this.doModelAdded({
					model: record,
					index: idx,
					collection: this
				});
			}
			return idx;
		},

		//*@public
		/**
			TODO: Not implemented
		*/
		addAt: function () {
			enyo.warn("enyo.Collection.addAt: not implemented yet");
		},

		//*@public
		/**
			Accepts an array of records to be added to the _collection_. If the
			records are object literals they will be converted to the _kind_ defined
			by the `model` property of the _collection_.
		*/
		addMany: function (records) {
			var added = [];
			this.silence();
			enyo.forEach(records, function (record) {
				var idx = this.add(record);
				if (!isNaN(idx)) {
					added.push({
						model: this.at(idx),
						index: idx,
						collection: this
					});
				}
			}, this);
			this.unsilence();
			if (added.length) {
				this.doModelsAdded({models: added});
			}
		},

		//*@public
		/**
			Removed `record` from the _collection_ if it exists and emits
			a `onModelRemoved` event.
		*/
		remove: function (record) {
			if (enyo.isArray(record)) {
				return this.removeMany.apply(this, arguments);
			}
			var idx = this.indexOf(record);
			if (!!~idx) {
				this._store.splice(idx, 1);
				this.set("length", this._store.length);
				record._removeCollection(this);
				if (!this._silenced) {
					this.doModelRemoved({
						model: record,
						index: idx,
						collection: this
					});
				}
				return idx;
			}
			return false;
		},

		//*@public
		/**
			Removes all models in the _collection_. Note these _models_ are not
			destroyed or removed from the `enyo.Store`.
		*/
		removeAll: function () {
			var $copy = enyo.clone(this._store);
			this.remove($copy);
		},

		//*@public
		/**
			TODO: Not implemented
		*/
		removeAt: function () {
			enyo.warn("enyo.Collection.removeAt: not implemented yet");
		},

		//*@public
		/**
			Removes all models in the `records` array if they are found in
			the _collection_. Note these _models_ are not destroyed or removed
			from the `enyo.Store`.
		*/
		removeMany: function (records) {
			var removed = [];
			this.silence();
			enyo.forEach(records, function (record) {
				var idx = this.remove(record);
				if (!isNaN(idx)) {
					removed.push({
						model: record,
						index: idx,
						collection: this
					});
				}
			}, this);
			this.unsilence();
			if (removed.length) {
				this.doModelsRemoved({models: removed});
			}
		},

		//*@public
		/**
			Overloaded `setter` that accepts an object literal and
			will apply all keys and values to the _collection_ also the
			normal key and value pair combination.
		*/
		set: function (prop, val) {
			if ("object" === typeof prop) {
				this.stopNotifications();
				for (var key in prop) {
					this.set(key, prop[key]);
				}
				this.startNotifications();
				return this;
			} else if (undefined === val) {
				return;
			} else {
				return this.inherited(arguments);
			}
		},

		//*@public
		/**
			Overloaded `ownerChanged` to prevent the normal handlers
			from executing. Overload with care.
		*/
		ownerChanged: function(old) {
			if (old && old.removeComponent) {
				old.removeComponent(this);
			}
			if (this.owner && this.owner.addComponent) {
				this.owner.addComponent(this);
			}
		},

		//*@public
		/**
			Accepts an array of models to add to the _collection_ at creation.
		*/
		constructor: function (props) {
			this._dirtyModels = [];
			this.inherited(arguments);
			// if the initial parameter is an array we use that as
			// our starting properties
			if (props && props instanceof Array) {
				this._store = this._store? this._store.concat(props): props;
			}
			// initialize our store
			this._store = this._store || [];
			this.length = this._store.length;
			this._initModel();
			if (this._store.length) {
				this._store = this.map(function (record) {
					var $rec = record instanceof this.model? record: new this.model(record);
					$rec._addCollection(this);
					return $rec;
				}, this);
			}
			this._initRelation();
		},

		// ...........................
		// PROTECTED METHODS

		//*@protected
		_initModel: function () {
			var $model = this.model;
			if ("string" === typeof $model) {
				$model = enyo.getPath($model);
			}
			this.model = $model;
			if ($model && this.relation) {
				$model = $model.prototype;
				if (this.relation.inverseKey) {
					if ($model._relations[this.relation.inverseKey]) {
						enyo.mixin(this.relation, $model._relations[this.relation.inverseKey]);
					}
				}
			}
		},

		//*@protected
		_initRelation: function () {
			var key, $rel = this.relation;
			if ($rel) {
				if ((key = $rel.relationKey)) {
					if (!this[key]) {
						this.addObserver(key, this._relationObserver, this);
					} else {
						this._relationObserver(key, null, this[key]);
					}
				}
			}
		},

		//*@protected
		_relationObserver: function (prop, prev, val) {
			var $rel = this.relation, key = $rel.inverseKey;
			if (key === prop && val) {
				this[$rel.relationKey].removeObserver(key, this._relationObserver);
				if ($rel.autoFetch) {
					this.fetch();
				}
			} else if ($rel.relationKey === prop) {
				if (val && $rel.autoFetch) {
					if (val[key]) {
						this.fetch();
					} else {
						val.addObserver(key, this._relationObserver, this);
					}
				}
			}
		},

		//*@protected
		_relationChanged: enyo.observer(function (prop, prev, val) {
			if (val) {
				this._initRelation();
			}
		}, "relation"),

		//*@protected
		_modelChanged: function (sender, event) {
			var idx = this.indexOf(sender);
			this.set("status", enyo.Model.DIRTY);
			if (!!~idx) {
				this.doModelChanged({
					model: sender,
					index: idx,
					collection: this
				});
			}
			return true;
		},

		//*@protected
		_modelDestroyed: function (sender, event) {
			var idx = this.indexOf(sender);
			if (!!~idx) {
				this.remove(sender);
				this.doModelDestroyed({
					model: sender,
					index: idx,
					collection: this
				});
			}
			return true;
		},

		// ...........................
		// OBSERVERS

		_statusChanged: enyo.observer(function (prop, prev, val) {
			if (prev == enyo.Model.DIRTY && val == enyo.Model.CLEAN) {
				enyo.forEach(enyo.clone(this._dirtyModels), function (rec) {
					rec.set("status", enyo.Model.CLEAN);
				});
			}
		}, "status")

	});

})(enyo);
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/data/Model.js"
Content-Type: application/octet-stream; x-encoding=base64
(function (enyo) {

	//*@public
	/**
		Takes an object and returns a _Boolean_ `true` | `false` if
		the object is a `enyo.Model` or _subkind_.
	*/
	enyo.isModel = function (obj) {
		return !! (obj && obj._isModel);
	};
	//*@protected
	var isModel = enyo.isModel;

	//*@protected
	var relationInitializers = {
		autoFetch: function (props) {
			if (!enyo.exists(props.autoFetch)) {
				props.autoFetch = true;
			}
		},
		model: function (props) {
			if (!enyo.exists(props.model)) {
				props.model = enyo.Model || "enyo.Model";
			}
		},
		isOwner: function (props) {
			if (!enyo.exists(props.isOwner)) {
				props.isOwner = true;
			}
		},
		inCommit: function (props) {
			if (!props.isOwner && !enyo.exists(props.inCommit)) {
				props.inCommit = true;
			} else if (!enyo.exists(props.inCommit)) {
				props.inCommit = false;
			}
		},
		inverseKey: function (props) {
			// nop for now
		},
		relationKey: function (props) {
			if (!enyo.exists(props.relationKey)) {
				props.relationKey = "id";
			}
		},
		collection: function (props) {
			if (props._kind == "toOne") {
				return;
			}
			if (!enyo.exists(props.collection)) {
				// we know there exists a model property because it
				// will have already been normalized
				props.collection = enyo.kind({kind: "enyo.Collection", model: props.model});
			}
		}
	};

	//*@protected
	var relationKeys = enyo.keys(relationInitializers);

	//*@protected
	/**
		Handles common features of all relations.
	*/
	var initRelation = function (props) {
		var $props = props;
		enyo.forEach(relationKeys, function (fn) {
			relationInitializers[fn]($props);
		});
	};

	//*@protected
	/**
		Executed in the context of the record.
	*/
	var toOneHandler = function (key, rel, val) {
		var $rec = this[key];
		if (!$rec) {
			if (rel.isOwner) {
				// if the incoming value is an object it is assumed to be
				// the property hash for the record so we create it in-place
				if (enyo.isObject(val)) {
					val[rel.inverseKey] = this;
					$rec = this[key] = new rel.model(val);
					$rec._relationKey = key;
				} else {
					// have to ask the store to look for the correct record first
					// just to be sure if it is already loaded that we use the same
					// one
					var $opts = {};
					$opts.params = {};
					$opts.params[rel.inverseKey] = this[rel.relationKey];
					// note this is a synchronous call because we don't supply
					// the asynchronous success/error callbacks and that it is
					// only a local find when executed this way
					$rec = enyo.store.findOne(rel.model, $opts);
					// if it isn't local already we create it and if its auto
					// fetchable call fetch
					if (!$rec) {
						$rec = new rel.model();
						$rec._relationKey = key;
						$rec.set(rel.inverseKey, this);
						if (rel.autoFetch) {
							$rec.fetch();
						}
					}
				}
			} else {
				// we aren't the owner but should have been handed the actual
				// record
				$rec = this[key] = val;
				// since we aren't the owner of the relation we register the owner
				// as a listener to our events
				this.addDispatchTarget($rec);
			}
		} else {
			// well, this would be odd but possible
			if (rel.isOwner) {
				$rec.didFetch({}, val);
			}
		}
	};

	//*@protected
	/**
		Executed in the context of the record.
	*/
	var toManyHandler = function (key, rel, val) {
		var $rec = this[key];
		if (!$rec) {
			var $pr = rel.collection.prototype.relation || rel;
			$rec = this[key] = enyo.singleton({kind: rel.collection, relation: enyo.clone($pr)});
			$rec._relationKey = key;
			$rec.addDispatchTarget(this);
			if (rel.inverseKey) {
				$rec.set(rel.inverseKey, this);
			}
			if (val) {
				if (enyo.isArray(val)) {
					$rec.add(val);
				} else if (enyo.isObject(val)) {
					$rec.didFetch({}, val);
				}
			}
		} else {
			if (val) {
				$rec.didFetch({}, val);
			}
		}
	};

	// TODO: For the resulting documentation to properly organize these methods
	// in association with enyo.Model we need a more powerful documentation tool,
	// enyo.Model.toOne, etc. is ugly and cumbersome.

	//*@public
	enyo.toOne = function (props) {
		var $props = props;
		$props._kind = "toOne";
		$props.handler = $props.handler || toOneHandler;
		initRelation($props);
		return $props;
	};

	//*@public
	enyo.toMany = function (props) {
		var $props = props;
		$props._kind = "toMany";
		$props.handler = $props.handler || toManyHandler;
		initRelation($props);
		return $props;
	};

	//*@protected
	var defaultFormatter = function (key, value, action, payload) {
		return value;
	};

	//*@protected
	var defaultTypeWrangler = function (attr, key, value, action, payload) {
		if (attr && attr.type && value) {
			if (enyo.isString(attr.type)) {
				// it hasn't been resolved yet
				var $type = enyo.getPath(attr.type);
				if ($type) {
					attr.type = $type;
				}
			}
			switch (action) {
			case "fetch":
				// TODO: All native types need to be added here and tested
				if (attr.type) {
					if (String === attr.type) {
						if (!enyo.isString(value)) {
							value = value && value.toString? value.toString(): String(value);
						}
					} else if (!(value && value instanceof attr.type)) {
						value = new attr.type(value);
					}
				}
				return value;
			case "commit":
				return value.toString();
			}
		}
		return value;
	};

	//*@protected
	var initializers = {
		/**
			If there is a remote key defined we track that and create
			the appropriate map between the local and remote key.
		*/
		remoteKey: function (proto, key, attr) {
			var $keys = proto._remoteKeys || (proto._remoteKeys = {});
			if (attr.remoteKey) {
				// so now we know it is a remote key and we have mapped
				// it to the local key
				$keys[attr.remoteKey] = key;
			}
		},
		type: function (proto, key, attr) {
			// nop for now
		},
		/**
			If a typeWrangler was defined as a string it is attempted to be
			grabbed from the prototype for the model and if it isn't found
			it uses the default.
		*/
		typeWrangler: function (proto, key, attr) {
			if (attr.typeWrangler) {
				if (enyo.isString(attr.typeWrangler)) {
					if (!(attr.typeWrangler = proto[attr.typeWrangler])) {
						enyo.warn("enyo.Model: attribute defined with typeWrangler but no " +
							"such typeWrangler was found on the model prototype -> " +
							key + " in " + proto.kindName + ", using the default typeWrangler");
					}
				}
				if (enyo.isFunction(attr.typeWrangler)) {
					return;
				}
			}
			// we bind it including the attribute has so it can attempt to use
			// the type if it was defined even if it couldn't be resolved at this
			// moment during initialization
			attr.typeWrangler = enyo.bind(null, defaultTypeWrangler, attr);
		},
		/**
			If a formatter was defined as a string it is attempted to be
			grabbed from the prototype for the model and if it isn't found
			it uses the default.
		*/
		formatter: function (proto, key, attr) {
			if (attr.formatter) {
				if (enyo.isString(attr.formatter)) {
					if (!(attr.formatter = proto[attr.formatter])) {
						enyo.warn("enyo.Model: attribute defined with formatter but no " +
							"such formatter was found on the model prototype -> " +
							key + " in " + proto.kindName + ", using the default formatter");
					}
				}
				if (enyo.isFunction(attr.formatter)) {
					return;
				}
			}
			// nothing we can do really except assume that the structure it
			// is coming in as will be correct
			attr.formatter = defaultFormatter;
		},
		/**
			If a relation is define for the attribute we go ahead and
			initialize what we can.
		*/
		relation: function (proto, key, attr) {
			var $rels = proto._relations || (proto._relations = {});
			if (attr.relation) {
				$rels[key] = attr.relation;
			}
		}
	};

	//*@protected
	/**
		The known and available attribute keys. All others are ignored (removed).
		A separate initializer is setup for each of these for modularity and
		(internal) extensibility in the future and easier debugging. They are retrieved
		from the initializers because for future options you only need to add the
		entry there and it will automatically be handled.
	*/
	var attributeKeys = enyo.keys(initializers);

	//*@protected
	var normalizeAttribute = function (proto, key, attr, attrs) {
		var $prop = attrs[key] = attr? enyo.only(attributeKeys, attr): {};
		var $proto = proto;
		var $key = key;
		enyo.forEach(attributeKeys, function (fn) {
			initializers[fn]($proto, $key, $prop);
		});
	};

	//*@protected
	/**
		Attempt to do all the necessary setup on our attributes in a single
		pass as opposed to several. Actions are broken up for modularity,
		better (internal) extensibility in the future and easier debugging.
	*/
	var normalizeAttributes = function (proto, attrs) {
		var key, $prop, $keys = proto._attributeKeys;
		if ($keys) {
			$keys = (proto._attributeKeys = enyo.clone($keys));
		} else {
			$keys = (proto._attributeKeys = []);
		}
		for (key in attrs) {
			if (!~enyo.indexOf(key, $keys)) {
				$keys.push(key);
			}
			$prop = attrs[key] = enyo.clone(attrs[key]);
			normalizeAttribute(proto, key, $prop, attrs);
		}
	};

	//*@protected

	var initRelations = function () {
		var key, $rel, $rels = this._relations;
		// since at load time we can't be sure all the constructors are loaded
		// we have to resort to doing this at runtime
		// TODO: Could register relations globally and have an enyo.ready call
		// that does this once for all known relations as opposed to every time
		// a new model is instanced (with potentially far more overhead)
		for (key in $rels) {
			$rel = $rels[key];
			if ($rel.isOwner) {
				switch ($rel._kind) {
				case "toOne":
					if (!enyo.isFunction($rel.model)) {
						$rel.model = enyo.getPath($rel.model);
					}
					break;
				case "toMany":
					if (!enyo.isFunction($rel.collection)) {
						$rel.collection = enyo.getPath($rel.collection);
					}
				}
			}
		}
	};

	//*@protected
	/**
		Used to breakdown a destroyed model.
	*/
	var breakdown = function (model) {
		// make sure to cleanup any relations that have been instantiated
		// for this where this model was the owner
		var key, $rel, $rels = model._relations;
		for (key in $rels) {
			$rel = $rels[key];
			if ($rel.isOwner) {
				// TODO: For now we just orphan related models but possibly
				// should determine a way to know whether or not to destroy
				// them as well
				// if the relation is a toMany than we should be looking at
				// a collection, since we own that collection, we destroy it
				// but the models will still exist
				if ($rel._kind == "toMany") {
					if (model[key]) {
						model[key].destroy();
						model[key] = null;
					}
				} else if ($rel._kind == "toOne") {
					if (model[key]) {
						model[key].removeDispatchTarget(model);
						if ($rel.inverseKey) {
							model[key].set($rel.inverseKey, null);
						}
						model[key] = null;
					}
				}
			}
		}
		// continue the normal component breakdown for the rest
		enyo.Component.prototype.destroy.call(model);
	};

	//*@protected
	/**
		Initialization routine for all model kinds. Is abstracted such that
		it can be used at run-time for generic models that are attempting to
		infer their own schemas from the data structure they encountered.
	*/
	var initModel = function (proto, props) {

		// the prototype of the given constructor
		var $proto = proto;

		// we break these operations for clarity
		// first we handle the attributes
		// if this is a subkind of another model the
		// attributes hash will already exist and we
		// we need to clone it for posterity
		if ($proto._attributes) {
			$proto._attributes = enyo.clone($proto._attributes);
		} else {
			// otherwise we need to create a new one to work with
			$proto._attributes = {};
		}
		if ($proto._relations) {
			$proto._relations = enyo.clone($proto._relations);
		}
		// the attributes serve as a way to preserve the original
		// definition of a model

		// we take the new attributes and preserve them along with
		// any current values
		enyo.mixin($proto._attributes, props);
		// we want to make sure that no matter what we include the primaryKey
		// attribute in the payload
		if (!($proto.primaryKey in $proto._attributes)) {
			$proto._attributes[$proto.primaryKey] = null;
		}
		// do the rest of the initialization routine on the attributes
		// in a single pass
		normalizeAttributes($proto, $proto._attributes);
	};

	//*@protected
	/**
		When the kind features chain comes across a model definition
		we need to ensure some setup so we do this once for any kind
		up front to be as efficient as possible.
	*/
	enyo.kind.features.push(function (ctor, props) {
		if (!isModel(ctor.prototype)) {
			return;
		}
		var $proto = ctor.prototype;
		// register this kind for use in a store
		enyo.models.add(ctor);
		initModel($proto, props.attributes || {});

		// we are no longer concerned with the attributes property of the
		// properties for this kind and do not wish them to be added to
		// to the kind body so we remove it
		delete $proto.attributes;

		// we attempt to auto generate a url when necessary
		$proto.url = props.url || !$proto.noUrl? ($proto.name || $proto.kindName)
			.replace(/^(.*)\./g, "").toLowerCase(): "";
	});

	//*@protected
	/**
		Without interfering with the construction chain we need to register
		the record with the store. This cannot be done in during the construction
		chain.
	*/
	enyo.kind.postConstructors.push(function () {
		if (this._isModel && enyo.store) {
			enyo.store.initModel(this);
		}
	});

	//*@public
	/**
		The _status_ of an `enyo.Model` will be one of those defined by values in
		`enyo.Model`. They can be checked explicitly against these statuc globals
		(e.g. `enyo.Model.ERROR.TYPE`).
	*/
	var STATES = {};

	//*@public
	/**
		There are multiple error states each with a different meaning.
	*/
	var ERROR = STATES.ERROR = {};

	/**
		When an attribute with a defined `type` is `set` on a model and fails
		to be or become (after being run through the `typeWrangler`) the correct
		`type` the model enters this state.
	*/
	ERROR.TYPE =						0x01;

	/**
		When an error occurs during initialization because the model could not
		determine the _schema_ for attributes based on their definition or implied
		from `defaults` or data supplied to the constructor the model enters this
		state.
	*/
	ERROR.SCHEMA =						0x02;

	/**
		When a model receives a bad response from the `enyo.Source` for the application
		it will enter this state.
	*/
	ERROR.RESPONSE =					0x03;

	/**
		When a model attempts to execute an action against a remote and the store does
		not exist or the source is unavailable it will enter this state.
	*/
	ERROR.SOURCE =						0x04;

	//*@public
	/**
		When the model is attempting to fetch, commit or destroy it will be in a busy
		state, one of the following.
	*/
	var BUSY = STATES.BUSY = {};

	/**
		When the model is in the process of fetching data from the `enyo.Source` for the
		application it will enter this state.
	*/
	BUSY.FETCHING		=				0x11;

	/**
		When the model is in the process of committing data to the `enyo.Source` for the
		application it will enter this state.
	*/
	BUSY.COMMITTING		=				0x12;

	/**
		When the model is in the process of being destroyed it will be in this state.
	*/
	BUSY.DESTROYING		=				0x13;

	//*@public
	/**
		The CLEAN state implies that the model has had values applie to it but they
		are synchronized with the given `source` and does not need to be committed.
		This is also true of a model whose values were set during construction or if
		it had default values defined and applied.
	*/
	var CLEAN = STATES.CLEAN =			0x21;

	//*@public
	/**
		If a model has been destroyed it will no longer exist in the `enyo.store` or
		in the remote. Changes will not be tracked.
	*/
	var DESTROYED = STATES.DESTROYED =	0x31;

	//*@public
	/**
		A model will be in the NEW state if it is recently created and has had no
		values applied to it that are in the defined schema (either via attributes
		explicitly or defaults/initial values implicitly). If default values are used
		during setup or values applied the model will not be in the NEW state.
	*/
	var NEW = STATES.NEW =				0x41;

	//*@public
	/**
		A model will be in the DIRTY state if it has a defined schema and an attribute
		of the schema has been modified and it needs to be synchronized. The only
		exception is if a model is created and uses default values it will not be in
		the DIRTY state until a modification takes place.
	*/
	var DIRTY = STATES.DIRTY =			0x51;

	//*@public
	/**
		The `enyo.Model` _kind_ is, simply put, an object that represents data. It
		has a common and abstracted interface by which it can be manipulated and
		persisted. When used in tandem with `enyo.Collection`, `enyo.Store` and
		`enyo.Source` it can be a well defined _schema_ representing _objects_ that
		in your application. They can have an implicitly derived _schema_ or an
		explicitly defined _schema_ according to your implementation and needs.

		[see enyo.Collection](#), [see enyo.Store](#), [see enyo.Source](#)

		__TODO: There are still features left to be implemented including but not
		limited to the following:__

		- field validators and error state based on validation
		- optional change set commits (as opposed to every field on every commit)

		A _schema_ for a _model_ is defined in the `attributes` property (hash). Each
		key in this hash represents an entry in the _schema_. It can be better defined
		with the _schema options_ hash. The _schema options_ are a set of properties
		that describe what this _attribute_ is and in some cases how it interacts with
		other _schemas_. The _schema options_ are as follows.

		###### [Schema Options](id:schema-options)

		##### [`remoteKey`](id:remoteKey) - _String_

		In cases where the remote key should differ from the local key specify what
		`remoteKey` should be paired with this local key. For example, a table in your
		RDBMS _schema_ has an index _my\_table\_id_ but this is ugly to reference in
		the client-side source you could map it to (local) _id_.

		```
		attributes: {
			// the key defined here is the attribute we want to use
			// in our application
			id: {
				// here we tell it that the remote key it should use
				// for this will have a different name
				remoteKey: "my_table_id"
			}
		}
		```

		##### [`type`](id:type) - _Function (object constructor)_ or _String_

		When specified the incoming data will be passed to the constructor defined by
		`type`. For example, if your database provides a value representing a date but
		is stored as a unix timestamp you could specify `Date` as the `type` and when
		data is retrieved/added for this key it will be passed to the _constructor_ for
		the named `type`. This can be a constructor or a String placeholder for a
		constructor. __If the `type` is a custom _kind_ that requires a different
		approach than simply supplying the data to the constructor you will have to use
		the `typeWrangler` ([see typeWrangler](#typeWrangler)) _schema option_ for type
		coercion.__ Note that if a `type` is specified but coercion fails the _model_
		will __throw an exception__. This should be used, however, as a sanity check and
		can help to find flaws in _schema_ designs both local and remote.

		```
		attributes: {
			published: {
				type: Date
			}
		}
		```

		##### [`typeWrangler`](id:typeWrangler) - _Function_ or _String_

		The `typeWrangler` _method_ or _string_ representing the _method_ on the model
		that, when supplied, will be called once for incoming data (via _fetch_) and
		once for outgoing data (via _commit_). The _function_ takes the form of
		`function (key, value, action, payload)` where `key` is the _attribute_ in the
		schema, `value` is the entry for that `key` in the `payload`, `action` is either
		"fetch" or "commit" where "fetch" means the data was just fetched and "commit"
		means the _model_ is about to be committed, and `payload` is the mutable full
		dataset (coming in or going out). In both cases it should return the data as
		the _correct type_ (specified by the `type` [[see type](#type)] _schema option_).

		##### [`formatter`](id:formatter) - _Function_ or _String_

		There are times when incoming data will be in an unuseful form or structure.
		While both the `enyo.Source` and `enyo.Model` _kinds_ supply a generic `filter`
		method for incoming data it may be more convenient to supply a separate data
		formatter for a particular key. For example, if you're using a public API to
		retrieve data and it comes back as a nested object in which the useful data is
		deeper in the tree you can use this method to retrieve just the useful
		information. This _method_ must return the desired data. If a `typeWrangler`
		([see typeWrangler](#typeWrangler)) exists for this _schema option_ the
		`formatter` will be executed first. The _function_ takes the form of
		`function (key, value, action, payload)` where `key` is the _attribute_ in the
		schema, `value` is the entry for that `key` in the `payload`, `action` is either
		"fetch" or "commit" where "fetch" means the data was just fetched and "commit"
		means the _model_ is about to be committed, and `payload` is the mutable full
		dataset (coming in or going out).


		##### [`relation`](id:relation) - _Enum_
		__TODO: While relationships do function their API is considered _volatile_ until
		a future release and is only partially implemented in `enyo.Store`
		([see enyo.Store](#))__

		Relationships between _schemas_ (_models_) are defined by supplying _relation
		options_ to the 2 known _relation types_ `enyo.toOne` and `enyo.toMany` and.
		These _relation types_ are _functions_ that take your
		_relation options_ and uses them to create an appropriate handler for the
		`relation`.

		```
		attributes: {
			comments: {
				relation: enyo.toMany({
					autoFetch: true,
					inCommit: false,
					inverseKey: "ownerId",
					relationKey: "id",
					model: "models.CommentModel"
				})
			}
		}
		```

		###### [Relation Options](id:relation-options)

		##### [`autoFetch`](id:autoFetch) - _Boolean_ (_default_ `true`)

		When the incoming data for a particular relation is not included in the
		_dataset_ and needs to be _fetched_ separately this _relation option_ can be set
		to `true`. Sometimes it is desirable to post-pone fetching of a relationship
		that is not included in the original payload and in such a case `autoFetch`
		should be explicitly set to `false`.

		##### [`model`](id:model) - _Function (object constructor)_ or _String_

		This is the _constructor_ or _string_ placeholder for the _constructor_ of the
		_kind_ of _model_ to use for this `relation`. If `model` is supplied for a
		`enyo.toMany` `relation` it will use a generic `enyo.Collection` with that
		_model kind_. If there is a custom _collection kind_ use the `collection`
		_relation option_ instead ([see collection](#collection)).

		##### [`collection`](id:collection) - _Function (object constructor)_ or _String_

		This is only used in `enyo.toMany` `relations` and will be
		ignored in `enyo.toOne` `relations`. This _constructor_ or _string_ placeholder
		for the _constructor_ of the _kind_ of _collection_ to use for this `relation`.
		When supplying a custom _collection kind_ that _kind_ is responsible for being
		configured to use the correct _model kind_.

		##### [`inCommit`](id:inCommit) - _Boolean_ (_default_ `false`)

		Set this _relation option_ to `true` to include this `relation` in a _commit_
		payload to the _source_.

		##### [`inverseKey`](id:inverseKey) - _String_

		The _string_ representing the key on the _related model_ to designate the
		relationship (if any).

		##### [`relationKey`](id:relationKey) - _String_ (_default_ `id`)

		The _string_ representing the key on this _model_ to compare against the
		`inverseKey` (if any).

		##### [`isOwner`](id:isOwner) - _Boolean_ (_default_ `true`)

		In relationships it is important to indicate the _direction of ownership_. The owner
		of the relationship will receive the _change events_ of the related _model(s)_
		whereas the _non-owner_ will not. Also note that if `isOwner` is `false` the `inCommit`
		option is automatically set to `false` even if it was explicitly set to `true`.

		---

		Here is an example of a completed `enyo.Model` _schema_ definition. __Note not
		every _model_ needs to use all of these features.__

		```
		enyo.kind({
			name: "models.Contact",
			kind: "enyo.Model",
			attributes: {
				// example of mapping a remote key to a different
				// local key and ensuring it will be a number
				id: {
					remoteKey: "contact_id",
					type: Number
				},
				// example of creating a meta attribute that has no
				// direct mapping to a remote key but combines two
				// remote keys into one locally and on commit will
				// parse them and map them back to the appropriate
				// structure in the payload, here we have the method
				// inlined in the attributes definition but it could
				// have been a string representing the name of the method
				// on the model itself
				name: {
					type: String,
					typeWrangler: function (field, action, payload) {
						switch (action) {
						case "fetch":
							// field will be undefined on fetch since it
							// isn't represented by an actual key in the dataset
							return payload.first_name + " " + payload.last_name;
							break;
						case "commit":
							// field will be defined as whatever the last state
							// of the record was in the application but returning
							// a value in this case will do nothing since we don't
							// have an actual key called "name" so we modify the
							// payload directly
							var parts = field.split(" ");
							payload.first_name = parts[0];
							payload.last_name = parts[1];
							// returning anything won't do anything but for clarity
							// we return undefined
							return undefined;
							break;
						}
					}
				},
				// example of a meta association with other data that is
				// not included in the payload associated with a fetch request
				// and note we do not wish to include it in a payload for commit
				// on this model
				comments: {
					relation: enyo.toMany({
						autoFetch: true,
						inCommit: false,
						inverseKey: "ownerId",
						relationKey: "id",
						model: "models.CommentModel"
					})
				}
			}
		});
		```
	*/
	enyo.kind({

		// ...........................
		// PUBLIC PROPERTIES

		//*@public
		name: "enyo.Model",

		//*@public
		kind: "enyo.MultipleDispatchComponent",

		//*@public
		/**
			The defined _schema_ for this _model_. If not defined the _model_
			will attempt to infer the the _schema_ from the structure of the
			data passed to it at construction time.
		*/
		attributes: null,

		//*@public
		/**
			The `defaults` hash of _default_ values to be supplied to the
			_model_ upon instantiation. When no _schema_ (attributes) are
			explicitly defined but defaults are, the _schema_ will be infered
			from these properties. They are in key-value pairs where the value
			can be of any type, even a _function_. If the _value_ is a _function_
			it will be executed during initialization under the context of the
			model (as `this`) and is expected to __return the default value for
			its assigned property__.
		*/
		defaults: null,

		//*@public
		/**
			The `url` property is a static root for this particular _model_.
			In a system with a simple REST backend with a 1 to 1 mapping of
			client _model_ to backend server/service the `url` could be of
			the form `/models/artist`. It can also be used in more complex
			scenarios. Note this `url` property is appended to the _domain url_
			genereated by the `enyo.Source` for the `enyo.Store` of the current
			application. If the `noUrl` property is set to `false` and no
			`url` is specified it will be automatically generated based on the
			name of the _model kind_.
		*/
		url: "",

		//*@public
		/**
			The `status` of the model is defined by a fixed-set of enumerated
			values. See documentation for `enyo.Model` states.
		*/
		status: NEW,

		//*@public
		/**
			A simple _Boolean_ flag indicating whether or not the current _model_
			is a locally created _new_ record. Once a _model_ has been _committed_
			or was _fetched_ this will be `false`.
		*/
		isNew: true,

		//*@public
		/**
			The _String_ representing which attribute to use as the indexable
			primary key for this _model_. Default is `id`.
		*/
		primaryKey: "id",

		//*@public
		/**
			In some cases correct remote calls are generated by some other means
			than the normal url generation. In those cases this _Boolean_ flag needs
			to be set to `true`. Default is `false`.
		*/
		noUrl: false,

		//*@public
		/**
			The `enyo.Model` will generate its own events. When a defined `attribute`
			of the _model schema_ is modified an event will be emitted to any
			listeners with an event payload with the `changed` property with the
			changeset, the `previous` property with the previous values of those in
			the changeset and the `model` property that is a reference to this _model_.

			When the _model_ is destroyed it will also emit an event with the property
			`model` that is a reference to this _model_ (even though it will already
			have been destroyed).
		*/
		events: {
			onChange: "",
			onDestroy: "",
			onError: ""
		},

		//*@public
		handlers: {
			onChange: "_relationChanged",
			onModelChanged: "_relationChanged"
		},

		//*@public
		statics: STATES,

		// ...........................
		// PROTECTED PROPERTIES

		_isModel: true,
		_collections: null,
		_relations: null,
		_attributes: null,
		_attributeKeys: null,
		_previous: null,
		_changed: null,
		_euuid: null,
		_isChild: false,
		_defaultModel: false,
		_noApplyMixinDestroy: true,

		// ...........................
		// COMPUTED PROPERTIES

		//*@public
		/**
			Used internally by `enyo.Source` to generate an appropriate request `url`.
			Overload this method in custom setups.
		*/
		query: enyo.computed(function () {
			return !this.noUrl? this.get("url") + "/" + this.get(this.primaryKey) || "": "";
		}),

		// ...........................
		// PUBLIC METHODS

		//*@public
		/**
			Overload the `filterData` method which takes a single parameter, the
			incoming payload from a _fetch_ request, and return it having made
			any necessary modifications to it. __This method is executed on the
			the top-level payload prior to individual formatters for defined
			attributes in a known schema__.
		*/
		filterData: function (data) {
			return data;
		},

		//*@public
		/**
			When the `enyo.Source` is constructing the request for this _model_
			(regardless of the action) it will call this method which you can
			overload to add custom parameters to the `queryParams` hash of the
			`options` parameter. These parameters are key-value pairs that are
			used to generate the options in a query string in default setups. It
			could also be used for other purposes in overloaded and custom setups.
		*/
		buildQueryParams: function (model, options) {
			// the options parameter will have a hash at property queryParams
			// that can be modified directly by adding properties or using
			// enyo.mixin for example
		},

		// ...........................
		// ASYNCHRONOUS METHODS

		//*@public
		/**
			The model should execute a _commit_ of its current state. The optional
			_options_ hash can have a `success` method and/or an `error` method
			to be executed on the appropriate result asynchronously.
		*/
		commit: function (options) {
			var $options = options? enyo.clone(options): {};
			$options.postBody = this.raw();
			this.set("status", BUSY.COMMITTING);
			this.exec("commit", $options);
		},

		//*@public
		/**
			The model should execute a _fetch_. The optional _options_ hash can
			have a `success` method and/or an `error` method to be executed on the
			appropriate result asynchronously.
		*/
		fetch: function (options) {
			this.set("status", BUSY.FETCHING);
			this.exec("fetch", options);
		},

		//*@public
		/**
			The model should execute a _destroy_ that destroys it in the client
			and also (by default) a _DELETE_ request to the _source_. The optional
			_options_ hash may have a `success` method and/or an `error` method to
			be executed on the appropriate result asynchronously.
		*/
		destroy: function (options) {
			this.set("status", BUSY.DESTROYING);
			this.exec("destroy", options);
		},

		//*@public
		/**
			While this method should not be executed directly it is overloadable
			in custom setups.
		*/
		exec: function (action, options) {
			if (enyo.store) {
				var $options = options? enyo.clone(options): {};
				$options.success = this.bindSafely("did" + enyo.cap(action), options || {});
				$options.error = this.bindSafely("didFail", action, options || {});
				enyo.store[action](this, $options);
			} else {
				this.set("status", ERROR.SOURCE);
			}
		},

		// ............................

		//*@public
		/**
			While this method should not be executed directly it is overloadable
			in custom setups. Note that many of the details of this implementation
			make _models_ work properly and great care should be taken when modifying
			it or you may encounter unexpected or unpredictable results. The third
			parameter can be set to true if you do not wish the result to be passed
			through the filter.
		*/
		didFetch: function (options, result, noFilter) {
			var $data = noFilter? (result || {}): this.filterData(result || {});
			var $attrs = this._attributes;
			var rem, loc, $prop, $val, $rel, $rels = this._relations;
			var queue = [], $fn;
			// ensure that no events or notifications propagate while we are
			// iterating over these entries in the result set
			this.silence();
			this.stopNotifications();
			for (loc in $attrs) {
				$prop = $attrs[loc];
				rem = $prop.remoteKey || loc;
				$val = $data[rem] || $data[loc];
				$rel = $rels[loc];

				// first we run it through its formatter to ensure it is
				// of the appropriate structure
				$val = $prop.formatter.call(this, loc, $val, "fetch", $data);
				// now we run it through the typeWrangler to ensure that
				// the type conversion takes place if necessary
				$val = $prop.typeWrangler.call(this, loc, $val, "fetch", $data);
				// note that validation of the type will take place when
				// the value is set, not here (single entry point of failure)

				// if this attribute is a relation it needs to be handled
				// separately and we need to postpone it until we have already
				// set all of the local props to ensure if one is needed by the
				// related model it will be available
				if ($rel) {
					queue.push(enyo.bind(this, $rel.handler, loc, $rel, $val));
					continue;
				}

				// otherwise we simply set the property directly
				this.set(loc, $val);
			}

			// now we need to make sure we are adding all of the extraneous data
			// (if any) as well from the payload
			for (rem in $data) {
				if (this.isAttribute(rem)) {
					continue;
				}
				this.set(rem, $data[rem]);
			}

			// now we execute any queued relation handlers
			while (queue.length) {
				$fn = queue.pop();
				$fn();
			}

			this.set("isNew", false);
			this.set("status", CLEAN);
			this.unsilence();
			this.startNotifications();
			this._flushChanges();
			if (options && options.success) {
				options.success(result);
			}
		},

		//*@public
		/**
			While this method should not be executed directly it is overloadable
			in custom setups.
		*/
		didCommit: function (options, result) {
			if (result && "object" === typeof result) {
				this.set(result);
			}
			this.set("isNew", false);
			this.set("status", CLEAN);
			this._changed = {};
			for (var key in this._relations) {
				if (this._relations[key].isOwner) {
					if (this[key]) {
						this[key].set("status", CLEAN);
					}
				}
			}
		},

		//*@public
		/**
			While this method should not be executed directly it is overloadable
			in custom setups.
		*/
		didDestroy: function (options, result) {
			this.set("status", DESTROYED);
			this.doDestroy({model: this});
			breakdown(this);
		},

		//*@public
		/**
			While this method should not be executed directly it is overloadable
			in custom setups. The `action` parameter is one of `"fetch"`, `"destroy"`,
			`"commit"` or `"update"` depicting which action failed.
		*/
		didFail: function (action, options, result) {
			this.set("status", ERROR.RESPONSE);
		},

		//*@public
		/**
			Returns an _Object_ literal that represents the JSON parsable form
			of the _model_ in its current state. Takes an optional _Boolean_ that
			if `true` will return the hash with local (client) keys but defaults
			to remote keys if they are defined such that it could be used as a
			payload to the _source_. If no remote keys are defined it defaults to
			using the local (client) keys.

			// TODO: Needs to use the typeWrangler and formatter
		*/
		raw: function (local) {
			var $attrs = this._attributes;
			var key, rem, $rel, $irel, $rels = this._relations, $ret = {}, $rkeys = enyo.keys($rels);
			if (local) {
				$ret = enyo.only(this._attributeKeys, enyo.except($rkeys), this);
			} else {
				for (key in $attrs) {
					rem = $attrs[key].remoteKey || key;
					if (!!~enyo.indexOf(key, $rkeys)) {
						continue;
					}
					$ret[rem] = this.get(key);
				}
			}
			// now to include any relations that are set to be included
			for (key in $rels) {
				$rel = $rels[key];
				$irel = this[key] && $rel.inverseKey? this[key]._relations[$rel.inverseKey]: {};
				if (!$rel.isOwner) {
					continue;
				}
				if (!enyo.exists(this[key]) || (!$rel.inCommit && !$irel.inCommit)) {
					continue;
				}
				$ret[local? key: $attrs[key].remoteKey || key] = this[key].raw(local);
			}
			return $ret;
		},

		//*@public
		/**
			Returns the JSON parsed _string_ value for this _model_ in its
			current state as would be appropriate to send in a payload to the _source_.
			If the optional _local_ is `true` then it will return with local keys
			instead of remote keys (if any).
		*/
		toJSON: function (local) {
			return enyo.json.stringify(this.raw(local));
		},

		//*@public
		/**
			Returns _Boolean_ `true` or `false` for whether the _prop_
			is an _attribute_ of the _schema_ for this _model_.
		*/
		isAttribute: function (prop) {
			return !!~enyo.indexOf(prop, this._attributeKeys);
		},

		//*@public
		/**
			Returns _Boolean_ `true` or `false` for whether the _prop_
			is known by the _model schema_ as a _remoteKey_.
		*/
		isRemoteAttribute: function (prop) {
			return prop in this._remoteKeys;
		},

		//*@public
		/**
			Returns _Boolean_ `true` or `false` for whether _prop_
			is a _relation_ or not.
		*/
		isRelation: function (prop) {
			return !! (prop in this._relations);
		},

		//*@public
		/**
			Retrieve the previous value for _prop_ (when it exists).
		*/
		previous: function (prop) {
			return this._previous[prop];
		},

		//*@public
		/**
			This is a specially overloaded version of `enyo.Object.set`. It
			takes either a _hash_ or a _string_ and _value_ combined as its
			parameters.
		*/
		set: function (prop, val) {
			if (enyo.isObject(prop)) {
				this.silence();
				this.stopNotifications();
				for (var key in prop) {
					this.set(key, prop[key]);
				}
				this.startNotifications();
				this.unsilence();
				this._flushChanges();
				return this;
			} else if (this.isRelation(prop)) {
				return this.setRelation(prop, val);
			} else {
				this.inherited(arguments);
				if (prop != this.primaryKey && this.isAttribute(prop)) {
					this.set("status", DIRTY);
				}
				return this;
			}
		},

		//*@public
		/**
			Used to set values for a given (and defined) _relation_ for an _attribute_
			of the _schema_. Do not call this method directly. It may be overloaded in
			non-standard use-cases.
		*/
		setRelation: function (prop, val) {
			var $rel = this._relations[prop];
			$rel.handler.call(this, prop, $rel, val);
			this.set("status", DIRTY);
			return this;
		},

		//*@public
		/**
			The constructor needs to be able to properly initialize the
			internal (protected) properties but is also responsible for
			determining if the _schema_ is defined and if not attempt to
			implicitly derive it from the passed in values. If there is
			no _schema_ defined and no initial _values_ passed in much of
			its functionality will not work as intended.
		*/
		constructor: function (values, recursing) {
			// first we have a local reference to the original values
			// passed into the object (if there were any)
			var $values = values;
			var $defaults = this.defaults;
			// here we handle defaults so we can attempt both scenarios
			// (incoming values and default values) at the same time
			if (!recursing) {
				if ($defaults) {
					this.defaults = null;
					// we go ahead and look for functions in the defaults and execute them
					// now so we can keep all of this handling in one place
					for (var key in $defaults) {
						if (enyo.isFunction($defaults[key])) {
							$defaults[key] = $defaults[key].call(this);
						}
					}
					$values = $values? enyo.mixin($values, $defaults, true): $defaults;
				}
				// initialize our relations (if any)
				initRelations.call(this);
			}
			// we set the object to undefined so we can allow the constructor
			// chain to continue without automatically applying these values
			// to this record
			if (!recursing) {
				values = undefined;
				// allow the remaining constructors to complete their initialization
				this.inherited(arguments);
				// initialize our variables
				this._collections = [];
				this._previous = {};
				this._changed = {};
			}
			// now we can safely apply any initial values if we have some
			// note that if there are values we are no longer a new status
			// we are clean because we aren't empty
			if ($values) {
				// if we have a defined structure we adhere to the structure
				// otherwise we implicitly derive the structure but no special
				// relationships
				if (this._attributeKeys.length > 1) {
					this.didFetch({}, $values, true);
				} else if (recursing) {
					// this is an error state because we did not determine any
					// schema for the values passed in
					this.set("state", ERROR.SCHEMA);
					return;
				} else {
					// will attempt to figure out what our schema should be with
					// all defaults
					var $schema = {};
					// we create an object with all of the keys but no values so
					// it does not interfere with the initialization
					enyo.forEach(enyo.keys($values), function (key) {
						$schema[key] = null;
					});
					// in case it is useful later we denote this record as being a model
					// whose schema was derived implicitly and not defined explicitly
					this._defaultModel = true;
					// run the new schema through initialization routine
					initModel(this, $schema);
					// now rerun this same constructor with the new schema already defined
					return this._constructor($values, true);
				}
				// we set our status to clean because we have values
				this.status = CLEAN;
				// clear our changed hash
				this._changed = {};
				// set our previous up initially
				this._previous = enyo.only(this._attributeKeys, this);
				// we flush notifications
				this.startNotifications();
			}
		},

		//*@public
		/**
			This method was deliberately overloaded to avoid a deeper nested
			call in the inheritance stack. It is not recommended to overload it
			again unless you know exactly what you are doing.
		*/
		ownerChanged: function(old) {
			if (old && old.removeComponent) {
				old.removeComponent(this);
			}
			if (this.owner && this.owner.addComponent) {
				this.owner.addComponent(this);
			}
		},

		// ...........................
		// PROTECTED METHODS

		//*@protected
		/**
			Used internally to register any collections that might have
			a reference to this model in it. Thus ensuring that each collection
			will receive any events propagated from this _model_.
		*/
		_addCollection: function (col) {
			if (!~enyo.indexOf(col, this._collections)) {
				this._collections.push(col);
				this.addDispatchTarget(col);
				for (var key in this._relations) {
					if (col[key]) {
						this.stopNotifications(true);
						this.set(key, col[key]);
						this.set("status", CLEAN);
						this.startNotifications(true);
					}
				}
			}
		},

		//*@protected
		/**
			Used internally to remove registered collections from the _model_.
		*/
		_removeCollection: function (col) {
			var idx = enyo.indexOf(col, this._collections);
			if (!!~idx) {
				this._collections.splice(idx, 1);
				this.removeDispatchTarget(col);
			}
		},

		//*@protected
		/**
			Used at the appropriate time to emit a _onChange_ event with the
			changeset, previous values and a reference to the model that changed
			(this _model_).
		*/
		_flushChanges: function () {
			if (!this._silenced) {
				this.doChange({
					previous: this._previous,
					changed: this._changed,
					model: this
				});
			}
		},

		//*@protected
		_relationChanged: function (sender, event) {
			var $m = event.model || sender, key = $m._relationKey;
			if (!key && event.collection) {
				$m = event.collection;
				key = $m._relationKey;
			}
			if (key) {
				if (this[key] == $m) {
					this._changed[key] = $m;
					this.set("status", DIRTY);
				}
			}
		},

		//*@protected
		/**
			Retrieves the attribute object for a given attribute `property`
			if it exists.
		*/
		_attributeFor: function (prop) {
			return this._attributes[prop];
		},

		// ...........................
		// OBSERVERS

		//*@protected
		/**
			Spies on all change notifications and adds to the changeset for
			the model. It will execute a _flushChanges_ that will emit the
			_onChange_ event if the _model_ isn't currently silenced.
		*/
		_attributeSpy: enyo.observer(function (prop, prev, val) {
			var attr;
			if ((attr = this._attributeFor(prop)) || this.isRelation(prop)) {
				// TODO: Type checking has been temporarily removed
				// and should probably be added as validation instead
				this._previous[prop] = prev;
				this._changed[prop] = val;
				this._flushChanges();
			}
		}, "*"),

		//*@protected
		_statusChanged: enyo.observer(function (prop, prev, val) {
			if (val == DIRTY) {
				enyo.forEach(this._collections, function (col) {
					if (!~enyo.indexOf(col._dirtyModels, this)) {
						col._dirtyModels.push(this);
					}
				}, this);
			} else if (val == CLEAN) {
				enyo.forEach(this._collections, function (col) {
					var idx = enyo.indexOf(col._dirtyModels, this);
					if (idx >= 0) {
						col._dirtyModels.splice(idx, 1);
					}
				}, this);
			}
		}, "status")

	});

})(enyo);
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/data/ModelController.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uIChlbnlvKSB7CgkKCXZhciBfc2VsZWN0ZWQgPSAvXnNlbGVjdGVkLzsKCQoJZW55by5raW5kKHsKCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gUFVCTElDIFBST1BFUlRJRVMKCgkJLy8qQHB1YmxpYwoJCW5hbWU6ICJlbnlvLk1vZGVsQ29udHJvbGxlciIsCgoJCS8vKkBwdWJsaWMKCQltb2RlbDogbnVsbCwKCQkKCQkvLypAcHVibGljCgkJZXZlbnRzOiB7CgkJCW9uTW9kZWxTZWxlY3RlZDogIiIsCgkJCW9uTW9kZWxEZXNlbGVjdGVkOiAiIgoJCX0sCgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIFBST1RFQ1RFRCBQUk9QRVJUSUVTCgoJCS8vKkBwdWJsaWMKCQlraW5kOiAiZW55by5Db250cm9sbGVyIiwKCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gQ09NUFVURUQgUFJPUEVSVElFUwoKCQkvLypAcHJvdGVjdGVkCgkJX2F0dHJpYnV0ZUtleXM6IGVueW8uY29tcHV0ZWQoZnVuY3Rpb24gKCkgewoJCQlyZXR1cm4gdGhpcy5tb2RlbD8gdGhpcy5tb2RlbC5fYXR0cmlidXRlS2V5czogbnVsbDsKCQl9LCAibW9kZWwiLCB7Y2FjaGVkOiB0cnVlLCBkZWZlcjogdHJ1ZX0pLAoKCQkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCQkvLyBQVUJMSUMgTUVUSE9EUwoKCQkvLypAcHVibGljCgkJaXNBdHRyaWJ1dGU6IGZ1bmN0aW9uIChwcm9wZXJ0eSkgewoJCQlyZXR1cm4gdGhpcy5tb2RlbD8gdGhpcy5tb2RlbC5pc0F0dHJpYnV0ZShwcm9wZXJ0eSk6IGZhbHNlOwoJCX0sCgoJCS8vKkBwdWJsaWMKCQlnZXQ6IGZ1bmN0aW9uIChwcm9wZXJ0eSkgewoJCQlpZiAoIm1vZGVsIiA9PT0gcHJvcGVydHkgfHwgIXRoaXMuaXNBdHRyaWJ1dGUocHJvcGVydHkpKSB7CgkJCQlyZXR1cm4gdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCQkJfQoJCQlyZXR1cm4gdGhpcy5tb2RlbD8gdGhpcy5tb2RlbC5nZXQocHJvcGVydHkpOiB1bmRlZmluZWQ7CgkJfSwKCgkJLy8qQHB1YmxpYwoJCXNldDogZnVuY3Rpb24gKHByb3BlcnR5LCB2YWx1ZSkgewoJCQlpZiAoIm1vZGVsIiAhPT0gcHJvcGVydHkgJiYgIXRoaXMuaXNBdHRyaWJ1dGUocHJvcGVydHkpKSB7CgkJCQlyZXR1cm4gdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCQkJfQoJCQlpZiAoIm1vZGVsIiA9PT0gcHJvcGVydHkpIHsKCQkJCXZhciAkbW9kZWwgPSB0aGlzLm1vZGVsOwoJCQkJaWYgKCRtb2RlbCkgewoJCQkJCXRoaXMuX3JlbW92ZU1vZGVsKCRtb2RlbCk7CgkJCQl9CgkJCQlpZiAoKCRtb2RlbCA9IHZhbHVlKSkgewoJCQkJCXRoaXMuc3RvcE5vdGlmaWNhdGlvbnMoKTsKCQkJCQl0aGlzLl9pbml0TW9kZWwoJG1vZGVsKTsKCQkJCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCQkJCXRoaXMuc3RhcnROb3RpZmljYXRpb25zKCk7CgkJCQkJcmV0dXJuIHRoaXM7CgkJCQl9CgkJCX0KCQkJcmV0dXJuIHRoaXMubW9kZWw/IHRoaXMubW9kZWwuc2V0KHByb3BlcnR5LCB2YWx1ZSk6IHRoaXM7CgkJfSwKCgkJLy8qQHB1YmxpYwoJCXN5bmM6IGZ1bmN0aW9uICgpIHsKCQkJZW55by5mb3JFYWNoKHRoaXMuYmluZGluZ3MsIGZ1bmN0aW9uIChiaW5kaW5nKSB7CgkJCQliaW5kaW5nLnN5bmMoKTsKCQkJfSk7CgkJCWVueW8uZm9yRWFjaCh0aGlzLmdldCgiX2F0dHJpYnV0ZUtleXMiKSwgZnVuY3Rpb24gKGtleSkgewoJCQkJdGhpcy5ub3RpZnlPYnNlcnZlcnMoa2V5LCBudWxsLCB0aGlzLm1vZGVsLmdldChrZXkpKTsKCQkJfSwgdGhpcyk7CgkJfSwKCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gUFJPVEVDVEVEIE1FVEhPRFMKCgkJY3JlYXRlOiBmdW5jdGlvbiAoKSB7CgkJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7CgkJCWlmICh0aGlzLm1vZGVsKSB7CgkJCQl0aGlzLl9pbml0TW9kZWwodGhpcy5tb2RlbCk7CgkJCX0KCQl9LAoJCQoJCV9yZW1vdmVNb2RlbDogZnVuY3Rpb24gKG1vZGVsKSB7CgkJCXZhciAkbW9kZWwgPSBtb2RlbCB8fCB0aGlzLm1vZGVsOwoJCQlpZiAoJG1vZGVsKSB7CgkJCQkkbW9kZWwucmVtb3ZlRGlzcGF0Y2hUYXJnZXQodGhpcyk7CgkJCQkkbW9kZWwucmVtb3ZlT2JzZXJ2ZXIoIioiLCB0aGlzLm5vdGlmeU9ic2VydmVycyk7CgkJCX0KCQl9LAoJCQoJCV9pbml0TW9kZWw6IGZ1bmN0aW9uIChtb2RlbCkgewoJCQl2YXIgJG1vZGVsID0gbW9kZWw7CgkJCWlmICgkbW9kZWwpIHsKCQkJCSRtb2RlbC5hZGREaXNwYXRjaFRhcmdldCh0aGlzKTsKCQkJCSRtb2RlbC5hZGRPYnNlcnZlcigiKiIsIHRoaXMubm90aWZ5T2JzZXJ2ZXJzLCB0aGlzKTsKCQkJCXRoaXMuc3RvcE5vdGlmaWNhdGlvbnMoKTsKCQkJCXRoaXMuc3luYygpOwoJCQkJdGhpcy5zdGFydE5vdGlmaWNhdGlvbnMoKTsKCQkJfQoJCX0sCgkJCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gT0JTRVJWRVJTCgoJCV9zZWxlY3Rpb25TcHk6IGVueW8ub2JzZXJ2ZXIoZnVuY3Rpb24gKHByb3BlcnR5LCBwcmV2aW91cywgdmFsdWUpIHsKCQkJaWYgKF9zZWxlY3RlZC50ZXN0KHByb3BlcnR5KSkgewoJCQkJaWYgKHRydWUgPT09IHZhbHVlKSB7CgkJCQkJdGhpcy5kb01vZGVsU2VsZWN0ZWQoKTsKCQkJCX0gZWxzZSBpZiAoZmFsc2UgPT09IHZhbHVlKSB7CgkJCQkJdGhpcy5kb01vZGVsRGVzZWxlY3RlZCgpOwoJCQkJfQoJCQl9CgkJfSwgIioiKQoJCQoJfSk7Cgp9KShlbnlvKTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/data/Source.js"
Content-Type: application/octet-stream; x-encoding=base64
(function (enyo) {

	/**
		The _enyo.Source_ kind is an pseudo-abstract API for communcating
		with a backend (could be local or remote). _enyo.Store_ requires a
		_source_ to function properly. An _enyo.Source_ can easily be overloaded
		to work with specific backend implementations and data formats.

		The build-in implementation is designed to work with a remote REST API
		relying on GET, POST, PUT and DELETE to communicate with (remote) source.
		For _fetch_ requests it will use GET. See _enyo.Collection_ and _enyo.Model_
		for specifics.
	*/

	var normalize = function (url) {
		return url.replace(/([^:]\/)(\/+)/g, "$1");
	};

	enyo.kind({

		// ...........................
		// PUBLIC PROPERTIES

		name: "enyo.Source",
		kind: "enyo.Component",

		//*@public
		/**
			The type of object to use for requests. As long as the kind has the
			same API as _enyo.Async_ it should be possible to use it with _enyo.Source_.
		*/
		requestKind: "enyo.Ajax",

		//*@public
		/**
			The root domain for all requests. Does not need the _http_ but may
			include it. For secure (_https_) see the _secure_ property.
		*/
		domain: "",

		//*@public
		/**
			An optional string to be appended after the _url_ is constructed.
		*/
		urlPostfix: "",

		//*@public
		/**
			The port to use when constructing the _url_. Only specify if the necessary
			port is different than the current domain port (e.g. it will automatically use
			document.location.port unless the _ignorePort_ option is set to _true_).
		*/
		port: null,

		//*@public
		/**
			If the current domain for the client is running on a _port_ different than
			80 (e.g. http://localhost:8080) this will construct requests without the
			port.
		*/
		ignorePort: false,

		//*@public
		/**
			If a secure url is necessary (_https_) set this to _true_.
		*/
		secure: false,

		//*@public
		/**
			If the (remote) backend is read-only set this to _true_ to dissallow any
			calls to POST, PUT or DELETE.
		*/
		readOnly: false,

		//*@public
		/**
			A bindable property to know when an asynchronous operation is taking place.
		*/
		busy: false,

		//*@public
		/**
			The default options that are passed to the request object. Presedence is given
			to _models_ or _collections_ that provide their own options that override any
			default options.

			Defaults with cacheBust true and contentType application/json.
		*/
		defaultOptions: {
			cacheBust: false,
			contentType: "application/json"
		},

		//*@public
		/**
			The default headers to be used in requests if they differ from those
			in _enyo.AjaxProperties_.
		*/
		defaultHeaders: null,

		// ...........................
		// PROTECTED PROPERTIES

		_noApplyMixinDestroy: true,

		// ...........................
		// COMPUTED PROPERTIES

		// ...........................
		// PUBLIC METHODS
		buildUrl: function (model, options) {
			if (!options.url) {
				var url = !~this.domain.indexOf("http")? "http" + (this.secure? "s": "") + "://" + this.domain: this.domain;
				if (!this.ignorePort && (this.port || location.port)) {
					url += (":" + (this.port? this.port: location.port) + "/");
				}
				url += "/" + this.urlPostfix + model.get("query");
				options.url = normalize(url);
			} else {
				options.urlProvided = true;
			}
		},
		buildHeaders: function (model, options) {
			options.headers = enyo.mixin(model.get("headers"), this.defaultHeaders, true);
		},
		buildQueryParams: function (model, options) {
			// add property to options called queryParams as
			// object literal to be appended to the query string
			options.queryParams = options.queryParams || {};
			model.buildQueryParams(model, options);
		},
		buildRequest: function (model, options) {
			this.buildUrl(model, options);
			this.buildHeaders(model, options);
			this.buildQueryParams(model, options);
			enyo.mixin(options, this.defaultOptions, true);
		},
		commit: function (model, options) {
			if (this.readOnly) {
				if (options && options.error) {
					options.error({});
				}
				return;
			}
			this.buildRequest(model, options);
			switch (model.isNew) {
			case true:
				options.method = "POST";
				break;
			case false:
				options.method = "PUT";
				break;
			}
			this.exec("commit", options);
		},
		fetch: function (model, options) {
			if (arguments.length > 1) {
				this.buildRequest(model, options);
				options.method = "GET";
			} else {
				options = model;
			}
			this.set("busy", true);
			this.exec("fetch", options);
		},
		destroy: function (model, options) {
			if (this.readOnly) {
				if (options && options.success) {
					options.success({});
				}
				return;
			}
			this.buildRequest(model, options);
			options.method = "DELETE";
			this.exec("destroy", options);
		},
		exec: function (which, options) {
			var $options = enyo.only(this._ajaxOptions, options);
			var $success = this.bindSafely("onSuccess", which, options);
			var $fail = this.bindSafely("onFail", which, options);
			var $com = new this.requestKind($options);
			var $params = options.queryParams;
			if (options.method !== "GET" && this.readOnly) {
				return $fail();
			}
			$com.response($success);
			$com.error($fail);
			$com.go($params);
		},
		filterData: function (data) {
			return data;
		},
		onSuccess: function (which, options, request, response) {
			if ("fetch" === which) {
				this.set("busy", false);
			}
			var result = this.filterData(response), $fn;

			if (($fn = this["did" + enyo.cap(which)])) {
				if (enyo.isFunction($fn)) {
					if (!$fn.call(this, options, result)) {
						return;
					}
				}
			}

			if (options.success) {
				options.success(result);
			}
		},
		onFail: function (which, options, request, error) {
			if ("fetch" === which) {
				this.set("busy", false);
			}
			if (options.error) {
				options.error(request, error);
			}
		},

		constructor: function () {
			this.inherited(arguments);
			this.defaultOptions = this.defaultOptions || {};
			this.defaultHeaders = this.defaultHeaders || {};
			this.domain = this.domain || (function () {
				return location.pathname.length > 1
					? location.href.split("/").slice(0,-1).join("/")
					: location.origin;
			}());
		},
		constructed: function () {
			var $kind = this.requestKind || enyo.Ajax;
			this.requestKind = "string" === typeof $kind? enyo.getPath($kind): $kind;
			this._ajaxOptions = enyo.keys(enyo.AjaxProperties);
		}

		// ...........................
		// PROTECTED METHODS

		// ...........................
		// OBSERVERS

	});

	//*@protected
	enyo.ready(function () {
		enyo.singleton({
			name: "enyo.Source.defaultSource",
			kind: "enyo.Source",
			fetch: function (model, options) {
				if (options && options.error) {
					var fn = options.error;
					options.error = null;
					fn(options);
				}
			},
			commit: function (model, options) {
				if (options && options.error) {
					var fn = options.error;
					options.error = null;
					fn(options);
				}
			},
			destroy: function (model, options) {
				if (options && options.success) {
					var fn = options.success;
					options.success = null;
					fn(options);
				}
			}
		});
	});

})(enyo);
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/data/Store.js"
Content-Type: application/octet-stream; x-encoding=base64
(function (enyo) {

	//*@public
	/**
		The _enyo.Store_ kind is a singleton (when used) that aids in
		managing localized data records at runtime. It tracks models and
		is an interface for interacting with the data schema for any application.

		An _enyo.Store_ must have an _enyo.Source_ to be fully functional. The
		_store_ is agnostic to how the _source_ retrieves or updates records. It
		provides an abstraction API for finding particular records. It is always
		accessible from the globally available _enyo.store_ variable.

		It is instantiated quite simply and would typically be done before instantiating
		the _enyo.Application_ for your app.

		new enyo.Store({source: "enyo.Source"}); // now available from enyo.store

		NOTE: Much of the public API of _enyo.Store_ needn't be called directly except
		in very complex remote-backend implementations requiring specialized handling.
		Most of the API is called by _enyo.Model_ and _enyo.Collection_ for you.

		TODO: Much of the implementation is working but in some cases is either absent
		or only partially completed. Please bear with us as we complete this moving
		forward.
	*/


	enyo.store = null;
	enyo.models = {
		kinds: [],
		add: function (ctor) {
			if (!~enyo.indexOf(ctor, this.kinds)) {
				this.kinds.push(ctor);
			}
			if (enyo.store) {
				enyo.store._addModelKind(ctor);
			}
		}
	};

	/**
		As seen https://gist.github.com/jcxplorer/823878, by jcxplorer.
		TODO: replace with faster implementation
	*/
	var uuid = function () {
		var uuid = "", idx = 0, rand;
		for (; idx < 32; ++idx) {
			rand = Math.random() * 16 | 0;
			if (idx == 8 || idx == 12 || idx == 16 || idx == 20) {
				uuid += "-";
			}
			uuid += (idx == 12? 4: (idx == 16? (rand & 3 | 8): rand)).toString(16);
		}
		return uuid;
	};

	enyo.kind({

		// ...........................
		// PUBLIC PROPERTIES

		name: "enyo.Store",
		kind: "enyo.MultipleDispatchComponent",
		source: null,
		busy: false,
		handlers: {
			onChange: "_modelChanged",
			onDestroy: "_modelDestroyed"
		},
		bindings: [
			{from: ".source.busy", to: ".busy"}
		],

		// ...........................
		// PROTECTED PROPERTIES

		_records: null,
		_noApplyMixinDestroy: true,

		// ...........................
		// COMPUTED PROPERTIES

		// ...........................
		// PUBLIC METHODS

		uuid: function () {
			return uuid();
		},
		/**
			A simple find mechanism to query valid records in the store.
			If the _options_ _remote_ property exists and is true it will
			execute asynchronously and run the query against the combined
			result set of records as returned from the remote source and
			any existing local records. If no _remote_ property exists or
			it is false (default) it will return synchronously resulting in
			an array of records if any were found or false otherwise. The
			first parameter can be either a string or constructor matching
			the name or kind of the model being queried with an optional
			_options_ parameter of an object literal. These options will be
			passed to the _queryResolver_ method. This method can be overridden
			to handle custom implementations for advanced querying needs.

			TODO: not implemented as stated
		*/
		find: function (ctor, options) {
			var ret = this._recordsForType(ctor);
			if (!ret) {
				return false;
			}
			return this.queryResolver(enyo.clone(ret.all), options);
		},
		/**
			Pass a constructor and options to this asynchronous method to retrieve
			a single record (will use locally if found) and will execute a fetch
			if needed. If no _success_ method is provided but the record was found
			locally it will be returned synchronously. The _options_ hash can have
			a _success_ method, _error_ method and a _params_ object with the properties
			used to find the correct record.

			TODO: not fully implemented
		*/
		findOne: function (ctor, options) {
			var $options = options? enyo.clone(options): {};
			var $params = options.params || {};
			var $models = this._recordsForType(ctor);
			var pk = ("string" === typeof ctor? enyo.getPath(ctor): ctor).prototype.primaryKey;
			var $ret;
			if (pk in $params) {
				if (($ret = $models.byPrimaryKey[$params[pk]])) {
					if (options.success) {
						options.success($ret);
					}
					return $ret;
				}
			}
			/* jshint -W055 */
			$ret = new ctor($params);
			/* jshint +W055 */
			$ret.fetch({success: this.bindSafely(function (options, model) {
				if (options.success) {
					options.success(model);
				}
			}, $options, $ret)});
		},
		/**
			TODO: not implemented
		*/
		findRemote: function (ctor, options) {
			// TODO: not implemented yet
			enyo.warn("enyo.Store.findRemote: this method is not implemented yet");
			return false;
		},
		/**
			TODO: not implemented
		*/
		queryResolver: function (models, options) {
			return models;
		},
		initModel: function (model) {
			var id = model.euuid = this.uuid();
			if (!model[model.primaryKey] && !model._didAttemptFetchId) {
				model._didAttemptFetchId = true;
				var $options = {
					success: this.bindSafely("initModel", model)
				};
				return this.fetchId(model, $options);
			}
			this._records[id] = model;
			this._records[model.kindName].all.push(model);
			model.addDispatchTarget(this);
			if (model[model.primaryKey]) {
				this._records[model.kindName].byPrimaryKey[model[model.primaryKey]] = model;
			}
		},
		fetch: function (model, options) {
			var $options = options? enyo.clone(options): {};
			$options.success = this.bindSafely("didFetch", model, options);
			$options.error = this.bindSafely("didFail", "fetch", model, options);

			// if the model is NEW we generate an id for it only
			if (model.status === "NEW") {
				return this.fetchId(model, options);
			}

			this.source.fetch(model, $options);
		},
		commit: function (model, options) {
			var $options = options? enyo.clone(options): {};
			$options.success = this.bindSafely("didCommit", model, options);
			$options.error = this.bindSafely("didFail", "commit", model, options);
			this.source.commit(model, $options);
		},
		destroy: function (model, options) {
			var $options = options? enyo.clone(options): {};
			$options.success = this.bindSafely("didDestroy", model, options);
			$options.error = this.bindSafely("didFail", "destroy", model, options);
			this.source.destroy(model, $options);
		},
		fetchId: function (model, options) {
			var $options = options? enyo.clone(options): {};
			$options.success = this.bindSafely("didFetchId", model, options);
			if (model.fetchId) {
				return model.fetchId($options);
			}
			// TODO: this may not be useful in and of itself since
			// most backends will have their own scheme for generating
			// an id via sequence etc.
			$options.success(model.euuid);
		},
		constructor: function () {
			// there can only be one store executing at a time
			if (enyo.store) {
				throw "There can only be one enyo.Store active";
			}
			enyo.store = this;
			this.inherited(arguments);
			this._records = {};
			enyo.forEach(enyo.models.kinds, this._addModelKind, this);
			if (!this.source) {
				this.source = enyo.Source.defaultSource;
			}
		},
		constructed: function () {
			this.inherited(arguments);
			this.findAndInstance("source");
		},
		sourceFindAndInstance: function (ctor, inst) {
			if (inst) {
				inst.set("owner", this);
			}
		},
		didFetch: function (model, options, result) {
			// TODO: ...
			if (options.success) {
				options.success(result);
			}
		},
		didCommit: function (model, options, result) {
			// TODO: ...
			if (options.success) {
				options.success(result);
			}
		},
		didDestroy: function (model, options, result) {
			// TODO: ...
			if (options.success) {
				options.success(result);
			}
		},
		didFetchId: function (model, options, result) {
			model.set(model.primaryKey, result);
			if (options.success) {
				options.success(result);
			}
		},
		didFail: function (which, model, options, request, error) {
			// TODO: ...
			if (options.error) {
				options.error(request, error);
			}
		},

		// ...........................
		// PROTECTED METHODS

		_addModelKind: function (ctor) {
			if (!this._records[ctor.prototype.kindName]) {
				this._records[ctor.prototype.kindName] = {
					all: [],
					byPrimaryKey: {}
				};
			}
		},
		_modelChanged: function (sender, event) {
			if (event.model) {
				var $model = event.model;
				var $changed = event.changed;
				var $prev = event.previous;
				var key = $model.primaryKey;
				var kind = $model.kindName;
				if (key in $changed) {
					// we have to remove the entry altogether
					delete this._records[kind].byPrimaryKey[$prev[key]];
					// only re-add it if there is actually a value
					if ($changed[key]) {
						this._records[kind].byPrimaryKey[$changed[key]] = $model;
					}
				}
			}
		},
		_modelDestroyed: function (sender, event) {
			var euuid = sender.euuid;
			var id = sender[sender.primaryKey];
			var kind = sender.kindName;
			var idx = enyo.indexOf(sender, this._records[kind].all);
			delete this._records[euuid];
			delete this._records[kind].byPrimaryKey[id];
			this._records[kind].all.splice(idx, 1);
			return true;
		},
		_recordsForType: function (ctor) {
			return this._records["string" === typeof ctor? ctor: ctor.prototype.kindName];
		},

		// ...........................
		// OBSERVERS

		_sourceChanged: enyo.observer(function (prop, prev, val) {
			if (val) {
				val.set("owner", this);
				this.rebuildBindings();
			}
		}, "source")

	});


})(enyo);
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/data/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJIlN0b3JlLmpzIiwKCSJTb3VyY2UuanMiLAoJIk1vZGVsLmpzIiwKCSJDb2xsZWN0aW9uLmpzIiwKCSJNb2RlbENvbnRyb2xsZXIuanMiCik7
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/dev.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uICgpIHsKCSJ1c2Ugc3RyaWN0IjsKCgkvLypAcHJvdGVjdGVkCgkvKioKCQlXaGVuIGF2YWlsYWJsZSwgc3VwcGx5IGEgaGlnaC1wcmVjaXNpb24sIGhpZ2ggcGVyZm9ybWFuY2UKCQltb25vdG9uaWMgYmVuY2htYXJrIGZvciBzb21lIGludGVybmFsIHVzYWdlIGFuZCBiZW5jaG1hcmsgdGVzdGluZy4KCSovCgllbnlvLmJlbmNoID0gKGZ1bmN0aW9uICgpIHsKCQkvLyB3ZSBoYXZlIHRvIGNoZWNrIHdoZXRoZXIgb3Igbm90IHRoZSBicm93c2VyIGhhcyBzdXBwbGllZCBhIHZhbGlkCgkJLy8gbWV0aG9kIHRvIHVzZQoJCXZhciBwZXJmID0gd2luZG93LnBlcmZvcm1hbmNlIHx8IHt9OwoJCS8vIHRlc3QgYWdhaW5zdCBhbGwga25vd24gdmVuZG9yLXNwZWNpZmljIGltcGxlbWVudGF0aW9ucywgYnV0IHVzZQoJCS8vIGEgZmFsbGJhY2sganVzdCBpbiBjYXNlCgkJcGVyZi5ub3cgPSBwZXJmLm5vdyB8fCBwZXJmLm1vek5vdyB8fCBwZXJmLm1zTm93IHx8IHBlcmYub05vdyB8fCBwZXJmLndlYmtpdE5vdyB8fCBlbnlvLm5vdzsKCQlyZXR1cm4gZnVuY3Rpb24gKCkgewoJCQlyZXR1cm4gcGVyZi5ub3coKTsKCQl9OwoJfSgpKTsKCgkvLypAcHJvdGVjdGVkCgkvKioKCQlUaGlzIGlzIGEgY29sbGVjdGlvbiBvZiBtZXRob2RzIHRvIGFzc2lzdCBpbiBzaW1wbGUgYmVuY2htYXJraW5nLgoJCVRoZSBnb2FsIHdhcyB0byBzdXBwbHkgdXNlZnVsIGZ1bmN0aW9uYWxpdHkgd2hpbGUgaW1wYWN0aW5nIHRoZSByZXN1bHRzCgkJYXMgbGl0dGxlIGFzIHBvc3NpYmxlICh0aGUgbW9yZSBjYWxjdWxhdGlvbnMgd2UgZG8gZHVyaW5nIGJlbmNobWFya2luZywKCQl0aGUgZ3JlYXRlciB0aGUgb3Bwb3J0dW5pdHkgdG8gc2tldyByZXN1bHRzKS4gVGhpcyBpcyBwYXJ0aWN1bGFybHkgaW1wb3J0YW50CgkJd2hlbiB1c2luZyBwb3RlbnRpYWxseS1uZXN0ZWQgYmVuY2htYXJrIHNlcmllcyAoaS5lLiwgYmVuY2htYXJraW5nIGEgbWV0aG9kCgkJdGhhdCBleGVjdXRlcyBvdGhlciBiZW5jaG1hcmtlZCBtZXRob2RzKS4KCSovCgoJLy8gVHJhY2sgdGhlIGFjdGl2ZSB0ZXN0cwoJdmFyIHRlc3RzID0ge307CgkvLyBUcmFjayBhdmVyYWdlcwoJdmFyIGF2ZXJhZ2VzID0ge307CgoJLy8gRGVmYXVsdCByZXBvcnQgdGVtcGxhdGUgc3RyaW5nCgl2YXIgcmVwb3J0X3RlbXBsYXRlID0gIi0gLSAtIC0gLSAtIC0gLSAtIC0gLSAtIC0gLSAtIC0gLVxuIiArCgkJCQkJIkJFTkNITUFSSyBSRVBPUlQgKCUuKTogJS5cbiIgKwoJCQkJCSJUT1RBTCBUSU1FIChtcyk6ICUuXG4iICsKCQkJCQkiQVZFUkFHRSBUSU1FIChtcyk6ICUuXG4iICsKCQkJCQkiTUlOSU1VTSBUSU1FIChtcyk6ICUuXG4iICsKCQkJCQkiTUFYSU1VTSBUSU1FIChtcyk6ICUuXG4iICsKCQkJCQkiTlVNQkVSIE9GIEVOVFJJRVM6ICUuXG4iICsKCQkJCQkiLSAtIC0gLSAtIC0gLSAtIC0gLSAtIC0gLSAtIC0gLSAtXG4iOwoKCS8vIENhbGN1bGF0ZXMgYXZlcmFnZSBhbmQgYmFzaWMgc3RhdGlzdGljcy4KCXZhciBjYWxjID0gZnVuY3Rpb24gKG51bWJlcnMpIHsKCQl2YXIgdG90YWwgPSAwOwoJCXZhciBtaW4gPSBJbmZpbml0eTsKCQl2YXIgbWF4ID0gLUluZmluaXR5OwoJCXZhciBudW1iZXIgPSBudW1iZXJzLmxlbmd0aDsKCQl2YXIgc3RhdHMgPSB7dG90YWw6IG51bGwsIGF2ZXJhZ2U6IG51bGwsIG51bWJlcjogbnVtYmVyLCBtaW46IG51bGwsIG1heDogbnVsbH07CgkJZW55by5mb3JFYWNoKG51bWJlcnMsIGZ1bmN0aW9uIChudW0pIHsKCQkJdG90YWwgKz0gbnVtOwoJCQltaW4gPSBNYXRoLm1pbihudW0sIG1pbik7CgkJCW1heCA9IE1hdGgubWF4KG51bSwgbWF4KTsKCQl9KTsKCQlzdGF0cy50b3RhbCA9IHRvdGFsOwoJCXN0YXRzLm1pbiA9IG1pbjsKCQlzdGF0cy5tYXggPSBtYXg7CgkJc3RhdHMuYXZlcmFnZSA9IE1hdGguYWJzKHRvdGFsLyhudW1iZXIgfHwgMSkpOwoJCXJldHVybiBzdGF0czsKCX07CgoJLy8qQHB1YmxpYwoJZW55by5kZXYgPSB7CgoJCS8vKkBwdWJsaWMKCQkvLyogY2FuIGJlIHNldCB0byBmYWxzZSB0byBkaXNhYmxlIGFsbCBiZW5jaG1hcmtpbmcgY29kZQoJCWVuYWJsZWQ6IHRydWUsCgoJCS8vKkBwdWJsaWMKCQkvKioKCQkJQ3JlYXRlIGEgbmV3IGJlbmNobWFyayB0ZXN0LgoJCQlUaGUgb3B0cyBvYmplY3QgcGFzc2VkIGluIGhhcyB0aGUgcHJvcGVydGllcwoKCQkJKiBuYW1lOiBvcHRpb25hbCBuYW1lIGZvciB0ZXN0CgkJCSogYXZlcmFnZTogaWYgdHJ1ZSwgY2FsY3VsYXRlIGFuIGF2ZXJhZ2Ugb2YgcmVwZWF0ZWQgc3RhcnQvc3RvcHMgZm9yIHRoZSB0ZXN0CgkJCSogbG9nZ2luZzogaWYgdHJ1ZSwgd3JpdGUgc3RhcnQgYW5kIHN0b3AgbWVzc2FnZXMgdG8gdGhlIGNvbnNvbGUgKGRlZmF1bHRzIHRvIHRydWUpCgkJCSogYXV0b3N0YXJ0OiBpZiB0cnVlLCBhdXRvbWF0aWNhbGx5IHN0YXJ0IHRoZSBiZW5jaG1hcmsgKGRlZmF1bHRzIHRvIHRydWUpCgoJCQlUaGlzIHJldHVybnMgYW4gb2JqZWN0IHRoYXQgaGFzIHN0YXJ0IGFuZCBzdG9wIG1ldGhvZHMgdXNlZAoJCQl0byB0cmFjayBhIHRlc3QuCgkJKi8KCQliZW5jaDogZnVuY3Rpb24gKG9wdHMpIHsKCQkJaWYgKHRydWUgIT09IHRoaXMuZW5hYmxlZCkgewoJCQkJcmV0dXJuIGZhbHNlOwoJCQl9CgkJCXZhciBvcHRpb25zID0gb3B0cyB8fCB7bmFtZTogZW55by51aWQoImJlbmNoIil9OwoJCQlyZXR1cm4gbmV3IEJlbmNobWFyayhvcHRpb25zKTsKCQl9LAoKCQkvLypAcHVibGljCgkJLy8qIE91dHB1dCB0byB0aGUgY29uc29sZSBpbmZvcm1hdGlvbiBhYm91dCBhIGJlbmNobWFyayBuYW1lZCBfbmFtZV8uCgkJcmVwb3J0OiBmdW5jdGlvbiAobmFtZSkgewoJCQl2YXIgYmVuY2ggPSBhdmVyYWdlc1tuYW1lXSB8fCB0ZXN0c1tuYW1lXTsKCQkJaWYgKCFiZW5jaCkgewoJCQkJcmV0dXJuIGZhbHNlOwoJCQl9CgkJCWlmIChiZW5jaC5yZXBvcnQgJiYgImZ1bmN0aW9uIiA9PT0gdHlwZW9mIGJlbmNoLnJlcG9ydCkgewoJCQkJcmV0dXJuIGJlbmNoLnJlcG9ydCgpOwoJCQl9IGVsc2UgewoJCQkJdmFyIHN0YXRzID0gY2FsYyhiZW5jaCk7CgkJCQllbnlvLmxvZygKCQkJCQllbnlvLmZvcm1hdCgKCQkJCQkJcmVwb3J0X3RlbXBsYXRlLAoJCQkJCQkiYXZlcmFnZXMiLAoJCQkJCQluYW1lLAoJCQkJCQlzdGF0cy50b3RhbCwKCQkJCQkJc3RhdHMuYXZlcmFnZSwKCQkJCQkJc3RhdHMubWluLAoJCQkJCQlzdGF0cy5tYXgsCgkJCQkJCXN0YXRzLm51bWJlcgoJCQkJCSkKCQkJCSk7CgkJCX0KCQl9LAoKCQkvLypAcHVibGljCgkJLy8qIFJlbW92ZSBzdG9yZWQgZGF0YSBmb3IgYSBiZW5jaG1hcmsgbmFtZWQgX25hbWVfLgoJCWNsZWFyOiBmdW5jdGlvbiAobmFtZSkgewoJCQl2YXIgc291cmNlID0gdGVzdHNbbmFtZV0/IHRlc3RzOiBhdmVyYWdlc1tuYW1lXT8gYXZlcmFnZXM6IG51bGw7CgkJCWlmICghc291cmNlKSB7CgkJCQlyZXR1cm4gZmFsc2U7CgkJCX0KCQkJaWYgKHNvdXJjZS5jb21wbGV0ZSkgewoJCQkJc291cmNlLmNvbXBsZXRlKCk7CgkJCX0KCQkJaWYgKHNvdXJjZVtuYW1lXSBpbnN0YW5jZW9mIEFycmF5KSB7CgkJCQlzb3VyY2VbbmFtZV0gPSBbXTsKCQkJfQoJCQllbHNlIHsKCQkJCWRlbGV0ZSBzb3VyY2VbbmFtZV07CgkJCX0KCQkJcmV0dXJuIHRydWU7CgkJfQoKCX07CgoJLy8qQHByb3RlY3RlZAoJZnVuY3Rpb24gQmVuY2htYXJrIChvcHRpb25zKSB7CgkJZW55by5taXhpbih0aGlzLCBvcHRpb25zKTsKCQl0ZXN0c1t0aGlzLm5hbWVdID0gdGhpczsKCQlpZiAodHJ1ZSA9PT0gdGhpcy5hdmVyYWdlICYmICFhdmVyYWdlc1t0aGlzLm5hbWVdKSB7CgkJCWF2ZXJhZ2VzW3RoaXMubmFtZV0gPSBbXTsKCQl9CgkJaWYgKGF2ZXJhZ2VzW3RoaXMubmFtZV0gJiYgZmFsc2UgIT09IHRoaXMuYXZlcmFnZSkgewoJCQl0aGlzLl9hdmVyYWdpbmcgPSB0cnVlOwoJCX0KCQlpZiAodHJ1ZSA9PT0gdGhpcy5hdXRvU3RhcnQpIHsKCQkJdGhpcy5zdGFydCgpOwoJCX0KCX0KCgkvLypAcHJvdGVjdGVkCglCZW5jaG1hcmsucHJvdG90eXBlID0gewoKCQkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCQkvLyBQVUJMSUMgUFJPUEVSVElFUwoKCQlsb2dnaW5nOiB0cnVlLAoJCWF1dG9TdGFydDogdHJ1ZSwKCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gUFJPVEVDVEVEIFBST1BFUlRJRVMKCgkJX3N0YXJ0ZWQ6IGZhbHNlLAoJCV9hdmVyYWdpbmc6IGZhbHNlLAoJCV9iZWdpbjogbnVsbCwKCQlfZW5kOiBudWxsLAoJCV90aW1lOiBudWxsLAoKCQkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCQkvLyBQVUJMSUMgTUVUSE9EUwoKCQlzdGFydDogZnVuY3Rpb24gKCkgewoJCQlpZiAodHJ1ZSA9PT0gdGhpcy5fc3RhcnRlZCkgewoJCQkJcmV0dXJuIGZhbHNlOwoJCQl9CgkJCXRoaXMuX2xvZygic3RhcnRpbmcgYmVuY2htYXJrIik7CgkJCXRoaXMuX2JlZ2luID0gZW55by5iZW5jaCgpOwoJCQl0aGlzLl9zdGFydGVkID0gdHJ1ZTsKCQkJcmV0dXJuIHRydWU7CgkJfSwKCgkJc3RvcDogZnVuY3Rpb24gKCkgewoJCQlpZiAoIXRoaXMuX3N0YXJ0ZWQpIHsKCQkJCXJldHVybiBmYWxzZTsKCQkJfQoJCQl0aGlzLl9lbmQgPSBlbnlvLmJlbmNoKCk7CgkJCXRoaXMuX3RpbWUgPSB0aGlzLl9lbmQgLSB0aGlzLl9iZWdpbjsKCQkJdGhpcy5fbG9nKCJiZW5jaG1hcmsgY29tcGxldGU6ICIgKyB0aGlzLl90aW1lKTsKCQkJaWYgKHRydWUgPT09IHRoaXMuX2F2ZXJhZ2luZykgewoJCQkJYXZlcmFnZXNbdGhpcy5uYW1lXS5wdXNoKHRoaXMuX3RpbWUpOwoJCQl9CgkJCXRoaXMuX3N0YXJ0ZWQgPSBmYWxzZTsKCQkJcmV0dXJuIHRydWU7CgkJfSwKCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gUFJPVEVDVEVEIE1FVEhPRFMKCgkJX2xvZzogZnVuY3Rpb24gKG1lc3NhZ2UpIHsKCQkJaWYgKCF0aGlzLmxvZ2dpbmcpIHsKCQkJCXJldHVybiBmYWxzZTsKCQkJfQoJCQllbnlvLmxvZygiYmVuY2ggKCIgKyB0aGlzLm5hbWUgKyAiKTogIiArIG1lc3NhZ2UpOwoJCX0KCgl9Owp9KCkpOw==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/job.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglJbnZva2VzIGZ1bmN0aW9uIF9pbkpvYl8gYWZ0ZXIgX2luV2FpdF8gbWlsbGlzZWNvbmRzIGhhdmUgZWxhcHNlZCBzaW5jZSB0aGUKCWxhc3QgdGltZSBfaW5Kb2JOYW1lXyB3YXMgcmVmZXJlbmNlZC4KCglKb2JzIGNhbiBiZSB1c2VkIHRvIHRocm90dGxlIGJlaGF2aW9ycy4gIElmIHNvbWUgZXZlbnQgbWF5IG9jY3VyIG9uY2Ugb3IKCW11bHRpcGxlIHRpbWVzLCBidXQgd2Ugd2FudCBhIHJlc3BvbnNlIHRvIG9jY3VyIG9ubHkgb25jZSBldmVyeSBfbl8Jc2Vjb25kcywKCXdlIGNhbiB1c2UgYSBqb2IuCgoJCW9uc2Nyb2xsOiBmdW5jdGlvbigpIHsKCQkJLy8gdXBkYXRlVGh1bWIgd2lsbCBiZSBjYWxsZWQsIGJ1dCBvbmx5IHdoZW4gMXMgaGFzIGVsYXBzZWQgc2luY2UgdGhlCgkJCS8vIGxhc3Qgb25zY3JvbGwKCQkJZW55by5qb2IoInVwZGF0ZVRodW1iIiwgdGhpcy5iaW5kU2FmZWx5KCJ1cGRhdGVUaHVtYiIpLCAxMDAwKTsKCQl9CiovCmVueW8uam9iID0gZnVuY3Rpb24oaW5Kb2JOYW1lLCBpbkpvYiwgaW5XYWl0KSB7CgllbnlvLmpvYi5zdG9wKGluSm9iTmFtZSk7CgllbnlvLmpvYi5fam9ic1tpbkpvYk5hbWVdID0gc2V0VGltZW91dChmdW5jdGlvbigpIHsKCQllbnlvLmpvYi5zdG9wKGluSm9iTmFtZSk7CgkJaW5Kb2IoKTsKCX0sIGluV2FpdCk7Cn07CgovKioKCUNhbmNlbHMgdGhlIG5hbWVkIGpvYiwgaWYgaXQgaGFzIG5vdCBhbHJlYWR5IGZpcmVkLgoqLwplbnlvLmpvYi5zdG9wID0gZnVuY3Rpb24oaW5Kb2JOYW1lKSB7CglpZiAoZW55by5qb2IuX2pvYnNbaW5Kb2JOYW1lXSkgewoJCWNsZWFyVGltZW91dChlbnlvLmpvYi5fam9ic1tpbkpvYk5hbWVdKTsKCQlkZWxldGUgZW55by5qb2IuX2pvYnNbaW5Kb2JOYW1lXTsKCX0KfTsKCi8qKgoJSW52b2tlIF9pbkpvYl8gaW1tZWRpYXRlbHksIHRoZW4gcHJldmVudCBhbnkgb3RoZXIgY2FsbHMgdG8KCWVueW8uam9iLnRocm90dGxlIHdpdGggdGhlIHNhbWUgX2luSm9iTmFtZV8gZnJvbSBydW5uaW5nIGZvcgoJdGhlIG5leHQgX2luV2FpdF8gbWlsbGlzZWNvbmRzLgoKCVRoaXMgaXMgdXNlZCBmb3IgdGhyb3R0bGluZyB1c2VyIGV2ZW50cyB3aGVuIHlvdSB3YW50IGFuCglpbW1lZGlhdGUgcmVzcG9uc2UsIGJ1dCBsYXRlciBpbnZvY2F0aW9ucyBtaWdodCBqdXN0IGJlIG5vaXNlCglpZiB0aGV5IGFycml2ZSB0b28gb2Z0ZW4uCiovCmVueW8uam9iLnRocm90dGxlID0gZnVuY3Rpb24oaW5Kb2JOYW1lLCBpbkpvYiwgaW5XYWl0KSB7CgkvLyBpZiB3ZSBzdGlsbCBoYXZlIGEgam9iIHdpdGggdGhpcyBuYW1lIHBlbmRpbmcsIHJldHVybiBpbW1lZGlhdGVseQoJaWYgKGVueW8uam9iLl9qb2JzW2luSm9iTmFtZV0pIHsKCQlyZXR1cm47Cgl9CglpbkpvYigpOwoJZW55by5qb2IuX2pvYnNbaW5Kb2JOYW1lXSA9IHNldFRpbWVvdXQoZnVuY3Rpb24oKSB7CgkJZW55by5qb2Iuc3RvcChpbkpvYk5hbWUpOwoJfSwgaW5XYWl0KTsKfTsKCmVueW8uam9iLl9qb2JzID0ge307Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/lang.js"
Content-Type: application/octet-stream; x-encoding=base64
(function(){
	//* @protected
	enyo.global = this;

	//*@protected
	/**
		Used internally by the enyo.uid method to be able to produce
		a runtime unique identifier.
	*/
	var uidCounter = 0;

	//*@public
	/**
		Returns a boolean value indicating whether a target is undefined.
	*/
	enyo.exists = function (target) {
		return (undefined !== target);
	};
	var exists = enyo.exists;

	//*@public
	/**
		Looks for last occurrence of a string _(needle)_ inside an array or string
		_(haystack)_. An IE8-safe fallback for the default _lastIndexOf_ method.
	*/
	enyo.lastIndexOf = function (needle, haystack, index) {
		if (haystack.lastIndexOf) {
			return haystack.lastIndexOf(needle, index || haystack.length);
		}
		// in IE8 there is no lastIndexOf for arrays or strings but we
		// treat them slightly differently, this is written for minimal-
		// code as a slight tradeoff in performance but should rarely be
		// hit as it is
		var string = ("string" === typeof haystack);
		var rev = (string? haystack.split(""): haystack).reverse();
		var cap = rev.length-1;
		var len = haystack.length;
		var idx;
		// if it is a string we need to make it a string again for
		// the indexOf method
		if (string) {
			rev = rev.join("");
		}
		idx = enyo.indexOf(needle, rev, len - (index || len));
		// put the array back the way it was
		if (!string) {
			rev.reverse();
		}
		return -1 === idx? idx: (cap - idx);
	};
	var lastIndexOf = enyo.lastIndexOf;

	//*@protected
	/**
		Internally-used method to strip leading '.' from string paths.
	*/
	var preparePath = function (path) {
		var idx = 0;
		while ("." === path[idx]) {
			++idx;
		}
		if (0 !== idx) {
			path = path.slice(idx);
		}
		return path;
	};

	//*@protected
	/**
		Internally-used method to detect if the given value exists,
		is a function and an overloaded getter. Returns true if these
		tests are successful; false otherwise.
	*/
	var isOverloaded = function (target) {
		return target && "function" === typeof target && true === target.overloaded;
	};

	//*@protected
	/**
		Internally-used method to detect deferred kind constructors.
	*/
	var isDeferredConstructor = function(target) {
		return target && ("function" === typeof target) &&
			(target._FinalCtor || target._finishKindCreation);
	};

	//*@public
	/**
		A fast-path enabled global getter that takes a string path that
		can be a full path (from context window/Enyo) or a relative path
		(to the execution context of the method). It knows how to check for
		and call the backwards-compatible generated getters as well as
		handle computed properties. Performs an optimized recursive search.
		Returns undefined if the object at the given path can not be
		found. Can safely be called on non-existent paths.
	*/
	enyo.getPath = function (path) {
		// if we don't have a path we can't do anything
		if (!exists(path) || null === path) {
			return undefined;
		}
		var idx = 0;
		var val;
		var part;
		var fn;
		var recursing = (true === arguments[1]) || ("object" === typeof path && path.recursing)? true: false;
		// on rare occasions this method would be called under the context
		// of enyo itself, the problem is detecting when this is intended since
		// under normal circumstances a general call would assume a window
		// context - here we see the _recursing_ parameter taking a double
		// meaning as enyo should _never be used as a reference on another object_
		// and as long as that is true this will never fail
		var cur = this === enyo && true !== recursing? window: this;
		// if we were recursing then we reassign path to the string part of the
		// object/parameter passed in
		if ("object" === typeof path) {
			if (path.path && "string" === typeof path.path) {
				path = path.path;
			}
			// otherwise it was an invalid request
			// but we return the object that was passed in
			else {
				return path;
			}
		}
		// clear any leading periods
		path = preparePath(path);
		// find the initial period if any
		idx = path.indexOf(".");

		if (path.length === 0) {
			return cur;
		}

		// if there isn't any try and find the path relative to our
		// current context, this is the fast path
		if (-1 === idx) {
			// figure out what our default/backwards-compatible getter
			// function would be
			fn = "get" + enyo.cap(path);
			// if that path exists relative to our context check to see
			// if it is an overloaded getter and call that if it is otherwise
			// just grab that path knowing if we get undefined that is ok
			val = isOverloaded(cur[fn])? cur[fn].call(this): cur[path];
		} else {
			// begin our recursive search
			part = path.substring(0, idx);
			path = path.slice(idx+1);

			var root = cur[part];
			if (isDeferredConstructor(root)) {
				root = enyo.checkConstructor(root);
			}
			if (root && typeof root in {"object":"","function":""}) {
				if (root._isObject) {
					return root.get(path);
				} else {
					val = enyo.getPath.call(root, {path: path, recursing: true});
				}
			}
		}

		if (isDeferredConstructor(val)) {
			val = enyo.checkConstructor(val);
		}
		// otherwise we've reached the end so return whatever we have
		return val;
	};

	//*@public
	/**
		A global setter that takes a string path (relative to the method's
		execution context) or a full path (relative to window). Attempts
		to automatically retrieve any previously existing value to supply
		to any observers. If the context is an _enyo.Object_ or subkind,
		the _notifyObservers_ method is used to notify listeners for the path's
		being set. If the previous value is the equivalent of the newly set
		value, observers will not be triggered by default. If the third
		parameter is present and is an explicit boolean true, it triggers
		the observers regardless. Optionally, the third parameter may be a
		function comparator that accepts two parameters and is expected to
		return a truthy-falsy value indicating whether or not the notifications
		will be fired. Returns the context from which the method was executed.
		Unlike its getter counterpart, this is not a recursive method.
	*/
	enyo.setPath = function (path, value, force) {
		// if there are less than 2 parameters we can't do anything
		if(!exists(path) || "string" !== typeof path || path.length === 0) {
			return this;
		}
		var cur = enyo === this? enyo.global: this;
		var idx;
		var target;
		var parts;
		var notify = true === force? true: false;
		var comparator = "function" === typeof force? force: undefined;
		// attempt to retrieve the previous value if it exists
		var prev = enyo.getPath.call(cur, path);
		// clear any leading periods
		path = preparePath(path);
		// find the inital index of any period in the path
		idx = path.indexOf(".");
		// if there wasn't one we can attempt to fast-path this setter
		if (-1 === idx) {
			// otherwise we just plain overwrite the method, this is the
			// expected behavior
			cur[path] = value;
		} else {
			// we have to walk the path until we find the end
			parts = path.split(".");
			// while we have any other parts to inspect
			while (parts.length) {
				target = parts.shift();
				// the rare case where the path could specify enyo
				// and is executed under the context of enyo
				if ("enyo" === target && enyo === cur) {
					continue;
				}
				// if this is the last piece we test to see if it is a computed
				// property and if it is we call it with the new value
				// as in the fast path
				if (0 === parts.length) {
					// otherwise we overwrite it just like in the fast-path
					cur[target] = value;
				} else {
					// we update our current reference context and if it does
					// not exist at the requested path it will be created
					if (!(typeof cur[target] in {"object":"","function":""})) {
						cur[target] = {};
					}
					if (true === cur[target]._isObject) {
						return cur[target].set(parts.join("."), value);
					}
					cur = cur[target];
				}
			}
		}
		// now we need to determine if we are going to issue notifications
		// first check to see if notify is already forced true
		if (true !== notify) {
			// now check to see if we have a comparator and if so use it
			// to determine if we're going to trigger observers
			if (comparator) {
				notify = comparator(prev, value);
			} else {
				// do the default which is to test the previous value
				// versus the new value
				notify = (prev !== value);
			}
		}
		if (true === notify) {
			if (cur.notifyObservers) {
				cur.notifyObservers(path, prev, value);
			}
		}
		// return the callee
		return cur;
	};

	//*@protected
	/**
		Called by instances of _enyo.Object_ in their own context via their
		local version of this method. Attempts to find the given property of
		the current context and instance the property if it is not already an
		instance. If it is a string, the method attempts to find the
		constructor for the named kind or the instance at the given path.
		When complete, it calls the callback method, passing it two
		parameters--the constructor (if it was found) and the instance (if it
		could be determined).
	*/
	enyo.findAndInstance = function (property, fn, context) {
		var Ctor;
		var inst;
		var path;
		fn = exists(fn) && "function" === typeof fn? fn: enyo.nop;
		// attempt to find the string path identifier on the kind
		// definition if possible
		path = enyo.getPath.call(this, property);
		// if there is nothing at the given property fast-path out
		// and return undefined everything
		if (!path) {
			return fn.call(context || this);
		}
		// if the path is a string (as in most cases) go ahead and
		// attempt to get the kind definition or instance at the
		// given path
		if ("string" === typeof path) {
			// we can fast-track this for relative paths that explicitly state
			// it is relative with a "." prefix, otherwise we have to guess
			Ctor = "." === path[0]? enyo.getPath.call(this, path):
				enyo.getPath(path) || enyo.getPath.call(this, path);
			// if it isn't a function we assume it is an instance
			if (exists(Ctor) && "function" !== typeof Ctor) {
				inst = Ctor;
				Ctor = undefined;
			}
		} else if ("function" === typeof path) {
			// instead of a string we were handed a constructor
			// so reassign that
			Ctor = path;
		} else {
			// the assumption here is that we were handed an
			// instance of the given object
			inst = path;
		}
		// if we have a constructor and no instance we need to
		// create an instance of the obejct
		if (exists(Ctor) && !exists(inst)) {
			inst = new Ctor();
		}
		// if we do have an instance assign it to the base object
		if (exists(inst)) {
			this[property] = inst;
		}
		// now use the calback and pass it the correct parameters
		return fn.call(context || this, Ctor, inst);
	};

	//*@public
	/**
		Creates a unique identifier (with an optional prefix) and returns
		the identifier as a string.
	*/
	enyo.uid = function (prefix) {
		return String((prefix? prefix: "") + uidCounter++);
	};

	//* @public
	//* Returns a random integer between 0 and a specified upper boundary;
	//* i.e., 0 <= return value < _inBound_.
	//
	//		var randomLetter = String.fromCharCode(enyo.irand(26) + 97);
	//
	enyo.irand = function(inBound) {
		return Math.floor(Math.random() * inBound);
	};

	//* Returns _inString_ with the first letter capitalized.
	enyo.cap = function(inString) {
		return inString.slice(0, 1).toUpperCase() + inString.slice(1);
	};

	//* Returns _inString_ with the first letter lower-cased.
	enyo.uncap = function(inString) {
		return inString.slice(0, 1).toLowerCase() + inString.slice(1);
	};

	enyo.format = function(inVarArgs) {
		var pattern = /\%./g;
		var arg = 0, template = inVarArgs, args = arguments;
		var replacer = function(inCode) {
			return args[++arg];
		};
		return template.replace(pattern, replacer);
	};

	var toString = Object.prototype.toString;

	//* Returns true if the argument is a string.
	enyo.isString = function(it) {
		return toString.call(it) === "[object String]";
	};

	//* Returns true if the argument is a function.
	enyo.isFunction = function(it) {
		return toString.call(it) === "[object Function]";
	};

	//* Returns true if the argument is an array.
	enyo.isArray = Array.isArray || function(it) {
		return toString.call(it) === "[object Array]";
	};

	//* Returns true if the argument is an object.
	enyo.isObject = Object.isObject || function (it) {
		return toString.call(it) === "[object Object]";
	};

	//* Returns true if the argument is true.
	enyo.isTrue = function(it) {
		return !(it === "false" || it === false || it === 0 || it === null || it === undefined);
	};

	//*@public
	/**
		Returns the index of any entry in _array_ whose _callback_ returns
		a truthy value. Can pass an optional _context_ for the _callback_. Each _callback_
		will receive 3 parameters, the _value_ at _index_ and an immutable copy
		of the original array. If no _callback_ returns true or _array_ is not
		an Array will return false.
	*/
	enyo.find = function (array, callback, context) {
		var $source = enyo.isArray(array) && array;
		var $ctx = context || enyo.global;
		var $fn = callback;
		var idx = 0, len, $copy, ret;
		if ($source && $fn && enyo.isFunction($fn)) {
			$copy = enyo.clone($source);
			len = $source.length;
			for (; idx < len; ++idx) {
				ret = $fn.call($ctx, $source[idx], idx, $copy);
				if (!! ret) {
					return idx;
				}
			}
		}
		return false;
	};

	//* Returns the index of the element in _inArray_ that is equivalent
	//* (==) to _inElement_, or -1 if no such element is found.
	enyo.indexOf = function(inElement, inArray, fromIndex) {
		if (inArray.indexOf) {
			return inArray.indexOf(inElement, fromIndex);
		}

		if (fromIndex) {
			if (fromIndex < 0) {
				fromIndex = 0;
			}

			if (fromIndex > inArray.length) {
				return -1;
			}
		}

		for (var i=fromIndex || 0, l=inArray.length, e; (e=inArray[i]) || (i<l); i++) {
			if (e == inElement) {
				return i;
			}
		}
		return -1;
	};

	//* Removes the first element in the passed-in array that is equivalent
	//* (==) to _inElement_.
	enyo.remove = function(inElement, inArray) {
		var i = enyo.indexOf(inElement, inArray);
		if (i >= 0) {
			inArray.splice(i, 1);
		}
	};

	/**
		Invokes _inFunc_ on each element of _inArray_.
		If _inContext_ is specified, _inFunc_ is called with _inContext_ as _this_.
	*/
	enyo.forEach = function(inArray, inFunc, inContext) {
		if (inArray) {
			var c = inContext || this;
			if (enyo.isArray(inArray) && inArray.forEach) {
				inArray.forEach(inFunc, c);
			} else {
				var a = Object(inArray);
				var al = a.length >>> 0;
				for (var i = 0; i < al; i++) {
					if (i in a) {
						inFunc.call(c, a[i], i, a);
					}
				}
			}
		}
	};

	/**
		Invokes _inFunc_ on each element of _inArray_, and returns the results as an Array.
		If _inContext_ is specified, _inFunc_ is called with _inContext_ as _this_.
	*/
	enyo.map = function(inArray, inFunc, inContext) {
		var c = inContext || this;
		if (enyo.isArray(inArray) && inArray.map) {
			return inArray.map(inFunc, c);
		} else {
			var results = [];
			var add = function(e, i, a) {
				results.push(inFunc.call(c, e, i, a));
			};
			enyo.forEach(inArray, add, c);
			return results;
		}
	};

	//*@public
	/**
		Concatenates a variable number of arrays, removing any duplicate
		entries.
	*/
	enyo.merge = function (/* _arrays_ */) {
		var merger = Array.prototype.concat.apply([], arguments);
		return unique(merger);
	};
	var merge = enyo.merge;

	//*@public
	/**
		Takes a variable number of arrays and returns an array of
		values that are unique across all of the arrays. Note that
		this is not a particularly cheap method and should never be
		called recursively.

		TODO: test in IE8
		TODO: figure out why the one-hit reversal wasn't working
	*/
	enyo.union = function (/* _arrays_ */) {
		// create one large array of all of the arrays passed to
		// the method for comparison
		var values = Array.prototype.concat.apply([], arguments);
		// the array of seen values
		var seen = [];
		// the array of values actually to be returned
		var ret = [];
		var idx = 0;
		var len = values.length;
		var value;
		for (; idx < len; ++idx) {
			value = values[idx];
			// if we haven't seen this value before go ahead and
			// push it to the seen array
			if (!~enyo.indexOf(value, seen)) {
				seen.push(value);
				// here we check against the entirety of any other values
				// in the values array starting from the end
				if (idx === lastIndexOf(value, values)) {
					// if this turned out to be true then it is a unique entry
					// so go ahead and push it to our union array
					ret.push(value);
				}
			}
		}
		// we should have a flattened/unique array now, return it
		return ret;
	};
	var union = enyo.union;
	//*@public
	/**
		Returns the unique values found in one or more arrays.
	*/
	enyo.unique = union;
	var unique = enyo.unique;

	//*@public
	/**
		Reduces one or more arrays, removing any duplicate entries
		across them.
	*/
	enyo.reduce = merge;

	//*@public
	/**
		Convenience method that takes an array of properties and an object
		as parameters. Returns a new object with just those properties named
		in the array that are found to exist on the base object. If the third
		parameter is true, falsy values will be ignored.
	*/
	enyo.only = function (properties, object, ignore) {
		var ret = {};
		var idx = 0;
		var len;
		var property;
		// sanity check the properties array
		if (!exists(properties) || !(properties instanceof Array)) {
			return ret;
		}
		// sanity check the object
		if (!exists(object) || "object" !== typeof object) {
			return ret;
		}
		// reduce the properties array to just unique entries
		properties = unique(properties);
		// iterate over the properties given and if the property exists on
		// the object copy its value to the return array
		for (len = properties.length; idx < len; ++idx) {
			property = properties[idx];
			if (property in object) {
				if (true === ignore && !object[property]) {
					continue;
				}
				ret[property] = object[property];
			}
		}
		// return the array of values we found for the given properties
		return ret;
	};

	//*@public
	/**
		Convenience method that takes two objects as parameters. For each key
		from the first object, if the key also exists in the second object, a
		mapping of the key from the first object to the key from the second
		object is added to a result object, which is eventually returned. In
		other words, the returned object maps the named properties of the
		first object to the named properties of the second object. The optional
		third parameter is a boolean designating whether to pass unknown key/value
		pairs through to the new object. If true, those keys will exist on the
		returned object.
	*/
	enyo.remap = function (map, obj, pass) {
		var $key, $val, $ret = pass? enyo.clone(obj): {};
		for ($key in map) {
			$val = map[$key];
			if ($key in obj) {
				$ret[$val] = obj.get? obj.get($key): obj[$key];
			}
		}
		return $ret;
	};

	//*@public
	/**
		Convenience method that takes an array of properties and an object
		as parameters. Returns a new object with all of the keys in the
		object except those specified in the _properties_ array. The values
		are shallow copies.
	*/
	enyo.except = function (properties, object) {
		// the new object to return with just the requested keys
		var ret = {};
		var keep;
		var idx = 0;
		var len;
		var key;
		// sanity check the properties array
		if (!exists(properties) || !(properties instanceof Array)) {
			return ret;
		}
		// sanity check the object
		if (!exists(object) || "object" !== typeof object) {
			return ret;
		}
		// we want to only use the union of the properties and the
		// available keys on the object
		keep = union(properties, keys(object));
		// for every property in the keep array now copy that to the new
		// hash
		for (len = keep.length; idx < len; ++idx) {
			key = keep[idx];
			// if the key was specified in the properties array but does not
			// exist in the object ignore it
			if (!(key in object)) {
				continue;
			}
			ret[key] = object[key];
		}
		// return the new hash
		return ret;
	};

	//*@public
	/**
		Helper method that accepts an array of objects and returns
		a hash of those objects indexed by the specified property. If a filter
		is provided, it should accept four parameters: the key, the value
		(object), the current mutable map reference, and an immutable
		copy of the original array of objects for comparison.
	*/
	enyo.indexBy = function (property, array, filter) {
		// the return value - indexed map from the given array
		var map = {};
		var value;
		var len;
		var idx = 0;
		// sanity check for the array with an efficient native array check
		if (!exists(array) || !(array instanceof Array)) {
			return map;
		}
		// sanity check the property as a string
		if (!exists(property) || "string" !== typeof property) {
			return map;
		}
		// the immutable copy of the array
		var copy = enyo.clone(array);
		// test to see if filter actually exsits
		filter = exists(filter) && "function" === typeof filter? filter: undefined;
		for (len = array.length; idx < len; ++idx) {
			// grab the value from the array
			value = array[idx];
			// make sure that it exists and has the requested property at all
			if (exists(value) && exists(value[property])) {
				if (filter) {
					// if there was a filter use it - it is responsible for
					// updating the map accordingly
					filter(property, value, map, copy);
				} else {
					// use the default behavior - check to see if the key
					// already exists on the map it will be overwritten
					map[value[property]] = value;
				}
			}
		}
		// go ahead and return our modified map
		return map;
	};

	//*@public
	/**
		Expects as parameters a string, _property_, and an array of objects
		that may have the named property. Returns an array of all the values
		of the named property in the objects in the array.
	*/
	enyo.pluck = function (property, array) {
		var ret = [];
		var idx = 0;
		var len;
		// if we don't have a property to look for or an array of
		// objects to search through we have to return an empty array
		if (!(exists(property) && exists(array))) {
			return ret;
		}
		// if it isn't actually an array, return an empty array
		if (!(array instanceof Array)) {
			return ret;
		}
		// if property isn't a string, then return an empty array
		if ("string" !== typeof property) {
			return ret;
		}
		// now that sanity is established to some extent, let's get
		// to work
		for (len = array.length; idx < len; ++idx) {
			// if the object in the array is actually undefined, skip
			if (!exists(array[idx])) {
				continue;
			}
			// if it was found, then check to see if the property
			// exists on it
			if (exists(array[idx][property])) {
				ret.push(array[idx][property]);
			}
		}
		// return whatever we found, if anything
		return ret;
	};

	/**
		Creates a new array with all elements of _inArray_ that pass the test
		defined by _inFunc_. If _inContext_ is specified, _inFunc_ is called
		with _inContext_ as _this_.
	*/
	enyo.filter = function(inArray, inFunc, inContext) {
		var c = inContext || this;
		if (enyo.isArray(inArray) && inArray.filter) {
			return inArray.filter(inFunc, c);
		} else {
			var results = [];
			var f = function(e, i, a) {
				var eo = e;
				if (inFunc.call(c, e, i, a)) {
					results.push(eo);
				}
			};
			enyo.forEach(inArray, f, c);
			return results;
		}
	};

	/**
		Returns an array of all own enumerable properties found on _inObject_.
	*/
	enyo.keys = Object.keys || function(inObject) {
		var results = [];
		var hop = Object.prototype.hasOwnProperty;
		for (var prop in inObject) {
			if (hop.call(inObject, prop)) {
				results.push(prop);
			}
		}
		// *sigh* IE 8
		if (!({toString: null}).propertyIsEnumerable("toString")) {
			var dontEnums = [
				'toString',
				'toLocaleString',
				'valueOf',
				'hasOwnProperty',
				'isPrototypeOf',
				'propertyIsEnumerable',
				'constructor'
			];
			for (var i = 0, p; (p = dontEnums[i]); i++) {
				if (hop.call(inObject, p)) {
					results.push(p);
				}
			}
		}
		return results;
	};
	var keys = enyo.keys;

	/**
		Clones an existing Array, or converts an array-like object into an Array.

		If _inOffset_ is non-zero, the cloning starts from that index in the source Array.
		The clone may be appended to an existing Array by passing the existing Array as _inStartWith_.

		Array-like objects have _length_ properties, and support square-bracket notation ([]).
		Often, array-like objects do not support Array methods, such as _push_ or _concat_, and
		so must be converted to Arrays before use.

		The special _arguments_ variable is an example of an array-like object.
	*/
	enyo.cloneArray = function(inArrayLike, inOffset, inStartWith) {
		var arr = inStartWith || [];
		for(var i = inOffset || 0, l = inArrayLike.length; i<l; i++){
			arr.push(inArrayLike[i]);
		}
		return arr;
	};
	enyo.toArray = enyo.cloneArray;

	/**
		Shallow-clones an object or an array.
	*/
	enyo.clone = function(obj) {
		return enyo.isArray(obj) ? enyo.cloneArray(obj) : enyo.mixin({}, obj);
	};

	//* @protected
	var empty = {};

	//* @public
	/**
		Will take a variety of options to ultimately mix a set of properties
		from objects into single object. All configurations accept a boolean as
		the final parameter to indicate whether or not to ignore _truthy_/_existing_
		values on any _objects_ prior.

		If `target` exists and is an object will be the base for all properties
		and the returned value. If the parameter is used but is _falsy_ a new
		object will be created and returned. If no such parameter exists the first
		parameter must be an array of objects and a new object will be created as
		the `target`.

		The `source` parameter may be an object or an array of objects. If no `target`
		parameter is provided `source` must be an array of objects.

		The `options` parameter allows you to set the `ignore` and/or `exists` flags
		such that if `ignore` is true it will not override any truthy values in the
		target and if `exists` is true it will only use truthy values from any of
		the sources.

		Setting `options` to true will set all options to true.
	*/
	enyo.mixin = function(target, source, options) {
		// the return object/target
		var $t;
		// the source or sources to use
		var $s;
		var $o, $i, $n, s$;
		if (enyo.isArray(target)) {
			$t = {};
			$s = target;
			if (source && enyo.isObject(source)) {
				$o = source;
			}
		} else {
			$t = target || {};
			$s = source;
			$o = options;
		}
		var release = false;
		if (!enyo.isObject($o)) {
			$o = enyo.pool.claimObject();
			release = true;
		}
		if (true === options) {
			$o.ignore = true;
			$o.exists = true;
		}
		// here we handle the array of sources
		if (enyo.isArray($s)) {
			for ($i=0; (s$=$s[$i]); ++$i) {
				enyo.mixin($t, s$, $o);
			}
		} else {
		// otherwise we execute singularly
			for ($n in $s) {
				s$ = $s[$n];
				if (empty[$n] !== s$) {
					if ((!$o.exists || s$) && (!$o.ignore || !$t[$n])) {
						$t[$n] = s$;
					}
				}
			}
		}
		if (release) {
			enyo.pool.releaseObject($o);
		}
		return $t;
	};

	//* @public
	/**
		Returns a function closure that will call (and return the value of)
		function _method_, with _scope_ as _this_.

		_method_ may be a function or the string name of a function-valued
		property on _scope_.

		Arguments to the closure are passed into the bound function.

			// a function that binds this to this.foo
			var fn = enyo.bind(this, "foo");
			// the value of this.foo(3)
			var value = fn(3);

		Optionally, any number of arguments may be prefixed to the bound function.

			// a function that binds this to this.bar, with arguments ("hello", 42)
			var fn = enyo.bind(this, "bar", "hello", 42);
			// the value of this.bar("hello", 42, "goodbye");
			var value = fn("goodbye");

		Functions may be bound to any scope.

			// binds function 'bar' to scope 'foo'
			var fn = enyo.bind(foo, bar);
			// the value of bar.call(foo);
			var value = fn();
	*/
	enyo.bind = function(scope, method/*, bound arguments*/){
		if (!method) {
			method = scope;
			scope = null;
		}
		scope = scope || enyo.global;
		if (enyo.isString(method)) {
			if (scope[method]) {
				method = scope[method];
			} else {
				throw(['enyo.bind: scope["', method, '"] is null (scope="', scope, '")'].join(''));
			}
		}
		if (enyo.isFunction(method)) {
			var args = enyo.cloneArray(arguments, 2);
			if (method.bind) {
				return method.bind.apply(method, [scope].concat(args));
			} else {
				return function() {
					var nargs = enyo.cloneArray(arguments);
					// invoke with collected args
					return method.apply(scope, args.concat(nargs));
				};
			}
		} else {
			throw(['enyo.bind: scope["', method, '"] is not a function (scope="', scope, '")'].join(''));
		}
	};

	/**
		Calls method _inMethod_ on _inScope_ asynchronously.

		Uses _window.setTimeout_ with minimum delay, usually around 10ms.

		Additional arguments are passed to _inMethod_ when it is invoked.
	*/
	enyo.asyncMethod = function(inScope, inMethod/*, inArgs*/) {
		return setTimeout(enyo.bind.apply(enyo, arguments), 1);
	};

	/**
		Calls named method _inMethod_ (String) on _inObject_ with optional
		arguments _inArguments_ (Array), if the object and method exist.

			enyo.call(myWorkObject, "doWork", [3, "foo"]);
	*/
	enyo.call = function(inObject, inMethod, inArguments) {
		var context = inObject || this;
		if (inMethod) {
			var fn = context[inMethod] || inMethod;
			if (fn && fn.apply) {
				return fn.apply(context, inArguments || []);
			}
		}
	};

	/**
		Returns the current time.

		The returned value is equivalent to _new Date().getTime()_.
	*/
	enyo.now = Date.now || function() {
		return new Date().getTime();
	};

	//* @protected

	enyo.nop = function(){};
	enyo.nob = {};
	enyo.nar = [];

	// this name is reported in inspectors as the type of objects created via delegate,
	// otherwise we would just use enyo.nop
	enyo.instance = function() {};

	// some platforms need alternative syntax (e.g., when compiled as a v8 builtin)
	if (!enyo.setPrototype) {
		enyo.setPrototype = function(ctor, proto) {
			ctor.prototype = proto;
		};
	}

	// boodman/crockford delegation w/cornford optimization
	enyo.delegate = function(proto) {
		enyo.setPrototype(enyo.instance, proto);
		return new enyo.instance();
	};

	//* @public

	/**
		Provides a stub function for _g11n_ string translation. This allows
		strings to be wrapped in preparation for localization. If the _g11n_
		library is not loaded, this function will return the string as is.

			$L('Welcome')

		If the _g11n_ library is loaded, this function will be replaced by the
		_g11n_ library version, which translates wrapped strings to strings from
		a developer-provided resource file corresponding to the current user
		locale.
	*/
	window.$L = function(string) {
		return string;
	};
})();

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/log.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qIEBwcm90ZWN0ZWQKCmVueW8ubG9nZ2luZyA9IHsKCS8vIGxvZyBsZXZlbHMgYXJlIGludGVnZXJzIGZyb20gMC05OQoJLy8gOTkgaXMgbWF4aW11bSBsb2dnaW5nCglsZXZlbDogOTksCgkvLyBzZXQgbGV2ZWwgdG8gLTEgdG8gZGlzYWJsZSBhbGwgbG9nZ2luZwoJbGV2ZWxzOiB7bG9nOiAyMCwgd2FybjogMTAsIGVycm9yOiAwfSwKCS8vIHJldHVybiB0cnVlIGlmIGxvZ2dpbmcgbGV2ZWwgaXMgbG93ZXIgdGhhbiB0aGUgY3VycmVudCBsb2cgbGV2ZWwKCXNob3VsZExvZzogZnVuY3Rpb24oaW5NZXRob2QpIHsKCQl2YXIgbGwgPSBwYXJzZUludCh0aGlzLmxldmVsc1tpbk1ldGhvZF0sIDApOwoJCXJldHVybiAobGwgPD0gdGhpcy5sZXZlbCk7Cgl9LAoJLyoKCWZvcm1hdEFyZ3M6IGZ1bmN0aW9uKGluTWV0aG9kLCBpbkFyZ3MpIHsKCQl2YXIgYSQgPSBbXTsKCQlmb3IgKHZhciBpPTAsIGw9aW5BcmdzLmxlbmd0aCwgYTsgKGE9aW5BcmdzW2ldKSB8fCBpPGw7IGkrKykgewoJCQlpZiAoU3RyaW5nKGEpID09ICJbb2JqZWN0IE9iamVjdF0iKSB7CgkJCQlhID0gZW55by5qc29uLnN0cmluZ2lmeShhKTsKCQkJfQoJCQlhJC5wdXNoKGEpOwoJCX0KCQlyZXR1cm4gYSQ7Cgl9LAoJKi8KCV9sb2c6IGZ1bmN0aW9uKGluTWV0aG9kLCBpbkFyZ3MpIHsKCQkvLyBhdm9pZCB0cnlpbmcgdG8gdXNlIGNvbnNvbGUgb24gSUUgaW5zdGFuY2VzIHdoZXJlIHRoZSBvYmplY3QgaGFzbid0IGJlZW4KCQkvLyBjcmVhdGVkIGR1ZSB0byB0aGUgZGV2ZWxvcGVyIHRvb2xzIGJlaW5nIHVub3BlbmVkCgkJdmFyIGNvbnNvbGUgPSB3aW5kb3cuY29uc29sZTsKCQlpZiAodHlwZW9mIGNvbnNvbGUgPT09ICJ1bmRlZmluZWQiKSB7CiAgICAgICAgICAgIHJldHVybjsKICAgICAgICB9CgkJLy92YXIgYSQgPSBlbnlvLmxvZ2dpbmcuZm9ybWF0QXJncyhpbk1ldGhvZCwgaW5BcmdzKTsKCQl2YXIgYSQgPSBlbnlvLmlzQXJyYXkoaW5BcmdzKSA/IGluQXJncyA6IGVueW8uY2xvbmVBcnJheShpbkFyZ3MpOwoJCWlmIChlbnlvLmR1bWJDb25zb2xlKSB7CgkJCS8vIGF0IGxlYXN0IGluIGVhcmx5IHZlcnNpb25zIG9mIHdlYm9zLCBjb25zb2xlLiogb25seSBhY2NlcHQgYSBzaW5nbGUgYXJndW1lbnQKCQkJYSQgPSBbYSQuam9pbigiICIpXTsKCQl9CgkJdmFyIGZuID0gY29uc29sZVtpbk1ldGhvZF07CgkJaWYgKGZuICYmIGZuLmFwcGx5KSB7CgkJCS8vIHNvbWUgY29uc29sZXMgc3VwcG9ydCAnd2FybicsICdpbmZvJywgYW5kIHNvIG9uCgkJCWZuLmFwcGx5KGNvbnNvbGUsIGEkKTsKCQl9IGVsc2UgaWYgKGNvbnNvbGUubG9nLmFwcGx5KSB7CgkJCS8vIHNvbWUgY29uc29sZXMgc3VwcG9ydCBjb25zb2xlLmxvZy5hcHBseQoJCQljb25zb2xlLmxvZy5hcHBseShjb25zb2xlLCBhJCk7CgkJfSBlbHNlIHsKCQkJLy8gb3RoZXJ3aXNlLCBkbyBvdXIgb3duIGZvcm1hdHRpbmcKCQkJY29uc29sZS5sb2coYSQuam9pbigiICIpKTsKCQl9Cgl9LAoJbG9nOiBmdW5jdGlvbihpbk1ldGhvZCwgaW5BcmdzKSB7CgkJdmFyIGNvbnNvbGUgPSB3aW5kb3cuY29uc29sZTsKCQlpZiAodHlwZW9mIGNvbnNvbGUgIT09ICJ1bmRlZmluZWQiKSB7CgkJCWlmICh0aGlzLnNob3VsZExvZyhpbk1ldGhvZCkpIHsKCQkJCXRoaXMuX2xvZyhpbk1ldGhvZCwgaW5BcmdzKTsKCQkJfQoJCX0KCX0KfTsKCi8vKiBAcHVibGljCgovKioKCVNldHMgdGhlIGxvZyBsZXZlbCBmb3IgdGhpcyB3aW5kb3cgaWYgdGhlIGlucHV0IGlzIGEgcmVhbCBudW1iZXIuCgoJVGhlIGxvZyBsZXZlbCBpcyB1c2VkIGFzIGEgd2F0ZXJtYXJrIHRvIGNvbnRyb2wgdGhlIGFtb3VudCBvZiBsb2dnaW5nLgoJU2V0dGluZyB0aGUgbG9nIGxldmVsIGxvd2VyIHdpbGwgcHJldmVudCBsb2dnaW5nIGZ1bmN0aW9ucyB3aXRoIGEgaGlnaGVyCglsZXZlbCBmcm9tIGJlaW5nIGV4ZWN1dGVkLgoKCVRoZSBkZWZhdWx0IGxvZyBsZXZlbCBpcyA5OS4gIDxhIGhyZWY9IiNlbnlvLmxvZyI+ZW55by5sb2c8L2E+IHdpbGwgb3V0cHV0CglpZiB0aGUgbGV2ZWwgaXMgMjAgb3IgYWJvdmUsIDxhIGhyZWY9IiNlbnlvLndhcm4iPmVueW8ud2FybjwvYT4gYXQgMTAsIGFuZAoJPGEgaHJlZj0iI2VueW8uZXJyb3IiPmVueW8uZXJyb3I8L2E+IGF0IDAuCiovCmVueW8uc2V0TG9nTGV2ZWwgPSBmdW5jdGlvbihpbkxldmVsKSB7Cgl2YXIgbGwgPSBwYXJzZUludChpbkxldmVsLCAwKTsKCWlmIChpc0Zpbml0ZShsbCkpIHsKCQllbnlvLmxvZ2dpbmcubGV2ZWwgPSBsbDsKCX0KfTsKCi8qKgoJU2VuZHMgYSBsb2cgbWVzc2FnZSB0byB0aGUgY29uc29sZSwgaWYgdGhlIGN1cnJlbnQgbG9nIGxldmVsIGFsbG93cyBmb3IgaXQuCgoJT2JqZWN0cyBhcmUgY29udmVydGVkIHRvIEpTT04gYXV0b21hdGljYWxseS4KCglNdWx0aXBsZSBhcmd1bWVudHMgYXJlIGNvZXJjZWQgdG8gU3RyaW5nIGFuZCBqb2luZWQgd2l0aCBzcGFjZXMuCiovCmVueW8ubG9nID0gZnVuY3Rpb24oKSB7CgllbnlvLmxvZ2dpbmcubG9nKCJsb2ciLCBhcmd1bWVudHMpOwp9OwoKLy8qIFNhbWUgYXMgX2xvZ18sIGV4Y2VwdCB1c2VzIHRoZSBjb25zb2xlJ3Mgd2FybiBtZXRob2QgKGlmIGl0IGV4aXN0cykuCmVueW8ud2FybiA9IGZ1bmN0aW9uKCkgewoJZW55by5sb2dnaW5nLmxvZygid2FybiIsIGFyZ3VtZW50cyk7Cn07CgovLyogU2FtZSBhcyBfbG9nXywgZXhjZXB0IHVzZXMgdGhlIGNvbnNvbGUncyBlcnJvciBtZXRob2QgKGlmIGl0IGV4aXN0cykuCmVueW8uZXJyb3IgPSBmdW5jdGlvbigpIHsKCWVueW8ubG9nZ2luZy5sb2coImVycm9yIiwgYXJndW1lbnRzKTsKfTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/mixins/ApplicationSupport.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qQHB1YmxpYwovKioKCVRoZSBfZW55by5BcHBsaWNhdGlvblN1cHBvcnRfIG1peGluIGFkZHMgZ2VuZXJpYyBzdXBwb3J0IHRvIGFueQoJX2VueW8uQ29tcG9uZW50XyBzdWNoIHRoYXQgYW55dGltZSB0aGUgX2NyZWF0ZUNvbXBvbmVudF8gKG9yCglfY3JlYXRlQ29tcG9uZW50c18pIG1ldGhvZCBpcyBjYWxsZWQsIGl0IHdpbGwgc3VwcGx5IGEgcmVmZXJlbmNlCgl0byB0aGUgYXBwbGljYXRpb24gaW5zdGFuY2UgaXQgYmVsb25ncyB0byBhcyB0aGUgX2FwcF8gcHJvcGVydHksIGZvcgoJdXNlIGluIGRldGVybWluaW5nIHJlbGF0aXZlIHBhdGhzIHRvIGFwcGxpY2F0aW9uLXNjb3BlZCBjb250cm9sbGVycwoJb3Igc3RhdGUuCiovCmVueW8uY3JlYXRlTWl4aW4oewoKCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJLy8gUFVCTElDIFBST1BFUlRJRVMKCgkvLypAcHVibGljCgluYW1lOiAiZW55by5BcHBsaWNhdGlvblN1cHBvcnQiLAoKCS8vKkBwdWJsaWMKCWFwcDogbnVsbCwKCgkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCS8vIFBST1RFQ1RFRCBQUk9QRVJUSUVTCgoJLy8qQHByb3RlY3RlZAoJX3N1cHBvcnRzQXBwbGljYXRpb25zOiB0cnVlLAoKCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJLy8gUFJPVEVDVEVEIE1FVEhPRFMKCgkvLypAcHJvdGVjdGVkCgkvKioKCQlPdmVybG9hZCB0aGlzIG1ldGhvZCB0byBhZGQgdGhlIF9hcHBfIHByb3BlcnR5IHRvCgkJYWxsIGNoaWxkIGNvbXBvbmVudHMuCgkqLwoJYWRqdXN0Q29tcG9uZW50UHJvcHM6IGZ1bmN0aW9uIChwcm9wcykgewoJCS8vIGNvcHkgdGhlIHJlZmVyZW5jZSBvZiB0aGlzIGNvbXBvbmVudCB0byB0aGUgY2hpbGQgY29tcG9uZW50CgkJLy8gcHJvcGVydGllcyBoYXNoCgkJcHJvcHMuYXBwID0gdGhpcy5hcHA7CgkJcmV0dXJuIHRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cywgcHJvcHMpOwoJfSwKCgkvLypAcHJvdGVjdGVkCglkZXN0cm95OiBmdW5jdGlvbiAoKSB7CgkJdGhpcy5hcHAgPSB1bmRlZmluZWQ7Cgl9Cgp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/mixins/AutoBindingSupport.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uIChlbnlvKSB7CgoJLy8qQHB1YmxpYwoJLyoqCgkJVGhpcyBoYXNoIGNvbnRhaW5zIGEgbWFwcGluZyBvZiB0aGUgYXV0byBiaW5kaW5nIEFQSSBwcm9wZXJ0aWVzCgkJdG8gdGhlaXIgZXF1aXZhbGVudHMgaW4gdGhlIGFjdHVhbCBiaW5kaW5ncyBBUEkuCgkqLwoJdmFyIHJlbWFwcGVkID0gewoJCWJpbmRGcm9tOiAiZnJvbSIsCgkJYmluZFRvOiAidG8iLAoJCWJpbmRUcmFuc2Zvcm06ICJ0cmFuc2Zvcm0iLAoJCWJpbmRPbmVXYXk6ICJvbmVXYXkiLAoJCWJpbmRUd29XYXk6ICJ0d29XYXkiLAoJCWJpbmRBdXRvU3luYzogImF1dG9TeW5jIiwKCQliaW5kRGVidWc6ICJkZWJ1ZyIsCgkJYmluZFNvdXJjZTogInNvdXJjZSIKCX07CgoJLy8qQHB1YmxpYwoJLyoqCgkJVGhpcyBoYXNoIGNvbnRhaW5zIHRoZSBkZWZhdWx0IHByb3BlcnRpZXMgYXBwbGllZCB0byBuZXcgYmluZGluZ3MKCQlnZW5lcmF0ZWQgYnkgdGhlIGF1dG8gYmluZGluZ3Mgc2V0dXAuIFRoZXNlIHByb3BlcnRpZXMgZmlsbCBpbiBnYXBzCgkJd2hlcmUgbmV3IHZhbHVlcyBhcmUgbm90IHN1cHBsaWVkLCBidXQgZG8gbm90IG92ZXJyaWRlIHNwZWNpZmllZAoJCXZhbHVlcy4KCSovCgl2YXIgZGVmYXVsdHMgPSB7CgkJdG86ICIuY29udGVudCIsCgkJdHJhbnNmb3JtOiBudWxsLAoJCW9uZVdheTogdHJ1ZSwKCQl0d29XYXk6IGZhbHNlLAoJCWF1dG9TeW5jOiB0cnVlLAoJCWRlYnVnOiBmYWxzZQoJfTsKCgkvLypAcHJvdGVjdGVkCgl2YXIgX3NldHVwQXV0b0JpbmRpbmdzID0gZnVuY3Rpb24gKCkgewoJCWlmICghdGhpcy5fc3VwcG9ydHNBdXRvQmluZGluZ3MpIHsKCQkJcmV0dXJuOwoJCX0KCQl2YXIgY29udHJvbHMgPSB0aGlzLmdldCgiX2JpbmRhYmxlQ29udHJvbHMiKTsKCQl2YXIgaWR4ID0gMDsKCQl2YXIgbGVuID0gY29udHJvbHMubGVuZ3RoOwoJCXZhciBiaW5kU291cmNlID0gdGhpcy5iaW5kU291cmNlIHx8IHRoaXMubW9kZWw7CgkJdmFyIGNvbnRyb2w7CgkJdmFyIHByb3BzOwoJCXZhciBiOwoJCXZhciB4dHJhOwoJCWZvciAoOyBpZHggPCBsZW47ICsraWR4KSB7CgkJCXh0cmEgPSBlbnlvLnBvb2wuY2xhaW1PYmplY3QoKTsKCQkJY29udHJvbCA9IGNvbnRyb2xzW2lkeF07CgkJCXByb3BzID0gdGhpcy5fYmluZFByb3BlcnRpZXMoY29udHJvbCk7CgkJCWlmIChwcm9wcy5zb3VyY2UgJiYgZW55by5pc1N0cmluZyhwcm9wcy5zb3VyY2UpKSB7CgkJCQlwcm9wcy5zb3VyY2UgPSBlbnlvLmdldFBhdGguY2FsbChjb250cm9sLCBwcm9wcy5zb3VyY2UpOwoJCQl9CgkJCXh0cmEuc291cmNlID0gcHJvcHMuc291cmNlIHx8IGJpbmRTb3VyY2U7CgkJCXh0cmEudGFyZ2V0ID0gY29udHJvbDsKCQkJZW55by5taXhpbihwcm9wcywgeHRyYSk7CgkJCWlmICgoYj1jb250cm9sLl9hdXRvQmluZGluZykpIHsKCQkJCWIuZGlzY29ubmVjdCgpOwoJCQkJZW55by5taXhpbihiLCBwcm9wcywge2V4aXN0czogdHJ1ZX0pOwoJCQkJYi5yZWZyZXNoKCk7CgkJCX0gZWxzZSB7CgkJCQliID0gdGhpcy5fYXV0b0JpbmRpbmcocHJvcHMpOwoJCQkJY29udHJvbC5fYXV0b0JpbmRpbmcgPSBiOwoJCQl9CgkJCWVueW8ucG9vbC5yZWxlYXNlT2JqZWN0KHh0cmEpOwoJCX0KCX07CgoJLy8qQHB1YmxpYwoJLyoqCgkJVGhlIF9lbnlvLkF1dG9CaW5kaW5nU3VwcG9ydF8gbWl4aW4gcHJvdmlkZXMgdGhlIGFiaWxpdHkgdG8KCQlkZWZpbmUgYmluZGluZ3MgZnJvbSBjaGlsZCBjb21wb25lbnRzIHJlbGF0aXZlIHRvIHRoZWlyIF9vd25lcl8uCgkJSW4gYSBjb21wb25lbnQgZGVjbGFyYXRpb24gKG9iamVjdCBsaXRlcmFsIGhhc2ggZm9yIGNvbXBvbmVudHMpLAoJCXNpbXBseSB1c2UgdGhlIGF2YWlsYWJsZSBhdXRvIGJpbmRpbmdzIEFQSSBhbmQgYSBiaW5kaW5nIHdpbGwKCQliZSBjcmVhdGVkIGFjY29yZGluZ2x5LgoJKi8KCWVueW8uY3JlYXRlTWl4aW4oewoKCQkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCQkvLyBQVUJMSUMgUFJPUEVSVElFUwoKCQkvLypAcHVibGljCgkJbmFtZTogImVueW8uQXV0b0JpbmRpbmdTdXBwb3J0IiwKCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gUFJPVEVDVEVEIFBST1BFUlRJRVMKCgkJLy8qQHByb3RlY3RlZAoJCV9zdXBwb3J0c0F1dG9CaW5kaW5nczogdHJ1ZSwKCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gQ09NUFVURUQgUFJPUEVSVElFUwoKCQkvLypAcHJvdGVjdGVkCgkJX2JpbmRhYmxlQ29udHJvbHM6IGVueW8uY29tcHV0ZWQoZnVuY3Rpb24gKGNvbnRyb2wpIHsKCQkJY29udHJvbCA9IGNvbnRyb2wgfHwgdGhpczsKCQkJdmFyIGJpbmRhYmxlID0gW107CgkJCXZhciBjb250cm9scyA9IGNvbnRyb2wuY29udHJvbHMgfHwgW107CgkJCXZhciBpZHggPSAwOwoJCQl2YXIgbGVuID0gY29udHJvbHMubGVuZ3RoOwoJCQlmb3IgKDsgaWR4IDwgbGVuOyArK2lkeCkgewoJCQkJYmluZGFibGUgPSBiaW5kYWJsZS5jb25jYXQodGhpcy5fYmluZGFibGVDb250cm9scyhjb250cm9sc1tpZHhdKSk7CgkJCX0KCQkJaWYgKCJiaW5kRnJvbSIgaW4gY29udHJvbCkgewoJCQkJYmluZGFibGUucHVzaChjb250cm9sKTsKCQkJfQoJCQlyZXR1cm4gYmluZGFibGU7CgkJfSwge2NhY2hlZDogdHJ1ZX0pLAoKCQkvLypAcHJvdGVjdGVkCgkJX2JpbmRpbmdEZWZhdWx0czogZW55by5jb21wdXRlZChmdW5jdGlvbiAoKSB7CgkJCXZhciBjdG9yID0gdGhpcy5nZXQoIl9iaW5kaW5nQ29uc3RydWN0b3IiKTsKCQkJdmFyIGtleXMgPSBlbnlvLmtleXMoZGVmYXVsdHMpOwoJCQlpZiAoZW55by5CaW5kaW5nICE9PSBjdG9yKSB7CgkJCQlyZXR1cm4gZW55by5taXhpbihlbnlvLmNsb25lKGRlZmF1bHRzKSwKCQkJCQllbnlvLm9ubHkoa2V5cywgY3Rvci5wcm90b3R5cGUsIHRydWUpKTsKCQkJfSBlbHNlIHsKCQkJCXJldHVybiBlbnlvLmNsb25lKGRlZmF1bHRzKTsKCQkJfQoJCX0sIHtjYWNoZWQ6IHRydWV9KSwKCgkJLy8qQHByb3RlY3RlZAoJCV9hdXRvQmluZGluZ3M6IGVueW8uY29tcHV0ZWQoZnVuY3Rpb24gKCkgewoJCQlyZXR1cm4gZW55by5maWx0ZXIodGhpcy5iaW5kaW5ncyB8fCBbXSwgZnVuY3Rpb24gKGJpbmQpIHsKCQkJCXJldHVybiBiaW5kICYmIGJpbmQuX2F1dG9CaW5kaW5nSWQ7CgkJCX0pOwoJCX0pLAoKCQkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCQkvLyBQUk9URUNURUQgTUVUSE9EUwoKCQkvLypAcHJvdGVjdGVkCgkJX2F1dG9CaW5kaW5nOiBmdW5jdGlvbiAoKSB7CgkJCXZhciBiaW5kID0gdGhpcy5iaW5kaW5nLmFwcGx5KHRoaXMsIGFyZ3VtZW50cyk7CgkJCWJpbmQuX2F1dG9CaW5kaW5nSWQgPSBlbnlvLnVpZCgiX2F1dG9CaW5kaW5nIik7CgkJCXJldHVybiBiaW5kOwoJCX0sCgoJCS8vKkBwcm90ZWN0ZWQKCQlfYmluZFByb3BlcnRpZXM6IGZ1bmN0aW9uIChjb250cm9sKSB7CgkJCXZhciBwcm9wcyA9IHRoaXMuZ2V0KCJfYmluZGluZ0RlZmF1bHRzIik7CgkJCXJldHVybiBlbnlvLm1peGluKGVueW8ucmVtYXAocmVtYXBwZWQsIGNvbnRyb2wpLCBwcm9wcywgdHJ1ZSk7CgkJfSwKCgkJLy8qQHB1YmxpYwoJCS8qKgoJCQlUaGlzIGlzIG92ZXJsb2FkZWQgc28gdGhhdCB3ZSBhbGxvdyB0aGUgZGVmYXVsdCBiZWhhdmlvciBmb3IKCQkJYW55IGJpbmRpbmdzIHRoYXQgYXJlIG5vdCBhdXRvLWJpbmRpbmdzIGFzIGEgbm9ybWFsIHJlYnVpbGQgd2lsbAoJCQlub3Qgd29yay4KCQkqLwoJCXJlYnVpbGRCaW5kaW5nczogZnVuY3Rpb24gKCkgewoJCQl2YXIgJGIgPSBlbnlvLmZpbHRlcih0aGlzLmJpbmRpbmdzIHx8IFtdLCBmdW5jdGlvbiAoYikgewoJCQkJcmV0dXJuIGIgJiYgIWIuX2F1dG9CaW5kaW5nOwoJCQl9KTsKCQkJdmFyICR0ID0gZW55by5wb29sLmNsYWltT2JqZWN0KCk7CgkJCSR0WzBdID0gJGI7CgkJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cywgJHQpOwoJCQlfc2V0dXBBdXRvQmluZGluZ3MuY2FsbCh0aGlzKTsKCQkJZW55by5wb29sLnJlbGVhc2VPYmplY3QoJHQpOwoJCX0sCgoJCS8vKkBwcm90ZWN0ZWQKCQljcmVhdGU6IGZ1bmN0aW9uICgpIHsKCQkJdmFyIHByb3AgPSB0aGlzLmJpbmRTb3VyY2UgfHwgIm1vZGVsIjsKCQkJLy8gTk9URTogV2UgZG9uJ3QgbmVlZCB0byByZWdpc3RlciBmb3IgdGhlIG1vZGVsIHByb3BlcnR5IGFzIGVueW8uQ29udHJvbAoJCQkvLyB3aWxsIGF1dG9tYXRpY2FsbHkgdHJpZ2dlciB0aGUgcmVidWlsZEJpbmRpbmdzIGNhbGwgd2hpY2ggd2UgaGF2ZSBtb2RpZmllZAoJCQkvLyB0byBoYW5kbGUgdGhpcyBwcm9wZXJseQoJCQlpZiAocHJvcCAhPSAibW9kZWwiKSB7CgkJCQl0aGlzLmFkZE9ic2VydmVyKHByb3AsIF9zZXR1cEF1dG9CaW5kaW5ncywgdGhpcyk7CgkJCX0KCQkJaWYgKHByb3AgPT0gIm1vZGVsIiAmJiB0aGlzLm1vZGVsICYmIHRoaXMubW9kZWwuX2lzTW9kZWwpIHsKCQkJCV9zZXR1cEF1dG9CaW5kaW5ncy5jYWxsKHRoaXMpOwoJCQl9CgkJfQoKCX0pOwoKfShlbnlvKSk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/mixins/BindingSupport.js"
Content-Type: application/octet-stream; x-encoding=base64
(function (enyo) {

	//*@public
	/**
		This mixin provides core support for bindings, bindings-arrays and a
		binding API to objects that implement it. It requires the computed
		property support and observer method support mixins.
	*/
	enyo.createMixin({

		// ...........................
		// PUBLIC PROPERTIES

		//*@public
		name: "enyo.BindingSupport",

		//*@public
		/**
			While binding kind may be overloaded on a per-binding basis
			for objects that intend to use a custom kind for all of their
			bindings, it may also be set here.
		*/
		defaultBindingKind: "enyo.Binding",

		//*@public
		/**
			An array of declared configurations for bindings that
			will be created on object instantiation.
		*/
		bindings: null,

		// ...........................
		// PROTECTED PROPERTIES

		//*@protected
		_supportsBindings: true,

		//*@protected
		_bindingsFromObservers: null,

		//*@protected
		_didSetupBindings: false,

		// ...........................
		// COMPUTED PROPERTIES

		//*@protected
		_bindingConstructor: enyo.computed(function () {
			return enyo.getPath(this.defaultBindingKind);
		}, {cached: true}),

		// ...........................
		// PUBLIC METHODS

		//*@public
		/**
			Accepts any number of hashes to be used to create a binding whose
			owner is this object by default.

			Returns a reference to the newly created binding and also adds the
			binding to this object's _bindings_ array.

			Note that calling the `destroy` method on a binding's owner will
			also clean up the binding itself.
		*/
		binding: function (/* _binding definitions_ */) {
			var definitions = arguments;
			var idx = 0;
			var len = definitions.length;
			var binding;
			var properties = {};
			var bindings = this.bindings;
			var def = this.get("_bindingConstructor");
			var Ctor;
			var kind;
			for (; idx < len; ++idx) {
				enyo.mixin(properties, definitions[idx]);
			}
			if ((kind = properties.kind)) {
				if ("string" === typeof kind) {
					Ctor = enyo.getPath(properties.kind);
				}
				else if ("function" === typeof kind) {
					Ctor = kind;
				}
			}
			if (!Ctor || "function" !== typeof Ctor) {
				Ctor = def;
			}
			binding = new Ctor({owner: this, autoConnect: true}, properties);
			bindings.push(binding);
			return binding;
		},

		//*@public
		/**
			Usually called when the object's `destroy` method is executed, but may
			be called at any time to properly clean up any bindings associated with
			this object (i.e., any bindings that have their _owner_ property set
			to this object).

			This method does not remove bindings that originated from another object
			but are	currently bound to a property on this object.

			If so desired, one may pass in an array of bindings, in which case only
			those bindings specified in the array will be destroyed.
		*/
		clearBindings: function (subset) {
			var $bindings = subset || this.bindings;
			if (!$bindings.length) {
				return;
			}
			do {
				$bindings.shift().destroy();
			} while ($bindings.length);
		},

		//*@public
		/**
			Calls the `refresh` method on the bindings associated with this
			object, or on a passed-in array of bindings.

			Differs from _rebuildBindings_ in that, instead of
			rediscovering the source and target of each binding, it
			remembers them from the most recent setup.

			In most scenarios, this method will be called automatically,
			with no need for explicit calls from the developer.
		*/
		refreshBindings: function (subset) {
			var $bindings = subset || this.bindings;
			var len = $bindings.length;
			var idx = 0;
			for (; idx < len; ++idx) {
				$bindings[idx].refresh();
			}
		},

		//*@public
		/**
			Calls the `rebuild` method on the bindings associated with this
			object, or on a passed-in array of bindings.

			Differs from _refreshBindings_ in that it forces the source and
			target of each binding to be rediscovered using the specified
			paths, rather than remembered from a previous setup.

			In most scenarios, this method will be called automatically,
			with no need for explicit calls from the developer.
		*/
		rebuildBindings: function (subset) {
			var $bindings = subset || this.bindings;
			var len = $bindings.length;
			var idx = 0;
			for (; idx < len; ++idx) {
				$bindings[idx].rebuild();
			}
		},

		//*@public
		/**
			This method is typically not called directly, but is called by the
			binding when it is destroyed. It accepts a single binding as its
			parameter; the binding is removed from the _bindings_ array if it
			exists there. This method does not destroy the binding or dereference
			its _owner_ property.
		*/
		removeBinding: function (binding) {
			// sanity check on binding
			if (!enyo.exists(binding) || !(binding instanceof enyo.Binding)) {
				return;
			}
			var bindings = this.bindings || [];
			var idx = enyo.indexOf(binding, bindings);
			if (!!~idx) {
				bindings.splice(idx, 1);
			}
		},

		//*@public
		/**
			We overload the _addObserver_ method from _ObserverMethodSupport_
			so that we can track which observers belong to bindings and, later,
			clean them up appropriately.
		*/
		addObserver: function (property, fn, context) {
			if (fn && fn.bindingId) {
				this._bindingsFromObservers.push(fn.bindingId);
			}
			return this.inherited(arguments);
		},

		// ...........................
		// PROTECTED METHODS

		//*@protected
		_constructor: function () {
			// initialize our bindings from observers array
			this._bindingsFromObservers = [];
			this.bindings = this.bindings || [];
			return this.inherited(arguments);
		},

		//*@protected
		create: function () {
			// we do a single pass at each of the binding declarations
			// and pass them to our binding creation method
			var $bindings = this.bindings || (this.bindings = []);
			var len = $bindings.length;
			var idx = 0;
			// we reset our bindings array because it will be used by our
			// binding method to store references to bindings owned by
			// this object
			this.bindings = [];
			for (; idx < len; ++idx) {
				this.binding($bindings[idx]);
			}
			// in case something needs to check to see if bindings have
			// already initialized
			this._didSetupBindings = true;
		},

		//*@protected
		destroy: function () {
			// we simply iterate over and destroy each of the bindings
			// in our bindings array
			var $bindings = this.bindings;
			var $ids = this._bindingsFromObservers;
			var $id;
			var $bind;
			if ($bindings.length) {
				do {
					$bindings.pop().destroy();
				} while ($bindings.length);
			}
			// check for any bindings associated with us that aren't
			// destroyed yet
			if ($ids.length) {
				while ($ids.length) {
					$id = $ids.pop();
					$bind = enyo.Binding.find($id);
					if ($bind && !$bind.destroyed) {
						$bind.destroy();
					}
				}
			}
		}

		// ...........................
		// OBSERVERS

	});

}(enyo));

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/mixins/ComputedPropertySupport.js"
Content-Type: application/octet-stream; x-encoding=base64
(function (enyo) {

	//*@public
	/**
		The possible/configurable options that may be passed in
		one or more hashes to the computed method when wrapping a
		function as a computed property of an object.
	*/
	var defaults = {

		//*@public
		/**
			If a computed property is marked as volatile, it will
			be executed on every request regardless of its dependencies.
			If both _volatile_ and _cached_ are set to true, the property
			is treated as _cached_, and _volatile_ is ignored.
		*/
		volatile: true,

		//*@public
		/**
			If a computed property is marked as cached, its value will be
			computed once and then reused, unless one of its dependencies
			has been flagged as being changed prior to the request. If
			there are no dependencies, the value will only ever be computed
			once.

			If this setting is true, it overrides	the value of the _volatile_
			property.
		*/
		cached: false,

		//*@public
		/**
			Most cacheable computed properties will not need to be evaluated
			until the first time their values are requested; however, you can
			force evaluation immediately after the associated object is
			instantiated by setting this flag to false.
		*/
		defer: true,

		//*@protected
		value: null,

		//*@protected
		dirty: 0
	};

	//*@public
	/**
		Wrapping a class method in _enyo.computed_ allows the method to be
		interpreted as a static property, bindable by an instance of
		_enyo.Binding_. The call to _enyo.computed_ accepts a configuration
		hash as an optional parameter, as well as any number of string
		parameters that will be evaluated as dependencies of the property.

		Notifications of changes to any of the dependencies will cause the
		computed property to be flagged as needing reevaluation the next
		time it is requested (if it is cached).

		Computed properties are _volatile_ by default and will be evaluated
		on each request unless marked otherwise.
	*/
	enyo.computed = function (fn /*, arguments */) {
		var deps = enyo.toArray(arguments).slice(1);
		var config;
		var properties;
		if (!enyo.exists(fn) || "function" !== typeof fn) {
			throw "enyo.computed: a computed property must be a function";
		}
		properties = fn.properties || (fn.properties = []);
		config = fn.config || enyo.clone(defaults);
		enyo.forEach(deps, function (dep) {
			if ("string" === typeof dep) {
				properties.push(dep);
			}
			else if ("object" === typeof dep) {
				// the assumption here is it must be a configuration
				// hash
				enyo.mixin(config, dep);
			}
		});
		if (false === config.volatile) {
			config.cached = true;
		}
		else if (true === config.cached) {
			config.volatile = false;
		}
		fn.config = config;
		fn._isProperty = true;
		config.properties = properties;
		return fn;
	};

	//*@protected
	var _isComputed = function (fn) {
		return fn && "function" === typeof fn && true === fn._isProperty;
	};

	//*@protected
	var _addDependent = function (proto, property, dependent) {
		var $map = proto._computedMap;
		if (!$map[dependent]) {
			$map[dependent] = [];
		}
		$map[dependent].push(property);
	};

	//*@protected
	var _addCacheable = function (proto, property) {
		var $cacheable = proto._computedCacheable;
		$cacheable.push(property);
	};

	//*@protected
	/**
		Adds an entry in the computed properties hash of the object so that
		it can be referenced easily later on. Also adds an entry for the
		property name of the computed property for each of its dependent
		properties, so that when they are mapped, the appropriate update
		can be triggered.
	*/
	var _addComputed = function (proto, property, fn) {
		var $computed = proto._computed;
		// TODO: this is assuming the desirable end is to override
		// any previous entry for a newer entry of the same property
		// name
		var $config = $computed[property] = fn.config;
		// if the property is configured as cacheable and not deferred
		// we add it to a special object to speed up initialization
		if ($config.cached && !$config.defer) {
			_addCacheable(proto, property);
		}
		// for every dependency we need to add it to the object
		enyo.forEach(fn.properties, function (dep) {
			_addDependent(proto, property, dep);
		});
	};

	//*@protected
	var _findComputed = function (proto, props, kind) {
		// no need to bother if this does not support computed properties
		if (!proto._supportsComputed) {
			return;
		}
		// otherwise we know it does and we need to make sure it has some
		// intial storage properties
		proto._computed = kind? enyo.clone(proto._computed || {}): proto._computed || {};
		proto._computedMap = kind? enyo.clone(proto._computedMap || {}) : proto._computedMap || {};
		proto._computedCacheable = kind? enyo.clone(proto._computedCacheable || []): proto._computedCacheable || [];
		// now we iterate over only the properties defined on this new
		// kind definition (or ones being added by a mixin) and check to
		// see if they are computed properties so we only have to do this
		// once and not at instance initialization time
		for (var prop in props) {
			if (props[prop] && _isComputed(props[prop])) {
				_addComputed(proto, prop, props[prop]);
			}
		}
	};

	//*@protected
	/**
		Called by the overloaded getter for objects using the mixin support
		feature; the object provides the context for the call.
	*/
	var _getComputed = function (path) {
		// we grab the current configuration for the computed property
		var $config = this._computed[path];
		// and a reference to the method in case we need it
		var fn = this[path];
		// the fast track is for computed properties that are volatile
		// and do not cache we simply execute them and return the value
		if (true === $config.volatile) {
			return fn.call(this);
		} else if (true === $config.cached) {
			if ($config.dirty || $config.defer) {
				$config.dirty = 0;
				$config.value = fn.call(this);
				$config.defer = null;
				return $config.value;
			} else {
				return $config.value;
			}
		}
	};

	//*@protected
	/**
		We pass the requested value into the computed property, which
		may or may not handle the value.
	*/
	var _setComputed = function (path, value) {
		// and a reference to the method because we will need it
		var fn = this[path];
		var prev = this.get(path);
		// we merely execute the computed property and pass it the
		// value, if it was capable of setting the value it will
		// handle it otherwise it will recompute
		fn.call(this, value);
		// mark it as dirty and push it to the queue
		_updateComputed.call(this, path, prev, value);
		// flush the queue immediately
		_flushQueue.call(this);
	};

	//*@protected
	var _updateComputed = function (prop, prev, value) {
		var $computed = this._computed;
		var $config = $computed[prop];
		if ($config) {
			// TODO: while not every computed property is cached
			// we have to check for it anyways and rather than
			// cause greater overhead to test whether or not it
			// is cachable and needs updating just do this as it
			// should be harmless otherwise
			++$config.dirty;
			this._computedQueue.push([prop, prev || $config.value, value]);
		}
	};

	//*@protected
	var _flushQueue = function () {
		var $queue = this._computedQueue;
		var values;
		if (!$queue.length) {
			return;
		}
		do {
			values = $queue.shift();
			if (!values[2]) {
				values[2] = _getComputed.call(this, values[0]);
			}
			this.notifyObservers.apply(this, values);
		} while ($queue.length);
	};

	//*@protected
	/**
		Called on every object, this method simply detects whether the
		object supports computed properties; if it does, we execute any
		cacheables that don't have _defer_ set to true.
	*/
	var _postConstructor = function () {
		if (!this._supportsComputed) {
			return;
		}
		// look for the special property created by the feature hook
		// for any cacheable non-deferred computed properties if
		// the kind even supports computed properties
		var $computed = this._computedCacheable;
		var prop;
		var idx;
		var len;
		// for each property that needs updating we arbitrarily flag
		// it as dirty and force it to be evaluated and cached
		for (idx = 0, len = $computed.length; idx < len; ++idx) {
			prop = $computed[idx];
			// mark it as dirty so it will actually be executed
			_updateComputed.call(this, prop);
			// evaluate the cacheable method
			_getComputed.call(this, prop);
		}
	};

	//*@protected
	/**
		Strictly for internal use, as the assumption is the structure of
		these protected properties will be safe and this method is not
		exposed. It should only ever be executed before the computation
		of any cacheable values.
	*/
	var _computedClone = function ($computed, recursing) {
		var copy = {};
		var prop;
		for (prop in $computed) {
			if (!$computed.hasOwnProperty(prop)) {
				continue;
			}
			if ("object" === typeof $computed[prop]	&& null !== $computed[prop]	&& !recursing) {
				copy[prop] = _computedClone($computed[prop], true);
			}
			else {
				copy[prop] = $computed[prop];
			}
		}
		return copy;
	};

	//*@protected
	/**
		Hooks the kind features to automate handling of computations when
		the kind is created.
	*/
	enyo.kind.features.push(function (ctor, props) {
		_findComputed(ctor.prototype, props, true);
	});

	//*@protected
	/**
		Hooks the kind post-initialization routines to make sure we can
		do setup for our cached computed properties that need it.
	*/
	enyo.kind.postConstructors.push(_postConstructor);

	//*@protected
	/**
		Adds a special handler for mixins to be aware of computed properties.
	*/
	enyo.mixins.features.push(_findComputed);

	//*@protected
	enyo.createMixin({

		// ...........................
		// PUBLIC PROPERTIES

		//*@public
		name: "enyo.ComputedSupport",

		// ...........................
		// PROTECTED PROPERTIES

		//*@protected
		_supportsComputed: true,

		// ...........................
		// COMPUTED PROPERTIES

		// ...........................
		// PUBLIC METHODS

		//*@public
		/**
			Overloads the getter so that it can retrieve computed
			property values properly.
		*/
		get: function (path) {
			if (_isComputed(this[path])) {
				return _getComputed.call(this, path);
			} else {
				return this.inherited(arguments);
			}
		},

		//*@public
		/**
			Overloads the setter so that it can attempt to call the
			computed property with the specified parameters. If it doesn't
			accept passed-in parameters, it may do nothing.
		*/
		set: function (path, value) {
			if (_isComputed(this[path])) {
				return _setComputed.call(this, path, value);
			} else {
				return this.inherited(arguments);
			}
		},

		//*@public
		/**
			Overloads the observer support method to hook when notifications
			are sent, so we can handle them the way we need to for computed
			properties.
		*/
		notifyObservers: function (prop, prev, value) {
			// any of the possible notifications we want to map
			// to computed property (by name)
			var $map = this._computedMap;
			var len;
			var idx;
			if ($map[prop]) {
				len = $map[prop].length;
				idx = 0;
				// iterate over each of the dependent properties
				// and mark them as dirty note that the update method
				// will queue the property to notify its own listeners
				// of its state change
				for (; idx < len; ++idx) {
					_updateComputed.call(this, $map[prop][idx]);
				}
			}
			this.inherited(arguments);
			// if there was anything queued lets flush it now
			_flushQueue.call(this);
		},

		// ...........................
		// PROTECTED METHODS

		//*@protected
		_constructor: function () {
			// we need to make sure that we have unique property hashes
			// for each computed property (something we cannot do when
			// applying them to prototypes)
			this._computed = _computedClone(this._computed);
			this._computedQueue = [];
			return this.inherited(arguments);
		},

		//*@protected
		destroy: function () {
			var $computed = this._computed;
			var $config;
			var prop;
			for (prop in $computed) {
				$config = $computed[prop];
				// make sure to fully release any reference this
				// might be holding onto
				delete $config.value;
				// release the entire object when we're done
				delete $computed[prop];
			}
		}

		// ...........................
		// OBSERVERS

	});

})(enyo);

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/mixins/ControllerSupport.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qQHB1YmxpYwovKioKKi8KZW55by5jcmVhdGVNaXhpbih7CgoJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkvLyBQVUJMSUMgUFJPUEVSVElFUwoKCS8vKkBwdWJsaWMKCW5hbWU6ICJlbnlvLkNvbnRyb2xsZXJTdXBwb3J0IiwKCgkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCS8vIFBST1RFQ1RFRCBQUk9QRVJUSUVTCgoJLy8qQHByb3RlY3RlZAoJX3N1cHBvcnRzQ29udHJvbGxlcnM6IHRydWUsCgoJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkvLyBDT01QVVRFRCBQUk9QRVJUSUVTCgoJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkvLyBQVUJMSUMgTUVUSE9EUwoKCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJLy8gUFJPVEVDVEVEIE1FVEhPRFMKCgkvLypAcHJvdGVjdGVkCgljcmVhdGU6IGZ1bmN0aW9uICgpIHsKCQlpZiAodGhpcy5jb250cm9sbGVyKSB7CgkJCXRoaXMubm90aWZ5T2JzZXJ2ZXJzKCJjb250cm9sbGVyIik7CgkJfQoJfSwKCgkvLypwcm90ZWN0ZWQKCWRlc3Ryb3k6IGZ1bmN0aW9uICgpIHsKCQlpZiAodGhpcy5jb250cm9sbGVyKSB7CgkJCWlmICh0aGlzLmNvbnRyb2xsZXIub3duZXIgJiYgdGhpcyA9PT0gdGhpcy5jb250cm9sbGVyLm93bmVyKSB7CgkJCQl0aGlzLmNvbnRyb2xsZXIuZGVzdHJveSgpOwoJCQl9CgkJCXRoaXMuY29udHJvbGxlciA9IG51bGw7CgkJfQoJfSwKCgkvLypAcHJvdGVjdGVkCglfY29udHJvbGxlckNoYW5nZWQ6IGVueW8ub2JzZXJ2ZXIoZnVuY3Rpb24gKHByb3BlcnR5LCBwcmV2aW91cywgdmFsdWUpIHsKCQlpZiAocHJldmlvdXMgJiYgdmFsdWUgJiYgcHJldmlvdXMgPT09IHZhbHVlKSB7CgkJCS8vIHNlZW1zIHRvIGJlIHRoZSBzYW1lIGNvbnRyb2xsZXIgd2UgYWxyZWFkeSBoYWQKCQkJcmV0dXJuOwoJCX0KCQkvLyBmaXJzdCBhdHRlbXB0IHRvIGZpbmQgdGhlIGNvbnRyb2xsZXIgZnJvbSB0aGUKCQkvLyBpbmZvcm1hdGlvbiB3ZSd2ZSBiZWVuIGhhbmRlZAoJCWlmICh0aGlzLmNvbnRyb2xsZXIpIHsKCQkJdGhpcy5maW5kQW5kSW5zdGFuY2UoImNvbnRyb2xsZXIiKTsKCQl9Cgl9LCAiY29udHJvbGxlciIpLAoKCS8vKkBwcm90ZWN0ZWQKCWNvbnRyb2xsZXJGaW5kQW5kSW5zdGFuY2U6IGZ1bmN0aW9uIChjdG9yLCBpbnN0KSB7CgkJLy8gaWYgdGhlcmUgaXMgbm8gY29uc3RydWN0b3Igb3IgaW5zdGFuY2UgaXQgd2FzIG5vdCBmb3VuZAoJCWlmICghKGN0b3IgfHwgaW5zdCkpIHsKCQkJZW55by5lcnJvcigiY2Fubm90IGZpbmQgY29udHJvbGxlcjogIiArIHRoaXMuY29udHJvbGxlcik7CgkJCXJldHVybjsKCQl9CgkJLy8gaWYgYSBjb25zdHJ1Y3RvciBleGlzdHMgd2UgaW5zdGFuY2VkIHRoZSBjbGFzcyBhbmQgY2FuCgkJLy8gY2xhaW0gaXQgYXMgb3VyIG93bgoJCWlmIChjdG9yKSB7CgkJCWluc3Quc2V0KCJvd25lciIsIHRoaXMpOwoJCX0KCQkvLyBsZXRzIGFkZCBvdXJzZWx2ZXMgYXMgYSBkaXNwYXRjaCBsaXN0ZW5lcgoJCWVsc2UgewoJCQlpbnN0LmFkZERpc3BhdGNoVGFyZ2V0KHRoaXMpOwoJCX0KCQkvLyB3ZSByZWJ1aWxkIChyYXRoZXIgdGhhbiByZWZyZXNoKSBvdXIgYmluZGluZ3MgYmVjYXVzZQoJCS8vIHRoZXkgYXJlIG5vdyBtb3N0IGxpa2VseSBjb25uZWN0ZWQgdG8gdGhlIHByZXZpb3VzIGNvbnRyb2xsZXIuCgkJLy8gVE9ETzogQXZvaWQgcmVidWlsZGluZyBiaW5kaW5ncyB0byBvYmplY3RzIG90aGVyIHRoYW4gdGhlIGNvbnRyb2xsZXI/CgkJdGhpcy5yZWJ1aWxkQmluZGluZ3MoKTsKCX0sCgoJLy8qQHByb3RlY3RlZAoJZGlzcGF0Y2hFdmVudDogZnVuY3Rpb24gKG5hbWUsIGV2ZW50LCBzZW5kZXIpIHsKCQkvLyBpZiB3ZSBoYXZlIGEgY29udHJvbGxlciBhdHRlbXB0IHRvIGRpc3BhdGNoIHRoZSBldmVudCB0aGVyZQoJCS8vIGFuZCBpZiBpdCByZXR1cm5zIHRydWUsIHN0b3AgdGhlIGRpc3BhdGNoCgkJaWYgKHRoaXMuY29udHJvbGxlciAmJiB0aGlzLmNvbnRyb2xsZXIuX2lzQ29udHJvbGxlcikgewoJCQlpZiAodGhpcy5jb250cm9sbGVyLmRpc3BhdGNoRXZlbnQobmFtZSwgZXZlbnQsIHNlbmRlcikpIHsKCQkJCXJldHVybiB0cnVlOwoJCQl9CgkJCS8vIHRoaXMgc2NlbmFyaW8gaXMgaGFuZGxlZCBjb21wbGV0ZWx5IGluIHRoZSBpbmhlcml0ZWQgbWV0aG9kCgkJCS8vIGhlcmUgd2Ugc2ltcGx5IHdhbnQgdG8gbWFrZSBzdXJlIHRoYXQgdGhlIGNvbnRyb2xsZXIgaGFzIGFuCgkJCS8vIG9wcG9ydHVuaXR5IHRvIGRlYWwgd2l0aCB0aGUgcmVtYXBwZWQgZXZlbnQKCQkJaWYgKHRoaXNbbmFtZV0gJiYgInN0cmluZyIgPT09IHR5cGVvZiB0aGlzW25hbWVdKSB7CgkJCQlpZiAodGhpcy5jb250cm9sbGVyLmRpc3BhdGNoRXZlbnQodGhpc1tuYW1lXSwgZXZlbnQsIHNlbmRlcikpIHsKCQkJCQlyZXR1cm4gdHJ1ZTsKCQkJCX0KCQkJfQoJCX0KCQlyZXR1cm4gdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCX0sCgoJLy8qQHByb3RlY3RlZAoJZGlzcGF0Y2g6IGZ1bmN0aW9uIChpbk1ldGhvZE5hbWUsIGluRXZlbnQsIGluU2VuZGVyKSB7CgkJLy8gYWxsb3cgYSBjb250cm9sbGVyIHRvIGhhbmRsZSB0aGUgZGVsZWdhdGVkIG5hbWVkIGV2ZW50IGZyb20KCQkvLyBhIGNoaWxkCgkJdmFyIGMgPSB0aGlzLmNvbnRyb2xsZXI7CgkJaWYgKGMpIHsKCQkJaWYgKGNbaW5NZXRob2ROYW1lXSAmJiBlbnlvLmlzRnVuY3Rpb24oY1tpbk1ldGhvZE5hbWVdKSkgewoJCQkJcmV0dXJuIGNbaW5NZXRob2ROYW1lXS5jYWxsKGMsIGluU2VuZGVyIHx8IHRoaXMsIGluRXZlbnQpOwoJCQl9CgkJfQoJCXJldHVybiB0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJfQoKCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJLy8gT0JTRVJWRVJTCgp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/mixins/MixinSupport.js"
Content-Type: application/octet-stream; x-encoding=base64
(function (enyo) {

	//*@protected
	/**
		When a kind is defined as an _enyo.Mixin_, it is actually
		instantiated immediately as a singleton to be reused by any
		other kinds that care to apply it.
	*/
	var store = enyo.mixins = {};

	//*@protected
	/**
		Used internally to keep a clearer abstraction between mixins'
		knowledge of available features of the framework.
	*/
	var features = enyo.mixins.features = [];

	//*@protected
	var register = function (name, mixin) {
		if (store[name]) {
			enyo.error("enyo.createMixin: cannot create " + name + " because " +
				"it already exists");
		}
		store[name] = mixin;
	};

	var proxyInheritedMethod = function (fn, inherited, nom) {
		return function () {
			var oldInherited = fn._inherited;
			var oldNom = fn.nom;
			fn._inherited = inherited;
			fn.nom = nom;
			var ret = fn.apply(this, arguments);
			fn._inherited =  oldInherited;
			fn.nom = oldNom;
			return ret;
		};
	};

	//*@public
	/**
		Creates a reusable hash of the properties to be applied for the
		the named mixin wherever it is requested. Returns a reference to
		the new mixin properties.
	*/
	var createMixin = enyo.createMixin = function (props) {
		if (!props) {
			return false;
		}
		// we need to grab the name but make sure it isn't stored as
		// a property on mixin hash
		var name = props.name;
		// if there isn't a name fail early
		// remove the name
		delete props.name;
		if (!name) {
			enyo.error("enyo.createMixin: cannot create mixin without name");
		}
		register(name, props);
		// move any mixin-constructor to a special name
		if ("function" === typeof props.create) {
			props._create = props.create;
			delete props.create;
			props._create.nom = name + ".create()";
		}
		if ("function" === typeof props.destroy) {
			props._destroy = props.destroy;
			delete props.destroy;
			props._destroy.nom = name + ".destroy()";
		}
		if ("object" === typeof props.handlers) {
			props._mixinHandlers = props.handlers;
			delete props.handlers;
		}
		if (enyo.exists(props["override"])) {
			props._mixinOverride = props.override;
			delete props.override;
		} else {
			props._mixinOverride = true;
		}
		return props;
	};

	//*@public
	/**
		Determine if the kind or instance has the requested _mixin_.
		The parameter must be a string representing the name of the mixin
		in question.
	*/
	enyo.hasMixin = function (target, mixin) {
		if (target && target.hasMixin) {
			return target.hasMixin(mixin);
		} else {
			return !!~enyo.indexOf(mixin, target._appliedMixins || []);
		}
	};

	//*@protected
	var _applyFeatures = function (base, mixin) {
		for (var idx = 0, len = features.length; idx < len; ++idx) {
			features[idx](base, mixin);
		}
	};

	//*@protected
	var _addMixin = function (proto, mixin, name) {

		// for the kind we clone the known mixin constructors
		var ctors = proto._mixinCreate = enyo.clone(proto._mixinCreate || []);
		// for the kind we clone the known mixin destructors
		var dtors = proto._mixin_destroy = enyo.clone(proto._mixin_destroy || []);
		// for the kind we clone the known applied mixins
		var applied = proto._appliedMixins = enyo.clone(proto._appliedMixins || []);

		// if this mixin is already applied, there is nothing we can do
		if (!!~enyo.indexOf(name, applied)) {
			return enyo.warn("enyo.applyMixin: " +
			"attempt to apply mixin " + name + " to " + proto.kindName +
			" multiple times");
		}
		// add the mixin constructor to the queue
		if ("function" === typeof mixin._create) {
			ctors.push(mixin._create);
		}
		// add the mixin destructor to the queue
		if ("function" === typeof mixin._destroy) {
			dtors.push(mixin._destroy);
		}
		// apply all of the properties and methods to the base kind prototype
		_applyProperties(proto, mixin, name);
		// add the name of this mixin to the applied mixins array for the kind
		applied.push(name);
		// give each available mixin feature the opportunity to handle properties
		_applyFeatures(proto, mixin);
	};

	//*@protected
	var _applyProperties = function (base, props, name) {
		// the name of any concatenatable properties
		var concat = base.concat || [];
		// whether or not the mixin wishes to override defined
		// properties (not functions) of the base if they exist
		var override = !enyo.exists(props._mixinOverride)? true: props._mixinOverride;
		// the name of the property to be applied
		var key;
		// the value for the property that will be applied
		var prop;
		// placeholder for functions if they already exist
		// on the target base
		var fn;
		var prev;
		// the name of the mixin or nothing
		name = name || "unnamed";
		for (key in props) {
			if (!props.hasOwnProperty(key)) {
				continue;
			}
			if ("_create" === key || "_destroy" === key) {
				continue;
			}

			prop = props[key];

			if ("_mixinHandlers" === key) {
				if (base._mixinHandlers) {
					// deliberate reuse of the key variable from the outer
					// for loop since we will be exiting this pass when we're
					// done here
					for (key in prop) {
						if (base._mixinHandlers[key] instanceof Array) {
							base._mixinHandlers.push(prop[key]);
						} else if (enyo.exists(base._mixinHandlers[key])) {
							prev = base._mixinHandlers[key];
							base._mixinHandlers[key] = [prev, prop[key]];
						} else {
							base._mixinHandlers[key] = prop[key];
						}
					}
				} else {
					base._mixinHandlers = enyo.clone(prop);
				}
				continue;
			}
			if ("function" === typeof prop && !prop.nom) {
				prop.nom = name + "." + key + "()";
			}

			// if the basetype has the property and it is a function, we
			// insert the props function but allow it to chain the original
			// if it wants
			if (base[key] && "function" === typeof base[key] && "function" === typeof prop && override) {
				fn = base[key];
				prop = base[key] = proxyInheritedMethod(prop, fn, name + "." + key + "()");
			} else if (!!~enyo.indexOf(key, concat)) {
				// we need to concatenate instead of blowing away the property
				// if they are both arrays
				if (base[key] instanceof Array && props[key] instanceof Array) {
					base[key] = enyo.merge(base[key], props[key]);
				} else if (props[key] instanceof Array) {
					base[key] = enyo.clone(prop);
				}
			} else {
				if (override || !enyo.exists(base[key])) {
					base[key] = prop;
				}
			}
		}
	};

	//*@public
	/**
		Used internally but made accessible to arbitrarily apply a mixin
		to a class prototype.
	*/
	var applyMixin = enyo.applyMixin = function (mixin, base) {
		// attempt to determine a name
		var name = "string" === typeof mixin? mixin: mixin.name || "unnamed";

		// determine the mixin from whether it is a string or a hash
		// of properties known as a mixin
		mixin = name !== "unnamed"? store[name]: mixin;

		// if there isn't a mixin we can't do anything
		if (!mixin || "object" !== typeof mixin) {
			return enyo.warn("enyo.applyMixin: could not find the requested mixin, '" +
				name + "'");
		}

		if ("function" === typeof base) {
			_addMixin(base.prototype, mixin, name);
		} else {
			_addMixin(base, mixin, name);
			_postConstructor.call(base);
		}
	};

	//*@protected
	var _createMixins = function () {
		var $mixins = this._mixinCreate;
		var len = $mixins.length;
		var idx = 0;
		var fn;
		for (; idx < len; ++idx) {
			fn = $mixins[idx];
			fn.call(this);
		}
	};

	//*@protected
	var _destroyMixins = function () {
		if (!this._mixinsDestroyed) {
			var $mixins = this._mixin_destroy;
			var len = $mixins.length;
			var idx = 0;
			var fn;
			for (; idx < len; ++idx) {
				fn = $mixins[idx];
				fn.call(this);
			}
		}
		this._mixinsDestroyed = true;
	};

	//*@protected
	var _postConstructor = function () {
		if (!this._supportsMixins) {
			return;
		}
		// we need to initialize all of the mixins registered to this
		// kind
		_createMixins.call(this);
	};

	//*@protected
	var _dispatchEvent = function (name, event, sender) {
		var $handlers = this._mixinHandlers || {};
		var idx;
		var len;
		var ret = false;
		if ($handlers[name]) {
			if ($handlers[name] instanceof Array) {
				for (idx = 0, len = $handlers[name].length; idx < len; ++idx) {
					ret = ret || this.dispatch($handlers[name][idx], event, sender);
				}
			} else {
				ret = this.dispatch($handlers[name], event, sender);
			}
		}
		return ret;
	};

	//*@protected
	/**
		We add a kind feature to snag and handle all mixins for a given
		kind.
	*/
	enyo.kind.features.push(function (ctor, props) {
		// see if there is a mixins array being applied to the kind
		var proto = ctor.prototype;
		var $mixins = proto.mixins || [];
		var len = $mixins.length;
		var idx = 0;
		// remove the array if it existed
		delete proto.mixins;
		for (; idx < len; ++idx) {
			applyMixin($mixins[idx], ctor);
		}
		// inject our special destructor that will enable the
		// other mixins to execute their own when the time is right
		if (!proto._noApplyMixinDestroy) {
			_applyProperties(proto, {destroy: function () {
				if (this._supportsMixins) {
					_destroyMixins.call(this);
				}
				return this.inherited(arguments);
			}}, "enyo.MixinSupport");
		}
	});

	//*@protected
	enyo.kind.postConstructors.push(_postConstructor);

	//*@public
	/**
		The _enyo.MixinSupport_ mixin allows instances of _enyo.Object_
		and its subkinds to have proper mixin support applied at kind
		initialization time, as well as appropriate cleanup when the
		object is destroyed.
	*/
	createMixin({

		// ...........................
		// PUBLIC PROPERTIES

		//*@public
		name: "enyo.MixinSupport",

		// ...........................
		// PROTECTED PROPERTIES

		//*@protected
		_supportsMixins: true,
		
		//*@protected
		_mixinsDestroyed: false,

		// ...........................
		// PUBLIC METHODS

		//*@public
		/**
			Apply a mixin to an instance of an enyo.kind that supports
			mixins. This applied mixin will ONLY be applied to this instance
			and not to other instances of the kind. The lone parameter is
			the name of an available mixin or a reference to a mixin type.
			Returns the callee.
		*/
		applyMixin: function (mixin) {
			enyo.applyMixin(mixin, this, true);
			return this;
		},

		//*@public
		/**
			Returns a boolean true | false whether or not this instance
			already has the requested _mixin_ applied to it. The parameter
			must be a string representing the name of the mixin in question.
		*/
		hasMixin: function (mixin) {
			return !!~enyo.indexOf(mixin, this._appliedMixins);
		}

	});

	//*@public
	/**
		A special mixin for supporting _enyo.Component_ events.
	*/
	enyo.createMixin({

		// ...........................
		// PUBLIC METHODS

		//*@public
		name: "enyo.MixinComponentSupport",

		// ...........................
		// PRIVATE METHODS

		//*@protected
		dispatchEvent: function () {
			if (_dispatchEvent.apply(this, arguments)) {
				return true;
			}
			return this.inherited(arguments);
		}

	});

}(enyo));

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/mixins/MultipleDispatchSupport.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qQHB1YmxpYwovKioKCUFsbG93cyBhIHNpbmdsZSBvYmplY3QgdG8gZGlzcGF0Y2ggaXRzIGV2ZW50cyB0byBtdWx0aXBsZSByZWdpc3RlcmVkCglsaXN0ZW5lcnMuIE5vdGUgdGhhdCBldmVudHMgZ2VuZXJhdGVkIGJ5IHRoZSBvYmplY3Qgd2lsbCBiZSBkaXNwYXRjaGVkCgl0byBhbGwgbGlzdGVuZXJzLCBidXQgZXZlbnRzIGJ1YmJsZWQgb3IgZGlzcGF0Y2hlZCB0byB0aGUgb2JqZWN0IHdpbGwKCW9ubHkgYmUgcHJvcGFnYXRlZCBpZiB0aGUgb2JqZWN0IGhhcyBhbiBfb3duZXJfLiBXaGlsZSB0aGUgX293bmVyXwoJcHJvcGVydHkgbWF5IGJlIHNldCBhdCBhbnkgdGltZSwgaWYgYSB2aWV3IGluaXRpYWxpemVzIGEgbmV3IGluc3RhbmNlCglvZiB0aGUga2luZCwgaXRzIF9vd25lcl8gd2lsbCBhdXRvbWF0aWNhbGx5IGJlIHNldC4KKi8KZW55by5jcmVhdGVNaXhpbih7CgoJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkvLyBQVUJMSUMgUFJPUEVSVElFUwoKCS8vKkBwdWJsaWMKCW5hbWU6ICJlbnlvLk11bHRpcGxlRGlzcGF0Y2hTdXBwb3J0IiwKCgkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCS8vIFBST1RFQ1RFRCBQUk9QRVJUSUVTCgoJLy8qQHByb3RlY3RlZAoJX3N1cHBvcnRzTXVsdGlwbGVEaXNwYXRjaDogdHJ1ZSwKCgkvLypAcHJvdGVjdGVkCglfZGlzcGF0Y2hUYXJnZXRzOiBudWxsLAoKCS8vKkBwcm90ZWN0ZWQKCS8qKgoJCUEgYm9vbGVhbiBmbGFnIHVzZWQgaW50ZXJuYWxseSB0byBjb21tdW5pY2F0ZSBob3cgdG8KCQloYW5kbGUgcmVxdWVzdHMgdG8gYnViYmxlIGV2ZW50cwoJKi8KCV9kZWZhdWx0RGlzcGF0Y2g6IGZhbHNlLAoKCS8vKkBwcm90ZWN0ZWQKCS8qKgoJCUNvbnRyb2xsZXJzIG92ZXJsb2FkIHRoZSBldmVudCBBUEkgYW5kIGhhbmRsZSBidWJibGluZyBkaWZmZXJlbnRseS4KCQlBbGwgY29udHJvbGxlcnMgc3VwcG9ydCBtdWx0aXBsZSBkaXNwYXRjaCBhbmQgYWxzbyBidWJibGluZyB0byBhIHNpbmdsZQoJCV9vd25lcl8gdW5kZXIgY2VydGFpbiBjaXJjdW1zdGFuY2VzLgoJKi8KCV9kZWZhdWx0VGFyZ2V0OiBudWxsLAoKCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJLy8gQ09NUFVURUQgUFJPUEVSVElFUwoKCS8vKkBwcm90ZWN0ZWQKCS8qKgoJCU92ZXJsb2FkcyB0aGUgYnViYmxlIHRhcmdldC4KCSovCglidWJibGVUYXJnZXQ6IGVueW8uY29tcHV0ZWQoZnVuY3Rpb24gKCkgewoJCS8vIGlmIHdlIGhhdmUgYSB2YWxpZCBkaXNwYXRjaCB0YXJnZXQgYW5kIGRlZmF1bHQgZGlzcGF0Y2hpbmcgZW5hYmxlZCwKCQkvLyB3ZSBnbyBoZWFkIGFuZCByZXR1cm4gdGhhdCBvYmplY3Q7IG90aGVyd2lzZSwgbm90aGluZywgc28gaXQgd2lsbAoJCS8vIG5vdCBwcm9wYWdhdGUgYW4gZXZlbnQKCQlpZiAodGhpcy5fZGVmYXVsdFRhcmdldCAmJiB0aGlzLl9kZWZhdWx0RGlzcGF0Y2gpIHsKCQkJcmV0dXJuIHRoaXMuX2RlZmF1bHRUYXJnZXQ7CgkJfQoJfSwgIl9kZWZhdWx0VGFyZ2V0IiwgIl9kZWZhdWx0RGlzcGF0Y2giLCB7Y2FjaGVkOiB0cnVlfSksCgoJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkvLyBQVUJMSUMgTUVUSE9EUwoKCS8vKkBwdWJsaWMKCS8qKgoJCVNldHMgYXMgZGlzcGF0Y2ggdGFyZ2V0IGFuIGluc3RhbmNlIG9mIF9lbnlvLkNvbXBvbmVudF8gb3IgYSBzdWJraW5kCgkJdGhhdCBzdXBwb3J0cyB0aGUgZXZlbnQgQVBJLgoJKi8KCWFkZERpc3BhdGNoVGFyZ2V0OiBmdW5jdGlvbiAodGFyZ2V0KSB7CgkJdmFyICRkaXN0ID0gdGhpcy5fZGlzcGF0Y2hUYXJnZXRzOwoJCS8vIGVuc3VyZSB0aGF0IHdlIGhhdmUgbm90IGFscmVhZHkgcmVnaXN0ZXJlZCB0aGUgdGFyZ2V0CgkJLy8gYW5kIHRoYXQgaXQgaXMgbm90IG91cnNlbHZlcyBvciBvdXIgb3duZXIgaWYgd2UgaGF2ZSBvbmUKCQlpZiAodGFyZ2V0ICYmIHRoaXMgIT09IHRhcmdldCAmJiAhfmVueW8uaW5kZXhPZih0YXJnZXQsICRkaXN0KQoJCQkmJiB0aGlzLm93bmVyICE9PSB0YXJnZXQpIHsKCQkJLy8gc2hvdWxkIGJlIHNhZmUgdG8gYWRkIHRoZSBsaXN0ZW5lcgoJCQkkZGlzdC5wdXNoKHRhcmdldCk7CgkJfQoJfSwKCgkvLypAcHVibGljCgkvKioKCQlBY2NlcHRzIGFuIGluc3RhbmNlIG9mIGEgcmVnaXN0ZXJlZCBsaXN0ZW5lciBvbiB0aGlzIG9iamVjdCBhcyBhCgkJcGFyYW1ldGVyOyBpZiB0aGUgcGFzc2VkLWluIGxpc3RlbmVyIGlzIHByZXNlbnQgdGhlIGFjdGl2ZSBkaXNwYXRjaAoJCXRhcmdldHMsIGl0IGlzIHRoZW4gcmVtb3ZlZC4KCSovCglyZW1vdmVEaXNwYXRjaFRhcmdldDogZnVuY3Rpb24gKHRhcmdldCkgewoJCXZhciAkZGlzdCA9IHRoaXMuX2Rpc3BhdGNoVGFyZ2V0czsKCQl2YXIgaWR4ID0gZW55by5pbmRleE9mKHRhcmdldCwgJGRpc3QpOwoJCWlmICgtMSAhPT0gaWR4KSB7CgkJCSRkaXN0LnNwbGljZShpZHgsIDEpOwoJCX0KCX0sCgoJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkvLyBQUk9URUNURUQgTUVUSE9EUwoKCS8vKkBwcm90ZWN0ZWQKCS8qKgoJCVdoZW4gdGhlIF9vd25lcl8gY2hhbmdlcywgd2UgaGF2ZSB0byBjaGVjayBvdXIgc3RhdGUgdG8gbWFrZQoJCXN1cmUgd2UgYXJlIHN0aWxsIGFwcHJvcHJpYXRlbHkgc2V0LgoJKi8KCW93bmVyQ2hhbmdlZDogZnVuY3Rpb24gKCkgewoJCS8vIGxldCB0aGUgbm9ybWFsIGNoYWluIG9mIG93bmVyIGNoYW5nZXMgb2NjdXIKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCS8vIGlmIHdlIGhhdmUgYW4gb3duZXIgYW5kIGl0IGlzIGEgY29tcG9uZW50IHdlIHNldCBvdXIgZGVmYXVsdAoJCS8vIGRpc3BhdGNoIHByb3BlcnR5IHRvIHRydWUgdG8gYWxsb3cgZXZlbnRzIHRvIHByb3BhZ2F0ZSB0byB0aGlzCgkJLy8gb2JqZWN0CgkJaWYgKHRoaXMub3duZXIgJiYgdHJ1ZSA9PT0gKHRoaXMub3duZXIgaW5zdGFuY2VvZiBlbnlvLkNvbXBvbmVudCkpIHsKCQkJdGhpcy5zZXQoIl9kZWZhdWx0VGFyZ2V0IiwgdGhpcy5vd25lcik7CgkJCXRoaXMuc2V0KCJfZGVmYXVsdERpc3BhdGNoIiwgdHJ1ZSk7CgkJfSBlbHNlIHsKCQkJLy8gb3RoZXJ3aXNlIHdlIGVpdGhlciBkb24ndCBoYXZlIGFuIG93bmVyIG9yIHRoZXkgY2Fubm90CgkJCS8vIGFjY2VwdCBldmVudHMgc28gd2UgcmVtb3ZlIG91ciBidWJibGUgdGFyZ2V0CgkJCXRoaXMuc2V0KCJfZGVmYXVsdFRhcmdldCIsIG51bGwpOwoJCX0KCX0sCgoJLy8qQHByb3RlY3RlZAoJZGlzcGF0Y2hGcm9tOiBmdW5jdGlvbiAoc2VuZGVyLCBldmVudCkgewoJCWlmIChldmVudC5kaXNwYXRjaGVkQnlDb250cm9sbGVyKSB7CgkJCWlmIChldmVudC5kaXNwYXRjaENvbnRyb2xsZXIgPT09IHRoaXMpIHsKCQkJCXJldHVybiB0cnVlOwoJCQl9CgkJfSBlbHNlIGlmIChzZW5kZXIgPT09IHRoaXMpIHsKCQkJZXZlbnQuZGlzcGF0Y2hlZEJ5Q29udHJvbGxlciA9IHRydWU7CgkJCWV2ZW50LmRpc3BhdGNoQ29udHJvbGxlciA9IHRoaXM7CgkJfQoJCXJldHVybiBmYWxzZTsKCX0sCgoJLy8qQHByb3RlY3RlZAoJYnViYmxlVXA6IGZ1bmN0aW9uIChuYW1lLCBldmVudCwgc2VuZGVyKSB7CgkJdmFyIHRhcmdldHM7CgoJCS8vIFRPRE86IGZvciBub3csIHRoaXMgaXMgc29sdmluZyBhIHByb2JsZW0gdGhhdCBpcyBub3Qgb2J2aW91cwoJCS8vIHdoZXRoZXIgb3Igbm90IHRoaXMgY2hhbmdlIHdpbGwgbWFrZSBhIGRpZmZlcmVuY2UgZm9yCgkJLy8gc29sZWx5IG93bmVkIGNvbnRyb2xsZXJzIHRoaXMgY2FuIHBvdGVudGlhbGx5IGNhdXNlIHRvcAoJCS8vIGxldmVsIGFwcGxpY2F0aW9uLWluc3RhbmNlcyB0byByZWNlaXZlIHRoZSBzYW1lIGJ1YmJsZWQgZXZlbnQKCQkvLyB0d2ljZSBpZiBpdCBpcyBub3QgZXhwbGljaXRseSBoYW5kbGVkIGFuZCBoYXMgYSB0cnV0aHkgdmFsdWUKCQkvLyByZXR1cm5lZCBzb21ld2hlcmUgdG8gc3RvcCBwcm9wYWdhdGlvbgoKCQlpZiAodGhpcy5fZGVmYXVsdERpc3BhdGNoKSB7CgkJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7CgkJfQoKCQl0YXJnZXRzID0gdGhpcy5fZGlzcGF0Y2hUYXJnZXRzOwoJCWVueW8uZm9yRWFjaChlbnlvLmNsb25lKHRhcmdldHMpLCBmdW5jdGlvbiAodGFyZ2V0KSB7CgkJCWlmICh0YXJnZXQpIHsKCQkJCWlmICh0YXJnZXQuZGVzdHJveWVkKSB7CgkJCQkJdGhpcy5yZW1vdmVEaXNwYXRjaFRhcmdldCh0YXJnZXQpOwoJCQkJfSBlbHNlIHsKCQkJCQl0YXJnZXQuZGlzcGF0Y2hCdWJibGUobmFtZSwgZXZlbnQsIHNlbmRlcik7CgkJCQl9CgkJCX0KCQl9LCB0aGlzKTsKCX0sCgoJLy8qQHByb3RlY3RlZAoJZGlzcGF0Y2hFdmVudDogZnVuY3Rpb24gKG5hbWUsIGV2ZW50LCBzZW5kZXIpIHsKCQlpZiAodGhpcy5kaXNwYXRjaEZyb20oc2VuZGVyLCBldmVudCkpIHsKCQkJcmV0dXJuIGZhbHNlOwoJCX0KCQlyZXR1cm4gdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCX0sCgoJLy8qQHByb3RlY3RlZAoJYnViYmxlRGVsZWdhdGlvbjogZnVuY3Rpb24gKGRlbGVnYXRlLCBwcm9wLCBuYW1lLCBldmVudCwgc2VuZGVyKSB7CgkJaWYgKHRoaXMuX2RlZmF1bHREaXNwYXRjaCkgewoJCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCX0KCQl2YXIgdGFyZ2V0cyA9IHRoaXMuZ2V0KCJfZGlzcGF0Y2hUYXJnZXRzIik7CgkJZW55by5mb3JFYWNoKGVueW8uY2xvbmUodGFyZ2V0cyksIGZ1bmN0aW9uICh0YXJnZXQpIHsKCQkJaWYgKHRhcmdldCkgewoJCQkJaWYgKHRhcmdldC5kZXN0cm95ZWQpIHsKCQkJCQl0aGlzLnJlbW92ZURpc3BhdGNoVGFyZ2V0KHRhcmdldCk7CgkJCQl9IGVsc2UgewoJCQkJCXRhcmdldC5kZWxlZ2F0ZUV2ZW50KGRlbGVnYXRlLCBwcm9wLCBuYW1lLCBldmVudCwgc2VuZGVyKTsKCQkJCX0KCQkJfQoJCX0pOwoJfSwKCgkvLypAcHJvdGVjdGVkCglfY29uc3RydWN0b3I6IGZ1bmN0aW9uICgpIHsKCQl0aGlzLl9kaXNwYXRjaFRhcmdldHMgPSBbXTsKCQlyZXR1cm4gdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCX0sCgkKCS8vKkBwcm90ZWN0ZWQKCWRlc3Ryb3k6IGZ1bmN0aW9uICgpIHsKCQl0aGlzLl9kaXNwYXRjaFRhcmdldHMgPSBbXTsKCX0KCgkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCS8vIE9CU0VSVkVSUwoKfSk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/mixins/ObserverMethodSupport.js"
Content-Type: application/octet-stream; x-encoding=base64
(function () {

	//*@public
	/**
		Accepts a function followed by one or more string parameters that are
		targets for the observer; returns a method with the appropriate
		properties to allow the system to notify it when the named properites
		have been modified.
	*/
	enyo.observer = function (fn /* arguments */) {
		var events = enyo.toArray(arguments).slice(1);
		if (!enyo.exists(fn) || "function" !== typeof fn) {
			// this is a necessary assert
			throw "enyo.observer: invalid observer, must have a function";
		}
		fn._isObserver = true;
		fn.events = (fn.events? fn.events: []).concat(events);
		return fn;
	};

	//*@public
	/**
		Registers an observer for the passed-in property, returning a reference
		to the handler function being registered, so that it can be stored (and,
		later, removed). In addition to the property that should trigger the
		observer/handler when changed, this method accepts an optional context,
		under which the handler function will be executed when triggered.

		An observer may be added for any property on the passed-in object
		_(base)_, but an observer may not be added for the same event more
		than once.
	*/
	var addObserver = function (base, property, fn, context) {
		var observers = base._observers || (base._observers = {});
		var handlers;
		// when there is name collision in an observer where one class
		// subclasses another while overloaded an observer method
		// it can call this.inherited as usual but we need to remove the
		// previous version of the method from observing the notifications
		// so it won't be handled twice
		if (fn._inherited && fn._inherited._isObserver) {
			removeObserver(base, property, fn._inherited);
		}
		// if a context is provided for the listener, we bind it
		// to that context now
		fn = context? enyo.bind(context, fn): fn;
		// if there are no registered handlers for this event
		// go ahead and create an array for them
		if (!enyo.exists(observers[property])) {
			handlers = observers[property] = [];
		}
		else {
			handlers = observers[property];
		}
		// only add it if it isn't already in the array
		if (!~enyo.indexOf(fn, handlers)) {
			handlers.push(fn);
		}
		// allow chaining
		return fn;
	};

	//*@public
	/**
		Attempts to remove the given listener/observer for the given property,
		if it exists. If no function is supplied, all listeners for the given
		property will be removed.

		Typically, this method will not be called directly.
	*/
	var removeObserver = function (base, property, fn) {
		var observers = base._observers;
		var idx;
		var handlers;
		if (!(handlers = observers[property])) {
			return this;
		}
		if (enyo.exists(fn) && "function" === typeof fn) {
			idx = enyo.indexOf(fn, handlers);
			if (!!~idx) {
				// remove it from the array
				handlers.splice(idx, 1);
			}
		} else {
			// we need to remove ALL the observers of this property
			delete observers[property];
		}
	};

	//*@public
	/**
		Convenience method to remove all observers on all properties.
		Returns a reference to this object for chaining.

		This will almost never need to be called by anything other than
		the _destroy_ method.
	*/
	var removeAllObservers = function (base) {
		var observers = base._observers;
		var handlers;
		var observer;
		var prop;
		var idx;
		var len;
		for (prop in observers) {
			if (!observers.hasOwnProperty(prop)) {
				continue;
			}
			handlers = observers[prop];
			// orphan the array so it will be cleaned up by the GC
			observers[prop] = null;
			for (idx = 0, len = handlers.length; idx < len; ++idx) {
				observer = handlers[idx];
			}
		}
		// reset our observers hash
		base._observers = {};
		base._notificationQueue = {};
		return base;
	};

	//*@public
	/**
		Notifies any observers for a given property. Accepts the previous
		value and the current value as parameters. Looks for a
		backwards-compatible function of the _propertyChanged_ form and
		will call that function, if it exists, while also notifying other
		observers.
	*/
	var notifyObservers = function (base, property, prev, value) {
		var observers = base._observers || {};
		var handlers = (observers[property] || []);
		var idx = 0;
		var fn;
		var ch = enyo.uncap(property) + "Changed";
		if ("*" !== property) {
			handlers = enyo.merge(handlers, observers["*"] || []);
		}
		if (handlers) {
			for (; idx < handlers.length; ++idx) {
				fn = handlers[idx];
				if (!enyo.exists(fn) || "function" !== typeof fn) {
					continue;
				}
				if (false === base._allowNotifications) {
					base.addNotificationToQueue(property, fn, [property, prev, value]);
				} else {
					fn.call(base, property, prev, value);
				}
			}
		}

		if (enyo.exists(base[ch]) && "function" === typeof base[ch]) {
			if (false === base._allowNotifications) {
				base.addNotificationToQueue(property, base[ch], [prev, value]);
			} else {
				base[ch].call(base, prev, value);
			}
		}
		return base;
	};

	//*@protected
	/**
		Used internally when a notification is queued.
	*/
	var addNotificationToQueue = function (base, property, fn, params) {
		var queue = base._notificationQueue || (base._notificationQueue = {});
		var handlers = queue[property];
		params = params || [];
		if (false === base._allowNotificationQueue) {
			return;
		}
		if (!enyo.exists(handlers)) {
			// create an entry for base property note that the queue for
			// every property uses the first array index as the parameters
			queue[property] = [params, fn];
		} else {
			// update the properties for base entry so if the value has
			// been updated before the queue is flushed it uses the most
			// recent values
			// TODO: replace me with something that will actually work!
			if (handlers[0] !== params) {
				handlers.splice(0, 1, params);
			}
			if (!~enyo.indexOf(fn, handlers)) {
				handlers.push(fn);
			}
		}
	};

	//*@public
	/**
		Prevents all notifications on this object from firing. Does not
		clear or flush the queue. Any new notifications fired while
		notifications are disabled will be added to the queue, which may be
		arbitrarily flushed or cleared when ready. To disable the queue, pass
		a boolean true as the second argument. Note that disabling the queue
		will immediately clear (but not flush) the queue.

		Also increments an internal counter that requires the
		_startNotifications_ method to be called an equal number of times
		before notifications will be enabled again. The queue cannot be flushed
		until the counter reaches 0.
	*/
	var stopNotifications = function (base, disableQueue) {
		base._allowNotifications = false;
		base._stopCount += 1;
		if (true === disableQueue) {
			base.disableNotificationQueue();
		}
	};

	//*@public
	/**
		Enables notifications for this object and immediately flushes the
		notification queue if the internal counter is 0. Has no effect if
		notifications are already enabled; otherwise, decrements the
		internal counter. If the counter reaches 0, will allow notifications
		and attempt to flush the queue (if there is one and it is enabled).

		This method must be called once for each time the _stopNotifications_
		method was called. Passing a boolean true as the second parameter
		will reenable the notification queue if it was disabled.
	*/
	var startNotifications = function (base, enableQueue) {
		if (0 !== base._stopCount) {
			--base._stopCount;
		}
		if (0 === base._stopCount) {
			base._allowNotifications = true;
			base.flushNotifications();
		}
		if (true === enableQueue) {
			base.enableNotificationQueue();
		}
	};

	//*@public
	/**
		Enables the notification queue. Has no effect if the queue is already
		enabled. If notifications are currently enabled, this method will have
		no effect until they are disabled.
	*/
	var enableNotificationQueue = function (base) {
		base._allowNotificationQueue = true;
	};

	//*@protected
	/**
		Used internally; flushes any notifications that have been queued.
	*/
	var flushNotifications = function (base) {
		if (0 !== base._stopCount) {
			return;
		}
		var queue = base._notificationQueue;
		var fn;
		var property;
		var handlers;
		var params;
		if (!enyo.exists(queue) || false === base._allowNotificationQueue) {
			return;
		}
		for (property in queue) {
			if (!queue.hasOwnProperty(property)) {
				continue;
			}
			handlers = queue[property];
			params = handlers.shift();
			// if an entry just so happens to be added improperly by someone
			// trying to bypass the default means by which to add something to
			// the queue...
			if ("function" === typeof params) {
				handlers.unshift(params);
				params = [];
			}
			while (handlers.length) {
				fn = handlers.shift();
				fn.apply(base, params);
			}
		}
	};

	//*@public
	/**
		Disables the notification queue. Has no effect if the queue is already
		disabled, or if notifications are currently enabled. If notifications
		are disabled, future notifications will not be queued and any items in
		the queue will be cleared (not flushed).
	*/
	var disableNotificationQueue = function (base) {
		base._allowNotificationQueue = false;
		base._notificationQueue = {};
	};

	//*@protected
	var _findObservers = function (proto, props, kind) {
		proto._observers = kind? _observerClone(proto._observers || {}): proto._observers || {};
		var addPropObserver = function (event, fn) {
			addObserver(proto, event, fn);
		};
		var prop;
		var idx;
		for (prop in props) {
			if ("function" === typeof props[prop] && true === props[prop]._isObserver) {
				for (idx = 0; idx < props[prop].events.length; ++idx) {
					// key note here, we use the function on the proto here
					// because it might have been proxied already
					addPropObserver(props[prop].events[idx], proto[prop]);
				}
			}
		}
	};

	//*@protected
	/**
		Strictly for internal use; copies observer hashes for kinds.
	*/
	var _observerClone = function ($observed, recursing) {
		var arrayCopy = function (orig) {
			return [].concat(orig);
		};
		var copy = {};
		for (var prop in $observed) {
			copy[prop] = arrayCopy($observed[prop]);
		}
		return copy;
	};

	enyo.kind.features.push(function (ctor, props) {
		_findObservers(ctor.prototype, props, true);
	});

	//*@protected
	/**
		Adds a special handler for mixins to be aware of how to handle
		observer properties of a kind.
	*/
	enyo.mixins.features.push(_findObservers);

	//*@protected
	enyo.createMixin({
		// ...........................
		// PUBLIC PROPERTIES

		//*@public
		name: "enyo.ObserverSupport",

		// ...........................
		// PROTECTED PROPERTIES

		//*@protected
		_supportsObservers: true,

		//*@protected
		_stopCount: 0,

		//*@protected
		_notificationQueue: null,

		//*@protected
		_allowNotifications: true,

		//*@protected
		_allowNotificationQueue: true,

		// ...........................
		// PUBLIC METHODS

		//*@public
		/**
			Registers an observer for the passed-in property, returning a
			reference to the handler function being registered, so that it
			can be stored (and, later, removed). In addition to the property
			that should trigger the observer/handler when changed, this
			method accepts an optional context, under which the handler
			function will be executed when triggered.

			An observer may be added for any property of the object, but an
			observer may not be added for the same event more than once.
		*/
		addObserver: function (property, fn, context) {
			return addObserver(this, property, fn, context);
		},

		//*@public
		/**
			Attempts to remove the given listener/observer for the given property,
			if it exists. If no function is supplied, all listeners for the given
			property will be removed.

			Typically, this method will not be called directly.
		*/
		removeObserver: function (property, fn) {
			return removeObserver(this, property, fn);
		},

		//*@public
		/**
			Convenience method to remove all observers on all properties.
			Returns a reference to this object for chaining.

			This will almost never need to be called by anything other than
			the _destroy_ method.
		*/
		removeAllObservers: function () {
			return removeAllObservers(this);
		},

		//*@public
		/**
			Notifies any observers for a given property. Accepts the previous
			value and the current value as parameters. Looks for a
			backwards-compatible function of the _propertyChanged_ form and will
			call that, if it exists, while also notifying other observers.
		*/
		notifyObservers: function (property, prev, value) {
			return notifyObservers(this, property, prev, value);
		},

		//*@public
		/**
			Prevents all notifications on this object from firing. Does not
			clear or flush the queue. Any new notifications fired while
			notifications are disabled will be added to the queue, which may be
			arbitrarily flushed or cleared when ready. To disable the queue, pass
			a boolean true as the second argument. Note that disabling the queue
			will immediately clear (but not flush) the queue.

			Also increments an internal counter that requires the
			_startNotifications_ method to be called an equal number of times
			before notifications will be enabled again. The queue cannot be
			flushed until the counter reaches 0.
		*/
		stopNotifications: function (disableQueue) {
			return stopNotifications(this, disableQueue);
		},

		//*@public
		/**
			Enables notifications for this object and immediately flushes the
			notification queue if the internal counter is 0. Has no effect if
			notifications are already enabled; otherwise, decrements the
			internal counter. If the counter reaches 0, will allow
			notifications and attempt to flush the queue (if there is one and
			it is enabled).

			This method must be called once for each time the _stopNotifications_
			method was called. Passing a boolean true as the second parameter
			will reenable the notification queue if it was disabled.
		*/
		startNotifications: function (enableQueue) {
			return startNotifications(this, enableQueue);
		},

		//*@public
		/**
			Enables the notification queue. Has no effect if the queue is already
			enabled. If notifications are currently enabled, this method will have
			no effect until they are disabled.
		*/
		enableNotificationQueue: function () {
			return enableNotificationQueue(this);
		},

		//*@public
		/**
			Disables the notification queue. Has no effect if the queue is
			already disabled, or if notifications are currently enabled. If
			notifications are disabled, future notifications will not be queued
			and any items in the queue will be cleared (not flushed).
		*/
		disableNotificationQueue: function () {
			return disableNotificationQueue(this);
		},

		// ...........................
		// PROTECTED METHODS

		//*@protected
		/**
			Used internally when a notification is queued
		*/
		addNotificationToQueue: function (property, fn, params) {
			return addNotificationToQueue(this, property, fn, params);
		},

		//*@protected
		/**
			Used internally; flushes any notifications that have been queued.
		*/
		flushNotifications: function () {
			return flushNotifications(this);
		},

		//*@protected
		_constructor: function () {
			// it is unfortunate, but we cannot share the observers block we inherited
			// from the kind since as an instance we can modify it at runtime so we're
			// forced to deep copy it
			this._observers = _observerClone(this._observers);
			this.inherited(arguments);
		},

		//*protected
		destroy: function () {
			this.removeAllObservers();
		}

	});

}());

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/mixins/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJIk1peGluU3VwcG9ydC5qcyIsCgkiQ29tcHV0ZWRQcm9wZXJ0eVN1cHBvcnQuanMiLAoJIk9ic2VydmVyTWV0aG9kU3VwcG9ydC5qcyIsCgkiQmluZGluZ1N1cHBvcnQuanMiLAoJIkFwcGxpY2F0aW9uU3VwcG9ydC5qcyIsCgkiQXV0b0JpbmRpbmdTdXBwb3J0LmpzIiwKCSJDb250cm9sbGVyU3VwcG9ydC5qcyIsCgkiTXVsdGlwbGVEaXNwYXRjaFN1cHBvcnQuanMiCik7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJImxvZy5qcyIsCgkibGFuZy5qcyIsCgkicG9vbC5qcyIsCgkiZGV2LmpzIiwKCSJqb2IuanMiLAoJIk9vcC5qcyIsCgkibWl4aW5zIiwKCSJCaW5kaW5nLmpzIiwKCSJPYmplY3QuanMiLAoJIkNvbXBvbmVudC5qcyIsCgkiVWlDb21wb25lbnQuanMiLAoJIkxheW91dC5qcyIsCgkiU2lnbmFscy5qcyIsCgkiTXVsdGlwbGVEaXNwYXRjaENvbXBvbmVudC5qcyIsCgkiQ29udHJvbGxlci5qcyIsCgkiUm91dGVyLmpzIiwKCSJWaWV3Q29udHJvbGxlci5qcyIsCgkiQXBwbGljYXRpb24uanMiLAoJImRhdGEiCik7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/kernel/pool.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uIChlbnlvKSB7CgoJLy8gaW50ZXJuYWxseSBtYWludGFpbiBhIHBvb2wgb2Ygb2JqZWN0cyB0byByZXVzZSB3aGVuZXZlciBwb3NzaWJsZQoJLy8gdG8gcHJldmVudCB0aGUgbWVtb3J5IGZvb3RwcmludCBleHBhbnNpb24gZm9yY2luZyB0aGUgR0MgdG8gZXhlY3V0ZSBtb3JlCgkvLyBvZnRlbgoJdmFyIHBvb2wgPSBbXSwgY2xhaW1lZCA9IFtdOwoJLy8gZm9yIGRlYnVnZ2luZyB0aGVzZSBhcmUgZXhwb3NlZAoJZW55by5wb29sID0ge307CgllbnlvLnBvb2wuYXZhaWxhYmxlID0gcG9vbDsKCWVueW8ucG9vbC5jbGFpbWVkID0gY2xhaW1lZDsKCS8vIHdlIHN0YXJ0IHdpdGggbm9uZSBzbyBpdCB3aWxsIGFsd2F5cyBjcmVhdGUgdGhlIGZld2VzdCBwb3NzaWJsZSBnaXZlbgoJLy8gYSBwYXJ0aWN1bGFyIGFwcGxpY2F0aW9ucyBuZWVkcwoJdmFyIHNjcnViID0gZnVuY3Rpb24gKG8pIHsKCQl2YXIgJG8gPSBvOwoJCWlmICgkbyAmJiAkby5fcG9vbGVkICYmICRvLl9yZWxlYXNlZCkgewoJCQlmb3IgKHZhciBrIGluICRvKSB7CgkJCQlpZiAoJG8uaGFzT3duUHJvcGVydHkoaykgJiYgIX5lbnlvLmluZGV4T2YoaywgWyJfY2xhaW1lZCIsIl9wb29sZWQiLCJfcmVsZWFzZWQiXSkpIHsKCQkJCQkkb1trXSA9IHVuZGVmaW5lZDsKCQkJCX0KCQkJfQoJCX0KCX07CgoJLy8gRk9SIERFQlVHR0lORyBVTkNPTU1FTlQgVEhJUwoJLyogdmFyIGl0ID0gc2V0SW50ZXJ2YWwoZnVuY3Rpb24gKCkgewoJCWVueW8ubG9nKAoJCQkiXG4tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tXG4iICsKCQkJIlBPT0wgU0laRTogIiArIHBvb2wubGVuZ3RoICsgIlxuIiArCgkJCSJDTEFJTUVEIFNJWkU6ICIgKyBjbGFpbWVkLmxlbmd0aCArICJcbiIgKwoJCQkiLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLVxuIgoJCSk7Cgl9LCA1MDAwKTsgKi8KCgkvLyBhZGQgYW4gb2JqZWN0IHRvIHRoZSBwb29sIGlmIGl0IHdhcyBlbmNvdW50ZXJlZCBpbiBhIHJldXNhYmxlCgkvLyB3YXkgYW5kIHJldHVybiBpdCBvciByZXR1cm4gYW4gb2JqZWN0IGZyb20gdGhlIHBvb2wsIGlmIG5vbmUKCS8vIGFyZSBhdmFpbGFibGUgaXQgd2lsbCBjcmVhdGUgYSBuZXcgb25lIGFuZCByZXR1cm4gaXQKCWVueW8ucG9vbC5jbGFpbU9iamVjdCA9IGZ1bmN0aW9uIChvKSB7CgkJdmFyICRvID0gbzsKCQlpZiAoJG8pIHsKCQkJY2xhaW1lZC5wdXNoKCRvKTsKCQl9IGVsc2UgaWYgKHBvb2wubGVuZ3RoKSB7CgkJCWNsYWltZWQucHVzaCgoJG8gPSBwb29sLnBvcCgpKSk7CgkJfSBlbHNlIHsKCQkJY2xhaW1lZC5wdXNoKCgkbyA9IHt9KSk7CgkJfQoJCSRvLl9wb29sZWQgPSB0cnVlOwoJCSRvLl9jbGFpbWVkID0gZW55by5iZW5jaCgpOwoJCSRvLl9yZWxlYXNlZCA9IG51bGw7CgkJcmV0dXJuICRvOwoJfTsKCS8vIHdoZW4gYW4gb2JqZWN0IGlzIGRvbmUgYmVpbmcgdXNlZCByZWxlYXNlIGl0IGJhY2sgdG8gdGhlIHBvb2wKCWVueW8ucG9vbC5yZWxlYXNlT2JqZWN0ID0gZnVuY3Rpb24gKG8pIHsKCQl2YXIgJG8gPSBvLCAkaTsKCQlpZiAoJG8pIHsKCQkJaWYgKCEhfigkaSA9IGVueW8uaW5kZXhPZigkbywgY2xhaW1lZCkpKSB7CgkJCQljbGFpbWVkLnNwbGljZSgkaSwgMSk7CgkJCQlwb29sLnB1c2goJG8pOwoJCQkJJG8uX3JlbGVhc2VkID0gZW55by5iZW5jaCgpOwoJCQkJc2NydWIoJG8pOwoJCQl9CgkJfQoJfTsKCgp9KShlbnlvKTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJImtlcm5lbCIsCgkiZXh0IiwKCSJhamF4IiwKCSJkb20iLAoJInRvdWNoIiwKCSJ1aSIsCgkiZW55by5kZXNpZ24iCik7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/ScrollMath.js"
Content-Type: application/octet-stream; x-encoding=base64
﻿/**
_enyo.ScrollMath_ implements a scrolling dynamics simulation.  It is a helper
kind used by other scroller kinds, such as
<a href="#enyo.TouchScrollStrategy">enyo.TouchScrollStrategy</a>.

_enyo.ScrollMath_ is not typically created in application code.
*/
enyo.kind({
	name: "enyo.ScrollMath",
	kind: "enyo.Component",
	published: {
		//* True if vertical scrolling is enabled
		vertical: true,
		//* True if horizontal scrolling is enabled
		horizontal: true
	},
	events: {
		//* Fires when scroll action starts.
		onScrollStart: "",
		//* Fires while scroll action is in progress.
		onScroll: "",
		//* Fires when scroll action stops.
		onScrollStop: ""
	},
	//* 'Spring' damping returns the scroll position to a value inside the
	//* boundaries.  Lower values provide _faster_ snapback.
	kSpringDamping: 0.93,
	//* 'Drag' damping resists dragging the scroll position beyond the
	//* boundaries.  Lower values provide _more_ resistance.
	kDragDamping: 0.5,
	//* 'Friction' damping reduces momentum over time.  Lower values provide
	//* _more_ friction.
	kFrictionDamping: 0.97,
	//* Additional 'friction' damping applied when momentum carries the viewport
	//* into overscroll.  Lower values provide _more_ friction.
	kSnapFriction: 0.9,
	//* Scalar applied to 'flick' event velocity
	kFlickScalar: 15,
	//* Limits the maximum allowable flick.  On Android > 2, we limit this to
	//* prevent compositing artifacts.
	kMaxFlick: enyo.platform.android > 2 ? 2 : 1e9,
	//* The value used in friction() to determine if the delta (e.g., y - y0) is
	//* close enough to zero to consider as zero
	kFrictionEpsilon: 1e-2,
	//* Top snap boundary, generally 0
	topBoundary: 0,
	//* Right snap boundary, generally (viewport width - content width)
	rightBoundary: 0,
	//* Bottom snap boundary, generally (viewport height - content height)
	bottomBoundary: 0,
	//* Left snap boundary, generally 0
	leftBoundary: 0,
	//* Animation time step
	interval: 20,
	//* Flag to enable frame-based animation; if false, time-based animation is used
	fixedTime: true,
	//* @protected
	// simulation state
	x0: 0,
	x: 0,
	y0: 0,
	y: 0,
	destroy: function() {
		this.stop();
		this.inherited(arguments);
	},
	/**
		Simple Verlet integrator for simulating Newtonian motion.
	*/
	verlet: function(p) {
		var x = this.x;
		this.x += x - this.x0;
		this.x0 = x;
		//
		var y = this.y;
		this.y += y - this.y0;
		this.y0 = y;
	},
	/**
		Boundary damping function.
		Returns damped 'value' based on 'coeff' on one side of 'origin'.
	*/
	damping: function(value, origin, coeff, sign) {
		var kEpsilon = 0.5;
		//
		// this is basically just value *= coeff (generally, coeff < 1)
		//
		// 'sign' and the conditional is to force the damping to only occur
		// on one side of the origin.
		//
		var dv = value - origin;
		// Force close-to-zero to zero
		if (Math.abs(dv) < kEpsilon) {
			return origin;
		}
		return value*sign > origin*sign ? coeff * dv + origin : value;
	},
	/**
		Dual-boundary damping function.
		Returns damped 'value' based on 'coeff' when exceeding either boundary.
	*/
	boundaryDamping: function(value, aBoundary, bBoundary, coeff) {
		return this.damping(this.damping(value, aBoundary, coeff, 1), bBoundary, coeff, -1);
	},
	/**
		Simulation constraints (spring damping occurs here)
	*/
	constrain: function() {
		var y = this.boundaryDamping(this.y, this.topBoundary, this.bottomBoundary, this.kSpringDamping);
		if (y != this.y) {
			// ensure snapping introduces no velocity, add additional friction
			this.y0 = y - (this.y - this.y0) * this.kSnapFriction;
			this.y = y;
		}
		var x = this.boundaryDamping(this.x, this.leftBoundary, this.rightBoundary, this.kSpringDamping);
		if (x != this.x) {
			this.x0 = x - (this.x - this.x0) * this.kSnapFriction;
			this.x = x;
		}
	},
	/**
		The friction function
	*/
	friction: function(inEx, inEx0, inCoeff) {
		// implicit velocity
		var dp = this[inEx] - this[inEx0];
		// let close-to-zero collapse to zero (i.e. smaller than epsilon is considered zero)
		var c = Math.abs(dp) > this.kFrictionEpsilon ? inCoeff : 0;
		// reposition using damped velocity
		this[inEx] = this[inEx0] + c * dp;
	},
	// one unit of time for simulation
	frame: 10,
	// piece-wise constraint simulation
	simulate: function(t) {
		while (t >= this.frame) {
			t -= this.frame;
			if (!this.dragging) {
				this.constrain();
			}
			this.verlet();
			this.friction('y', 'y0', this.kFrictionDamping);
			this.friction('x', 'x0', this.kFrictionDamping);
		}
		return t;
	},
	animate: function() {
		this.stop();
		// time tracking
		var t0 = enyo.now(), t = 0;
		// delta tracking
		var x0, y0;
		// animation handler
		var fn = this.bindSafely(function() {
			// wall-clock time
			var t1 = enyo.now();
			// schedule next frame
			this.job = enyo.requestAnimationFrame(fn);
			// delta from last wall clock time
			var dt = t1 - t0;
			// record the time for next delta
			t0 = t1;
			// user drags override animation
			if (this.dragging) {
				this.y0 = this.y = this.uy;
				this.x0 = this.x = this.ux;
			}
			// frame-time accumulator
			// min acceptable time is 16ms (60fps)
			t += Math.max(16, dt);
			// alternate fixed-time step strategy:
			if (this.fixedTime && !this.isInOverScroll()) {
				t = this.interval;
			}
			// consume some t in simulation
			t = this.simulate(t);
			// scroll if we have moved, otherwise the animation is stalled and we can stop
			if (y0 != this.y || x0 != this.x) {
				//this.log(this.y, y0);
				this.scroll();
			} else if (!this.dragging) {
				this.stop(true);
				this.scroll();
			}
			y0 = this.y;
			x0 = this.x;
		});
		this.job = enyo.requestAnimationFrame(fn);
	},
	//* @protected
	start: function() {
		if (!this.job) {
			this.animate();
			this.doScrollStart();
		}
	},
	stop: function(inFireEvent) {
		this.job = enyo.cancelRequestAnimationFrame(this.job);
		if (inFireEvent) {
			this.doScrollStop();
		}
	},
	stabilize: function() {
		this.start();
		var y = Math.min(this.topBoundary, Math.max(this.bottomBoundary, this.y));
		var x = Math.min(this.leftBoundary, Math.max(this.rightBoundary, this.x));
		this.y = this.y0 = y;
		this.x = this.x0 = x;
		this.scroll();
		this.stop(true);
	},
	startDrag: function(e) {
		this.dragging = true;
		//
		this.my = e.pageY;
		this.py = this.uy = this.y;
		//
		this.mx = e.pageX;
		this.px = this.ux = this.x;
	},
	drag: function(e) {
		if (this.dragging) {
			var dy = this.vertical ? e.pageY - this.my : 0;
			this.uy = dy + this.py;
			// provides resistance against dragging into overscroll
			this.uy = this.boundaryDamping(this.uy, this.topBoundary, this.bottomBoundary, this.kDragDamping);
			//
			var dx = this.horizontal ? e.pageX - this.mx : 0;
			this.ux = dx + this.px;
			// provides resistance against dragging into overscroll
			this.ux = this.boundaryDamping(this.ux, this.leftBoundary, this.rightBoundary, this.kDragDamping);
			//
			this.start();
			return true;
		}
	},
	dragDrop: function(e) {
		if (this.dragging && !window.PalmSystem) {
			var kSimulatedFlickScalar = 0.5;
			this.y = this.uy;
			this.y0 = this.y - (this.y - this.y0) * kSimulatedFlickScalar;
			this.x = this.ux;
			this.x0 = this.x - (this.x - this.x0) * kSimulatedFlickScalar;
		}
		this.dragFinish();
	},
	dragFinish: function() {
		this.dragging = false;
	},
	flick: function(e) {
		var v;
		if (this.vertical) {
			v = e.yVelocity > 0 ? Math.min(this.kMaxFlick, e.yVelocity) : Math.max(-this.kMaxFlick, e.yVelocity);
			this.y = this.y0 + v * this.kFlickScalar;
		}
		if (this.horizontal) {
			v = e.xVelocity > 0 ? Math.min(this.kMaxFlick, e.xVelocity) : Math.max(-this.kMaxFlick, e.xVelocity);
			this.x = this.x0 + v * this.kFlickScalar;
		}
		this.start();
	},
	mousewheel: function(e) {
		var dy = this.vertical ? e.wheelDeltaY || e.wheelDelta: 0;
		if ((dy > 0 && this.y < this.topBoundary) || (dy < 0 && this.y > this.bottomBoundary)) {
			this.stop(true);
			this.y = this.y0 = this.y0 + dy;
			this.start();
			return true;
		}
	},
	scroll: function() {
		this.doScroll();
	},
	// NOTE: Yip/Orvell method for determining scroller instantaneous velocity
	// FIXME: incorrect if called when scroller is in overscroll region
	// because does not account for additional overscroll damping.
	/**
		Animates a scroll to the specified position.
	*/
	scrollTo: function(inX, inY) {
		if (inY !== null) {
			this.y = this.y0 - (inY + this.y0) * (1 - this.kFrictionDamping);
		}
		if (inX !== null) {
			this.x = this.x0 - (inX + this.x0) * (1 - this.kFrictionDamping);
		}
		this.start();
	},
	setScrollX: function(inX) {
		this.x = this.x0 = inX;
	},
	setScrollY: function(inY) {
		this.y = this.y0 = inY;
	},
	setScrollPosition: function(inPosition) {
		this.setScrollY(inPosition);
	},
	isScrolling: function() {
		return Boolean(this.job);
	},
	isInOverScroll: function() {
		return this.job && (this.x > this.leftBoundary || this.x < this.rightBoundary ||
			this.y > this.topBoundary || this.y < this.bottomBoundary);
	}
});

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/ScrollStrategy.js"
Content-Type: application/octet-stream; x-encoding=base64
﻿/**
	_enyo.ScrollStrategy_ is a helper kind that implements a default scrolling
	strategy for an <a href="#enyo.Scroller">enyo.Scroller</a>.

	_enyo.ScrollStrategy_ is not typically created in application code.
	Instead, it is specified as the value of the `strategyKind` property of an
	`enyo.Scroller` or <a href="#enyo.List">enyo.List</a>, or is used by the
	framework implicitly.
*/
enyo.kind({
	name: "enyo.ScrollStrategy",
	tag: null,
	published: {
		/**
			Specifies how to vertically scroll.  Acceptable values are:

			* "scroll": Always shows a scrollbar; sets _overflow: scroll_.
			* "auto": Scrolls only if needed; sets _overflow: auto_.
			* "hidden": Never scrolls; sets _overflow: hidden_.
			* "default": Same as "auto".
		*/
		vertical: "default",
		/**
			Specifies how to horizontally scroll.  Acceptable values are:

			* "scroll": Always shows a scrollbar; sets _overflow: scroll_.
			* "auto": Scrolls only if needed; sets _overflow: auto_.
			* "hidden": Never scrolls; sets _overflow: hidden_.
			* "default": Same as "auto".
		*/
		horizontal: "default",
		//* Scroll position along horizontal axis
		scrollLeft: 0,
		//* Scroll position along vertical axis
		scrollTop: 0,
		//* Maximum height of scroll content
		maxHeight: null,
		//* Use mouse wheel to move scroller
		useMouseWheel: true
	},
	//* @protected
	handlers: {
		ondragstart: "dragstart",
		ondragfinish: "dragfinish",
		ondown: "down",
		onmove: "move",
		onmousewheel: "mousewheel"
	},
	create: function() {
		this.inherited(arguments);
		this.horizontalChanged();
		this.verticalChanged();
		this.maxHeightChanged();
	},
	rendered: function() {
		this.inherited(arguments);
		enyo.makeBubble(this.container, "scroll");
		this.scrollNode = this.calcScrollNode();
	},
	teardownRender: function() {
		this.inherited(arguments);
		this.scrollNode = null;
	},
	calcScrollNode: function() {
		return this.container.hasNode();
	},
	horizontalChanged: function() {
		this.container.applyStyle("overflow-x", this.horizontal == "default" ? "auto" : this.horizontal);
	},
	verticalChanged: function() {
		this.container.applyStyle("overflow-y", this.vertical == "default" ? "auto" : this.vertical);
	},
	maxHeightChanged: function() {
		this.container.applyStyle("max-height", this.maxHeight);
	},
	scrollTo: function(inX, inY) {
		if (this.scrollNode) {
			this.setScrollLeft(inX);
			this.setScrollTop(inY);
		}
	},
	scrollToNode: function(inNode, inAlignWithTop) {
		if (this.scrollNode) {
			var sb = this.getScrollBounds();
			var n = inNode;
			var b = {height: n.offsetHeight, width: n.offsetWidth, top: 0, left: 0};
			while (n && n.parentNode && n.id != this.scrollNode.id) {
				b.top += n.offsetTop;
				b.left += n.offsetLeft;
				n = n.parentNode;
			}
			// By default, the element is scrolled to align with the top of the scroll area.
			this.setScrollTop(Math.min(sb.maxTop, inAlignWithTop === false ? b.top - sb.clientHeight + b.height : b.top));
			this.setScrollLeft(Math.min(sb.maxLeft, inAlignWithTop === false ? b.left - sb.clientWidth + b.width : b.left));
		}
	},
	scrollIntoView: function(inControl, inAlignWithTop) {
		if (inControl.hasNode()) {
			inControl.node.scrollIntoView(inAlignWithTop);
		}
	},
	isInView: function(inNode) {
		var sb = this.getScrollBounds();
		var ot = inNode.offsetTop;
		var oh = inNode.offsetHeight;
		var ol = inNode.offsetLeft;
		var ow = inNode.offsetWidth;
		return (ot >= sb.top && ot + oh <= sb.top + sb.clientHeight) && (ol >= sb.left && ol + ow <= sb.left + sb.clientWidth);
	},
	setScrollTop: function(inTop) {
		this.scrollTop = inTop;
		if (this.scrollNode) {
			this.scrollNode.scrollTop = this.scrollTop;
		}
	},
	setScrollLeft: function(inLeft) {
		this.scrollLeft = inLeft;
		if (this.scrollNode) {
			this.scrollNode.scrollLeft = this.scrollLeft;
		}
	},
	getScrollLeft: function() {
		return this.scrollNode ? this.scrollNode.scrollLeft : this.scrollLeft;
	},
	getScrollTop: function() {
		return this.scrollNode ? this.scrollNode.scrollTop : this.scrollTop;
	},
	_getScrollBounds: function() {
		var s = this.getScrollSize(), cn = this.container.hasNode();
		var b = {
			left: this.getScrollLeft(),
			top: this.getScrollTop(),
			clientHeight: cn ? cn.clientHeight : 0,
			clientWidth: cn ? cn.clientWidth : 0,
			height: s.height,
			width: s.width
		};
		b.maxLeft = Math.max(0, b.width - b.clientWidth);
		b.maxTop = Math.max(0, b.height - b.clientHeight);
		return b;
	},
	getScrollSize: function() {
		var n = this.scrollNode;
		return {width: n ? n.scrollWidth : 0, height: n ? n.scrollHeight : 0};
	},
	getScrollBounds: function() {
		return this._getScrollBounds();
	},
	calcStartInfo: function() {
		var sb = this.getScrollBounds();
		var y = this.getScrollTop(), x = this.getScrollLeft();
		this.canVertical = sb.maxTop > 0 && this.vertical != "hidden";
		this.canHorizontal = sb.maxLeft > 0 && this.horizontal != "hidden";
		this.startEdges = {
			top: y === 0,
			bottom: y === sb.maxTop,
			left: x === 0,
			right: x === sb.maxLeft
		};
	},
	// NOTE: down, move, and drag handlers are needed only for native touch scrollers
	shouldDrag: function(inEvent) {
		var requestV = inEvent.vertical;
		return (requestV && this.canVertical  || !requestV && this.canHorizontal) /*&& !this.isOobVerticalScroll(inEvent)*/;
	},
	dragstart: function(inSender, inEvent) {
		this.dragging = this.shouldDrag(inEvent);
		if (this.dragging) {
			return this.preventDragPropagation;
		}
	},
	dragfinish: function(inSender, inEvent) {
		if (this.dragging) {
			this.dragging = false;
			inEvent.preventTap();
		}
	},
	// avoid allowing scroll when starting at a vertical boundary to prevent ios from window scrolling.
	down: function(inSender, inEvent) {
		this.calcStartInfo();
	},
	// NOTE: mobile native scrollers need touchmove. Indicate this by
	// setting the requireTouchmove property to true.
	move: function(inSender, inEvent) {
		if (inEvent.which && (this.canVertical && inEvent.vertical || this.canHorizontal && inEvent.horizontal)) {
			inEvent.disablePrevention();
		}
	},
	mousewheel: function(inSender, inEvent) {
		//* We disable mouse wheel scrolling by preventing the default
		if (!this.useMouseWheel) {
			inEvent.preventDefault();
		}
	}
});
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/Scroller.css"
Content-Type: application/octet-stream; x-encoding=base64
LmVueW8tc2Nyb2xsZXIgewoJcG9zaXRpb246IHJlbGF0aXZlOwp9CgouZW55by1maXQuZW55by1zY3JvbGxlciB7Cglwb3NpdGlvbjogYWJzb2x1dGU7Cn0KCi5lbnlvLXRvdWNoLXNjcm9sbGVyIHsKCW92ZXJmbG93OiBoaWRkZW47Cn0KCi5lbnlvLXRvdWNoLXN0cmF0ZWd5LWNvbnRhaW5lciB7CglvdmVyZmxvdzogaGlkZGVuOwp9CgouZW55by1zY3JvbGxlZS1maXQgewoJaGVpZ2h0OiAxMDAlOwp9
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/Scroller.js"
Content-Type: application/octet-stream; x-encoding=base64
/**
_enyo.Scroller_ is a scroller suitable for use in both desktop and mobile
applications.

In some mobile environments, a default scrolling solution is not implemented for
DOM elements.  In such cases, _enyo.Scroller_ implements a touch-based scrolling
solution, which may be opted into either globally (by setting the flag
_enyo.Scroller.touchScrolling = true;_) or on a per-instance basis (by
specifying a _strategyKind_ of "TouchScrollStrategy").

For more information, see the documentation on
[Scrollers](https://github.com/enyojs/enyo/wiki/Scrollers) in the Enyo Developer
Guide.
*/
enyo.kind({
	name: "enyo.Scroller",
	published: {
		/**
			Specifies how to horizontally scroll.  Acceptable values are
			"scroll", "auto," "hidden," and "default".  The precise
			effect of the setting is determined by the scroll strategy.
		*/
		horizontal: "default",
		/**
			Specifies how to vertically scroll.  Acceptable values are "scroll",
			"auto," "hidden," and "default".  The precise effect of the setting
			is determined by the scroll strategy.
		*/
		vertical: "default",
		/**
			The vertical scroll position
		*/
		scrollTop: 0,
		/**
			The horizontal scroll position
		*/
		scrollLeft: 0,
		/**
			Maximum height of the scroll content
		*/
		maxHeight: null,
		/**
			Set to true to make this scroller select a platform-appropriate
			touch-based scrolling strategy. Note that if you specify a value for
			_strategyKind_, that will take precedence over this setting.
		*/
		touch: false,
		/**
			Specifies a type of scrolling. The scroller will attempt to
			automatically select a strategy compatible with the runtime
			environment. Alternatively, you may choose to use a specific
			strategy:

			* <a href="#enyo.ScrollStrategy">ScrollStrategy</a> is the default
				and implements no scrolling, relying instead on the environment
				to scroll properly.

			* <a href="#enyo.TouchScrollStrategy">TouchScrollStrategy</a>
				implements a touch scrolling mechanism.

			* <a href="#enyo.TranslateScrollStrategy">TranslateScrollStrategy</a>
				implements a touch scrolling mechanism using translations; it is
				currently recommended only for Android 3 and 4 & Windows Phone 8.

			* <a href="#enyo.TransitionScrollStrategy">TransitionScrollStrategy</a>
				implements a touch scrolling mechanism using CSS transitions; it is
				currently recommended only for iOS 5 and later.
		*/
		strategyKind: "ScrollStrategy",
		//* Set to true to display a scroll thumb in touch scrollers
		thumb: true,
		//* Use mouse wheel to move scroller
		useMouseWheel: true
	},
	events: {
		//* Fires when a scrolling action starts.
		onScrollStart: "",
		//* Fires while a scrolling action is in progress.
		onScroll: "",
		//* Fires when a scrolling action stops.
		onScrollStop: ""
	},
	/**
		If true (the default) and a touch scroller, the scroller will overscroll
		and bounce back at the edges
	*/
	touchOverscroll: true,
	/**
		If true (the default), the scroller will not propagate _dragstart_
		events that cause it to start scrolling
	*/
	preventDragPropagation: true,
	/**
		If true, the scroller will not propagate scroll events
	*/
	preventScrollPropagation: true,
	//* @protected
	handlers: {
		onscroll: "domScroll",
		onScrollStart: "scrollStart",
		onScroll: "scroll",
		onScrollStop: "scrollStop"
	},
	classes: "enyo-scroller",
	statics: {
		osInfo: [
			{os: "android", version: 3},
			{os: "androidChrome", version: 18},
			{os: "androidFirefox", version: 16},
			{os: "firefoxOS", version: 16},
			{os: "ios", version: 5},
			{os: "webos", version: 1e9},
			{os: "blackberry", version:1e9},
			{os: "tizen", version: 2}
		],
		//* Returns true if platform should have touch events.
		hasTouchScrolling: function() {
			for (var i=0, t; (t=this.osInfo[i]); i++) {
				if (enyo.platform[t.os]) {
					return true;
				}
			}
			// special detection for IE10+ on touch devices
			if ((enyo.platform.ie >= 10 || enyo.platform.windowsPhone >= 8) && enyo.platform.touch) {
				return true;
			}
		},
		/**
			Returns true if the platform has native div scrollers (desktop
			browsers always have them).
		*/
		hasNativeScrolling: function() {
			for (var i=0, t; (t=this.osInfo[i]); i++) {
				if (enyo.platform[t.os] < t.version) {
					return false;
				}
			}
			return true;
		},
		getTouchStrategy: function() {
			return (enyo.platform.android >= 3) || (enyo.platform.windowsPhone === 8)
				? "TranslateScrollStrategy"
				: "TouchScrollStrategy";
		}
	},
	controlParentName: "strategy",
	create: function() {
		this.inherited(arguments);
		this.horizontalChanged();
		this.verticalChanged();
		this.useMouseWheelChanged();
	},
	importProps: function(inProps) {
		this.inherited(arguments);
		// allow global overriding of strategy kind
		if (inProps && inProps.strategyKind === undefined && (enyo.Scroller.touchScrolling || this.touch)) {
			this.strategyKind = enyo.Scroller.getTouchStrategy();
		}
	},
	initComponents: function() {
		this.strategyKindChanged();
		this.inherited(arguments);
	},
	teardownChildren: function() {
		this.cacheScrollPosition();
		this.inherited(arguments);
	},
	rendered: function() {
		this.inherited(arguments);
		this.restoreScrollPosition();
	},
	strategyKindChanged: function() {
		if (this.$.strategy) {
			this.$.strategy.destroy();
			this.controlParent = null;
		}
		// note: createComponents automatically updates controlParent.
		this.createStrategy();
		if (this.hasNode()) {
			this.render();
		}
	},
	createStrategy: function() {
		this.createComponents([{name: "strategy", maxHeight: this.maxHeight,
			kind: this.strategyKind, thumb: this.thumb,
			preventDragPropagation: this.preventDragPropagation,
			overscroll:this.touchOverscroll, isChrome: true}]);
	},
	getStrategy: function() {
		return this.$.strategy;
	},
	maxHeightChanged: function() {
		this.$.strategy.setMaxHeight(this.maxHeight);
	},
	showingChanged: function() {
		if (!this.showing) {
			this.cacheScrollPosition();
			this.setScrollLeft(0);
			this.setScrollTop(0);
		}
		this.inherited(arguments);
		if (this.showing) {
			this.restoreScrollPosition();
		}
	},
	thumbChanged: function() {
		this.$.strategy.setThumb(this.thumb);
	},
	cacheScrollPosition: function() {
		this.cachedPosition = {left: this.getScrollLeft(), top: this.getScrollTop()};
	},
	restoreScrollPosition: function() {
		if (this.cachedPosition) {
			this.setScrollLeft(this.cachedPosition.left);
			this.setScrollTop(this.cachedPosition.top);
			this.cachedPosition = null;
		}
	},
	horizontalChanged: function() {
		this.$.strategy.setHorizontal(this.horizontal);
	},
	verticalChanged: function() {
		this.$.strategy.setVertical(this.vertical);
	},
	// FIXME: these properties are virtual; property changed methods are fired only if
	// property value changes, not if getter changes.
	//* Sets scroll position along horizontal axis.
	setScrollLeft: function(inLeft) {
		this.scrollLeft = inLeft;
		this.$.strategy.setScrollLeft(this.scrollLeft);
	},
	//* Sets scroll position along vertical axis.
	setScrollTop: function(inTop) {
		this.scrollTop = inTop;
		this.$.strategy.setScrollTop(inTop);
	},
	//* Gets scroll position along horizontal axis.
	getScrollLeft: function() {
		return this.$.strategy.getScrollLeft();
	},
	//* Gets scroll position along vertical axis.
	getScrollTop: function() {
		return this.$.strategy.getScrollTop();
	},
	//* @public
	/**
		Returns an object describing the scroll boundaries with _height_ and
		_width_ properties.
	*/
	getScrollBounds: function() {
		return this.$.strategy.getScrollBounds();
	},
	/**
		Scrolls the given control (_inControl_) into view. If _inAlignWithTop_
		is true, _inControl_ is aligned with the top of the scroller.
	*/
	scrollIntoView: function(inControl, inAlignWithTop) {
		this.$.strategy.scrollIntoView(inControl, inAlignWithTop);
	},
	//* Scrolls to the position specified by _inX_ and _inY_ in pixel units.
	scrollTo: function(inX, inY) {
		this.$.strategy.scrollTo(inX, inY);
	},
	/**
		Ensures that the given control is visible in the scroller's viewport.
		Unlike _scrollIntoView_, which uses DOM's _scrollIntoView_, this only
		affects the current scroller.
	*/
	scrollToControl: function(inControl, inAlignWithTop) {
		this.scrollToNode(inControl.hasNode(), inAlignWithTop);
	},
	//* Ensures that the given node is visible in the scroller's viewport.
	scrollToNode: function(inNode, inAlignWithTop) {
		this.$.strategy.scrollToNode(inNode, inAlignWithTop);
	},
	//* @protected
	//* Normalizes scroll event to _onScroll_.
	domScroll: function(inSender, e) {
		// if a scroll event originated here, pass it to our strategy to handle
		if (this.$.strategy.domScroll && e.originator == this) {
			this.$.strategy.scroll(inSender, e);
		}
		this.doScroll(e);
		return true;
	},
	/**
		Returns true if the current scroll event should be stopped; false if it
		should be allowed to propagate.
	*/
	shouldStopScrollEvent: function(inEvent) {
		return (this.preventScrollPropagation &&
			inEvent.originator.owner != this.$.strategy);
	},
	/**
		Calls _shouldStopScrollEvent_ to determine whether current scroll event
		should be stopped.
	*/
	scrollStart: function(inSender, inEvent) {
		return this.shouldStopScrollEvent(inEvent);
	},
	//* Either propagates or stops the current scroll event.
	scroll: function(inSender, inEvent) {
		// note: scroll event can be native dom or generated.
		if (inEvent.dispatchTarget) {
			// allow a dom event if it orignated with this scroller or its strategy
			return this.preventScrollPropagation && !(inEvent.originator == this ||
				inEvent.originator.owner == this.$.strategy);
		} else {
			return this.shouldStopScrollEvent(inEvent);
		}
	},
	/**
		Calls _shouldStopScrollEvent_ to determine whether current scroll event
		should be stopped.
	*/
	scrollStop: function(inSender, inEvent) {
		return this.shouldStopScrollEvent(inEvent);
	},
	//* @public
	//* Scroll to the top of the scrolling region.
	scrollToTop: function() {
		this.setScrollTop(0);
	},
	//* Scroll to the bottom of the scrolling region.
	scrollToBottom: function() {
		this.setScrollTop(this.getScrollBounds().maxTop);
	},
	//* Scroll to the right edge of the scrolling region.
	scrollToRight: function() {
		this.setScrollLeft(this.getScrollBounds().maxLeft);
	},
	//* Scroll to the left edge of the scrolling region.
	scrollToLeft: function() {
		this.setScrollLeft(0);
	},
	//* Ensures scroll position is in bounds.
	stabilize: function() {
		var s = this.getStrategy();
		if (s.stabilize) {
			s.stabilize();
		}
	},
	//* Send the useMouseWheel propert to the scroll strategy
	useMouseWheelChanged: function() {
		this.$.strategy.setUseMouseWheel(this.useMouseWheel);
	}
});

// provide a touch scrolling solution by default when the environment is mobile
if (enyo.Scroller.hasTouchScrolling()) {
	enyo.Scroller.prototype.strategyKind = enyo.Scroller.getTouchStrategy();
}

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/Thumb.css"
Content-Type: application/octet-stream; x-encoding=base64
LmVueW8tdGh1bWIgewoJcG9zaXRpb246IGFic29sdXRlOwoJLW1vei1ib3gtc2l6aW5nOiBib3JkZXItYm94OwoJYm94LXNpemluZzogYm9yZGVyLWJveDsKCWJvcmRlci1yYWRpdXM6IDRweDsKCWJhY2tncm91bmQ6ICMzMzM7Cglib3JkZXI6IDFweCBzb2xpZCAjNjY2OwoJb3BhY2l0eTogMC43NTsKCXotaW5kZXg6IDE7Cn0KCi5lbnlvLXZ0aHVtYiB7Cgl0b3A6IDA7CglyaWdodDogMnB4OwoJd2lkdGg6IDRweDsKfQoKLmVueW8taHRodW1iIHsKCWxlZnQ6IDA7Cglib3R0b206IDJweDsKCWhlaWdodDogNHB4Owp9Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/Thumb.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCl9lbnlvLlNjcm9sbFRodW1iXyBpcyBhIGhlbHBlciBraW5kIHVzZWQgYnkKPGEgaHJlZj0iI2VueW8uVG91Y2hTY3JvbGxTdHJhdGVneSI+ZW55by5Ub3VjaFNjcm9sbFN0cmF0ZWd5PC9hPiBhbmQKPGEgaHJlZj0iI2VueW8uVHJhbnNsYXRlU2Nyb2xsU3RyYXRlZ3kiPmVueW8uVHJhbnNsYXRlU2Nyb2xsU3RyYXRlZ3k8L2E+IHRvCmRpc3BsYXkgYSBzbWFsbCB2aXN1YWwgc2Nyb2xsIGluZGljYXRvci4KCl9lbnlvLlNjcm9sbFRodW1iXyBpcyBub3QgdHlwaWNhbGx5IGNyZWF0ZWQgaW4gYXBwbGljYXRpb24gY29kZS4KKi8KCmVueW8ua2luZCh7CgluYW1lOiAiZW55by5TY3JvbGxUaHVtYiIsCgkvLyogVGhlIG9yaWVudGF0aW9uIG9mIHRoZSBzY3JvbGwgaW5kaWNhdG9yIGJhcjsgInYiIGZvciB2ZXJ0aWNhbCBvciAiaCIgZm9yIGhvcml6b250YWwKCWF4aXM6ICJ2IiwKCS8vKiBAcHJvdGVjdGVkCgkvLyogTWluaW11bSBzaXplIG9mIHRoZSBpbmRpY2F0b3IKCW1pblNpemU6IDQsCgkvLyogU2l6ZSBvZiB0aGUgY29ybmVycyBvZiB0aGUgaW5kaWNhdG9yCgljb3JuZXJTaXplOiA2LAoJY2xhc3NlczogImVueW8tdGh1bWIiLAoJY3JlYXRlOiBmdW5jdGlvbigpIHsKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCXZhciB2ID0gdGhpcy5heGlzID09ICJ2IjsKCQl0aGlzLmRpbWVuc2lvbiA9IHYgPyAiaGVpZ2h0IiA6ICJ3aWR0aCI7CgkJdGhpcy5vZmZzZXQgPSB2ID8gInRvcCIgOiAibGVmdCI7CgkJdGhpcy50cmFuc2xhdGlvbiA9IHYgPyAidHJhbnNsYXRlWSIgOiAidHJhbnNsYXRlWCI7CgkJdGhpcy5wb3NpdGlvbk1ldGhvZCA9IHYgPyAiZ2V0U2Nyb2xsVG9wIiA6ICJnZXRTY3JvbGxMZWZ0IjsKCQl0aGlzLnNpemVEaW1lbnNpb24gPSB2ID8gImNsaWVudEhlaWdodCIgOiAiY2xpZW50V2lkdGgiOwoJCXRoaXMuYWRkQ2xhc3MoImVueW8tIiArIHRoaXMuYXhpcyArICJ0aHVtYiIpOwoJCXRoaXMudHJhbnNmb3JtID0gZW55by5kb20uY2FuVHJhbnNmb3JtKCk7CgkJaWYgKGVueW8uZG9tLmNhbkFjY2VsZXJhdGUoKSkgewoJCQllbnlvLmRvbS50cmFuc2Zvcm1WYWx1ZSh0aGlzLCAidHJhbnNsYXRlWiIsIDApOwoJCX0KCX0sCgkvLyogU3luY3MgdGhlIHNjcm9sbCBpbmRpY2F0b3IgYmFyIHRvIHRoZSBzY3JvbGxlciBzaXplIGFuZCBwb3NpdGlvbiwKCS8vKiBhcyBkZXRlcm1pbmVkIGJ5IHRoZSBwYXNzZWQtaW4gc2Nyb2xsIHN0cmF0ZWd5LgoJc3luYzogZnVuY3Rpb24oaW5TdHJhdGVneSkgewoJCXRoaXMuc2Nyb2xsQm91bmRzID0gaW5TdHJhdGVneS5fZ2V0U2Nyb2xsQm91bmRzKCk7CgkJdGhpcy51cGRhdGUoaW5TdHJhdGVneSk7Cgl9LAoJdXBkYXRlOiBmdW5jdGlvbihpblN0cmF0ZWd5KSB7CgkJaWYgKHRoaXMuc2hvd2luZykgewoJCQl2YXIgZCA9IHRoaXMuZGltZW5zaW9uLCBvID0gdGhpcy5vZmZzZXQ7CgkJCXZhciBiZCA9IHRoaXMuc2Nyb2xsQm91bmRzW3RoaXMuc2l6ZURpbWVuc2lvbl0sIHNiZCA9IHRoaXMuc2Nyb2xsQm91bmRzW2RdOwoJCQl2YXIgb3ZlcnMgPSAwLCBvdmVycCA9IDAsIG92ZXIgPSAwOwoJCQlpZiAoYmQgPj0gc2JkKSB7CgkJCQl0aGlzLmhpZGUoKTsKCQkJCXJldHVybjsKCQkJfQoJCQlpZiAoaW5TdHJhdGVneS5pc092ZXJzY3JvbGxpbmcoKSkgewoJCQkJb3ZlciA9IGluU3RyYXRlZ3kuZ2V0T3ZlclNjcm9sbEJvdW5kcygpWyJvdmVyIiArIG9dOwoJCQkJb3ZlcnMgPSBNYXRoLmFicyhvdmVyKTsKCQkJCW92ZXJwID0gTWF0aC5tYXgob3ZlciwgMCk7CgkJCX0KCQkJdmFyIHNibyA9IGluU3RyYXRlZ3lbdGhpcy5wb3NpdGlvbk1ldGhvZF0oKSAtIG92ZXI7CgkJCS8vIGNhbGMgc2l6ZSAmIHBvc2l0aW9uCgkJCXZhciBiZGMgPSBiZCAtIHRoaXMuY29ybmVyU2l6ZTsKCQkJdmFyIHMgPSBNYXRoLmZsb29yKChiZCAqIGJkIC8gc2JkKSAtIG92ZXJzKTsKCQkJcyA9IE1hdGgubWF4KHRoaXMubWluU2l6ZSwgcyk7CgkJCXZhciBwID0gTWF0aC5mbG9vcigoYmRjICogc2JvIC8gc2JkKSArIG92ZXJwKTsKCQkJcCA9IE1hdGgubWF4KDAsIE1hdGgubWluKGJkYyAtIHRoaXMubWluU2l6ZSwgcCkpOwoJCQkvLyBhcHBseSB0aHVtYiBzdHlsaW5nCgkJCXRoaXMubmVlZGVkID0gcyA8IGJkOwoJCQlpZiAodGhpcy5uZWVkZWQgJiYgdGhpcy5oYXNOb2RlKCkpIHsKCQkJCWlmICh0aGlzLl9wb3MgIT09IHApIHsKCQkJCQl0aGlzLl9wb3MgPSBwOwoJCQkJCWlmKCF0aGlzLnRyYW5zZm9ybSkgewoJCQkJCQkvL2FkanVzdCB0b3AvbGVmdCBmb3IgYnJvd3NlcnMgdGhhdCBkb24ndCBzdXBwb3J0IHRyYW5zbGF0aW9ucwoJCQkJCQlpZih0aGlzLmF4aXM9PSJ2IikgewoJCQkJCQkJdGhpcy5zZXRCb3VuZHMoe3RvcDpwICsgInB4In0pOwoJCQkJCQl9IGVsc2UgewoJCQkJCQkJdGhpcy5zZXRCb3VuZHMoe2xlZnQ6cCArICJweCJ9KTsKCQkJCQkJfQoJCQkJCX0gZWxzZSB7CgkJCQkJCWVueW8uZG9tLnRyYW5zZm9ybVZhbHVlKHRoaXMsIHRoaXMudHJhbnNsYXRpb24sIHAgKyAicHgiKTsKCQkJCQl9CgkJCQl9CgkJCQlpZiAodGhpcy5fc2l6ZSAhPT0gcykgewoJCQkJCXRoaXMuX3NpemUgPSBzOwoJCQkJCXRoaXMubm9kZS5zdHlsZVtkXSA9IHRoaXMuZG9tU3R5bGVzW2RdID0gcyArICJweCI7CgkJCQl9CgkJCX0gZWxzZSB7CgkJCQl0aGlzLmhpZGUoKTsKCQkJfQoJCX0KCX0sCgkvLyBpbXBsZW1lbnQgc2V0IGJlY2F1c2Ugc2hvd2luZyBpcyBub3QgY2hhbmdlZCB3aGlsZQoJLy8gd2UgZGVsYXlIaWRlIGJ1dCB3ZSB3YW50IHRvIGNhbmNlbCB0aGUgaGlkZS4KCXNldFNob3dpbmc6IGZ1bmN0aW9uKGluU2hvd2luZykgewoJCWlmIChpblNob3dpbmcgJiYgaW5TaG93aW5nICE9IHRoaXMuc2hvd2luZykgewoJCQlpZiAodGhpcy5zY3JvbGxCb3VuZHNbdGhpcy5zaXplRGltZW5zaW9uXSA+PSB0aGlzLnNjcm9sbEJvdW5kc1t0aGlzLmRpbWVuc2lvbl0pIHsKCQkJCXJldHVybjsKCQkJfQoJCX0KCQlpZiAodGhpcy5oYXNOb2RlKCkpIHsKCQkJdGhpcy5jYW5jZWxEZWxheUhpZGUoKTsKCQl9CgkJaWYgKGluU2hvd2luZyAhPSB0aGlzLnNob3dpbmcpIHsKCQkJdmFyIGxhc3QgPSB0aGlzLnNob3dpbmc7CgkJCXRoaXMuc2hvd2luZyA9IGluU2hvd2luZzsKCQkJdGhpcy5zaG93aW5nQ2hhbmdlZChsYXN0KTsKCQl9Cgl9LAoJZGVsYXlIaWRlOiBmdW5jdGlvbihpbkRlbGF5KSB7CgkJaWYgKHRoaXMuc2hvd2luZykgewoJCQllbnlvLmpvYih0aGlzLmlkICsgImhpZGUiLCB0aGlzLmJpbmRTYWZlbHkoImhpZGUiKSwgaW5EZWxheSB8fCAwKTsKCQl9Cgl9LAoJY2FuY2VsRGVsYXlIaWRlOiBmdW5jdGlvbigpIHsKCQllbnlvLmpvYi5zdG9wKHRoaXMuaWQgKyAiaGlkZSIpOwoJfQp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/TouchScrollStrategy.js"
Content-Type: application/octet-stream; x-encoding=base64
/**
	_enyo.TouchScrollStrategy_ is a helper kind for implementing a touch-based
	scroller. It integrates the scrolling simulation provided by
	<a href="#enyo.ScrollMath">enyo.ScrollMath</a> into an
	<a href="#enyo.Scroller">enyo.Scroller</a>.

	_enyo.TouchScrollStrategy_ is not typically created in application code.
	Instead, it is specified as the value of the `strategyKind` property of an
	`enyo.Scroller` or <a href="#enyo.List">enyo.List</a>, or is used by the
	framework implicitly.
*/
enyo.kind({
	name: "enyo.TouchScrollStrategy",
	kind: "ScrollStrategy",
	/**
		If true (the default), the scroller will overscroll and bounce back at the edges
	*/
	overscroll: true,
	/**
		If true (the default), the scroller will not propagate _dragstart_
		events that cause it to start scrolling
	*/
	preventDragPropagation: true,
	published: {
		/**
			Specifies how to vertically scroll.  Acceptable values are:

			* "scroll": Always scroll.
			* "auto": Scroll only if the content overflows the scroller.
			* "hidden": Never scroll.
			* "default": In touch environments, the default vertical scrolling
				behavior is to always scroll. If the content does not overflow
				the scroller, the scroller will overscroll and snap back.
		*/
		vertical: "default",
		/**
			Specifies how to horizontally scroll.  Acceptable values are:

			* "scroll": Always scroll.
			* "auto":  Scroll only if the content overflows the scroller.
			* "hidden": Never scroll.
			* "default": Same as "auto".
		*/
		horizontal: "default",
		//* Set to true to display a scroll thumb
		thumb: true,
		/**
			Set to true to display a transparent overlay while scrolling. This
			can help improve performance of complex, large scroll regions on
			some platforms (e.g., Android).
		*/
		scrim: false,
		//*	Allow drag events sent when gesture events are happening simultaneously
		dragDuringGesture: true
	},
	events: {
		onShouldDrag: ""
	},
	//* @protected
	handlers: {
		onscroll: "domScroll",
		onflick: "flick",
		onhold: "hold",
		ondragstart: "dragstart",
		onShouldDrag: "shouldDrag",
		ondrag: "drag",
		ondragfinish: "dragfinish",
		onmousewheel: "mousewheel"
	},
	tools: [
		{kind: "ScrollMath", onScrollStart: "scrollMathStart", onScroll: "scrollMathScroll", onScrollStop: "scrollMathStop"},
		{name: "vthumb", kind: "ScrollThumb", axis: "v", showing: false},
		{name: "hthumb", kind: "ScrollThumb", axis: "h", showing: false}
	],
	scrimTools: [{name: "scrim", classes: "enyo-fit", style: "z-index: 1;", showing: false}],
	components: [
		{name: "client", classes: "enyo-touch-scroller"}
	],
	// flag telling us whether the list is currently reordering
	listReordering: false,
	create: function() {
		this.inherited(arguments);
		this.transform = enyo.dom.canTransform();
		if(!this.transform) {
			if(this.overscroll) {
				//so we can adjust top/left if browser can't handle translations
				this.$.client.applyStyle("position", "relative");
			}
		}
		this.accel = enyo.dom.canAccelerate();
		var containerClasses = "enyo-touch-strategy-container";
		// note: needed for ios to avoid incorrect clipping of thumb
		// and need to avoid on Android as it causes problems hiding the thumb
		if (enyo.platform.ios && this.accel) {
			containerClasses += " enyo-composite";
		}
		this.scrimChanged();
		this.container.addClass(containerClasses);
		this.translation = this.accel ? "translate3d" : "translate";
	},
	initComponents: function() {
		this.createChrome(this.tools);
		this.inherited(arguments);
	},
	destroy: function() {
		this.container.removeClass("enyo-touch-strategy-container");
		this.inherited(arguments);
	},
	rendered: function() {
		this.inherited(arguments);
		enyo.makeBubble(this.$.client, "scroll");
		this.calcBoundaries();
		this.syncScrollMath();
		if (this.thumb) {
			this.alertThumbs();
		}
	},
	scrimChanged: function() {
		if (this.scrim && !this.$.scrim) {
			this.makeScrim();
		}
		if (!this.scrim && this.$.scrim) {
			this.$.scrim.destroy();
		}
	},
	makeScrim: function() {
		// reset control parent so scrim doesn't go into client.
		var cp = this.controlParent;
		this.controlParent = null;
		this.createChrome(this.scrimTools);
		this.controlParent = cp;
		var cn = this.container.hasNode();
		// render scrim in container, strategy has no dom.
		if (cn) {
			this.$.scrim.parentNode = cn;
			this.$.scrim.render();
		}
	},
	//* Whether or not the scroller is actively moving
	isScrolling: function() {
		var m = this.$.scrollMath;
		return m ? m.isScrolling() : this.scrolling;
	},
	//* Whether or not the scroller is in overscrolling
	isOverscrolling: function() {
		var m = this.$.scrollMath || this;
		return (this.overscroll) ? m.isInOverScroll() : false;
	},
	domScroll: function() {
		if (!this.isScrolling()) {
			this.calcBoundaries();
			this.syncScrollMath();
			if (this.thumb) {
				this.alertThumbs();
			}
		}
	},
	horizontalChanged: function() {
		this.$.scrollMath.horizontal = (this.horizontal != "hidden");
	},
	verticalChanged: function() {
		this.$.scrollMath.vertical = (this.vertical != "hidden");
	},
	maxHeightChanged: function() {
		this.$.client.applyStyle("max-height", this.maxHeight);
		// note: previously used enyo-fit here but IE would reset scroll position when the scroll thumb
		// was hidden; in general IE resets scrollTop when there are 2 abs position siblings, one has
		// scrollTop and the other is hidden.
		this.$.client.addRemoveClass("enyo-scrollee-fit", !this.maxHeight);
	},
	thumbChanged: function() {
		this.hideThumbs();
	},
	stop: function() {
		if (this.isScrolling()) {
			this.$.scrollMath.stop(true);
		}
	},
	stabilize: function() {
		if(this.$.scrollMath) {
			this.$.scrollMath.stabilize();
		}
	},
	//* Scrolls to specific x/y positions within the scroll area.
	scrollTo: function(inX, inY) {
		this.stop();
		this.$.scrollMath.scrollTo(inX, inY || inY === 0 ? inY : null);
	},
	scrollIntoView: function() {
		this.stop();
		this.inherited(arguments);
	},
	//* Sets the left scroll position within the scroller.
	setScrollLeft: function() {
		this.stop();
		this.inherited(arguments);
	},
	//* Sets the top scroll position within the scroller.
	setScrollTop: function() {
		this.stop();
		this.inherited(arguments);
	},
	//* Gets the left scroll position within the scroller.
	getScrollLeft: function() {
		return this.isScrolling() ? this.scrollLeft : this.inherited(arguments);
	},
	//* Gets the top scroll position within the scroller.
	getScrollTop: function() {
		return this.isScrolling() ? this.scrollTop : this.inherited(arguments);
	},
	calcScrollNode: function() {
		return this.$.client.hasNode();
	},
	calcAutoScrolling: function() {
		var v = (this.vertical == "auto");
		var h = (this.horizontal == "auto") || (this.horizontal == "default");
		if ((v || h) && this.scrollNode) {
			var b = this.getScrollBounds();
			if (v) {
				this.$.scrollMath.vertical = b.height > b.clientHeight;
			}
			if (h) {
				this.$.scrollMath.horizontal = b.width > b.clientWidth;
			}
		}
	},
	shouldDrag: function(inSender, e) {
		this.calcAutoScrolling();
		var requestV = e.vertical;
		var canH = this.$.scrollMath.horizontal && !requestV;
		var canV = this.$.scrollMath.vertical && requestV;
		var down = e.dy < 0, right = e.dx < 0;
		var oobV = (!down && this.startEdges.top || down && this.startEdges.bottom);
		var oobH = (!right && this.startEdges.left || right && this.startEdges.right);
		// we would scroll if not at a boundary
		if (!e.boundaryDragger && (canH || canV)) {
			e.boundaryDragger = this;
		}
		// include boundary exclusion
		if ((!oobV && canV) || (!oobH && canH)) {
			e.dragger = this;
			return true;
		}
	},
	flick: function(inSender, e) {
		var onAxis = Math.abs(e.xVelocity) > Math.abs(e.yVelocity) ? this.$.scrollMath.horizontal : this.$.scrollMath.vertical;
		if (onAxis && this.dragging) {
			this.$.scrollMath.flick(e);
			return this.preventDragPropagation;
		}
	},
	hold: function(inSender, e) {
		if (this.isScrolling() && !this.isOverscrolling()) {
			var m = this.$.scrollMath || this;
			m.stop(e);
			return true;
		}
	},
	move: function(inSender, inEvent) {
	},
	// Special synthetic DOM events served up by the Gesture system
	dragstart: function(inSender, inEvent) {
		// Ignore drags sent from multi-touch events
		if(!this.dragDuringGesture && inEvent.srcEvent.touches && inEvent.srcEvent.touches.length > 1) {
			return true;
		}
		// note: allow drags to propagate to parent scrollers via data returned in the shouldDrag event.
		this.doShouldDrag(inEvent);
		this.dragging = (inEvent.dragger == this || (!inEvent.dragger && inEvent.boundaryDragger == this));
		if (this.dragging) {
			inEvent.preventDefault();
			// note: needed because show/hide changes
			// the position so sync'ing is required when
			// dragging begins (needed because show/hide does not trigger onscroll)
			this.syncScrollMath();
			this.$.scrollMath.startDrag(inEvent);
			if (this.preventDragPropagation) {
				return true;
			}
		}
	},
	drag: function(inSender, inEvent) {
		// if the list is doing a reorder, don't scroll
		if(this.listReordering) {
			return false;
		}
		if (this.dragging) {
			inEvent.preventDefault();
			this.$.scrollMath.drag(inEvent);
			if (this.scrim) {
				this.$.scrim.show();
			}
		}
	},
	dragfinish: function(inSender, inEvent) {
		if (this.dragging) {
			inEvent.preventTap();
			this.$.scrollMath.dragFinish();
			this.dragging = false;
			if (this.scrim) {
				this.$.scrim.hide();
			}
		}
	},
	mousewheel: function(inSender, e) {
		if (!this.dragging && this.useMouseWheel) {
			this.calcBoundaries();
			this.syncScrollMath();
			this.stabilize();
			if (this.$.scrollMath.mousewheel(e)) {
				e.preventDefault();
				return true;
			}
		}
	},
	scrollMathStart: function(inSender) {
		if (this.scrollNode) {
			this.calcBoundaries();
			if (this.thumb) {
				this.showThumbs();
			}
		}
	},
	scrollMathScroll: function(inSender) {
		if(!this.overscroll) {
			//don't overscroll past edges
			this.effectScroll(-Math.min(inSender.leftBoundary, Math.max(inSender.rightBoundary, inSender.x)),
					-Math.min(inSender.topBoundary, Math.max(inSender.bottomBoundary, inSender.y)));
		} else {
			this.effectScroll(-inSender.x, -inSender.y);
		}
		if (this.thumb) {
			this.updateThumbs();
		}
	},
	scrollMathStop: function(inSender) {
		this.effectScrollStop();
		if (this.thumb) {
			this.delayHideThumbs(100);
		}
	},
	calcBoundaries: function() {
		var s = this.$.scrollMath || this, b = this._getScrollBounds();
		s.bottomBoundary = b.clientHeight - b.height;
		s.rightBoundary = b.clientWidth - b.width;
	},
	syncScrollMath: function() {
		var m = this.$.scrollMath;
		if(m) {
			m.setScrollX(-this.getScrollLeft());
			m.setScrollY(-this.getScrollTop());
		}
	},
	effectScroll: function(inX, inY) {
		if (this.scrollNode) {
			this.scrollLeft = this.scrollNode.scrollLeft = inX;
			this.scrollTop = this.scrollNode.scrollTop = inY;
			this.effectOverscroll(Math.round(inX), Math.round(inY));
		}
	},
	effectScrollStop: function() {
		this.effectOverscroll(null, null);
	},
	effectOverscroll: function(inX, inY) {
		var n = this.scrollNode;
		var x = "0", y = "0", z = this.accel ? ",0" : "";
		if (inY !== null && Math.abs(inY - n.scrollTop) > 1) {
			y = (n.scrollTop - inY);
		}
		if (inX !== null && Math.abs(inX - n.scrollLeft) > 1) {
			x = (n.scrollLeft - inX);
		}
		if(!this.transform) {
			//adjust top/left if browser can't handle translations
			this.$.client.setBounds({left:x + "px", top:y + "px"});
		} else {
			enyo.dom.transformValue(this.$.client, this.translation, x + "px, " + y + "px" + z);
		}
	},
	//* Returns the values of _overleft_ and _overtop_, if any.
	getOverScrollBounds: function() {
		var m = this.$.scrollMath || this;
		return {
			overleft: Math.min(m.leftBoundary - m.x, 0) || Math.max(m.rightBoundary - m.x, 0),
			overtop: Math.min(m.topBoundary - m.y, 0) || Math.max(m.bottomBoundary - m.y, 0)
		};
	},
	_getScrollBounds: function() {
		var r = this.inherited(arguments);
		enyo.mixin(r, this.getOverScrollBounds());
		return r;
	},
	getScrollBounds: function() {
		this.stop();
		return this.inherited(arguments);
	},
	// Thumb processing
	alertThumbs: function() {
		this.showThumbs();
		this.delayHideThumbs(500);
	},
	//* Syncs the vertical and horizontal scroll indicators.
	syncThumbs: function() {
		this.$.vthumb.sync(this);
		this.$.hthumb.sync(this);
	},
	updateThumbs: function() {
		this.$.vthumb.update(this);
		this.$.hthumb.update(this);
	},
	//* Syncs and shows both the vertical and horizontal scroll indicators.
	showThumbs: function() {
		this.syncThumbs();
		if (this.horizontal != "hidden") {
			this.$.hthumb.show();
		}
		if (this.vertical != "hidden") {
			this.$.vthumb.show();
		}
	},
	//* Hides the vertical and horizontal scroll indicators.
	hideThumbs: function() {
		this.$.vthumb.hide();
		this.$.hthumb.hide();
	},
	//* Hides the vertical and horizontal scroll indicators asynchronously.
	delayHideThumbs: function(inDelay) {
		this.$.vthumb.delayHide(inDelay);
		this.$.hthumb.delayHide(inDelay);
	}
});

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/TransitionScrollStrategy.js"
Content-Type: application/octet-stream; x-encoding=base64
/**
	_enyo.TransitionScrollStrategy_ is a helper kind that extends
	<a href="#enyo.TouchScrollStrategy">enyo.TouchScrollStrategy</a>, optimizing
	it for scrolling environments in which effecting scroll changes with
	transforms using CSS transitions is fastest.

	_enyo.TransitionScrollStrategy_ is not typically created in application code.
	Instead, it is specified as the value of the `strategyKind` property of an
	`enyo.Scroller` or <a href="#enyo.List">enyo.List</a>, or is used by the
	framework implicitly.
*/
enyo.kind({
	name: "enyo.TransitionScrollStrategy",
	kind: "enyo.TouchScrollStrategy",
	//* @protected
	components: [
		{name: "clientContainer", classes: "enyo-touch-scroller", components: [
			{name: "client"}
		]}
	],
	events: {
		onScrollStart: "",
		onScroll: "",
		onScrollStop: ""
	},
	handlers: {
		ondown: "down",
		ondragfinish: "dragfinish",
		onwebkitTransitionEnd: "transitionComplete"
	},
	//* No scrollMath tool for this strategy
	tools: [
		{name: "vthumb", kind: "ScrollThumb", axis: "v", showing: true},
		{name: "hthumb", kind: "ScrollThumb", axis: "h", showing: false}
	],
	//* Scalar applied to 'flick' event velocity
	kFlickScalar: 600,
	//* Top snap boundary, generally 0
	topBoundary: 0,
	//* Right snap boundary, generally (viewport width - content width)
	rightBoundary: 0,
	//* Bottom snap boundary, generally (viewport height - content height)
	bottomBoundary: 0,
	//* Left snap boundary, generally 0
	leftBoundary: 0,
	//* Flag to specify whether scrolling is in progress
	scrolling: false,
	//* Event listener for webkit transition completion
	listener: null,
	//* X Distance to scroll into overscroll space before bouncing back
	boundaryX:0,
	//* Y Distance to scroll into overscroll space before bouncing back
	boundaryY:0,
	//* Timeout used to stop scrolling on mousedown
	stopTimeout: null,
	//* MS delay used to stop scrolling on mousedown
	stopTimeoutMS: 80,
	//* Interval used to update scroll values and bubble scroll events during scroll animation
	scrollInterval: null,
	//* MS delay used to update scroll values and bubble scroll events during scroll animation
	scrollIntervalMS: 50,
	//* Transition animations
	transitions: {
		//* None - used for dragging, etc.
		none   : "",
		//* Scroll - basic scrolling behavior
		scroll : "3.8s cubic-bezier(.19,1,.28,1.0) 0s",
		//* Bounce - overscroll bounceback behavior
		bounce : "0.5s cubic-bezier(0.06,.5,.5,.94) 0s"
	},

	//* @public

	//* Sets the left scroll position within the scroller.
	setScrollLeft: function(inLeft) {
		var prevLeft = this.scrollLeft;
		this.stop();
		this.scrollLeft = inLeft;
		if(this.isInLeftOverScroll() || this.isInRightOverScroll()) {
			this.scrollLeft = prevLeft;
		}
		this.effectScroll();
	},
	//* Sets the top scroll position within the scroller.
	setScrollTop: function(inTop) {
		var prevTop = this.scrollTop;
		this.stop();
		this.scrollTop = inTop;
		if(this.isInTopOverScroll() || this.isInBottomOverScroll()) {
			this.scrollTop = prevTop;
		}
		this.effectScroll();
	},
	setScrollX: function(inLeft) {
		this.scrollLeft = -1*inLeft;
	},
	setScrollY: function(inTop) {
		this.scrollTop = -1*inTop;
	},

	//* Gets the left scroll position within the scroller.
	getScrollLeft: function() {
		return this.scrollLeft;
	},
	//* Gets the top scroll position within the scroller.
	getScrollTop: function() {
		return this.scrollTop;
	},

	//* @protected

	// apply initial transform so we're always composited
	create: function() {
		this.inherited(arguments);
		enyo.dom.transformValue(this.$.client, this.translation, "0,0,0");
	},
	destroy: function() {
		this.clearCSSTransitionInterval();
		this.inherited(arguments);
	},
	getScrollSize: function() {
		var n = this.$.client.hasNode();
		return {width: n ? n.scrollWidth : 0, height: n ? n.scrollHeight : 0};
	},
	horizontalChanged: function() {
		if(this.horizontal == "hidden") {
			this.scrollHorizontal = false;
		}
	},
	verticalChanged: function() {
		if(this.vertical == "hidden") {
			this.scrollVertical = false;
		}
	},
	calcScrollNode: function() {
		return this.$.clientContainer.hasNode();
	},
	calcBoundaries: function() {
		var b = this._getScrollBounds();
		this.bottomBoundary = b.clientHeight - b.height;
		this.rightBoundary = b.clientWidth - b.width;
	},
	maxHeightChanged: function() {
		// content should cover scroller at a minimum if there's no max-height.
		this.$.client.applyStyle("min-height", this.maxHeight ? null : "100%");
		this.$.client.applyStyle("max-height", this.maxHeight);
		this.$.clientContainer.addRemoveClass("enyo-scrollee-fit", !this.maxHeight);
	},
	calcAutoScrolling: function() {
		var b = this.getScrollBounds();
		if(this.vertical) {
			this.scrollVertical = b.height > b.clientHeight;
		}
		if(this.horizontal) {
			this.scrollHorizontal = b.width > b.clientWidth;
		}
	},
	isInOverScroll: function() {
		return (this.isInTopOverScroll() || this.isInBottomOverScroll() || this.isInLeftOverScroll() || this.isInRightOverScroll());
	},
	isInLeftOverScroll: function() {
		return (this.getScrollLeft() < this.leftBoundary);
	},
	isInRightOverScroll: function() {
		if(this.getScrollLeft <= 0) {
			return false;
		} else {
			return (this.getScrollLeft()*-1 < this.rightBoundary);
		}
	},
	isInTopOverScroll: function() {
		return (this.getScrollTop() < this.topBoundary);
	},
	isInBottomOverScroll: function() {
		if(this.getScrollTop() <= 0) {
			return false;
		} else {
			return (this.getScrollTop()*-1 < this.bottomBoundary);
		}
	},
	calcStartInfo: function() {
		var sb = this.getScrollBounds(), y = this.getScrollTop(), x = this.getScrollLeft();
		this.startEdges = {
			top: y === 0,
			bottom: y === sb.maxTop,
			left: x === 0,
			right: x === sb.maxLeft
		};
	},
	mousewheel: function(inSender, e) {
		if (!this.dragging && this.useMouseWheel) {
			this.calcBoundaries();
			this.syncScrollMath();
			this.stabilize();
			var dy = this.vertical ? e.wheelDeltaY || e.wheelDelta : 0;
			var y = parseFloat(this.getScrollTop()) + -1*parseFloat(dy);
			y = (y*-1 < this.bottomBoundary) ? -1*this.bottomBoundary : (y < this.topBoundary) ? this.topBoundary : y;
			this.setScrollTop(y);
			this.doScroll();
			e.preventDefault();
			return true;
		}
	},
	// Update thumbs, recalculate boundaries, and bubble scroll event
	scroll: function(inSender, inEvent) {
		if(this.thumb) {
			this.updateThumbs();
		}
		this.calcBoundaries();
		this.doScroll();
	},
	// Scroll to current x,y coordinates and bubble scrollstart event
	start: function() {
		this.startScrolling();
		this.doScrollStart();
	},
	// If scrolling, stop. Hide thumbs and bubble scrollstop event.
	stop: function() {
		if(this.isScrolling()) {
			this.stopScrolling();
		}
		if (this.thumb) {
			this.delayHideThumbs(100);
		}
		this.doScrollStop();
	},

	// Set scroll x value to the current computed style
	updateX: function() {
		var x = window.getComputedStyle(this.$.client.node,null).getPropertyValue(enyo.dom.getCssTransformProp()).split('(')[1];
		x = (x === undefined) ? 0 : x.split(')')[0].split(',')[4];
		if(-1*parseFloat(x) === this.scrollLeft) {
			return false;
		}
		this.scrollLeft = -1*parseFloat(x);
		return true;
	},
	// Set scroll y value to the current computed style
	updateY: function() {
		var y = window.getComputedStyle(this.$.client.node,null).getPropertyValue(enyo.dom.getCssTransformProp()).split('(')[1];
		y = (y === undefined) ? 0 : y.split(')')[0].split(',')[5];
		if(-1*parseFloat(y) === this.scrollTop) {
			return false;
		}
		this.scrollTop = -1*parseFloat(y);
		return true;
	},
	// Apply transform to scroll the scroller
	effectScroll: function() {
		var o = (-1*this.scrollLeft) + "px, " + (-1*this.scrollTop) + "px" + (this.accel ? ", 0" : "");
		enyo.dom.transformValue(this.$.client, this.translation, o);
	},
	// On touch, stop transition by setting transform values to current computed style, and
	// changing transition time to 0s. TODO
	down: function(inSender, inEvent) {
		var _this = this;
		if (this.isScrolling() && !this.isOverscrolling()) {
			this.stopTimeout = setTimeout(function() {
				_this.stop();
			}, this.stopTimeoutMS);
			return true;
		}
	},
	// Special synthetic DOM events served up by the Gesture system
	dragstart: function(inSender, inEvent) {
		if(this.stopTimeout) {
			clearTimeout(this.stopTimeout);
		}
		// Ignore drags sent from multi-touch events
		if(!this.dragDuringGesture && inEvent.srcEvent.touches && inEvent.srcEvent.touches.length > 1) {
			return true;
		}
		this.shouldDrag(inEvent);
		this.dragging = (inEvent.dragger == this || (!inEvent.dragger && inEvent.boundaryDragger == this));
		if (this.dragging) {
			if(this.isScrolling()) {
				this.stopScrolling();
			}
			if(this.thumb) {
				this.showThumbs();
			}
			inEvent.preventDefault();
			this.prevY = inEvent.pageY;
			this.prevX = inEvent.pageX;
			if (this.preventDragPropagation) {
				return true;
			}
		}
	},
	// Determine if we should allow dragging
	shouldDrag: function(e) {
		this.calcStartInfo();
		this.calcBoundaries();
		this.calcAutoScrolling();
		if(!this.scrollHorizontal) {
			return this.shouldDragVertical(e);
		} else {
			if(!this.scrollVertical) {
				return this.shouldDragHorizontal(e);
			} else {
				return (this.shouldDragVertical(e) || this.shouldDragHorizontal(e));
			}
		}
	},
	shouldDragVertical: function(e) {
		var canV = this.canDragVertical(e);
		var oobV = this.oobVertical(e);
		// scroll if not at a boundary
		if (!e.boundaryDragger && canV) {
			e.boundaryDragger = this;
		}
		// include boundary exclusion
		if (!oobV && canV) {
			e.dragger = this;
			return true;
		}
	},
	shouldDragHorizontal: function(e) {
		var canH = this.canDragHorizontal(e);
		var oobH = this.oobHorizontal(e);
		// scroll if not at a boundary
		if (!e.boundaryDragger && canH) {
			e.boundaryDragger = this;
		}
		// include boundary exclusion
		if (!oobH && canH) {
			e.dragger = this;
			return true;
		}
	},
	canDragVertical: function(e) {
		return (this.scrollVertical && e.vertical);
	},
	canDragHorizontal: function(e) {
		return (this.scrollHorizontal && !e.vertical);
	},
	oobVertical: function(e) {
		var down = e.dy < 0;
		return (!down && this.startEdges.top || down && this.startEdges.bottom);
	},
	oobHorizontal: function(e) {
		var right = e.dx < 0;
		return (!right && this.startEdges.left || right && this.startEdges.right);
	},
	// Move scroller based on user's dragging
	drag: function(inSender, e) {
		// if the list is doing a reorder, don't scroll
		if(this.listReordering) {
			return false;
		}
		// if shouldDrag() set this.dragging to true
		if(this.dragging) {
			e.preventDefault();
			// calculate new scroll values
			this.scrollLeft = this.scrollHorizontal ?
				this.calculateDragDistance(parseInt(this.getScrollLeft(), 10), (-1*(e.pageX-this.prevX)), this.leftBoundary, this.rightBoundary) :
				this.getScrollLeft();
			this.scrollTop = this.scrollVertical ?
				this.calculateDragDistance(this.getScrollTop(), (-1*(e.pageY-this.prevY)), this.topBoundary, this.bottomBoundary) :
				this.getScrollTop();
			// apply new scroll values
			this.effectScroll();
			this.scroll();
			// save values for next drag
			this.prevY = e.pageY;
			this.prevX = e.pageX;
			this.resetBoundaryX();
			this.resetBoundaryY();
		}
	},
	//* Figure how far the drag should go based on pointer movement (delta)
	calculateDragDistance: function(currentPosition, delta, aBoundary, bBoundary) {
		var newPosition = currentPosition + delta;
		return this.overscrollDragDamping(currentPosition, newPosition,delta,aBoundary,bBoundary);
	},
	//* Provides resistance against dragging into overscroll
	overscrollDragDamping: function(currentPosition, newPosition, delta, aBoundary, bBoundary) {
		if(newPosition < aBoundary || newPosition*-1 < bBoundary) {
			delta /= 2;
			newPosition = currentPosition + delta;
		}
		return newPosition;
	},
	resetBoundaryX: function() {
		this.boundaryX = 0;
	},
	resetBoundaryY: function() {
		this.boundaryY = 0;
	},
	// When user releases the drag, set this.dragging to false, bounce overflow back, and hide scrim.
	dragfinish: function(inSender, inEvent) {
		if (this.dragging) {
			inEvent.preventTap();
			this.dragging = false;
			if(!this.isScrolling()) {
				this.correctOverflow();
			}
			if (this.scrim) {
				this.$.scrim.hide();
			}
		}
	},
	// Bounce back from overscroll region
	correctOverflow: function() {
		if(this.isInOverScroll()) {
			var x = (this.scrollHorizontal) ? this.correctOverflowX() : false;
			var y = (this.scrollVertical) ? this.correctOverflowY() : false;
			if(x !== false && y !== false) {
				this.scrollLeft = (x !== false) ? x : this.getScrollLeft();
				this.scrollTop = (y !== false) ? y : this.getScrollTop();
				this.startOverflowScrolling();
			} else if(x !== false) {
				this.scrollLeft = x;
				this.scrollTop = this.targetScrollTop || this.scrollTop;
				this.targetScrollLeft = this.getScrollLeft();
				if(!this.vertical) {
					this.startOverflowScrolling();
				} else {
					this.startScrolling();
				}
			} else if(y !== false) {
				this.scrollTop = y;
				this.scrollLeft = this.targetScrollLeft || this.scrollLeft;
				this.targetScrollTop = this.getScrollTop();
				if(!this.scrollHorizontal) {
					this.startOverflowScrolling();
				} else {
					this.startScrolling();
				}
			}
		}
	},
	// Determine if we're overscrolled on the x axis and if so return proper edge value
	correctOverflowX: function() {
		if(this.isInLeftOverScroll()) {
			if(this.beyondBoundary(this.getScrollLeft(), this.leftBoundary, this.boundaryX)) {
				return this.leftBoundary;
			}
		} else if(this.isInRightOverScroll()) {
			if(this.beyondBoundary(this.getScrollLeft(), this.rightBoundary, this.boundaryX)) {
				return -1*this.rightBoundary;
			}
		}
		return false;
	},
	// Determine if we're overscrolled on the y axis and if so return proper edge value
	correctOverflowY: function() {
		if(this.isInTopOverScroll()) {
			if(this.beyondBoundary(this.getScrollTop(), this.topBoundary, this.boundaryY)) {
				return this.topBoundary;
			}
		} else if(this.isInBottomOverScroll()) {
			if(this.beyondBoundary(this.getScrollTop(), this.bottomBoundary, this.boundaryY)) {
				return -1*this.bottomBoundary;
			}
		}
		return false;
	},
	// If we've crossed the determined delta, bounce back
	beyondBoundary: function(current,boundary,max) {
		return (Math.abs(Math.abs(boundary) - Math.abs(current)) > Math.abs(max));
	},
	// When user flicks/throws scroller, figure the distance to be travelled and whether we will end up
	// in the overscroll region.
	flick: function(inSender, e) {
		if(this.dragging && this.flickOnEnabledAxis(e)) {
			this.scrollLeft = this.scrollHorizontal ? this.calculateFlickDistance(this.scrollLeft, -1*e.xVelocity) : this.getScrollLeft();
			this.scrollTop = this.scrollVertical ? this.calculateFlickDistance(this.scrollTop, -1*e.yVelocity) : this.getScrollTop();
			this.targetScrollLeft = this.scrollLeft;
			this.targetScrollTop = this.scrollTop;
			this.boundaryX = null;
			this.boundaryY = null;
			// if flick will put the x axis into overscroll, figure where we should bounce back (boundary)
			if(this.isInLeftOverScroll()) {
				this.boundaryX = this.figureBoundary(this.getScrollLeft());
			} else if(this.isInRightOverScroll()) {
				this.boundaryX = this.figureBoundary(-1*this.bottomBoundary - this.getScrollLeft());
			}
			// if flick will put the y axis into overscroll, figure where we should bounce back (boundary)
			if(this.isInTopOverScroll()) {
				this.boundaryY = this.figureBoundary(this.getScrollTop());
			} else if(this.isInBottomOverScroll()) {
				this.boundaryY = this.figureBoundary(-1*this.bottomBoundary - this.getScrollTop());
			}
			// kickoff scrolling animation
			this.startScrolling();
			return this.preventDragPropagation;
		}
	},
	flickOnEnabledAxis: function(e) {
		return Math.abs(e.xVelocity) > Math.abs(e.yVelocity) ? this.scrollHorizontal : this.scrollVertical;
	},
	calculateFlickDistance: function(currentPosition, flickVelocity) {
		return (currentPosition + (flickVelocity * this.kFlickScalar));
	},
	// Apply the 'scroll' transition, apply new transform based on x and y, and begin
	// this.scrollInterval to update the scrollTop/Left values while scrolling
	startScrolling: function() {
		this.applyTransition("scroll");
		this.effectScroll();
		this.setCSSTransitionInterval();
		this.scrolling = true;
	},
	// Apply the 'bounce' transition, apply new transform based on x and y, and begin
	// this.scrollInterval to update the scrollTop/Left values while scrolling
	startOverflowScrolling: function() {
		this.applyTransition("bounce");
		this.effectScroll();
		this.setOverflowTransitionInterval();
		this.scrolling = true;
	},
	// Apply the given transition to this.$.client
	applyTransition: function(which) {
		this.$.client.applyStyle("-webkit-transition", this.transitions[which]);
	},
	// Turn off CSS transition and clear this.scrollInterval
	stopScrolling: function() {
		this.resetCSSTranslationVals();
		this.clearCSSTransitionInterval();
		this.scrolling = false;
	},
	// Create an interval to: update the x/y values while scrolling is happening, check for
	// crossing into the overflow region, and bubble a scroll event
	setCSSTransitionInterval: function() {
		this.clearCSSTransitionInterval();
		this.scrollInterval = setInterval(this.bindSafely(function() {
			this.updateScrollPosition();
			this.correctOverflow();
		}), this.scrollIntervalMS);
	},
	// Create an interval to: update the x/y values while scrolling is happening, and bubble
	// a scroll event (don't check for crossing into overflow since we're there already)
	setOverflowTransitionInterval: function() {
		this.clearCSSTransitionInterval();
		this.scrollInterval = setInterval(this.bindSafely(function() {
			this.updateScrollPosition();
		}), this.scrollIntervalMS);
	},
	// Save current x/y position and bubble scroll event
	updateScrollPosition: function() {
		var yChanged = this.updateY();
		var xChanged = this.updateX();
		this.scroll();
		if(!yChanged && !xChanged) {
			this.stop();
		}
	},
	// Clear this.scrollInterval
	clearCSSTransitionInterval: function() {
		if(this.scrollInterval) {
			clearInterval(this.scrollInterval);
			this.scrollInterval = null;
		}
	},
	// Set scroller translation to current position and turn transition off. This effectively
	// stops scrolling
	resetCSSTranslationVals: function() {
		var prop = enyo.dom.getCssTransformProp();
		var transformStyle = window.getComputedStyle(this.$.client.node,null).getPropertyValue(prop).split('(')[1].split(')')[0].split(',');
		this.applyTransition("none");
		this.scrollLeft = -1*transformStyle[4];
		this.scrollTop = -1*transformStyle[5];
		this.effectScroll();
	},
	// Figure how far into the overscroll region we should go before bouncing back
	figureBoundary: function(target) {
		var absTarget = Math.abs(target);
		var retVal = absTarget - absTarget/Math.pow(absTarget,0.02);
		retVal = target < 0 ? -1*retVal : retVal;
		return retVal;
	},
	// When transition animation is complete, check if we need to bounce back from overscroll
	// region. If not, stop.
	transitionComplete: function(inSender, inEvent) {
		// Only process transition complete if sent from this container
		if(inEvent.originator !== this.$.client) {
			return;
		}

		var posChanged = false;

		if(this.isInTopOverScroll()) {
			posChanged = true;
			this.scrollTop = this.topBoundary;
		} else if (this.isInBottomOverScroll()) {
			posChanged = true;
			this.scrollTop = -1*this.bottomBoundary;
		}

		if(this.isInLeftOverScroll()) {
			posChanged = true;
			this.scrollLeft = this.leftBoundary;
		} else if (this.isInRightOverScroll()) {
			posChanged = true;
			this.scrollLeft = -1*this.rightBoundary;
		}

		if(posChanged) {
			this.startOverflowScrolling();
		} else {
			this.stop();
		}
	},
	//* Scroll to the specified x and y coordinates
	scrollTo: function(inX, inY) {
		this.setScrollTop(inY);
		this.setScrollLeft(inX);
		this.start();
	},
	//* Returns the values of _overleft_ and _overtop_, if any.
	getOverScrollBounds: function() {
		return {
			overleft: Math.min(this.leftBoundary + this.scrollLeft, 0) || Math.max(this.rightBoundary + this.scrollLeft, 0),
			overtop: Math.min(this.topBoundary + this.scrollTop, 0) || Math.max(this.bottomBoundary + this.scrollTop, 0)
		};
	}
});

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/TranslateScrollStrategy.js"
Content-Type: application/octet-stream; x-encoding=base64
/**
	_enyo.TranslateScrollStrategy_ is a helper kind that extends
	<a href="#enyo.TouchScrollStrategy">enyo.TouchScrollStrategy</a>, optimizing
	it for scrolling environments in which effecting scroll changes with
	transforms is fastest.

	_enyo.TranslateScrollStrategy_ is not typically created in application code.
	Instead, it is specified as the value of the `strategyKind` property of an
	`enyo.Scroller` or <a href="#enyo.List">enyo.List</a>, or is used by the
	framework implicitly.
*/
enyo.kind({
	name: "enyo.TranslateScrollStrategy",
	kind: "TouchScrollStrategy",
	//* Set to true to optimize the strategy to only use translation to scroll; this increases fluidity of
	//* scrolling animation. It should not be used when the scroller contains controls that require keyboard
	//* input. This is because when _translateOptimized_ is true, it is possible to position inputs such that
	//* they will not become visibile when focused.
	translateOptimized: false,
	//* @protected
	components: [
		{name: "clientContainer", classes: "enyo-touch-scroller", components: [
			{name: "client"}
		]}
	],
	rendered: function() {
		this.inherited(arguments);
		enyo.makeBubble(this.$.clientContainer, "scroll");
	},
	getScrollSize: function() {
		var n = this.$.client.hasNode();
		return {width: n ? n.scrollWidth : 0, height: n ? n.scrollHeight : 0};
	},
	create: function() {
		this.inherited(arguments);
		// apply initial transform so we're always composited
		enyo.dom.transformValue(this.$.client, this.translation, "0,0,0");
	},
	calcScrollNode: function() {
		return this.$.clientContainer.hasNode();
	},
	maxHeightChanged: function() {
		// content should cover scroller at a minimum if there's no max-height.
		this.$.client.applyStyle("min-height", this.maxHeight ? null : "100%");
		this.$.client.applyStyle("max-height", this.maxHeight);
		this.$.clientContainer.addRemoveClass("enyo-scrollee-fit", !this.maxHeight);
	},
	shouldDrag: function(inSender, inEvent) {
		// stop and update drag info before checking drag status
		this.stop();
		this.calcStartInfo();
		return this.inherited(arguments);
	},
	syncScrollMath: function() {
		if (!this.translateOptimized) {
			this.inherited(arguments);
		}
	},
	//* @public
	//* Sets the left scroll position within the scroller.
	setScrollLeft: function(inLeft) {
		this.stop();
		if (this.translateOptimized) {
			var m = this.$.scrollMath;
			m.setScrollX(-inLeft);
			m.stabilize();
		} else {
			this.inherited(arguments);
		}
	},
	//* Sets the top scroll position within the scroller.
	setScrollTop: function(inTop) {
		this.stop();
		if (this.translateOptimized) {
			var m = this.$.scrollMath;
			m.setScrollY(-inTop);
			m.stabilize();
		} else {
			this.inherited(arguments);
		}
	},
	//* Gets the left scroll position within the scroller.
	getScrollLeft: function() {
		return this.translateOptimized ? this.scrollLeft: this.inherited(arguments);
	},
	//* Gets the top scroll position within the scroller.
	getScrollTop: function() {
		return this.translateOptimized ? this.scrollTop : this.inherited(arguments);
	},
	//* @protected
	scrollMathStart: function(inSender) {
		this.inherited(arguments);
		this.scrollStarting = true;
		this.startX = 0;
		this.startY = 0;
		if (!this.translateOptimized && this.scrollNode) {
			this.startX = this.getScrollLeft();
			this.startY = this.getScrollTop();
		}
	},
	scrollMathScroll: function(inSender) {
		if(!this.overscroll) { //don't overscroll past edges
			this.scrollLeft = -Math.min(inSender.leftBoundary, Math.max(inSender.rightBoundary, inSender.x));
			this.scrollTop = -Math.min(inSender.topBoundary, Math.max(inSender.bottomBoundary, inSender.y));
		} else {
			this.scrollLeft = -inSender.x;
			this.scrollTop = -inSender.y;
		}
		if (this.isScrolling()) {
			if (this.$.scrollMath.isScrolling()) {
				this.effectScroll(this.startX - this.scrollLeft, this.startY - this.scrollTop);
			}
			if (this.thumb) {
				this.updateThumbs();
			}
		}
	},
	// While moving, scroller uses translate.
	effectScroll: function(inX, inY) {
		var o = inX + "px, " + inY + "px" + (this.accel ? ",0" : "");
		enyo.dom.transformValue(this.$.client, this.translation, o);
	},
	// When stopped, we use scrollLeft/Top (makes cursor positioning automagic).
	effectScrollStop: function() {
		if (!this.translateOptimized) {
			var t = "0,0" + (this.accel ? ",0" : "");
			// FIXME: normally translate3d changes not effect scrollHeight; however
			// there appear to be some dom changes (e.g. showing a node inside the scroller,
			// which do cause the scrollHeight to be changed from the translate3d.
			// In this case setting the translate3d back to 0 does not restore scrollHeight.
			// This causes a problem because setting scrollTop can produced an unexpected result if
			// scrollHeight is less than expected.
			// We detect this fault by validating scroll bounds and (1) un-apply the translate3d,
			// (2) update scrollTop/Left, and (3) re-apply a 0,0,0 translate3d to ensure compositing.
			// Luckily this corrects the problem (which appears to be a webkit bug). Note that
			// it's important to maintain a composited state (translate3d 0,0,0) or Android 4 is
			// slow to start scrolling.
			var m = this.$.scrollMath, sb = this._getScrollBounds();
			var needsBoundsFix = Boolean((sb.maxTop + m.bottomBoundary) || (sb.maxLeft + m.rightBoundary));
			enyo.dom.transformValue(this.$.client, this.translation, needsBoundsFix ? null : t);
			// note: this asynchronously triggers dom scroll event
			this.setScrollLeft(this.scrollLeft);
			this.setScrollTop(this.scrollTop);
			if (needsBoundsFix) {
				enyo.dom.transformValue(this.$.client, this.translation, t);
			}
		}
	},
	// FIXME: we can fix scrolling artifacts BUGS on Android 4.04 with this heinous incantation.
	twiddle: function() {
		if (this.translateOptimized && this.scrollNode) { // this.scrollNode is not always defined and makes Motorola XOOM crash
			this.scrollNode.scrollTop = 1;
			this.scrollNode.scrollTop = 0;
		}
	},
	down: enyo.nop
});
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/gesture.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglHZXN0dXJlIGV2ZW50czogZW11bGF0ZXMgaU9TIGdlc3R1cmUgZXZlbnRzIG9uIG5vbiBpT1MgZGV2aWNlcwoJRXZlbnQgQ29udGVudHMKCQktIHBhZ2VYIC8gcGFnZVk6IGNlbnRlciBwb2ludCBiZXR3ZWVuIGZpbmdlcnMKCQktIHJvdGF0aW9uOiBkZWdyZWVzIG9mIHJvdGF0aW9uIGZyb20gYmVnaW5uaW5nIG9mIGdlc3R1cmUKCQktIHNjYWxlOiAlIGNoYW5nZSBvZiBkaXN0YW5jZSBiZXR3ZWVuIGZpbmdlcnMKKi8KLy8qIEBwcm90ZWN0ZWQKKGZ1bmN0aW9uKCkgewoJaWYgKCFlbnlvLnBsYXRmb3JtLmdlc3R1cmUgJiYgZW55by5wbGF0Zm9ybS50b3VjaCkgewoJCWVueW8uZGlzcGF0Y2hlci5mZWF0dXJlcy5wdXNoKGZ1bmN0aW9uKGUpIHsKCQkJaWYgKGhhbmRsZXJzW2UudHlwZV0pIHsKCQkJCXRvdWNoR2VzdHVyZXNbZS50eXBlXShlKTsKCQkJfQoJCX0pOwoJfQoJdmFyIGhhbmRsZXJzID0gewoJCXRvdWNoc3RhcnQ6IHRydWUsCgkJdG91Y2htb3ZlOiB0cnVlLAoJCXRvdWNoZW5kOiB0cnVlCgl9OwoJdmFyIHRvdWNoR2VzdHVyZXMgPSB7CgkJb3JkZXJlZFRvdWNoZXM6IFtdLAoJCWdlc3R1cmU6IG51bGwsCgkJdG91Y2hzdGFydDogZnVuY3Rpb24oaW5FdmVudCkgewoJCQkvLyBzb21lIGRldmljZXMgY2FuIHNlbmQgbXVsdGlwbGUgY2hhbmdlZCB0b3VjaGVzIG9uIHN0YXJ0IGFuZCBlbmQKCQkJZW55by5mb3JFYWNoKGluRXZlbnQuY2hhbmdlZFRvdWNoZXMsIGZ1bmN0aW9uKHQpIHsKCQkJCXZhciBpZCA9IHQuaWRlbnRpZmllcjsKCQkJCS8vIHNvbWUgZGV2aWNlcyBjYW4gc2VuZCBtdWx0aXBsZSB0b3VjaHN0YXJ0cwoJCQkJaWYgKGVueW8uaW5kZXhPZihpZCwgdGhpcy5vcmRlcmVkVG91Y2hlcykgPCAwKSB7CgkJCQkJdGhpcy5vcmRlcmVkVG91Y2hlcy5wdXNoKGlkKTsKCQkJCX0KCQkJfSwgdGhpcyk7CgkJCWlmIChpbkV2ZW50LnRvdWNoZXMubGVuZ3RoID49IDIgJiYgIXRoaXMuZ2VzdHVyZSkgewoJCQkJdmFyIHAgPSB0aGlzLmdlc3R1cmVQb3NpdGlvbnMoaW5FdmVudCk7CgkJCQl0aGlzLmdlc3R1cmUgPSB0aGlzLmdlc3R1cmVWZWN0b3IocCk7CgkJCQl0aGlzLmdlc3R1cmUuYW5nbGUgPSB0aGlzLmdlc3R1cmVBbmdsZShwKTsKCQkJCXRoaXMuZ2VzdHVyZS5zY2FsZSA9IDE7CgkJCQl0aGlzLmdlc3R1cmUucm90YXRpb24gPSAwOwoJCQkJdmFyIGcgPSB0aGlzLm1ha2VHZXN0dXJlKCJnZXN0dXJlc3RhcnQiLCBpbkV2ZW50LCB7dmVjdG9yOiB0aGlzLmdlc3R1cmUsIHNjYWxlOiAxLCByb3RhdGlvbjogMH0pOwoJCQkJZW55by5kaXNwYXRjaChnKTsKCQkJfQoJCX0sCgkJdG91Y2hlbmQ6IGZ1bmN0aW9uKGluRXZlbnQpIHsKCQkJLy8gc29tZSBkZXZpY2VzIGNhbiBzZW5kIG11bHRpcGxlIGNoYW5nZWQgdG91Y2hlcyBvbiBzdGFydCBhbmQgZW5kCgkJCWVueW8uZm9yRWFjaChpbkV2ZW50LmNoYW5nZWRUb3VjaGVzLCBmdW5jdGlvbih0KSB7CgkJCQllbnlvLnJlbW92ZSh0LmlkZW50aWZpZXIsIHRoaXMub3JkZXJlZFRvdWNoZXMpOwoJCQl9LCB0aGlzKTsKCQkJaWYgKGluRXZlbnQudG91Y2hlcy5sZW5ndGggPD0gMSAmJiB0aGlzLmdlc3R1cmUpIHsKCQkJCXZhciB0ID0gaW5FdmVudC50b3VjaGVzWzBdIHx8IGluRXZlbnQuY2hhbmdlZFRvdWNoZXNbaW5FdmVudC5jaGFuZ2VkVG91Y2hlcy5sZW5ndGggLSAxXTsKCQkJCS8vIGdlc3R1cmUgZW5kIHNlbmRzIGxhc3Qgcm90YXRpb24gYW5kIHNjYWxlLCB3aXRoIHRoZSB4L3kgb2YgdGhlIGxhc3QgZmluZ2VyCgkJCQllbnlvLmRpc3BhdGNoKHRoaXMubWFrZUdlc3R1cmUoImdlc3R1cmVlbmQiLCBpbkV2ZW50LCB7dmVjdG9yOiB7eGNlbnRlcjogdC5wYWdlWCwgeWNlbnRlcjogdC5wYWdlWX0sIHNjYWxlOiB0aGlzLmdlc3R1cmUuc2NhbGUsIHJvdGF0aW9uOiB0aGlzLmdlc3R1cmUucm90YXRpb259KSk7CgkJCQl0aGlzLmdlc3R1cmUgPSBudWxsOwoJCQl9CgkJfSwKCQl0b3VjaG1vdmU6IGZ1bmN0aW9uKGluRXZlbnQpIHsKCQkJaWYgKHRoaXMuZ2VzdHVyZSkgewoJCQkJdmFyIGcgPSB0aGlzLm1ha2VHZXN0dXJlKCJnZXN0dXJlY2hhbmdlIiwgaW5FdmVudCk7CgkJCQl0aGlzLmdlc3R1cmUuc2NhbGUgPSBnLnNjYWxlOwoJCQkJdGhpcy5nZXN0dXJlLnJvdGF0aW9uID0gZy5yb3RhdGlvbjsKCQkJCWVueW8uZGlzcGF0Y2goZyk7CgkJCX0KCQl9LAoJCWZpbmRJZGVudGlmaWVkVG91Y2g6IGZ1bmN0aW9uKGluVG91Y2hlcywgaW5JZCkgewoJCQlmb3IgKHZhciBpID0gMCwgdDsgKHQgPSBpblRvdWNoZXNbaV0pOyBpKyspIHsKCQkJCWlmICh0LmlkZW50aWZpZXIgPT09IGluSWQpIHsKCQkJCQlyZXR1cm4gdDsKCQkJCX0KCQkJfQoJCX0sCgkJZ2VzdHVyZVBvc2l0aW9uczogZnVuY3Rpb24oaW5FdmVudCkgewoJCQl2YXIgZmlyc3QgPSB0aGlzLmZpbmRJZGVudGlmaWVkVG91Y2goaW5FdmVudC50b3VjaGVzLCB0aGlzLm9yZGVyZWRUb3VjaGVzWzBdKTsKCQkJdmFyIGxhc3QgPSB0aGlzLmZpbmRJZGVudGlmaWVkVG91Y2goaW5FdmVudC50b3VjaGVzLCB0aGlzLm9yZGVyZWRUb3VjaGVzW3RoaXMub3JkZXJlZFRvdWNoZXMubGVuZ3RoIC0gMV0pOwoJCQl2YXIgZnggPSBmaXJzdC5wYWdlWCwgbHggPSBsYXN0LnBhZ2VYLCBmeSA9IGZpcnN0LnBhZ2VZLCBseSA9IGxhc3QucGFnZVk7CgkJCS8vIGNlbnRlciB0aGUgZmlyc3QgdG91Y2ggYXMgMCwwCgkJCXZhciB4ID0gbHggLSBmeCwgeSA9IGx5IC0gZnk7CgkJCXZhciBoID0gTWF0aC5zcXJ0KHgqeCArIHkqeSk7CgkJCXJldHVybiB7eDogeCwgeTogeSwgaDogaCwgZng6IGZ4LCBseDogbHgsIGZ5OiBmeSwgbHk6IGx5fTsKCQl9LAoJCS8vIGZpbmQgcm90YXRpb24gYW5nbGUKCQlnZXN0dXJlQW5nbGU6IGZ1bmN0aW9uKGluUG9zaXRpb25zKSB7CgkJCXZhciBwID0gaW5Qb3NpdGlvbnM7CgkJCS8vIHlheSBtYXRoISwgcmFkIC0+IGRlZwoJCQl2YXIgYSA9IE1hdGguYXNpbihwLnkgLyBwLmgpICogKDE4MCAvIE1hdGguUEkpOwoJCQkvLyBmaXggZm9yIHJhbmdlIGxpbWl0cyBvZiBhc2luICgtOTAgdG8gOTApCgkJCS8vIFF1YWRyYW50cyBJSSBhbmQgSUlJCgkJCWlmIChwLnggPCAwKSB7CgkJCQlhID0gMTgwIC0gYTsKCQkJfQoJCQkvLyBRdWFkcmFudCBJVgoJCQlpZiAocC54ID4gMCAmJiBwLnkgPCAwKSB7CgkJCQlhICs9IDM2MDsKCQkJfQoJCQlyZXR1cm4gYTsKCQl9LAoJCS8vIGZpbmQgYm91bmRpbmcgYm94CgkJZ2VzdHVyZVZlY3RvcjogZnVuY3Rpb24oaW5Qb3NpdGlvbnMpIHsKCQkJLy8gdGhlIGxlYXN0IHJlY2VudCB0b3VjaCBhbmQgdGhlIG1vc3QgcmVjZW50IHRvdWNoIGRldGVybWluZSB0aGUgYm91bmRpbmcgYm94IG9mIHRoZSBnZXN0dXJlIGV2ZW50CgkJCXZhciBwID0gaW5Qb3NpdGlvbnM7CgkJCS8vIGNlbnRlciB0aGUgZmlyc3QgdG91Y2ggYXMgMCwwCgkJCXJldHVybiB7CgkJCQltYWduaXR1ZGU6IHAuaCwKCQkJCXhjZW50ZXI6IE1hdGguYWJzKE1hdGgucm91bmQocC5meCArIChwLnggLyAyKSkpLAoJCQkJeWNlbnRlcjogTWF0aC5hYnMoTWF0aC5yb3VuZChwLmZ5ICsgKHAueSAvIDIpKSkKCQkJfTsKCQl9LAoJCW1ha2VHZXN0dXJlOiBmdW5jdGlvbihpblR5cGUsIGluRXZlbnQsIGluQ2FjaGUpIHsKCQkJdmFyIHZlY3Rvciwgc2NhbGUsIHJvdGF0aW9uOwoJCQlpZiAoaW5DYWNoZSkgewoJCQkJdmVjdG9yID0gaW5DYWNoZS52ZWN0b3I7CgkJCQlzY2FsZSA9IGluQ2FjaGUuc2NhbGU7CgkJCQlyb3RhdGlvbiA9IGluQ2FjaGUucm90YXRpb247CgkJCX0gZWxzZSB7CgkJCQl2YXIgcCA9IHRoaXMuZ2VzdHVyZVBvc2l0aW9ucyhpbkV2ZW50KTsKCQkJCXZlY3RvciA9IHRoaXMuZ2VzdHVyZVZlY3RvcihwKTsKCQkJCXNjYWxlID0gdmVjdG9yLm1hZ25pdHVkZSAvIHRoaXMuZ2VzdHVyZS5tYWduaXR1ZGU7CgkJCQkvLyBnZXN0dXJlRXZlbnQucm90YXRpb24gaXMgZGlmZmVyZW5jZSBmcm9tIHRoZSBzdGFydGluZyBhbmdsZSwgY2xvY2t3aXNlCgkJCQlyb3RhdGlvbiA9ICgzNjAgKyB0aGlzLmdlc3R1cmVBbmdsZShwKSAtIHRoaXMuZ2VzdHVyZS5hbmdsZSkgJSAzNjA7CgkJCX0KCQkJdmFyIGUgPSBlbnlvLmNsb25lKGluRXZlbnQpOwoJCQlyZXR1cm4gZW55by5taXhpbihlLCB7CgkJCQl0eXBlOiBpblR5cGUsCgkJCQlzY2FsZTogc2NhbGUsCgkJCQlwYWdlWDogdmVjdG9yLnhjZW50ZXIsCgkJCQlwYWdlWTogdmVjdG9yLnljZW50ZXIsCgkJCQlyb3RhdGlvbjogcm90YXRpb24KCQkJfSk7CgkJfQoJfTsKfSkoKTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/msevents.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qIEBwcm90ZWN0ZWQKKGZ1bmN0aW9uKCkgewoJLy8gYWRkIHRvdWNoLXNwZWNpZmljIGdlc3R1cmUgZmVhdHVyZQoJdmFyIGdlc3R1cmUgPSBlbnlvLmdlc3R1cmU7CglpZiAod2luZG93Lm5hdmlnYXRvci5tc1BvaW50ZXJFbmFibGVkKSB7CgkJdmFyIG1zRXZlbnRzID0gWwoJCQkiTVNQb2ludGVyRG93biIsCgkJCSJNU1BvaW50ZXJVcCIsCgkJCSJNU1BvaW50ZXJNb3ZlIiwKCQkJIk1TUG9pbnRlck92ZXIiLAoJCQkiTVNQb2ludGVyT3V0IiwKCQkJIk1TUG9pbnRlckNhbmNlbCIsCgkJCSJNU0dlc3R1cmVUYXAiLAoJCQkiTVNHZXN0dXJlRG91YmxlVGFwIiwKCQkJIk1TR2VzdHVyZUhvbGQiLAoJCQkiTVNHZXN0dXJlU3RhcnQiLAoJCQkiTVNHZXN0dXJlQ2hhbmdlIiwKCQkJIk1TR2VzdHVyZUVuZCIKCQldOwoJCWVueW8uZm9yRWFjaChtc0V2ZW50cywgZnVuY3Rpb24oZSkgewoJCQllbnlvLmRpc3BhdGNoZXIubGlzdGVuKGRvY3VtZW50LCBlKTsKCQl9KTsKCQkvLyBhZGQgb3VyIG93biBNU1BvaW50ZXIgZXZlbnQgaGFuZGxlcgoJCWVueW8uZGlzcGF0Y2hlci5mZWF0dXJlcy5wdXNoKGZ1bmN0aW9uKGUpIHsKCQkJaWYgKGhhbmRsZXJzW2UudHlwZV0gJiYgZS5pc1ByaW1hcnkpIHsKCQkJCWhhbmRsZXJzW2UudHlwZV0oZSk7CgkJCX0KCQl9KTsKCQkvLyByZW1vdmUgdGhlIGRlZmF1bHQgbW91c2UgZXZlbnQgaGFuZGxlcnMKCQllbnlvLmdlc3R1cmUuZXZlbnRzID0ge307Cgl9CgoJdmFyIG1ha2VFdmVudCA9IGZ1bmN0aW9uKGluRXZlbnQpIHsKCQl2YXIgZSA9IGVueW8uY2xvbmUoaW5FdmVudCk7CgkJZS5zcmNFdmVudCA9IGluRXZlbnQ7CgkJLy8gbm9ybWFsaXplICJtb3VzZSBidXR0b24iIGluZm8KCQllLndoaWNoID0gMTsKCQlyZXR1cm4gZTsKCX07CgoJdmFyIGhhbmRsZXJzID0gewoJCS8vIEZJWE1FOiBuZWVkIHRvIHJlZ2lzdGVyIGZvciBnZXN0dXJlcyBpbiBNU1BvaW50ZXJEb3duCgkJLy8gYWNjb3JkaW5nIHRvIE1pY3Jvc29mdCBkb2NzCgkJLypNU0dlc3R1cmVTdGFydDogZnVuY3Rpb24oaW5FdmVudCkgewoJCQllbnlvLmRpc3BhdGNoKGdlc3R1cmVOb3JtYWxpemUoImdlc3R1cmVzdGFydCIsIGluRXZlbnQpKTsKCQl9LAoJCU1TR2VzdHVyZUNoYW5nZTogZnVuY3Rpb24oaW5FdmVudCkgewoJCQllbnlvLmRpc3BhdGNoKGdlc3R1cmVOb3JtYWxpemUoImdlc3R1cmVjaGFuZ2UiLCBpbkV2ZW50KSk7CgkJfSwKCQlNU0dlc3R1cmVFbmQ6IGZ1bmN0aW9uKGluRXZlbnQpIHsKCQkJZW55by5kaXNwYXRjaChnZXN0dXJlTm9ybWFsaXplKCJnZXN0dXJlZW5kIiwgaW5FdmVudCkpOwoJCX0sKi8KCQlNU1BvaW50ZXJEb3duOiBmdW5jdGlvbihpbkV2ZW50KSB7CgkJCXZhciBlID0gbWFrZUV2ZW50KGluRXZlbnQpOwoJCQlnZXN0dXJlLmRvd24oZSk7CgkJfSwKCQlNU1BvaW50ZXJVcDogZnVuY3Rpb24oaW5FdmVudCkgewoJCQl2YXIgZSA9IG1ha2VFdmVudChpbkV2ZW50KTsKCQkJZ2VzdHVyZS51cChlKTsKCQl9LAoJCU1TUG9pbnRlck1vdmU6IGZ1bmN0aW9uKGluRXZlbnQpIHsKCQkJdmFyIGUgPSBtYWtlRXZlbnQoaW5FdmVudCk7CgkJCWdlc3R1cmUubW92ZShlKTsKCQl9LAoJCU1TUG9pbnRlckNhbmNlbDogZnVuY3Rpb24oaW5FdmVudCkgewoJCQkvLyBGSVhNRTogbm90IHJlYWxseSB0aGUgc2FtZSBhcyB0b3VjaGVuZCwgYXMgdG91Y2ggYWN0aW9uCgkJCS8vIHdhcyBjYW5jZWxsZWQsIGJ1dCBFbnlvIGRvZXNuJ3QgaGF2ZSB0aGF0IGNvbmNlcHQKCQkJdmFyIGUgPSBtYWtlRXZlbnQoaW5FdmVudCk7CgkJCWdlc3R1cmUudXAoZSk7CgkJfSwKCQlNU1BvaW50ZXJPdmVyOiBmdW5jdGlvbihpbkV2ZW50KSB7CgkJCXZhciBlID0gbWFrZUV2ZW50KGluRXZlbnQpOwoJCQlnZXN0dXJlLm92ZXIoZSk7CgkJfSwKCQlNU1BvaW50ZXJPdXQ6IGZ1bmN0aW9uKGluRXZlbnQpIHsKCQkJdmFyIGUgPSBtYWtlRXZlbnQoaW5FdmVudCk7CgkJCWdlc3R1cmUub3V0KGUpOwoJCX0KCX07Cn0pKCk7
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJInRvdWNoLmpzIiwKCSJtc2V2ZW50cy5qcyIsCgkiZ2VzdHVyZS5qcyIsCgkiU2Nyb2xsTWF0aC5qcyIsCgkiU2Nyb2xsU3RyYXRlZ3kuanMiLAoJIlRodW1iLmNzcyIsCgkiVGh1bWIuanMiLAoJIlRvdWNoU2Nyb2xsU3RyYXRlZ3kuanMiLAoJIlRyYW5zbGF0ZVNjcm9sbFN0cmF0ZWd5LmpzIiwKCSJUcmFuc2l0aW9uU2Nyb2xsU3RyYXRlZ3kuanMiLAoJIlNjcm9sbGVyLmpzIiwKCSJTY3JvbGxlci5jc3MiCik7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/touch/touch.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qIEBwcm90ZWN0ZWQKZW55by5yZXF1aXJlc1dpbmRvdyhmdW5jdGlvbigpIHsKCS8vIGFkZCB0b3VjaC1zcGVjaWZpYyBnZXN0dXJlIGZlYXR1cmUKCXZhciBnZXN0dXJlID0gZW55by5nZXN0dXJlOwoJdmFyIG9sZGV2ZW50cyA9IGdlc3R1cmUuZXZlbnRzOwoJLy8KCWdlc3R1cmUuZXZlbnRzLnRvdWNoc3RhcnQgPSBmdW5jdGlvbihlKSB7CgkJLy8gZm9yIGR1cmF0aW9uIG9mIHRoaXMgdG91Y2gsIG9ubHkgaGFuZGxlIHRvdWNoIGV2ZW50cy4gIE9sZCBldmVudAoJCS8vIHN0cnVjdHVyZSB3aWxsIGJlIHJlc3RvcmVkIGR1cmluZyB0b3VjaGVuZC4KCQlnZXN0dXJlLmV2ZW50cyA9IHRvdWNoR2VzdHVyZTsKCQlnZXN0dXJlLmV2ZW50cy50b3VjaHN0YXJ0KGUpOwoJfTsKCS8vCgl2YXIgdG91Y2hHZXN0dXJlID0gewoJCV90b3VjaENvdW50OiAwLAoJCXRvdWNoc3RhcnQ6IGZ1bmN0aW9uKGluRXZlbnQpIHsKCQkJdGhpcy5fdG91Y2hDb3VudCArPSBpbkV2ZW50LmNoYW5nZWRUb3VjaGVzLmxlbmd0aDsKCQkJdGhpcy5leGNsdWRlZFRhcmdldCA9IG51bGw7CgkJCXZhciBlID0gdGhpcy5tYWtlRXZlbnQoaW5FdmVudCk7CgkJCWdlc3R1cmUuZG93bihlKTsKCQkJLy8gZ2VuZXJhdGUgYSBuZXcgZXZlbnQgb2JqZWN0IHNpbmNlIG92ZXIgaXMgYSBkaWZmZXJlbnQgZXZlbnQKCQkJZSA9IHRoaXMubWFrZUV2ZW50KGluRXZlbnQpOwoJCQl0aGlzLm92ZXJFdmVudCA9IGU7CgkJCWdlc3R1cmUub3ZlcihlKTsKCQl9LAoJCXRvdWNobW92ZTogZnVuY3Rpb24oaW5FdmVudCkgewoJCQllbnlvLmpvYi5zdG9wKCJyZXNldEdlc3R1cmVFdmVudHMiKTsKCQkJLy8gTk9URTogYWxsb3cgdXNlciB0byBzdXBwbHkgYSBub2RlIHRvIGV4Y2x1ZGUgZnJvbSBldmVudAoJCQkvLyB0YXJnZXQgZmluZGluZyB2aWEgdGhlIGRyYWcgZXZlbnQuCgkJCXZhciBkZSA9IGdlc3R1cmUuZHJhZy5kcmFnRXZlbnQ7CgkJCXRoaXMuZXhjbHVkZWRUYXJnZXQgPSBkZSAmJiBkZS5kcmFnSW5mbyAmJiBkZS5kcmFnSW5mby5ub2RlOwoJCQl2YXIgZSA9IHRoaXMubWFrZUV2ZW50KGluRXZlbnQpOwoJCQlnZXN0dXJlLm1vdmUoZSk7CgkJCS8vIHByZXZlbnQgZGVmYXVsdCBkb2N1bWVudCBzY3JvbGxpbmcgaWYgZW55by5ib2R5SXNGaXR0aW5nID09IHRydWUKCQkJLy8gYXZvaWQgd2luZG93IHNjcm9sbGluZyBieSBwcmV2ZW50aW5nIGRlZmF1bHQgb24gdGhpcyBldmVudAoJCQkvLyBub3RlOiB0aGlzIGV2ZW50IGNhbiBiZSBtYWRlIHVucHJldmVudGFibGUgKG5hdGl2ZSBzY3JvbGxlcnMgZG8gdGhpcykKCQkJaWYgKGVueW8uYm9keUlzRml0dGluZykgewoJCQkJaW5FdmVudC5wcmV2ZW50RGVmYXVsdCgpOwoJCQl9CgkJCS8vIHN5bnRoZXNpemUgb3ZlciBhbmQgb3V0IChub3JtYWxseSBnZW5lcmF0ZWQgdmlhIG1vdXNlb3V0KQoJCQlpZiAodGhpcy5vdmVyRXZlbnQgJiYgdGhpcy5vdmVyRXZlbnQudGFyZ2V0ICE9IGUudGFyZ2V0KSB7CgkJCQl0aGlzLm92ZXJFdmVudC5yZWxhdGVkVGFyZ2V0ID0gZS50YXJnZXQ7CgkJCQllLnJlbGF0ZWRUYXJnZXQgPSB0aGlzLm92ZXJFdmVudC50YXJnZXQ7CgkJCQlnZXN0dXJlLm91dCh0aGlzLm92ZXJFdmVudCk7CgkJCQlnZXN0dXJlLm92ZXIoZSk7CgkJCX0KCQkJdGhpcy5vdmVyRXZlbnQgPSBlOwoJCX0sCgkJdG91Y2hlbmQ6IGZ1bmN0aW9uKGluRXZlbnQpIHsKCQkJZ2VzdHVyZS51cCh0aGlzLm1ha2VFdmVudChpbkV2ZW50KSk7CgkJCS8vIE5PVEU6IGluIHRvdWNoIGxhbmQsIHRoZXJlIGlzIG5vIGRpc3RpbmN0aW9uIGJldHdlZW4KCQkJLy8gYSBwb2ludGVyIGVudGVyL2xlYXZlIGFuZCBhIGRyYWcgb3Zlci9vdXQuCgkJCS8vIFdoaWxlIGl0IG1heSBtYWtlIHNlbnNlIHRvIHNlbmQgYSBsZWF2ZSBldmVudCB3aGVuIGEgdG91Y2gKCQkJLy8gZW5kcywgaXQgZG9lcyBub3QgbWFrZSBzZW5zZSB0byBzZW5kIGEgZHJhZ291dC4KCQkJLy8gV2UgYXZvaWQgdGhpcyBieSBwcm9jZXNzaW5nIG91dCBhZnRlciB1cCwgYnV0CgkJCS8vIHRoaXMgb3JkZXJpbmcgaXMgYWQgaG9jLgoJCQlnZXN0dXJlLm91dCh0aGlzLm92ZXJFdmVudCk7CgkJCS8vIHJlc2V0IHRoZSBldmVudCBoYW5kbGVycyBiYWNrIHRvIHRoZSBtb3VzZS1mcmllbmRseSBvbmVzIGFmdGVyCgkJCS8vIGEgc2hvcnQgdGltZW91dC4gV2UgY2FuJ3QgZG8gdGhpcyBkaXJlY3RseSBpbiB0aGlzIGhhbmRsZXIKCQkJLy8gYmVjYXVzZSBpdCBtZXNzZXMgdXAgQW5kcm9pZCB0byBoYW5kbGUgdGhlIG1vdXNldXAgZXZlbnQuCgkJCS8vIEZJWE1FOiBmb3IgMi4xIHJlbGVhc2UsIGNvbmRpdGlvbmFsIG9uIHBsYXRmb3JtIGJlaW5nCgkJCS8vIGRlc2t0b3AgQ2hyb21lLCBzaW5jZSB3ZSdyZSBzZWVpbmcgaXNzdWVzIGluIFBob25lR2FwIHdpdGggdGhpcwoJCQkvLyBjb2RlLgoJCQl0aGlzLl90b3VjaENvdW50IC09IGluRXZlbnQuY2hhbmdlZFRvdWNoZXMubGVuZ3RoOwoJCX0sCgkJLy8gdXNlIG1vdXNldXAgYWZ0ZXIgdG91Y2hlcyBhcmUgZG9uZSB0byByZXNldCBldmVudCBoYW5kbGluZyBiYWNrIHRvIGRlZmF1bHQKCQkvLyAtLXRoaXMgd29ya3MgYXMgbG9uZyBhcyBubyBvbmUgZGlkIGEgcHJldmVudERlZmF1bHQgb24gdGhlIHRvdWNoIGV2ZW50cwoJCW1vdXNldXA6IGZ1bmN0aW9uKGUpIHsKCQkJaWYgKHRoaXMuX3RvdWNoQ291bnQgPT09IDApIHsKCQkJCXRoaXMuc2F3TW91c2Vkb3duID0gZmFsc2U7CgkJCQlnZXN0dXJlLmV2ZW50cyA9IG9sZGV2ZW50czsKCQkJfQoJCX0sCgkJbWFrZUV2ZW50OiBmdW5jdGlvbihpbkV2ZW50KSB7CgkJCXZhciBlID0gZW55by5jbG9uZShpbkV2ZW50LmNoYW5nZWRUb3VjaGVzWzBdKTsKCQkJZS5zcmNFdmVudCA9IGluRXZlbnQ7CgkJCWUudGFyZ2V0ID0gdGhpcy5maW5kVGFyZ2V0KGUpOwoJCQkvLyBub3JtYWxpemUgIm1vdXNlIGJ1dHRvbiIgaW5mbwoJCQllLndoaWNoID0gMTsKCQkJLy9lbnlvLmxvZygidGFyZ2V0IGZvciAiICsgaW5FdmVudC50eXBlICsgIiBhdCAiICsgZS5wYWdlWCArICIsICIgKyBlLnBhZ2VZICsgIiBpcyAiICsgKGUudGFyZ2V0ID8gZS50YXJnZXQuaWQgOiAibm9uZSIpKTsKCQkJcmV0dXJuIGU7CgkJfSwKCQljYWxjTm9kZU9mZnNldDogZnVuY3Rpb24oaW5Ob2RlKSB7CgkJCWlmIChpbk5vZGUuZ2V0Qm91bmRpbmdDbGllbnRSZWN0KSB7CgkJCQl2YXIgbyA9IGluTm9kZS5nZXRCb3VuZGluZ0NsaWVudFJlY3QoKTsKCQkJCXJldHVybiB7CgkJCQkJbGVmdDogby5sZWZ0LAoJCQkJCXRvcDogby50b3AsCgkJCQkJd2lkdGg6IG8ud2lkdGgsCgkJCQkJaGVpZ2h0OiBvLmhlaWdodAoJCQkJfTsKCQkJfQoJCX0sCgkJZmluZFRhcmdldDogZnVuY3Rpb24oZSkgewoJCQlyZXR1cm4gZG9jdW1lbnQuZWxlbWVudEZyb21Qb2ludChlLmNsaWVudFgsIGUuY2xpZW50WSk7CgkJfSwKCQkvLyBOT1RFOiB3aWxsIGZpbmQgb25seSAxIGVsZW1lbnQgdW5kZXIgdGhlIHRvdWNoIGFuZAoJCS8vIHdpbGwgZmFpbCBpZiBhbiBlbGVtZW50IGlzIHBvc2l0aW9uZWQgb3V0c2lkZSB0aGUgYm91bmRpbmcgYm94IG9mIGl0cyBwYXJlbnQKCQlmaW5kVGFyZ2V0VHJhdmVyc2U6IGZ1bmN0aW9uKGluTm9kZSwgaW5YLCBpblkpIHsKCQkJdmFyIG4gPSBpbk5vZGUgfHwgZG9jdW1lbnQuYm9keTsKCQkJdmFyIG8gPSB0aGlzLmNhbGNOb2RlT2Zmc2V0KG4pOwoJCQlpZiAobyAmJiBuICE9IHRoaXMuZXhjbHVkZWRUYXJnZXQpIHsKCQkJCXZhciB4ID0gaW5YIC0gby5sZWZ0OwoJCQkJdmFyIHkgPSBpblkgLSBvLnRvcDsKCQkJCS8vZW55by5sb2coInRlc3Q6ICIgKyBuLmlkICsgIiAobGVmdDogIiArIG8ubGVmdCArICIsIHRvcDogIiArIG8udG9wICsgIiwgd2lkdGg6ICIgKyBvLndpZHRoICsgIiwgaGVpZ2h0OiAiICsgby5oZWlnaHQgKyAiKSIpOwoJCQkJaWYgKHg+MCAmJiB5PjAgJiYgeDw9by53aWR0aCAmJiB5PD1vLmhlaWdodCkgewoJCQkJCS8vZW55by5sb2coIklOOiAiICsgbi5pZCArICIgLT4gWyIgKyB4ICsgIiwiICsgeSArICIgaW4gIiArIG8ud2lkdGggKyAieCIgKyBvLmhlaWdodCArICJdIChjaGlsZHJlbjogIiArIG4uY2hpbGROb2Rlcy5sZW5ndGggKyAiKSIpOwoJCQkJCXZhciB0YXJnZXQ7CgkJCQkJZm9yICh2YXIgbiQ9bi5jaGlsZE5vZGVzLCBpPW4kLmxlbmd0aC0xLCBjOyAoYz1uJFtpXSk7IGktLSkgewoJCQkJCQl0YXJnZXQgPSB0aGlzLmZpbmRUYXJnZXRUcmF2ZXJzZShjLCBpblgsIGluWSk7CgkJCQkJCWlmICh0YXJnZXQpIHsKCQkJCQkJCXJldHVybiB0YXJnZXQ7CgkJCQkJCX0KCQkJCQl9CgkJCQkJcmV0dXJuIG47CgkJCQl9CgkJCX0KCQl9LAoJCWNvbm5lY3Q6IGZ1bmN0aW9uKCkgewoJCQllbnlvLmZvckVhY2goWydvbnRvdWNoc3RhcnQnLCAnb250b3VjaG1vdmUnLCAnb250b3VjaGVuZCcsICdvbmdlc3R1cmVzdGFydCcsICdvbmdlc3R1cmVjaGFuZ2UnLCAnb25nZXN0dXJlZW5kJ10sIGZ1bmN0aW9uKGUpIHsKCQkJCWRvY3VtZW50W2VdID0gZW55by5kaXNwYXRjaDsKCQkJfSk7CgkJCWlmIChlbnlvLnBsYXRmb3JtLmFuZHJvaWRDaHJvbWUgPD0gMTggfHwgZW55by5wbGF0Zm9ybS5zaWxrID09PSAyKSB7CgkJCQkvLyBIQUNLOiBvbiBDaHJvbWUgZm9yIEFuZHJvaWQgdjE4IG9uIGRldmljZXMgd2l0aCBoaWdoZXIgZGVuc2l0eSBkaXNwbGF5cywKCQkJCS8vIGRvY3VtZW50LmVsZW1lbnRGcm9tUG9pbnQgZXhwZWN0cyBzY3JlZW4gY29vcmRpbmF0ZXMsIG5vdCBkb2N1bWVudCBvbmVzCgkJCQkvLyBidWcgYWxzbyBhcHBlYXJzIG9uIEtpbmRsZSBGaXJlIEhECgkJCQl0aGlzLmZpbmRUYXJnZXQgPSBmdW5jdGlvbihlKSB7CgkJCQkJcmV0dXJuIGRvY3VtZW50LmVsZW1lbnRGcm9tUG9pbnQoZS5zY3JlZW5YLCBlLnNjcmVlblkpOwoJCQkJfTsKCQkJfSBlbHNlIGlmICghZG9jdW1lbnQuZWxlbWVudEZyb21Qb2ludCkgewoJCQkJdGhpcy5maW5kVGFyZ2V0ID0gZnVuY3Rpb24oZSkgewoJCQkJCXJldHVybiB0aGlzLmZpbmRUYXJnZXRUcmF2ZXJzZShudWxsLCBlLmNsaWVudFgsIGUuY2xpZW50WSk7CgkJCQl9OwoJCQl9CgkJfQoJfTsKCS8vCgl0b3VjaEdlc3R1cmUuY29ubmVjdCgpOwp9KTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Animator.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5BbmltYXRvcl8gaXMgYSBiYXNpYyBhbmltYXRpb24gY29tcG9uZW50LiAgQ2FsbCBfcGxheV8gdG8gc3RhcnQgdGhlCglhbmltYXRpb24uIFRoZSBhbmltYXRpb24gd2lsbCBydW4gZm9yIHRoZSBwZXJpb2QgKGluIG1pbGxpc2Vjb25kcykgc3BlY2lmaWVkCglieSBpdHMgX2R1cmF0aW9uXyBwcm9wZXJ0eS4gIFRoZSBfb25TdGVwXyBldmVudCB3aWxsIGZpcmUgaW4gcXVpY2sKCXN1Y2Nlc3Npb24gYW5kIHNob3VsZCBiZSBoYW5kbGVkIHRvIGRvIHNvbWV0aGluZyBiYXNlZCBvbiB0aGUgX3ZhbHVlXwoJcHJvcGVydHkuCgoJVGhlIF92YWx1ZV8gcHJvcGVydHkgd2lsbCBwcm9ncmVzcyBmcm9tIF9zdGFydFZhbHVlXyB0byBfZW5kVmFsdWVfIGR1cmluZwoJdGhlIGFuaW1hdGlvbiBiYXNlZCBvbiB0aGUgZnVuY3Rpb24gcmVmZXJlbmNlZCBieSB0aGUgX2Vhc2luZ0Z1bmN0aW9uXwoJcHJvcGVydHkuICBUaGUgX3N0b3BfIG1ldGhvZCBtYXkgYmUgY2FsbGVkIHRvIG1hbnVhbGx5IHN0b3AgYW4gaW4tcHJvZ3Jlc3MKCWFuaW1hdGlvbjsgY2FsbGluZyBpdCB3aWxsIGZpcmUgdGhlIF9vblN0b3BfIGV2ZW50LiAgV2hlbiBhbiBhbmltYXRpb24KCWNvbXBsZXRlcyBub3JtYWxseSwgdGhlIF9vbkVuZF8gZXZlbnQgaXMgZmlyZWQuCgoJRXZlbnQgaGFuZGxlcnMgbWF5IGJlIHNwZWNpZmllZCBhcyBmdW5jdGlvbnMuICBJZiBzcGVjaWZpZWQsIHRoZSBoYW5kbGVyCglmdW5jdGlvbiB3aWxsIGJlIHVzZWQgdG8gaGFuZGxlIHRoZSBldmVudCBkaXJlY3RseSwgd2l0aG91dCBzZW5kaW5nIHRoZQoJZXZlbnQgdG8gaXRzIG93bmVyIG9yIGJ1YmJsaW5nIGl0LiAgVGhlIF9jb250ZXh0XyBwcm9wZXJ0eSBjYW4gYmUgdXNlZCB0bwoJY2FsbCB0aGUgc3VwcGxpZWQgZXZlbnQgZnVuY3Rpb25zIGluIGEgcGFydGljdWxhciAidGhpcyIgY29udGV4dC4KKi8KZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLkFuaW1hdG9yIiwKCWtpbmQ6ICJDb21wb25lbnQiLAoJcHVibGlzaGVkOiB7CgkJLy8qIEFuaW1hdGlvbiBkdXJhdGlvbiBpbiBtaWxsaXNlY29uZHMKCQlkdXJhdGlvbjogMzUwLAoJCS8vKiBWYWx1ZSBvZiBfdmFsdWVfIHByb3BlcnR5IGF0IHRoZSBiZWdpbm5pbmcgb2YgYW4gYW5pbWF0aW9uCgkJc3RhcnRWYWx1ZTogMCwKCQkvLyogVmFsdWUgb2YgX3ZhbHVlXyBwcm9wZXJ0eSBhdCB0aGUgZW5kIG9mIGFuIGFuaW1hdGlvbgoJCWVuZFZhbHVlOiAxLAoJCS8vKiBOb2RlIHRoYXQgbXVzdCBiZSB2aXNpYmxlIGluIG9yZGVyIGZvciB0aGUgYW5pbWF0aW9uIHRvIGNvbnRpbnVlLgoJCS8vKiBUaGlzIHJlZmVyZW5jZSBpcyBkZXN0cm95ZWQgd2hlbiB0aGUgYW5pbWF0aW9uIGNlYXNlcy4KCQlub2RlOiBudWxsLAoJCS8vKiBGdW5jdGlvbiB0aGF0IGRldGVybWluZXMgaG93IHRoZSBhbmltYXRpb24gcHJvZ3Jlc3NlcyBmcm9tCgkJLy8qIF9zdGFydFZhbHVlXyB0byBfZW5kVmFsdWVfCgkJZWFzaW5nRnVuY3Rpb246IGVueW8uZWFzaW5nLmN1YmljT3V0Cgl9LAoJZXZlbnRzOiB7CgkJLy8qIEZpcmVzIHdoZW4gYW4gYW5pbWF0aW9uIHN0ZXAgb2NjdXJzLgoJCW9uU3RlcDogIiIsCgkJLy8qIEZpcmVzIHdoZW4gdGhlIGFuaW1hdGlvbiBmaW5pc2hlcyBub3JtYWxseS4KCQlvbkVuZDogIiIsCgkJLy8qIEZpcmVzIHdoZW4gdGhlIGFuaW1hdGlvbiBpcyBwcmVtYXR1cmVseSBzdG9wcGVkLgoJCW9uU3RvcDogIiIKCX0sCgkvLyogQHByb3RlY3RlZAoJY29uc3RydWN0ZWQ6IGZ1bmN0aW9uKCkgewoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7CgkJdGhpcy5fbmV4dCA9IHRoaXMuYmluZFNhZmVseSgibmV4dCIpOwoJfSwKCWRlc3Ryb3k6IGZ1bmN0aW9uKCkgewoJCXRoaXMuc3RvcCgpOwoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7Cgl9LAoJLy8qIEBwdWJsaWMKCS8vKiBQbGF5cyB0aGUgYW5pbWF0aW9uLgoJLy8qIEZvciBjb252ZW5pZW5jZSwgX2luUHJvcHNfIHdpbGwgYmUgbWl4ZWQgZGlyZWN0bHkgaW50byB0aGlzIG9iamVjdC4KCXBsYXk6IGZ1bmN0aW9uKGluUHJvcHMpIHsKCQl0aGlzLnN0b3AoKTsKCQl0aGlzLnJldmVyc2VkID0gZmFsc2U7CgkJaWYgKGluUHJvcHMpIHsKCQkJZW55by5taXhpbih0aGlzLCBpblByb3BzKTsKCQl9CgkJdGhpcy50MCA9IHRoaXMudDEgPSBlbnlvLm5vdygpOwoJCXRoaXMudmFsdWUgPSB0aGlzLnN0YXJ0VmFsdWU7CgkJdGhpcy5qb2IgPSB0cnVlOwoJCXRoaXMubmV4dCgpOwoJCXJldHVybiB0aGlzOwoJfSwKCS8vKiBTdG9wcyB0aGUgYW5pbWF0aW9uIGFuZCBmaXJlcyB0aGUgX29uU3RvcF8gZXZlbnQuCglzdG9wOiBmdW5jdGlvbigpIHsKCQlpZiAodGhpcy5pc0FuaW1hdGluZygpKSB7CgkJCXRoaXMuY2FuY2VsKCk7CgkJCXRoaXMuZmlyZSgib25TdG9wIik7CgkJCXJldHVybiB0aGlzOwoJCX0KCX0sCgkvLyogUmV2ZXJzZXMgdGhlIGRpcmVjdGlvbiBvZiBhIHJ1bm5pbmcgYW5pbWF0aW9uOyByZXR1cm5zIHNlbGYgaWYgYW5pbWF0aW5nLgoJcmV2ZXJzZTogZnVuY3Rpb24oKSB7CgkJaWYgKHRoaXMuaXNBbmltYXRpbmcoKSkgewoJCQl0aGlzLnJldmVyc2VkID0gIXRoaXMucmV2ZXJzZWQ7CgkJCXZhciBub3cgPSB0aGlzLnQxID0gZW55by5ub3coKTsKCQkJLy8gYWRqdXN0IHN0YXJ0IHRpbWUgKHQwKSB0byBhbGxvdyBmb3IgYW5pbWF0aW9uIGRvbmUgc28gZmFyIHRvIHJlcGxheQoJCQl2YXIgZWxhcHNlZCA9IG5vdyAtIHRoaXMudDA7CgkJCXRoaXMudDAgPSBub3cgKyBlbGFwc2VkIC0gdGhpcy5kdXJhdGlvbjsKCQkJLy8gc3dhcCBzdGFydCBhbmQgZW5kIHZhbHVlcwoJCQl2YXIgc3RhcnRWYWx1ZSA9IHRoaXMuc3RhcnRWYWx1ZTsKCQkJdGhpcy5zdGFydFZhbHVlID0gdGhpcy5lbmRWYWx1ZTsKCQkJdGhpcy5lbmRWYWx1ZSA9IHN0YXJ0VmFsdWU7CgkJCXJldHVybiB0aGlzOwoJCX0KCX0sCgkvLyogUmV0dXJucyB0cnVlIGlmIGFuaW1hdGlvbiBpcyBpbiBwcm9ncmVzcy4KCWlzQW5pbWF0aW5nOiBmdW5jdGlvbigpIHsKCQlyZXR1cm4gQm9vbGVhbih0aGlzLmpvYik7Cgl9LAoJLy8qIEBwcm90ZWN0ZWQKCXJlcXVlc3ROZXh0OiBmdW5jdGlvbigpIHsKCQl0aGlzLmpvYiA9IGVueW8ucmVxdWVzdEFuaW1hdGlvbkZyYW1lKHRoaXMuX25leHQsIHRoaXMubm9kZSk7Cgl9LAoJY2FuY2VsOiBmdW5jdGlvbigpIHsKCQllbnlvLmNhbmNlbFJlcXVlc3RBbmltYXRpb25GcmFtZSh0aGlzLmpvYik7CgkJdGhpcy5ub2RlID0gbnVsbDsKCQl0aGlzLmpvYiA9IG51bGw7Cgl9LAoJc2hvdWxkRW5kOiBmdW5jdGlvbigpIHsKCQlyZXR1cm4gKHRoaXMuZHQgPj0gdGhpcy5kdXJhdGlvbik7Cgl9LAoJbmV4dDogZnVuY3Rpb24oKSB7CgkJdGhpcy50MSA9IGVueW8ubm93KCk7CgkJdGhpcy5kdCA9IHRoaXMudDEgLSB0aGlzLnQwOwoJCXZhciBhcmdzID0gdGhpcy5lYXNpbmdGdW5jdGlvbi5sZW5ndGg7CgkJdmFyIGY7CgoJCWlmIChhcmdzID09PSAxKSB7CgkJCS8vIHRpbWUgaW5kZXBlbmRlbnQKCQkJZiA9IHRoaXMuZnJhY3Rpb24gPSBlbnlvLmVhc2VkTGVycCh0aGlzLnQwLCB0aGlzLmR1cmF0aW9uLCB0aGlzLmVhc2luZ0Z1bmN0aW9uLCB0aGlzLnJldmVyc2VkKTsKCQkJdGhpcy52YWx1ZSA9IHRoaXMuc3RhcnRWYWx1ZSArIGYgKiAodGhpcy5lbmRWYWx1ZSAtIHRoaXMuc3RhcnRWYWx1ZSk7CgkJfSBlbHNlIHsKCQkJdGhpcy52YWx1ZSA9IGVueW8uZWFzZWRDb21wbGV4TGVycCh0aGlzLnQwLCB0aGlzLmR1cmF0aW9uLCB0aGlzLmVhc2luZ0Z1bmN0aW9uLCB0aGlzLnJldmVyc2VkLAoJCQkJdGhpcy5kdCwgdGhpcy5zdGFydFZhbHVlLCAodGhpcy5lbmRWYWx1ZSAtIHRoaXMuc3RhcnRWYWx1ZSkpOwoJCX0KCQlpZiAoKChmID49IDEpICYmIChhcmdzID09PSAxKSkgfHwgdGhpcy5zaG91bGRFbmQoKSkgewoJCQl0aGlzLnZhbHVlID0gdGhpcy5lbmRWYWx1ZTsKCQkJdGhpcy5mcmFjdGlvbiA9IDE7CgkJCXRoaXMuZmlyZSgib25TdGVwIik7CgkJCXRoaXMuZmlyZSgib25FbmQiKTsKCQkJdGhpcy5jYW5jZWwoKTsKCQl9IGVsc2UgewoJCQl0aGlzLmZpcmUoIm9uU3RlcCIpOwoJCQl0aGlzLnJlcXVlc3ROZXh0KCk7CgkJfQoJfSwKCWZpcmU6IGZ1bmN0aW9uKGluRXZlbnROYW1lKSB7CgkJdmFyIGZuID0gdGhpc1tpbkV2ZW50TmFtZV07CgkJaWYgKGVueW8uaXNTdHJpbmcoZm4pKSB7CgkJCXRoaXMuYnViYmxlKGluRXZlbnROYW1lKTsKCQl9IGVsc2UgaWYgKGZuKSB7CgkJCWZuLmNhbGwodGhpcy5jb250ZXh0IHx8IHdpbmRvdywgdGhpcyk7CgkJfQoJfQp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Audio.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5BdWRpb18gaW1wbGVtZW50cyBhbiBIVE1MIDUgTWVkaWEgZWxlbWVudCBieSBleHRlbmRpbmcgX2VueW8uTWVkaWFfLAoJYWxsb3dpbmcgeW91IHRvIHBsYXkgYXVkaW8uCgoJSW5pdGlhbGl6ZSBhbiBhdWRpbyBjb21wb25lbnQgYXMgZm9sbG93czoKCgkJe2tpbmQ6ICJlbnlvLkF1ZGlvIiwgc3JjOiAiaHR0cDovL3d3dy53M3NjaG9vbHMuY29tL3RhZ3MvaG9yc2UubXAzIn0KCQoJVG8gcGxheSB0aGUgYXVkaW8sIGNhbGwgJ3RoaXMuJC5hdWRpby5wbGF5KCknLgoKCVRvIGdldCBhIHJlZmVyZW5jZSB0byB0aGUgYWN0dWFsIEhUTUwgNSBNZWRpYSBlbGVtZW50LCBjYWxsCgkndGhpcy4kLmF1ZGlvLmhhc05vZGUoKScuCiovCmVueW8ua2luZCh7CgluYW1lOiAiZW55by5BdWRpbyIsCglraW5kOiAiZW55by5NZWRpYSIsCgl0YWc6ICJhdWRpbyIsCglwdWJsaXNoZWQ6IHsKCQkvLyogSW5kaWNhdGVzIGhvdyBkYXRhIHNob3VsZCBiZSBwcmVsb2FkZWQsIHJlZmxlY3RpbmcgdGhlIHByZWxvYWQgSFRNTCBhdHRyaWJ1dGUgKG5vbmUsIG1ldGFkYXRhLCBhdXRvKQoJCXByZWxvYWQ6ICJhdXRvIgoJfQp9KTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/BaseLayout.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5CYXNlTGF5b3V0XyBwcm92aWRlcyBhIGJhc2ljIGxheW91dCBzdHJhdGVneSwgcG9zaXRpb25pbmcgY29udGFpbmVkCgljb21wb25lbnRzIHdpdGggdGhlIF9lbnlvLXBvc2l0aW9uZWRfIGxheW91dENsYXNzLiBJbiBhZGRpdGlvbiwgaXQgYWRqdXN0cwoJdGhlIGxheW91dCB3aGVuIF9yZWZsb3dfIGlzIGNhbGxlZCwgcmVtb3Zpbmcgb3IgYWRkaW5nIHRoZSBfZW55by1maXRfIGNsYXNzCglmb3IgY29tcG9uZW50cyB0aGF0IGhhdmUgc2V0IHRoZSBfZml0XyBwcm9wZXJ0eS4KKi8KZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLkJhc2VMYXlvdXQiLAoJa2luZDogImVueW8uTGF5b3V0IiwKCWxheW91dENsYXNzOiAiZW55by1wb3NpdGlvbmVkIiwKCS8vKiBBZGRzIG9yIHJlbW92ZXMgdGhlIF9lbnlvLWZpdF8gY2xhc3MgZm9yIGNvbXBvbmVudHMgd2hvc2UgX2ZpdF8gcHJvcGVydHkKCS8vKiBoYXMgYmVlbiBzZXQuCglyZWZsb3c6IGZ1bmN0aW9uKCkgewoJCWVueW8uZm9yRWFjaCh0aGlzLmNvbnRhaW5lci5jaGlsZHJlbiwgZnVuY3Rpb24oYykgewoJCQlpZiAoYy5maXQgIT09IG51bGwpIHsKCQkJCWMuYWRkUmVtb3ZlQ2xhc3MoImVueW8tZml0IiwgYy5maXQpOwoJCQl9CgkJfSwgdGhpcyk7Cgl9Cn0pOwoKLy9lbnlvLkNvbnRyb2wucHJvdG90eXBlLmxheW91dEtpbmQgPSAiZW55by5CYXNlTGF5b3V0Ijs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Button.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5CdXR0b25fIGltcGxlbWVudHMgYW4gSFRNTCBidXR0b24sIHdpdGggc3VwcG9ydCBmb3IgZ3JvdXBpbmcgdXNpbmcKCTxhIGhyZWY9IiNlbnlvLkdyb3VwIj5lbnlvLkdyb3VwPC9hPi4KCglGb3IgbW9yZSBpbmZvcm1hdGlvbiwgc2VlIHRoZSBkb2N1bWVudGF0aW9uIG9uCgk8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vZW55b2pzL2VueW8vd2lraS9CdXR0b25zIj5CdXR0b25zPC9hPiBpbiB0aGUKCUVueW8gRGV2ZWxvcGVyIEd1aWRlLgoqLwplbnlvLmtpbmQoewoJbmFtZTogImVueW8uQnV0dG9uIiwKCS8vKiBAcHJvdGVjdGVkCglraW5kOiAiZW55by5Ub29sRGVjb3JhdG9yIiwKCXRhZzogImJ1dHRvbiIsCglhdHRyaWJ1dGVzOiB7CgkJLy8gc2V0IHRvIGJ1dHRvbiwgYXMgZGVmYXVsdCBpcyAic3VibWl0IiB3aGljaCBjYW4gY2F1c2UgdW5leHBlY3RlZAoJCS8vIHByb2JsZW1zIHdoZW4gY29udHJvbHMgYXJlIHVzZWQgaW5zaWRlIGEgZm9ybQoJCXR5cGU6ICJidXR0b24iCgl9LAoJLy8qIEBwdWJsaWMKCXB1Ymxpc2hlZDogewoJCS8vKiBXaGVuIHRydWUsIGJ1dHRvbiBpcyBzaG93biBhcyBkaXNhYmxlZCBhbmQgZG9lcyBub3QgZ2VuZXJhdGUgdGFwCgkJLy8qIGV2ZW50cwoJCWRpc2FibGVkOiBmYWxzZQoJfSwKCS8vKiBAcHJvdGVjdGVkCgljcmVhdGU6IGZ1bmN0aW9uKCkgewoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7CgkJdGhpcy5kaXNhYmxlZENoYW5nZWQoKTsKCX0sCglkaXNhYmxlZENoYW5nZWQ6IGZ1bmN0aW9uKCkgewoJCXRoaXMuc2V0QXR0cmlidXRlKCJkaXNhYmxlZCIsIHRoaXMuZGlzYWJsZWQpOwoJfSwKCXRhcDogZnVuY3Rpb24oKSB7CgkJaWYgKHRoaXMuZGlzYWJsZWQpIHsKCQkJLy8gd29yayBhcm91bmQgZm9yIHBsYXRmb3JtcyBsaWtlIENocm9tZSBvbiBBbmRyb2lkIG9yIE9wZXJhIHRoYXQgc2VuZAoJCQkvLyBtb3VzZXVwIHRvIGRpc2FibGVkIGZvcm0gY29udHJvbHMKCQkJcmV0dXJuIHRydWU7CgkJfSBlbHNlIHsKCQkJdGhpcy5zZXRBY3RpdmUodHJ1ZSk7CgkJfQoJfQp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Checkbox.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5DaGVja2JveF8gaW1wbGVtZW50cyBhbiBIVE1MIGNoZWNrYm94IGlucHV0LCB3aXRoIHN1cHBvcnQgZm9yIGdyb3VwaW5nLgoqLwplbnlvLmtpbmQoewoJbmFtZTogImVueW8uQ2hlY2tib3giLAoJa2luZDogImVueW8uSW5wdXQiLAoJY2xhc3NlczogImVueW8tY2hlY2tib3giLAoJZXZlbnRzOiB7CgkJLy8qIEZpcmVzIHdoZW4gY2hlY2tib3ggaXMgdGFwcGVkLgoJCW9uQWN0aXZhdGU6ICIiCgl9LAoJcHVibGlzaGVkOiB7CgkJLy8qIFZhbHVlIG9mIGNoZWNrYm94OyB0cnVlIGlmIGNoZWNrZWQKCQljaGVja2VkOiBmYWxzZSwKCQkvLyogR3JvdXAgQVBJIHJlcXVpcmVtZW50IGZvciBkZXRlcm1pbmluZyBzZWxlY3RlZCBpdGVtCgkJYWN0aXZlOiBmYWxzZSwKCQkvLyogQHByb3RlY3RlZAoJCXR5cGU6ICJjaGVja2JveCIKCX0sCgkvLyogQHByb3RlY3RlZAoJLy8gZGlzYWJsZSBjbGFzc2VzIGluaGVyaXRlZCBmcm9tIGVueW8uSW5wdXQKCWtpbmRDbGFzc2VzOiAiIiwKCWhhbmRsZXJzOiB7CgkJb25jaGFuZ2U6ICJjaGFuZ2UiLAoJCW9uY2xpY2s6ICJjbGljayIKCX0sCgljcmVhdGU6IGZ1bmN0aW9uKCkgewoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7Cgl9LAoJcmVuZGVyZWQ6IGZ1bmN0aW9uKCkgewoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7CgkJaWYgKHRoaXMuYWN0aXZlKSB7CgkJCXRoaXMuYWN0aXZlQ2hhbmdlZCgpOwoJCX0KCQl0aGlzLmNoZWNrZWRDaGFuZ2VkKCk7Cgl9LAoJY2hlY2tlZENoYW5nZWQ6IGZ1bmN0aW9uKCkgewoJCXRoaXMuc2V0Tm9kZVByb3BlcnR5KCJjaGVja2VkIiwgdGhpcy5jaGVja2VkKTsKCQl0aGlzLnNldEF0dHJpYnV0ZSgiY2hlY2tlZCIsIHRoaXMuY2hlY2tlZCA/ICJjaGVja2VkIiA6ICIiKTsKCQl0aGlzLnNldEFjdGl2ZSh0aGlzLmNoZWNrZWQpOwoJfSwKCS8vIGFjdGl2ZSBwcm9wZXJ0eSwgYW5kIG9uQWN0aXZhdGUgZXZlbnQsIGFyZSBwYXJ0IG9mICJHcm91cEl0ZW0iIGludGVyZmFjZQoJLy8gdGhhdCB3ZSBzdXBwb3J0IGluIHRoaXMgb2JqZWN0CglhY3RpdmVDaGFuZ2VkOiBmdW5jdGlvbigpIHsKCQl0aGlzLmFjdGl2ZSA9IGVueW8uaXNUcnVlKHRoaXMuYWN0aXZlKTsKCQl0aGlzLnNldENoZWNrZWQodGhpcy5hY3RpdmUpOwoJCXRoaXMuYnViYmxlKCJvbkFjdGl2YXRlIik7Cgl9LAoJLy8gYWxsIGlucHV0IHR5cGUgY29udHJvbHMgc3VwcG9ydCAndmFsdWUnIHByb3BlcnR5CglzZXRWYWx1ZTogZnVuY3Rpb24oaW5WYWx1ZSkgewoJCXRoaXMuc2V0Q2hlY2tlZChlbnlvLmlzVHJ1ZShpblZhbHVlKSk7Cgl9LAoJZ2V0VmFsdWU6IGZ1bmN0aW9uKCkgewoJCXJldHVybiB0aGlzLmdldENoZWNrZWQoKTsKCX0sCgl2YWx1ZUNoYW5nZWQ6IGZ1bmN0aW9uKCkgewoJCS8vIGluaGVyaXRlZCBiZWhhdmlvciBpcyB0byBzZXQgInZhbHVlIiBhdHRyaWJ1dGUgYW5kIG5vZGUtcHJvcGVydHkKCQkvLyB3aGljaCBkb2VzIG5vdCBhcHBseSB0byBjaGVja2JveCAodXNlcyAiY2hlY2tlZCIpIHNvCgkJLy8gd2Ugc3F1ZWxjaCB0aGUgaW5oZXJpdGVkIG1ldGhvZAoJfSwKCWNoYW5nZTogZnVuY3Rpb24oKSB7CgkJdmFyIG5vZGVDaGVja2VkID0gZW55by5pc1RydWUodGhpcy5nZXROb2RlUHJvcGVydHkoImNoZWNrZWQiKSk7CgkJdGhpcy5zZXRBY3RpdmUobm9kZUNoZWNrZWQpOwoJfSwKCWNsaWNrOiBmdW5jdGlvbihpblNlbmRlciwgaW5FdmVudCkgewoJCS8vIFZhcmlvdXMgdmVyc2lvbnMgb2YgSUUgKG5vdGFibHkgSUU4KSBkbyBub3QgZmlyZSAnb25jaGFuZ2UnIGZvcgoJCS8vIGNoZWNrYm94ZXMsIHNvIHdlIGRpc2Nlcm4gY2hhbmdlIHZpYSAnY2xpY2snLgoJCS8vIE5vdGU6IGtleWJvYXJkIGludGVyYWN0aW9uIChlLmcuIHByZXNzaW5nIHNwYWNlIHdoZW4gZm9jdXNlZCkgZmlyZXMKCQkvLyBhIGNsaWNrIGV2ZW50LgoJCWlmIChlbnlvLnBsYXRmb3JtLmllIDw9IDgpIHsKCQkJdGhpcy5idWJibGUoIm9uY2hhbmdlIiwgaW5FdmVudCk7CgkJfQoJfQp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/DragAvatar.js"
Content-Type: application/octet-stream; x-encoding=base64
Ly8qIEBwcm90ZWN0ZWQKZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLl9EcmFnQXZhdGFyIiwKCXN0eWxlOiAicG9zaXRpb246IGFic29sdXRlOyB6LWluZGV4OiAxMDsgcG9pbnRlci1ldmVudHM6IG5vbmU7IGN1cnNvcjogbW92ZTsiLAoJc2hvd2luZzogZmFsc2UsCglzaG93aW5nQ2hhbmdlZDogZnVuY3Rpb24oKSB7CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCQlkb2N1bWVudC5ib2R5LnN0eWxlLmN1cnNvciA9IHRoaXMuc2hvd2luZyA/ICJtb3ZlIiA6IG51bGw7Cgl9Cn0pOwoKLy8qIEBwdWJsaWMKLyoqCglfZW55by5EcmFnQXZhdGFyXyBjcmVhdGVzIGEgY29udHJvbCB0byBmb2xsb3cgdGhlIHBvaW50ZXIgd2hlbiBkcmFnZ2luZy4gSXQKCWF1dG9tYXRpY2FsbHkgZGlzcGxheXMgdGhlIGF2YXRhciBjb250cm9sIHdoZW4gdGhlIHVzZXIgZHJhZ3MsIGFuZCB1cGRhdGVzCglpdHMgcG9zaXRpb24gcmVsYXRpdmUgdG8gdGhlIGN1cnJlbnQgcG9pbnRlciBsb2NhdGlvbi4KCgkJZW55by5raW5kKHsKCQkJbmFtZTogIkFwcCIsCgkJCWhhbmRsZXJzOiB7CgkJCQlvbmRyYWc6ICJkcmFnIiwKCQkJCW9uZHJhZ2ZpbmlzaDogImRyYWdGaW5pc2giLAoJCQl9LAoJCQljb21wb25lbnRzOiBbCgkJCQl7bmFtZToiZHJhZ0F2YXRhciIsIGtpbmQ6IkRyYWdBdmF0YXIiLAoJCQkJCWNvbXBvbmVudHM6IFt7dGFnOiAiaW1nIiwgc3JjOiAiaW1hZ2VzL2ljb24ucG5nIn1dCgkJCQl9CgkJCV0sCgkJCWRyYWc6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJCQl0aGlzLiQuZHJhZ0F2YXRhci5kcmFnKGluRXZlbnQpOwoJCQl9LAoJCQlkcmFnRmluaXNoOiBmdW5jdGlvbihpblNlbmRlciwgaW5FdmVudCkgewoJCQkJdGhpcy4kLmRyYWdBdmF0YXIuaGlkZSgpOwoJCQl9CgkJfSk7CiovCmVueW8ua2luZCh7CgluYW1lOiAiZW55by5EcmFnQXZhdGFyIiwKCWtpbmQ6ICJlbnlvLkNvbXBvbmVudCIsCglwdWJsaXNoZWQ6IHsKCQkvLyogQ3VycmVudCB2aXNpYmlsaXR5IHN0YXRlIG9mIHRoZSBEcmFnQXZhdGFyCgkJc2hvd2luZzogZmFsc2UsCgkJLyoqCgkJCURpc3RhbmNlIChpbiBwaXhlbHMpIGFsb25nIHRoZSBob3Jpem9udGFsIGF4aXMgYmV0d2VlbiB0aGUgY3VycmVudAoJCQlkcmFnIHBvc2l0aW9uIGFuZCB3aGVyZSB0aGUgYXZhdGFyIGNvbnRyb2wgaXMgZGlzcGxheWVkLgoJCSovCgkJb2Zmc2V0WDogMjAsCgkJLyoqCgkJCURpc3RhbmNlIChpbiBwaXhlbHMpIGFsb25nIHRoZSB2ZXJ0aWNhbCBheGlzIGJldHdlZW4gdGhlIGN1cnJlbnQKCQkJZHJhZyBwb3NpdGlvbiBhbmQgd2hlcmUgdGhlIGF2YXRhciBjb250cm9sIGlzIGRpc3BsYXllZC4KCQkqLwoJCW9mZnNldFk6IDMwCgl9LAoJLy8qIEBwcm90ZWN0ZWQKCWluaXRDb21wb25lbnRzOiBmdW5jdGlvbigpIHsKCQl0aGlzLmF2YXRhckNvbXBvbmVudHMgPSB0aGlzLmNvbXBvbmVudHM7CgkJdGhpcy5jb21wb25lbnRzID0gbnVsbDsKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJfSwKCXJlcXVpcmVBdmF0YXI6IGZ1bmN0aW9uKCkgewoJCS8vIEZJWE1FOiB0aGVyZSBpcyBub2JvZHkgdG8gY2FsbCB0ZWFyZG93blJlbmRlciBvbiB0aGlzLmF2YXRhcgoJCS8vIGlmIGRvY3VtZW50LmJvZHkuaW5uZXJIVE1MIGhhcyBiZWVuIHdyaXR0ZW4gb3ZlciwgaGlzIG5vZGUgaXMgaW52YWxpZAoJCS8vIHdlIHNob3VsZCBoYXZlIGEgdHJhcCBmb3IgdGhpcyBjb25kaXRpb24gaGVyZQoJCWlmICghdGhpcy5hdmF0YXIpIHsKCQkJdGhpcy5hdmF0YXIgPSB0aGlzLmNyZWF0ZUNvbXBvbmVudCh7a2luZDogZW55by5fRHJhZ0F2YXRhciwgcGFyZW50Tm9kZTogZG9jdW1lbnQuYm9keSwgc2hvd2luZzogZmFsc2UsIGNvbXBvbmVudHM6IHRoaXMuYXZhdGFyQ29tcG9uZW50c30pLnJlbmRlcigpOwoJCX0KCX0sCglzaG93aW5nQ2hhbmdlZDogZnVuY3Rpb24oKSB7CgkJdGhpcy5hdmF0YXIuc2V0U2hvd2luZyh0aGlzLnNob3dpbmcpOwoJCWRvY3VtZW50LmJvZHkuc3R5bGUuY3Vyc29yID0gdGhpcy5zaG93aW5nID8gIm1vdmUiIDogbnVsbDsKCX0sCgkvLyogQHB1YmxpYwoJLyoqCgkJSW5zdGFudGlhdGVzIHRoZSBhdmF0YXIgY29udHJvbCAoaWYgbmVjZXNzYXJ5KSwgZGV0ZXJtaW5lcyBjb3JyZWN0CgkJcG9zaXRpb24sIGFuZCBjYWxscyBfc2hvd18gdG8gbWFrZSBpdCB2aXNpYmxlLgoJKi8KCWRyYWc6IGZ1bmN0aW9uKGluRXZlbnQpIHsKCQl0aGlzLnJlcXVpcmVBdmF0YXIoKTsKCQl0aGlzLmF2YXRhci5zZXRCb3VuZHMoe3RvcDogaW5FdmVudC5wYWdlWSAtIHRoaXMub2Zmc2V0WSwgbGVmdDogaW5FdmVudC5wYWdlWCArIHRoaXMub2Zmc2V0WH0pOwoJCXRoaXMuc2hvdygpOwoJfSwKCS8vKiBTaG93cyB0aGUgRHJhZ0F2YXRhci4KCXNob3c6IGZ1bmN0aW9uKCkgewoJCXRoaXMuc2V0U2hvd2luZyh0cnVlKTsKCX0sCgkvLyogSGlkZXMgdGhlIERyYWdBdmF0YXIuCgloaWRlOiBmdW5jdGlvbigpIHsKCQl0aGlzLnNldFNob3dpbmcoZmFsc2UpOwoJfQp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Drawer.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqDQoJX2VueW8uRHJhd2VyXyBpcyBhIGNvbnRyb2wgdGhhdCBhcHBlYXJzIG9yIGRpc2FwcGVhcnMgYmFzZWQgb24gaXRzIF9vcGVuXw0KCXByb3BlcnR5LiBCeSBkZWZhdWx0LCB0aGUgZHJhd2VyIGFwcGVhcnMgb3IgZGlzYXBwZWFycyB3aXRoIGEgc2xpZGluZw0KCWFuaW1hdGlvbiB3aG9zZSBkaXJlY3Rpb24gaXMgZGV0ZXJtaW5lZCBieSB0aGUgX29yaWVudF8gcHJvcGVydHkuDQoNCglGb3IgbW9yZSBpbmZvcm1hdGlvbiwgc2VlIHRoZSBkb2N1bWVudGF0aW9uIG9uDQoJPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2VueW9qcy9lbnlvL3dpa2kvRHJhd2VycyI+RHJhd2VyczwvYT4gaW4gdGhlDQoJRW55byBEZXZlbG9wZXIgR3VpZGUuDQoqLw0KDQplbnlvLmtpbmQoew0KCW5hbWU6ICJlbnlvLkRyYXdlciIsDQoJcHVibGlzaGVkOiB7DQoJCS8vKiBUaGUgdmlzaWJpbGl0eSBzdGF0ZSBvZiB0aGUgZHJhd2VyJ3MgYXNzb2NpYXRlZCBjb250cm9sDQoJCW9wZW46IHRydWUsDQoJCS8qKg0KCQkJRGlyZWN0aW9uIG9mIHRoZSBvcGVuaW5nL2Nsb3NpbmcgYW5pbWF0aW9uLS1laXRoZXIgInYiIGZvciB2ZXJ0aWNhbA0KCQkJb3IgImgiIGZvciBob3Jpem9udGFsDQoJCSovDQoJCW9yaWVudDogInYiLA0KCQkvLyogSWYgdHJ1ZSwgdGhlIG9wZW5pbmcvY2xvc2luZyB0cmFuc2l0aW9uIHdpbGwgYmUgYW5pbWF0ZWQNCgkJYW5pbWF0ZWQ6IHRydWUNCgl9LA0KCS8vKiBAcHJvdGVjdGVkDQoJc3R5bGU6ICJvdmVyZmxvdzogaGlkZGVuOyBwb3NpdGlvbjogcmVsYXRpdmU7IiwNCgl0b29sczogWw0KCQl7a2luZDogIkFuaW1hdG9yIiwgb25TdGVwOiAiYW5pbWF0b3JTdGVwIiwgb25FbmQ6ICJhbmltYXRvckVuZCJ9LA0KCQl7bmFtZTogImNsaWVudCIsIHN0eWxlOiAicG9zaXRpb246IHJlbGF0aXZlOyIsIGNsYXNzZXM6ICJlbnlvLWJvcmRlci1ib3gifQ0KCV0sDQoJY3JlYXRlOiBmdW5jdGlvbigpIHsNCgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsNCgkJdGhpcy5hbmltYXRlZENoYW5nZWQoKTsNCgkJdGhpcy5vcGVuQ2hhbmdlZCgpOw0KCX0sDQoJaW5pdENvbXBvbmVudHM6IGZ1bmN0aW9uKCkgew0KCQl0aGlzLmNyZWF0ZUNocm9tZSh0aGlzLnRvb2xzKTsNCgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsNCgl9LA0KCWFuaW1hdGVkQ2hhbmdlZDogZnVuY3Rpb24oKSB7DQoJCWlmICghdGhpcy5hbmltYXRlZCAmJiB0aGlzLmhhc05vZGUoKSAmJiB0aGlzLiQuYW5pbWF0b3IuaXNBbmltYXRpbmcoKSkgew0KCQkJdGhpcy4kLmFuaW1hdG9yLnN0b3AoKTsNCgkJCXRoaXMuYW5pbWF0b3JFbmQoKTsNCgkJfQ0KCX0sDQoJb3BlbkNoYW5nZWQ6IGZ1bmN0aW9uKCkgew0KCQl0aGlzLiQuY2xpZW50LnNob3coKTsNCgkJaWYgKHRoaXMuaGFzTm9kZSgpKSB7DQoJCQlpZiAodGhpcy4kLmFuaW1hdG9yLmlzQW5pbWF0aW5nKCkpIHsNCgkJCQl0aGlzLiQuYW5pbWF0b3IucmV2ZXJzZSgpOw0KCQkJfSBlbHNlIHsNCgkJCQl2YXIgdiA9IHRoaXMub3JpZW50ID09ICJ2IjsNCgkJCQl2YXIgZCA9IHYgPyAiaGVpZ2h0IiA6ICJ3aWR0aCI7DQoJCQkJdmFyIHAgPSB2ID8gInRvcCIgOiAibGVmdCI7DQoJCQkJLy8gdW5maXhpbmcgdGhlIGhlaWdodC93aWR0aCBpcyBuZWVkZWQgdG8gcHJvcGVybHkNCgkJCQkvLyBtZWFzdXJlIHRoZSBzY3JvbGxIZWlnaHQvV2lkdGggRE9NIHByb3BlcnR5LCBidXQNCgkJCQkvLyBjYW4gY2F1c2UgYSBtb21lbnRhcnkgZmxhc2ggb2YgY29udGVudCBvbiBzb21lIGJyb3dzZXJzDQoJCQkJdGhpcy5hcHBseVN0eWxlKGQsIG51bGwpOw0KCQkJCXZhciBzID0gdGhpcy5oYXNOb2RlKClbdiA/ICJzY3JvbGxIZWlnaHQiIDogInNjcm9sbFdpZHRoIl07DQoJCQkJaWYgKHRoaXMuYW5pbWF0ZWQpIHsNCgkJCQkJdGhpcy4kLmFuaW1hdG9yLnBsYXkoew0KCQkJCQkJc3RhcnRWYWx1ZTogdGhpcy5vcGVuID8gMCA6IHMsDQoJCQkJCQllbmRWYWx1ZTogdGhpcy5vcGVuID8gcyA6IDAsDQoJCQkJCQlkaW1lbnNpb246IGQsDQoJCQkJCQlwb3NpdGlvbjogcA0KCQkJCQl9KTsNCgkJCQl9IGVsc2Ugew0KCQkJCQkvLyBkaXJlY3RseSBydW4gbGFzdCBmcmFtZSBpZiBub3QgYW5pbWF0aW5nDQoJCQkJCXRoaXMuYW5pbWF0b3JFbmQoKTsNCgkJCQl9DQoJCQl9DQoJCX0gZWxzZSB7DQoJCQl0aGlzLiQuY2xpZW50LnNldFNob3dpbmcodGhpcy5vcGVuKTsNCgkJfQ0KCX0sDQoJYW5pbWF0b3JTdGVwOiBmdW5jdGlvbihpblNlbmRlcikgew0KCQkvLyB0aGUgYWN0dWFsIGRyYXdlciBET00gbm9kZSBhZGp1c3RzIGl0cyBoZWlnaHQNCgkJaWYgKHRoaXMuaGFzTm9kZSgpKSB7DQoJCQl2YXIgZCA9IGluU2VuZGVyLmRpbWVuc2lvbjsNCgkJCXRoaXMubm9kZS5zdHlsZVtkXSA9IHRoaXMuZG9tU3R5bGVzW2RdID0gaW5TZW5kZXIudmFsdWUgKyAicHgiOw0KCQl9DQoJCS8vIHdoaWxlIHRoZSBjbGllbnQgaW5zaWRlIHRoZSBkcmF3ZXIgYWRqdXN0cyBpdHMgcG9zaXRpb24gdG8gbW92ZSBvdXQgb2YgdGhlIHZpc2libGUgYXJlYQ0KCQl2YXIgY24gPSB0aGlzLiQuY2xpZW50Lmhhc05vZGUoKTsNCgkJaWYgKGNuKSB7DQoJCQl2YXIgcCA9IGluU2VuZGVyLnBvc2l0aW9uOw0KCQkJdmFyIG8gPSAodGhpcy5vcGVuID8gaW5TZW5kZXIuZW5kVmFsdWUgOiBpblNlbmRlci5zdGFydFZhbHVlKTsNCgkJCWNuLnN0eWxlW3BdID0gdGhpcy4kLmNsaWVudC5kb21TdHlsZXNbcF0gPSAoaW5TZW5kZXIudmFsdWUgLSBvKSArICJweCI7DQoJCX0NCgkJaWYgKHRoaXMuY29udGFpbmVyKSB7DQoJCQl0aGlzLmNvbnRhaW5lci5yZXNpemVkKCk7DQoJCX0NCgkJcmV0dXJuIHRydWU7DQoJfSwNCglhbmltYXRvckVuZDogZnVuY3Rpb24oKSB7DQoJCWlmICghdGhpcy5vcGVuKSB7DQoJCQl0aGlzLiQuY2xpZW50LmhpZGUoKTsNCgkJfQ0KCQllbHNlIHsNCgkJCS8vIHNhdmUgY2hhbmdlcyB0byB0aGlzLmRvbUNzc1RleHQgLS0+IHNlZSBFTllPLTE1NjENCgkJCXRoaXMuJC5jbGllbnQuZG9tQ3NzVGV4dCA9IGVueW8uQ29udHJvbC5kb21TdHlsZXNUb0Nzc1RleHQodGhpcy4kLmNsaWVudC5kb21TdHlsZXMpOw0KCQkJLy8gYXQgZW5kIG9mIG9wZW4gYW5pbWF0aW9uLCBjbGVhbiBsaW1pdCBvbiBoZWlnaHQvd2lkdGgNCgkJCXZhciB2ID0gKHRoaXMub3JpZW50ID09ICJ2Iik7DQoJCQl2YXIgZCA9IHYgPyAiaGVpZ2h0IiA6ICJ3aWR0aCI7DQoJCQl2YXIgcCA9IHYgPyAidG9wIiA6ICJsZWZ0IjsNCgkJCXZhciBjbiA9IHRoaXMuJC5jbGllbnQuaGFzTm9kZSgpOw0KCQkJLy8gY2xlYXIgb3V0IGNoYW5nZXMgdG8gY29udGFpbmVyIHBvc2l0aW9uICYgbm9kZSBkaW1lbnNpb24NCgkJCWlmIChjbikgew0KCQkJCWNuLnN0eWxlW3BdID0gdGhpcy4kLmNsaWVudC5kb21TdHlsZXNbcF0gPSBudWxsOw0KCQkJfQ0KCQkJaWYgKHRoaXMubm9kZSkgew0KCQkJCXRoaXMubm9kZS5zdHlsZVtkXSA9IHRoaXMuZG9tU3R5bGVzW2RdID0gbnVsbDsNCgkJCX0NCgkJfQ0KCQlpZiAodGhpcy5jb250YWluZXIpIHsNCgkJCXRoaXMuY29udGFpbmVyLnJlc2l6ZWQoKTsNCgkJfQ0KCQlyZXR1cm4gdHJ1ZTsNCgl9DQp9KTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/FloatingLayer.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5GbG9hdGluZ0xheWVyXyBpcyBhIGNvbnRyb2wgdGhhdCBwcm92aWRlcyBhIGxheWVyIGZvciBjb250cm9scyB0aGF0CglzaG91bGQgYmUgZGlzcGxheWVkIGFib3ZlIGFuIGFwcGxpY2F0aW9uLgoJVGhlIEZsb2F0aW5nTGF5ZXIgc2luZ2xldG9uIGNhbiBiZSBzZXQgYXMgYSBjb250cm9sJ3MgcGFyZW50IHRvIGhhdmUgdGhlCgljb250cm9sIGZsb2F0IGFib3ZlIGFuIGFwcGxpY2F0aW9uLCBlLmcuOgoKCQljcmVhdGU6IGZ1bmN0aW9uKCkgewoJCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCQl0aGlzLnNldFBhcmVudChlbnlvLmZsb2F0aW5nTGF5ZXIpOwoJCX0KCglOb3RlOiBJdCdzIG5vdCBpbnRlbmRlZCB0aGF0IHVzZXJzIGNyZWF0ZSBpbnN0YW5jZXMgb2YgX2VueW8uRmxvYXRpbmdMYXllcl8uCiovCi8vKiBAcHJvdGVjdGVkCmVueW8ua2luZCh7CgluYW1lOiAiZW55by5GbG9hdGluZ0xheWVyIiwKCS8vKiBAcHJvdGVjdGVkCgljcmVhdGU6IGZ1bmN0aW9uKCkgewoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7CgkJdGhpcy5zZXRQYXJlbnQobnVsbCk7Cgl9LAoJLy8gZGV0ZWN0IHdoZW4gbm9kZSBpcyBkZXRhdGNoZWQgZHVlIHRvIGRvY3VtZW50LmJvZHkgYmVpbmcgc3RvbXBlZAoJaGFzTm9kZTogZnVuY3Rpb24oKSB7CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCQlpZiAodGhpcy5ub2RlICYmICF0aGlzLm5vZGUucGFyZW50Tm9kZSkgewoJCQl0aGlzLnRlYXJkb3duUmVuZGVyKCk7CgkJfQoJCXJldHVybiB0aGlzLm5vZGU7Cgl9LAoJcmVuZGVyOiBmdW5jdGlvbigpIHsKCQl0aGlzLnBhcmVudE5vZGUgPSBkb2N1bWVudC5ib2R5OwoJCXJldHVybiB0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJfSwKCWdlbmVyYXRlSW5uZXJIdG1sOiBmdW5jdGlvbigpIHsKCQlyZXR1cm4gIiI7Cgl9LAoJYmVmb3JlQ2hpbGRSZW5kZXI6IGZ1bmN0aW9uKCkgewoJCWlmICghdGhpcy5oYXNOb2RlKCkpIHsKCQkJdGhpcy5yZW5kZXIoKTsKCQl9Cgl9LAoJdGVhcmRvd25DaGlsZHJlbjogZnVuY3Rpb24oKSB7Cgl9Cn0pOwoKZW55by5mbG9hdGluZ0xheWVyID0gbmV3IGVueW8uRmxvYXRpbmdMYXllcigpOw==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Group.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5Hcm91cF8gcHJvdmlkZXMgYSB3cmFwcGVyIGFyb3VuZCBtdWx0aXBsZSBlbGVtZW50cy4gIEl0IGVuYWJsZXMgdGhlCgljcmVhdGlvbiBvZiByYWRpbyBncm91cHMgZnJvbSBhcmJpdHJhcnkgY29tcG9uZW50cyBzdXBwb3J0aW5nIHRoZQoJW0dyb3VwSXRlbV0oI2VueW8uR3JvdXBJdGVtKSBBUEkuCiovCmVueW8ua2luZCh7CgluYW1lOiAiZW55by5Hcm91cCIsCglwdWJsaXNoZWQ6IHsKCQkvKioKCQkJSWYgdHJ1ZSwgb25seSBvbmUgR3JvdXBJdGVtIGluIHRoZSBjb21wb25lbnQgbGlzdCBtYXkgYmUgYWN0aXZlIGF0CgkJCWEgZ2l2ZW4gdGltZS4KCQkqLwoJCWhpZ2hsYW5kZXI6IHRydWUsCgkJLy8qIFRoZSBjb250cm9sIHRoYXQgd2FzIGxhc3Qgc2VsZWN0ZWQKCQlhY3RpdmU6IG51bGwKCX0sCgkvLyogQHByb3RlY3RlZAoJaGFuZGxlcnM6IHsKCQlvbkFjdGl2YXRlOiAiYWN0aXZhdGUiCgl9LAoJYWN0aXZhdGU6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJaWYgKHRoaXMuaGlnaGxhbmRlcikgewoJCQkvLyBkZWFjdGl2YXRpb24gbWVzc2FnZXMgYXJlIGlnbm9yZWQgdW5sZXNzIGl0J3MgYW4gYXR0ZW1wdAoJCQkvLyB0byBkZWFjdGl2YXRlIHRoZSBoaWdobGFuZGVyCgkJCWlmICghaW5FdmVudC5vcmlnaW5hdG9yLmFjdGl2ZSkgewoJCQkJLy8gdGhpcyBjbGF1c2UgcHJldmVudHMgZGVhY3RpdmF0aW5nIGEgZ3JvdXBlZCBpdGVtIG9uY2UgaXQncyBiZWVuIGFjdGl2ZS4KCQkJCS8vIHRoZSBvbmx5IHByb3BlciB3YXkgdG8gZGVhY3RpdmF0ZSBhIGdyb3VwZWQgaXRlbSBpcyB0byBjaG9vc2UgYSBuZXcKCQkJCS8vIGhpZ2hsYW5kZXIuCgkJCQlpZiAoaW5FdmVudC5vcmlnaW5hdG9yID09IHRoaXMuYWN0aXZlKSB7CgkJCQkJdGhpcy5hY3RpdmUuc2V0QWN0aXZlKHRydWUpOwoJCQkJfQoJCQl9IGVsc2UgewoJCQkJdGhpcy5zZXRBY3RpdmUoaW5FdmVudC5vcmlnaW5hdG9yKTsKCQkJfQoJCX0KCX0sCglhY3RpdmVDaGFuZ2VkOiBmdW5jdGlvbihpbk9sZCkgewoJCWlmIChpbk9sZCkgewoJCQlpbk9sZC5zZXRBY3RpdmUoZmFsc2UpOwoJCQlpbk9sZC5yZW1vdmVDbGFzcygiYWN0aXZlIik7CgkJfQoJCWlmICh0aGlzLmFjdGl2ZSkgewoJCQl0aGlzLmFjdGl2ZS5hZGRDbGFzcygiYWN0aXZlIik7CgkJfQoJfQp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/GroupItem.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5Hcm91cEl0ZW1fIGlzIHRoZSBiYXNlIGtpbmQgZm9yIHRoZSBHcm91cGluZyBBUEkuIEl0IG1hbmFnZXMgdGhlCglhY3RpdmUgc3RhdGUgb2YgdGhlIGNvbXBvbmVudCAob3IgdGhlIGluaGVyaXRpbmcgY29tcG9uZW50KS4gQSBzdWJraW5kIG1heQoJY2FsbCBfc2V0QWN0aXZlXyB0byBzZXQgdGhlIF9hY3RpdmVfIHByb3BlcnR5IHRvIHRoZSBkZXNpcmVkIHN0YXRlOyB0aGlzCgl3aWxsIGFkZGl0aW9uYWxseSBidWJibGUgYW4gX29uQWN0aXZhdGVfIGV2ZW50LCB3aGljaCBjYW4gYmUgaGFuZGxlZCBhcwoJbmVlZGVkIGJ5IHRoZSBjb250YWluaW5nIGNvbXBvbmVudHMuIFRoaXMgaXMgdXNlZnVsIGZvciBjcmVhdGluZyBncm91cHMgb2YKCWl0ZW1zIHdob3NlIHN0YXRlIHNob3VsZCBiZSBtYW5hZ2VkIGFzIGEgZ3JvdXAuCgoJRm9yIGFuIGV4YW1wbGUgb2YgaG93IHRoaXMgd29ya3MsIHNlZSB0aGUKCTxhIGhyZWY9IiNlbnlvLkdyb3VwIj5lbnlvLkdyb3VwPC9hPiBraW5kLCB3aGljaCBlbmFibGVzIHRoZSBjcmVhdGlvbiBvZgoJcmFkaW8gZ3JvdXBzIGZyb20gYXJiaXRyYXJ5IGNvbXBvbmVudHMgdGhhdAlzdXBwb3J0IHRoZSBHcm91cGluZyBBUEkuCiovCgplbnlvLmtpbmQoewoJbmFtZTogImVueW8uR3JvdXBJdGVtIiwKCXB1Ymxpc2hlZDogewoJCS8vKiBUcnVlIGlmIHRoZSBpdGVtIGlzIGN1cnJlbnRseSBzZWxlY3RlZAoJCWFjdGl2ZTogZmFsc2UKCX0sCgkvLyogQHByb3RlY3RlZAoJcmVuZGVyZWQ6IGZ1bmN0aW9uKCkgewoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7CgkJdGhpcy5hY3RpdmVDaGFuZ2VkKCk7Cgl9LAoJYWN0aXZlQ2hhbmdlZDogZnVuY3Rpb24oKSB7CgkJdGhpcy5idWJibGUoIm9uQWN0aXZhdGUiKTsKCX0KfSk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Image.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5JbWFnZV8gaW1wbGVtZW50cyBhbiBIVE1MICZsdDtpbWcmZ3Q7IGVsZW1lbnQgYW5kLCBvcHRpb25hbGx5LCBidWJibGVzCgl0aGUgX29ubG9hZF8gYW5kIF9vbmVycm9yXyBldmVudHMuIEltYWdlIGRyYWdnaW5nIGlzIHN1cHByZXNzZWQgYnkgZGVmYXVsdCwKCXNvIGFzIG5vdCB0byBpbnRlcmZlcmUgd2l0aCB0b3VjaCBpbnRlcmZhY2VzLgoqLwplbnlvLmtpbmQoewoJbmFtZTogImVueW8uSW1hZ2UiLAoJLy8qIFdoZW4gdHJ1ZSwgbm8gX29ubG9hZF8gb3IgX29uZXJyb3JfIGV2ZW50IGhhbmRsZXJzIHdpbGwgYmUgY3JlYXRlZAoJbm9FdmVudHM6IGZhbHNlLAoJLy8qIEBwcm90ZWN0ZWQKCXRhZzogImltZyIsCglhdHRyaWJ1dGVzOiB7CgkJLy8gbm90ZTogZHJhZ2dhYmxlIGF0dHJpYnV0ZSB0YWtlcyBvbmUgb2YgdGhlc2UgU3RyaW5nIHZhbHVlczogInRydWUiLCAiZmFsc2UiLCAiYXV0byIKCQkvLyAoQm9vbGVhbiBfZmFsc2VfIHdvdWxkIHJlbW92ZSB0aGUgYXR0cmlidXRlKQoJCWRyYWdnYWJsZTogImZhbHNlIgoJfSwKCWNyZWF0ZTogZnVuY3Rpb24oKSB7CgkJaWYgKHRoaXMubm9FdmVudHMpIHsKCQkJZGVsZXRlIHRoaXMuYXR0cmlidXRlcy5vbmxvYWQ7CgkJCWRlbGV0ZSB0aGlzLmF0dHJpYnV0ZXMub25lcnJvcjsKCQl9CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCX0sCglyZW5kZXJlZDogZnVuY3Rpb24oKSB7CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCQllbnlvLm1ha2VCdWJibGUodGhpcywgImxvYWQiLCAiZXJyb3IiKTsKCX0KfSk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Input.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5JbnB1dF8gaW1wbGVtZW50cyBhbiBIVE1MICZsdDtpbnB1dCZndDsgZWxlbWVudCB3aXRoIGNyb3NzLXBsYXRmb3JtCglzdXBwb3J0IGZvciBjaGFuZ2UgZXZlbnRzLgoKCVlvdSBjYW4gbGlzdGVuIGZvciBfb25pbnB1dF8gYW5kIF9vbmNoYW5nZV8gRE9NIGV2ZW50cyBmcm9tIHRoaXMgY29udHJvbAoJdG8ga25vdyB3aGVuIHRoZSB0ZXh0IGluc2lkZSBoYXMgYmVlbiBtb2RpZmllZC4gX29uaW5wdXRfIGZpcmVzIGltbWVkaWF0ZWx5LAoJd2hpbGUgX29uY2hhbmdlXyBmaXJlcyB3aGVuIHRoZSB0ZXh0IGhhcyBjaGFuZ2VkIGFuZCB0aGUgaW5wdXQgc3Vic2VxdWVudGx5Cglsb3NlcyBmb2N1cy4KCglGb3IgbW9yZSBpbmZvcm1hdGlvbiwgc2VlIHRoZSBkb2N1bWVudGF0aW9uIG9uCglbVGV4dCBGaWVsZHNdKGh0dHBzOi8vZ2l0aHViLmNvbS9lbnlvanMvZW55by93aWtpL1RleHQtRmllbGRzKSBpbiB0aGUgRW55bwoJRGV2ZWxvcGVyIEd1aWRlLgoqLwplbnlvLmtpbmQoewoJbmFtZTogImVueW8uSW5wdXQiLAoJcHVibGlzaGVkOiB7CgkJLyoqCgkJCVZhbHVlIG9mIHRoZSBpbnB1dC4gVXNlIHRoaXMgcHJvcGVydHkgb25seSB0byBpbml0aWFsaXplIHRoZSB2YWx1ZS4KCQkJQ2FsbCBfZ2V0VmFsdWVfIGFuZCBfc2V0VmFsdWVfIHRvIG1hbmlwdWxhdGUgdGhlIHZhbHVlIGF0IHJ1bnRpbWUuCgkJKi8KCQl2YWx1ZTogIiIsCgkJLy8qIFRleHQgdG8gZGlzcGxheSB3aGVuIHRoZSBpbnB1dCBpcyBlbXB0eQoJCXBsYWNlaG9sZGVyOiAiIiwKCQkvKioKCQkJVHlwZSBvZiBpbnB1dDsgaWYgbm90IHNwZWNpZmllZCwgaXQncyB0cmVhdGVkIGFzICJ0ZXh0Ii4gSXQgY2FuCgkJCWJlIGFueXRoaW5nIHNwZWNpZmllZCBmb3IgdGhlIF90eXBlXyBhdHRyaWJ1dGUgaW4gdGhlIEhUTUwKCQkJc3BlY2lmaWNhdGlvbiwgaW5jbHVkaW5nICJ1cmwiLCAiZW1haWwiLCAic2VhcmNoIiwgb3IgIm51bWJlciIuCgkJKi8KCQl0eXBlOiAiIiwKCQkvKioKCQkJV2hlbiB0cnVlLCBwcmV2ZW50cyBpbnB1dCBpbnRvIHRoZSBjb250cm9sLiBUaGlzIG1hcHMgdG8gdGhlCgkJCV9kaXNhYmxlZF8gRE9NIGF0dHJpYnV0ZS4KCQkqLwoJCWRpc2FibGVkOiBmYWxzZSwKCQkvLyogV2hlbiB0cnVlLCBzZWxlY3QgdGhlIGNvbnRlbnRzIG9mIHRoZSBpbnB1dCB3aGVuIGl0IGdhaW5zIGZvY3VzLgoJCXNlbGVjdE9uRm9jdXM6IGZhbHNlCgl9LAoJZXZlbnRzOiB7CgkJLy8qIEZpcmVzIHdoZW4gdGhlIGlucHV0IGlzIGRpc2FibGVkIG9yIGVuYWJsZWQuCgkJb25EaXNhYmxlZENoYW5nZTogIiIKCX0sCgkvLyogU2V0IHRvIHRydWUgdG8gZm9jdXMgdGhpcyBjb250cm9sIHdoZW4gaXQgaXMgcmVuZGVyZWQKCWRlZmF1bHRGb2N1czogZmFsc2UsCgkvLyogQHByb3RlY3RlZAoJdGFnOiAiaW5wdXQiLAoJY2xhc3NlczogImVueW8taW5wdXQiLAoJaGFuZGxlcnM6IHsKCQlvbmZvY3VzOiAiZm9jdXNlZCIsCgkJb25pbnB1dDogImlucHV0IiwKCQlvbmNsZWFyOiAiY2xlYXIiLAoJCW9uZHJhZ3N0YXJ0OiAiZHJhZ3N0YXJ0IgoJfSwKCWNyZWF0ZTogZnVuY3Rpb24oKSB7CgkJaWYgKGVueW8ucGxhdGZvcm0uaWUpIHsKCQkJdGhpcy5oYW5kbGVycy5vbmtleXVwID0gImlla2V5dXAiOwoJCX0KCQlpZiAoZW55by5wbGF0Zm9ybS53aW5kb3dzUGhvbmUpIHsKCQkJdGhpcy5oYW5kbGVycy5vbmtleWRvd24gPSAiaWVrZXlkb3duIjsKCQl9CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCQl0aGlzLnBsYWNlaG9sZGVyQ2hhbmdlZCgpOwoJCS8vIHByZXZlbnQgb3ZlcnJpZGluZyBhIGN1c3RvbSBhdHRyaWJ1dGUgd2l0aCBudWxsCgkJaWYgKHRoaXMudHlwZSkgewoJCQl0aGlzLnR5cGVDaGFuZ2VkKCk7CgkJfQoJCXRoaXMudmFsdWVDaGFuZ2VkKCk7Cgl9LAoJcmVuZGVyZWQ6IGZ1bmN0aW9uKCkgewoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7CgoJCWVueW8ubWFrZUJ1YmJsZSh0aGlzLCAiZm9jdXMiLCAiYmx1ciIpOwoKCQkvL0ZvcmNlIG9uY2hhbmdlIGV2ZW50IHRvIGJlIGJ1YmJsZWQgaW5zaWRlIEVueW8gZm9yIElFOAoJCWlmKGVueW8ucGxhdGZvcm0uaWUgPT0gOCl7CgkJCXRoaXMuc2V0QXR0cmlidXRlKCJvbmNoYW5nZSIsIGVueW8uYnViYmxlcik7CgkJfQoKCQl0aGlzLmRpc2FibGVkQ2hhbmdlZCgpOwoJCWlmICh0aGlzLmRlZmF1bHRGb2N1cykgewoJCQl0aGlzLmZvY3VzKCk7CgkJfQoJfSwKCXR5cGVDaGFuZ2VkOiBmdW5jdGlvbigpIHsKCQl0aGlzLnNldEF0dHJpYnV0ZSgidHlwZSIsIHRoaXMudHlwZSk7Cgl9LAoJcGxhY2Vob2xkZXJDaGFuZ2VkOiBmdW5jdGlvbigpIHsKCQl0aGlzLnNldEF0dHJpYnV0ZSgicGxhY2Vob2xkZXIiLCB0aGlzLnBsYWNlaG9sZGVyKTsKCX0sCglkaXNhYmxlZENoYW5nZWQ6IGZ1bmN0aW9uKCkgewoJCXRoaXMuc2V0QXR0cmlidXRlKCJkaXNhYmxlZCIsIHRoaXMuZGlzYWJsZWQpOwoJCXRoaXMuYnViYmxlKCJvbkRpc2FibGVkQ2hhbmdlIik7Cgl9LAoJdmFsdWVDaGFuZ2VkOiBmdW5jdGlvbigpIHsKCQl0aGlzLnNldEF0dHJpYnV0ZSgidmFsdWUiLCB0aGlzLnZhbHVlKTsKCQlpZiAodGhpcy5nZXROb2RlUHJvcGVydHkoInZhbHVlIiwgdGhpcy52YWx1ZSkgIT09IHRoaXMudmFsdWUpIHsKCQkJdGhpcy5zZXROb2RlUHJvcGVydHkoInZhbHVlIiwgdGhpcy52YWx1ZSk7CgkJfQoJfSwKCWlla2V5dXA6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJdmFyIGllID0gZW55by5wbGF0Zm9ybS5pZSwga2MgPSBpbkV2ZW50LmtleUNvZGU7CgkJLy8gaW5wdXQgZXZlbnQgbWlzc2luZyBvbiBpZSA4LCBmYWlscyB0byBmaXJlIG9uIGJhY2tzcGFjZSBhbmQgZGVsZXRlIGtleXMgaW4gaWUgOQoJCWlmIChpZSA8PSA4IHx8IChpZSA9PSA5ICYmIChrYyA9PSA4IHx8IGtjID09IDQ2KSkpIHsKCQkJdGhpcy5idWJibGUoIm9uaW5wdXQiLCBpbkV2ZW50KTsKCQl9Cgl9LAoJaWVrZXlkb3duOiBmdW5jdGlvbihpblNlbmRlciwgaW5FdmVudCkgewoJCXZhciB3cCA9IGVueW8ucGxhdGZvcm0ud2luZG93c1Bob25lLCBrYyA9IGluRXZlbnQua2V5Q29kZSwgZHQgPSBpbkV2ZW50LmRpc3BhdGNoVGFyZ2V0OwoJCS8vIG9uY2hhbmdlIGV2ZW50IGZhaWxzIHRvIGZpcmUgb24gZW50ZXIga2V5IGZvciBXaW5kb3dzIFBob25lIDgsIHNvIHdlIGZvcmNlIGJsdXIKCQlpZiAod3AgPD0gOCAmJiBrYyA9PSAxMyAmJiB0aGlzLnRhZyA9PSAiaW5wdXQiICYmIGR0Lmhhc05vZGUoKSkgewoJCQlkdC5ub2RlLmJsdXIoKTsKCQl9Cgl9LAoJY2xlYXI6IGZ1bmN0aW9uKCkgewoJCXRoaXMuc2V0VmFsdWUoIiIpOwoJfSwKCS8vIG5vdGU6IHdlIGRpc2FsbG93IGRyYWdnaW5nIG9mIGFuIGlucHV0IHRvIGFsbG93IHRleHQgc2VsZWN0aW9uIG9uIGFsbCBwbGF0Zm9ybXMKCWRyYWdzdGFydDogZnVuY3Rpb24oKSB7CgkJcmV0dXJuIHRoaXMuaGFzRm9jdXMoKTsKCX0sCglmb2N1c2VkOiBmdW5jdGlvbigpIHsKCQlpZiAodGhpcy5zZWxlY3RPbkZvY3VzKSB7CgkJCWVueW8uYXN5bmNNZXRob2QodGhpcywgInNlbGVjdENvbnRlbnRzIik7CgkJfQoJfSwKCXNlbGVjdENvbnRlbnRzOiBmdW5jdGlvbigpIHsKCQl2YXIgbiA9IHRoaXMuaGFzTm9kZSgpOwoKCQlpZiAobiAmJiBuLnNldFNlbGVjdGlvblJhbmdlKSB7CgkJCW4uc2V0U2VsZWN0aW9uUmFuZ2UoMCwgbi52YWx1ZS5sZW5ndGgpOwoJCX0gZWxzZSBpZiAobiAmJiBuLmNyZWF0ZVRleHRSYW5nZSkgewoJCQl2YXIgciA9IG4uY3JlYXRlVGV4dFJhbmdlKCk7CgkJCXIuZXhwYW5kKCJ0ZXh0ZWRpdCIpOwoJCQlyLnNlbGVjdCgpOwoJCX0KCX0sCglpbnB1dDogZnVuY3Rpb24oKSB7CgkJdmFyIHZhbCA9IHRoaXMuZ2V0Tm9kZVByb3BlcnR5KCJ2YWx1ZSIpOwoJCXRoaXMuc2V0VmFsdWUodmFsKTsKCX0KfSk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Media.js"
Content-Type: application/octet-stream; x-encoding=base64
/**
	_enyo.Media_ implements an HTML 5 Media element. It is not intended to 
	be used explicitly, as it is a base control for _enyo.Audio_ & _enyo.Video_

	Initialize an audio component as follows:

		{kind: "enyo.Audio", src: "http://www.w3schools.com/tags/horse.mp3"}
	
	To play the audio, call 'this.$.audio.play()'.

	To get a reference to the actual HTML 5 Media element, call
	'this.$.audio.hasNode()'.
*/
enyo.kind({
	name: "enyo.Media",
	//* @public
	published: {
		//* URL of the sound file to play, can be relative to the application HTML file
		src: "",
		//* if true, media will automatically start playback when loaded
		autoplay: false,
		//* The desired speed at which the media resource is to play
		defaultPlaybackRate: 1.0,
		//* The effective playback rate
		playbackRate: 1.0,
		//* Indicates how data should be preloaded, reflecting the preload HTML attribute (none, metadata, auto)
		preload: "none",
		//* if true, restart media from beginning when finished
		loop: false,
		//* If true, media volume will be muted
		muted: false,
		//* if true, show default media controls
		showControls: false,
		//* Current playback volume, as a number in the range from 0.0 to 1.0
		volume: 1.0
	},
	events: {
		//* Fires when element stops fetching media data before it is completely downloaded, but not due to an error
		onAbort: "",
		//* Fires when element can resume playback of the media data, but may need to stop for further buffering of content
		onCanPlay: "",
		//* Fires when element can resume playback of the media data without needing to stop for further buffering of content
		onCanPlayThrough: "",
		//* Fires when the duration attribute has been changed
		onDurationChange: "",
		//* Fires when networkState was previously not in the NETWORK_EMPTY state and has just switched to that state
		onEmptied: "",
		//* Fires when the media finishes normally
		onEnded: "",
		//* Fires when an error occurs while fetching media data
		onError: "",
		//* Fires when the media data is rendered
		onLoadedData: "",
		//* Fires when the media duration & dimensions of the media resource/text tracks are ready
		onLoadedMetaData: "",
		//* Fires when the media element begins looking for media data
		onLoadStart: "",
		//* Fires when playback is paused
		onPause: "",
		//* Fires when playback is no longer paused
		onPlay: "",
		//* Fires when playback is ready to start after having been paused or delayed due to lack of media data
		onPlaying: "",
		//* Fires when fetching media data
		onProgress: "",
		//* Fires when the _this.defaultPlaybackRate_ or _this.playbackRate_ properties have been updated
		onRateChange: "",
		//* Fires when the seeking IDL attribute changes to false
		onSeeked: "",
		//* Fires when the seeking IDL attribute changes to true
		onSeeking: "",
		//* Fires when media fetching is interrupted
		onStalled: "",
		//* Fires when the media controller position changes
		onTimeUpdate: "",
		//* Fires when either the _this.volume_ or _this.muted_ properties is updated
		onVolumeChange: "",
		//* Fires when playback has stopped because the next frame is not available, but is expected to be
		onWaiting: ""
	},
	//* @protected
	handlers: {
		onabort: "_abort",
		oncanplay: "_canPlay",
		oncanplaythrough: "_canPlayThrough",
		ondurationchange: "_durationChange",
		onemptied: "_emptied",
		onended: "_ended",
		onerror: "_error",
		onloadeddata: "_loadedData",
		onloadedmetadata: "_loadedMetaData",
		onloadstart: "_loadStart",
		onpause: "_pause",
		onplay: "_play",
		onplaying: "_playing",
		onprogress: "_progress",
		onratechange: "_rateChange",
		onseeked: "_seeked",
		onseeking: "_seeking",
		onstalled: "_stalled",
		ontimeupdate: "_timeUpdate",
		onvolumechange: "_volumeChange",
		onwaiting: "_waiting"
	},
	create: function() {
		this.inherited(arguments);
		this.autoplayChanged();
		this.loopChanged();
		this.preloadChanged();
		this.showControlsChanged();
		this.srcChanged();
	},
	rendered: function() {
		this.inherited(arguments);
		enyo.makeBubble(this, "abort");
		enyo.makeBubble(this, "canplay");
		enyo.makeBubble(this, "canplaythrough");
		enyo.makeBubble(this, "durationchange");
		enyo.makeBubble(this, "emptied");
		enyo.makeBubble(this, "ended");
		enyo.makeBubble(this, "error");
		enyo.makeBubble(this, "loadeddata");
		enyo.makeBubble(this, "loadedmetadata");
		enyo.makeBubble(this, "loadstart");
		enyo.makeBubble(this, "pause");
		enyo.makeBubble(this, "play");
		enyo.makeBubble(this, "playing");
		enyo.makeBubble(this, "progress");
		enyo.makeBubble(this, "ratechange");
		enyo.makeBubble(this, "seeked");
		enyo.makeBubble(this, "seeking");
		enyo.makeBubble(this, "stalled");
		enyo.makeBubble(this, "timeupdate");
		enyo.makeBubble(this, "volumechange");
		enyo.makeBubble(this, "waiting");
		this.defaultPlaybackRateChanged();
		this.mutedChanged();
		this.playbackRateChanged();
		this.volumeChanged();
	},
	srcChanged: function() {
		var path = enyo.path.rewrite(this.src);
		this.setAttribute("src", path);
		if (this.hasNode()) {
			this.node.load();
		}
	},
	autoplayChanged: function() {
		this.setAttribute("autoplay", this.autoplay ? "autoplay" : null);
	},
	loopChanged: function() {
		this.setAttribute("loop", this.loop ? "loop" : null);
	},
	mutedChanged: function() {
		this.setAttribute("muted", this.muted ? "muted" : null);
	},
	preloadChanged: function() {
		this.setAttribute("preload", this.preload);
	},
	defaultPlaybackRateChanged: function() {
		if (this.hasNode()) {
			this.node.defaultPlaybackRate = this.defaultPlaybackRate;
		}
	},
	playbackRateChanged: function() {
		if (this.hasNode()) {
			this.node.playbackRate = this.playbackRate;
		}
	},
	showControlsChanged: function() {
		this.setAttribute("controls", this.showControls ? "controls" : null);
	},
	volumeChanged: function() {
		if (this.hasNode()) {
			this.node.volume = this.volume;
		}
	},
	//* When element stops fetching media data before it is completely downloaded, but not due to an error
	_abort: function() {
		this.doEnded();
	},
	//* When element can resume playback of the media data, but may need to stop for further buffering of content
	_canPlay: function() {
		this.doCanPlay();
	},
	//* When element can resume playback of the media data without needing to stop for further buffering of content
	_canPlayThrough: function() {
		this.doCanPlayThrough();
	},
	//* When the duration attribute has been changed
	_durationChange: function() {
		this.doDurationChange();
	},
	//* When networkState was previously not in the NETWORK_EMPTY state and has just switched to that state
	_emptied: function() {
		this.doEmptied();
	},
	//* When audio playback reaches the end of the media
	_ended: function() {
		this.doEnded();
	},
	//* When error occurs while fetching media data
	_error: function() {
		this.doError();
	},
	//* When we can render the media data at the current playback position for the first time
	_loadedData: function() {
		this.doLoadedData();
	},
	//* When the media duration & dimensions of the media resource/text tracks are ready
	_loadedMetaData: function() {
		this.doLoadedMetaData();
	},
	//* When the media element begins looking for media data
	_loadStart: function() {
		this.doLoadStart();
	},
	//* When playback is paused
	_pause: function() {
		this.doPause();
	},
	//* When playback is no longer paused
	_play: function() {
		this.doPlay();
	},
	//* When playback is ready to start after having been paused or delayed due to lack of media data
	_playing: function() {
		this.doPlaying();
	},
	//* When the media element is fetching media data
	_progress: function() {
		this.doProgress();
	},
	//* When the _this.defaultPlaybackRate_ or _this.playbackRate_ properties have been updated
	_rateChange: function() {
		this.doRateChange();
	},
	//* When fetching media data is interrupted
	_stalled: function() {
		this.doStalled();
	},
	//* When the seeking IDL attribute changes to false
	_seeked: function() {
		this.doSeeked();
	},
	//* When the seeking IDL attribute changes to true
	_seeking: function() {
		this.doSeeking();
	},
	//* When the media controller position has changed
	_timeUpdate: function() {
		this.doTimeUpdate();
	},
	//* When either the _this.volume_ or _this.muted_ properties is updated
	_volumeChange: function() {
		this.doVolumeChange();
	},
	//* When playback has stopped because the next frame is not available, but is expected to be
	_waiting: function() {
		this.doWaiting();
	},
	//* @public
	//* Initiates playback of the audio referenced in _this.src_
	play: function() {
		if (this.hasNode()) {
			this.node.play();
		}
	},
	//* Pauses the audio playback
	pause: function() {
		if (this.hasNode()) {
			this.node.pause();
		}
	},
	//* Seeks to the specified value of _inValue_ (in seconds)
	seekTo: function(inValue) {
		if (this.hasNode()) {
			this.node.currentTime = inValue;
		}
	},
	//* Returns the ranges of the media source that has buffered
	getBuffered: function() {
		if (this.hasNode()) {
			return this.node.buffered;
		}
		return 0;
	},
	//* Returns the current playback position (in seconds)
	getCurrentTime: function() {
		if (this.hasNode()) {
			return this.node.currentTime;
		}
		return 0;
	},
	//* Returns the total duration time of the loaded media (in seconds)
	getDuration: function() {
		if (this.hasNode()) {
			return this.node.duration;
		}
		return 0;
	},
	//* Represents whether the media element is paused or not
	getPaused: function() {
		if (this.hasNode()) {
			return this.node.paused;
		}
	},
	//* Returns the ranges of the media source that have been played, if any
	getPlayed: function() {
		if (this.hasNode()) {
			return this.node.played;
		}
	},
	//* Returns the readiness state of the media
	getReadyState: function() {
		if (this.hasNode()) {
			return this.node.readyState;
		}
	},
	//* Returns the ranges of the media source the user is able to seek to, if any
	getSeekable: function() {
		if (this.hasNode()) {
			return this.node.seekable;
		}
	},
	//* Indicates whether the media is currently seeking to a new position
	getSeeking: function() {
		if (this.hasNode()) {
			return this.node.seeking;
		}
	}
});

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Popup.js"
Content-Type: application/octet-stream; x-encoding=base64
/**
	_enyo.Popup_ is a control used to display certain content on top of other
	content.

	Popups are initially hidden on creation; they can be shown by calling the
	_show_ method and re-hidden by calling _hide_.  Popups may be centered using
	the	_centered_ property; if not centered, they should be given a specific
	position.

	A popup may be optionally floated above all application content by setting
	its _floating_ property to _true_.  This has the advantage of guaranteeing
	that the popup will be displayed on top of other content.  This usage is
	appropriate when the popup does not need to scroll along with other content.

	For more information, see the documentation on
	[Popups](https://github.com/enyojs/enyo/wiki/Popups) in the Enyo Developer
	Guide.
 */
enyo.kind({
	name: "enyo.Popup",
	classes: "enyo-popup enyo-no-touch-action",
	published: {
		//* Set to true to prevent controls outside the popup from receiving
		//* events while the popup is showing
		modal: false,
		//* By default, the popup will hide when the user taps outside it or
		//* presses ESC.  Set to false to prevent this behavior.
		autoDismiss: true,
		//* Set to true to render the popup in a floating layer outside of other
		//* controls.  This can be used to guarantee that the popup will be
		//* shown on top of other controls.
		floating: false,
		//* Set to true to automatically center the popup in the middle of the viewport
		centered: false,
		//* Set to true to be able to show transition on the style modifications otherwise
		//* the transition is invisible (visibility: hidden)
		showTransitions: false
	},
	//* @protected
	showing: false,
	handlers: {
		ondown: "down",
		onkeydown: "keydown",
		ondragstart: "dragstart",
		onfocus: "focus",
		onblur: "blur",
		onRequestShow: "requestShow",
		onRequestHide: "requestHide"
	},
	captureEvents: true,
	//* @public
	events: {
		//* Fires after the popup is shown.
		onShow: "",
		//* Fires after the popup is hidden.
		onHide: ""
	},
	//* @protected
	tools: [
		{kind: "Signals", onKeydown: "keydown"}
	],
	create: function() {
		this.inherited(arguments);
		this.canGenerate = !this.floating;
	},
	render: function() {
		if (this.floating) {
			if (!enyo.floatingLayer.hasNode()) {
				enyo.floatingLayer.render();
			}
			this.parentNode = enyo.floatingLayer.hasNode();
		}
		this.inherited(arguments);
	},
	destroy: function() {
		if (this.showing) {
			this.release();
		}
		this.inherited(arguments);
	},

	reflow: function() {
		this.updatePosition();
		this.inherited(arguments);
	},
	calcViewportSize: function() {
		if (window.innerWidth) {
			return {
				width: window.innerWidth,
				height: window.innerHeight
			};
		} else {
			var e = document.documentElement;
			return {
				width: e.offsetWidth,
				height: e.offsetHeight
			};
		}
	},
	updatePosition: function() {
		var d = this.calcViewportSize();
		var b = this.getBounds();

		if (this.targetPosition) {
			// For brevity's sake...
			var p = this.targetPosition;

			// Test and optionally adjust our target bounds (only first is commented, because logic is effectively identical for all scenarios)
			if (typeof p.left === 'number') {
				// If popup will be outside window bounds, switch anchor
				if (p.left + b.width > d.width) {
					if (p.left - b.width >= 0) {
						// Switching to right corner will fit in window
						p.right = d.width - p.left;
					} else {
						// Neither corner will work; stick at side of window
						p.right = 0;
					}
					p.left = null;
				} else {
					p.right = null;
				}
			} else if (typeof p.right === 'number') {
				if (p.right + b.width > d.width) {
					if (p.right - b.width >= 0) {
						p.left = d.width - p.right;
					} else {
						p.left = 0;
					}
					p.right = null;
				} else {
					p.left = null;
				}
			}

			if (typeof p.top === 'number') {
				if (p.top + b.height > d.height) {
					if (p.top - b.height >= 0) {
						p.bottom = d.height - p.top;
					} else {
						p.bottom = 0;
					}
					p.top = null;
				} else {
					p.bottom = null;
				}
			} else if (typeof p.bottom === 'number') {
				if (p.bottom + b.height > d.height) {
					if (p.bottom - b.height >= 0) {
						p.top = d.height - p.bottom;
					} else {
						p.top = 0;
					}
					p.bottom = null;
				} else {
					p.top = null;
				}
			}

			// 'initial' values are necessary to override positioning rules in the CSS
			this.addStyles('left: ' + (p.left !== null ? p.left + 'px' : 'initial') + '; right: ' + (p.right !== null ? p.right + 'px' : 'initial') + '; top: ' + (p.top !== null ? p.top + 'px' : 'initial') + '; bottom: ' + (p.bottom !== null ? p.bottom + 'px' : 'initial') + ';');
		} else if (this.centered) {
			this.addStyles( "top: " + Math.max( ( ( d.height - b.height ) / 2 ), 0 ) + "px; left: " + Math.max( ( ( d.width - b.width ) / 2 ), 0 ) + "px;" );
		}
	},
	showingChanged: function() {
		// auto render when shown.
		if (this.floating && this.showing && !this.hasNode()) {
			this.render();
		}
		// hide while sizing, and move to top corner for accurate sizing
		if (this.centered || this.targetPosition) {
			if (!this.showTransitions) {
				this.applyStyle("visibility", "hidden");
			}
			this.addStyles("top: 0px; left: 0px; right: initial; bottom: initial;");
		}
		this.inherited(arguments);
		if (this.showing) {
			this.resized();
			if (this.captureEvents) {
				this.capture();
			}
		} else {
			if (this.captureEvents) {
				this.release();
			}
		}
		// show after sizing
		if (this.centered || this.targetPosition && !this.showTransitions) {
			this.applyStyle("visibility", null);
		}
		// events desired due to programmatic show/hide
		if (this.hasNode()) {
			this[this.showing ? "doShow" : "doHide"]();
		}
	},
	capture: function() {
		enyo.dispatcher.capture(this, !this.modal);
	},
	release: function() {
		enyo.dispatcher.release(this);
	},
	down: function(inSender, inEvent) {
		//record the down event to verify in tap
		this.downEvent = inEvent;

		// prevent focus from shifting outside the popup when modal.
		if (this.modal && !inEvent.dispatchTarget.isDescendantOf(this)) {
			inEvent.preventDefault();
		}
	},
	tap: function(inSender, inEvent) {
		// dismiss on tap if property is set and click started & ended outside the popup
		if (this.autoDismiss && (!inEvent.dispatchTarget.isDescendantOf(this)) && this.downEvent &&
			(!this.downEvent.dispatchTarget.isDescendantOf(this))) {
			this.downEvent = null;
			this.hide();
			return true;
		}
	},
	// if a drag event occurs outside a popup, hide
	dragstart: function(inSender, inEvent) {
		var inScope = (inEvent.dispatchTarget === this || inEvent.dispatchTarget.isDescendantOf(this));
		if (inSender.autoDismiss && !inScope) {
			inSender.setShowing(false);
		}
		return true;
	},
	keydown: function(inSender, inEvent) {
		if (this.showing && this.autoDismiss && inEvent.keyCode == 27 /* escape */) {
			this.hide();
		}
	},
	// If something inside the popup blurred, keep track of it.
	blur: function(inSender, inEvent) {
		if (inEvent.dispatchTarget.isDescendantOf(this)) {
			this.lastFocus = inEvent.originator;
		}
	},
	// When something outside the popup focuses (e.g., due to tab key), focus our last focused control.
	focus: function(inSender, inEvent) {
		var dt = inEvent.dispatchTarget;
		if (this.modal && !dt.isDescendantOf(this)) {
			if (dt.hasNode()) {
				dt.node.blur();
			}
			var n = (this.lastFocus && this.lastFocus.hasNode()) || this.hasNode();
			if (n) {
				n.focus();
			}
		}
	},
	requestShow: function(inSender, inEvent) {
		this.show();
		return true;
	},
	requestHide: function(inSender, inEvent) {
		this.hide();
		return true;
	},

	//* @public
	/**
		Open at the location of a mouse event (_inEvent_). The popup's
		position is automatically constrained so that it does not
		display outside the viewport, and defaults to anchoring the top
		left corner of the popup to the mouse event.

		_inOffset_ is an optional object which may contain left and top
		properties to specify an offset relative to the location the
		popup would otherwise be positioned.
	*/
	showAtEvent: function(inEvent, inOffset) {
		// Calculate our ideal target based on the event position and offset
		var p = {
			left: inEvent.centerX || inEvent.clientX || inEvent.pageX,
			top: inEvent.centerY || inEvent.clientY || inEvent.pageY
		};
		if (inOffset) {
			p.left += inOffset.left || 0;
			p.top += inOffset.top || 0;
		}

		this.showAtPosition(p);
	},

	/**
		Open the popup at a specific position. The final location
		of the popup will be automatically constrained so that it does not
		display outside the viewport.

		_inPosition_ is an object which may contain left, top, bottom,
		and right properties to specify where the popup will be anchored.
		If both left and right are included, the popup will preference left
		(same for top vs. bottom).
	*/
	showAtPosition: function(inPosition) {
		// Save our target position for later processing
		this.targetPosition = inPosition;

		// Show the dialog
		this.show();
	}
});

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Repeater.js"
Content-Type: application/octet-stream; x-encoding=base64
77u/LyoqCglfZW55by5SZXBlYXRlcl8gaXMgYSBzaW1wbGUgY29udHJvbCBmb3IgbWFraW5nIGxpc3RzIG9mIGl0ZW1zLgoKCVRoZSBjb21wb25lbnRzIG9mIGEgcmVwZWF0ZXIgYXJlIGNvcGllZCBmb3IgZWFjaCBpdGVtIGNyZWF0ZWQsIGFuZCBhcmUKCXdyYXBwZWQJaW4gYSBjb250cm9sIHRoYXQga2VlcHMgdGhlIHN0YXRlIG9mIHRoZSBpdGVtIGluZGV4LgoKCUV4YW1wbGU6CgoJCXtraW5kOiAiUmVwZWF0ZXIiLCBjb3VudDogMiwgb25TZXR1cEl0ZW06ICJzZXRJbWFnZVNvdXJjZSIsIGNvbXBvbmVudHM6IFsKCQkJe2tpbmQ6ICJJbWFnZSJ9CgkJXX0KCgkJc2V0SW1hZ2VTb3VyY2U6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJCXZhciBpbmRleCA9IGluRXZlbnQuaW5kZXg7CgkJCXZhciBpdGVtID0gaW5FdmVudC5pdGVtOwoJCQlpdGVtLiQuaW1hZ2Uuc2V0U3JjKHRoaXMuaW1hZ2VTb3VyY2VzW2luZGV4XSk7CgkJCXJldHVybiB0cnVlOwoJCX0KCglCZSBzdXJlIHRvIHJldHVybiBfdHJ1ZV8gZnJvbSB5b3VyIF9vblNldHVwSXRlbV8gaGFuZGxlciB0byBhdm9pZCBoYXZpbmcKCW90aGVyIGV2ZW50IGhhbmRsZXJzIGZ1cnRoZXIgdXAgdGhlIHRyZWUgdHJ5IHRvIG1vZGlmeSB5b3VyIGl0ZW0gY29udHJvbC4KCglGb3IgbW9yZSBpbmZvcm1hdGlvbiwgc2VlIHRoZSBkb2N1bWVudGF0aW9uIG9uCglbTGlzdHNdKGh0dHBzOi8vZ2l0aHViLmNvbS9lbnlvanMvZW55by93aWtpL0xpc3RzKSBpbiB0aGUgRW55byBEZXZlbG9wZXIKCUd1aWRlLgoqLwplbnlvLmtpbmQoewoJbmFtZTogImVueW8uUmVwZWF0ZXIiLAoJcHVibGlzaGVkOiB7CgkJLy8qIE51bWJlciBvZiBpdGVtcwoJCWNvdW50OiAwCgl9LAoJZXZlbnRzOiB7CgkJLyoqCgkJCUZpcmVzIHdoZW4gZWFjaCBpdGVtIGlzIGNyZWF0ZWQuCgoJCQlfaW5FdmVudC5pbmRleF8gY29udGFpbnMgdGhlIGl0ZW0ncyBpbmRleC4KCgkJCV9pbkV2ZW50Lml0ZW1fIGNvbnRhaW5zIHRoZSBpdGVtIGNvbnRyb2wsIGZvciBkZWNvcmF0aW9uLgoJCSovCgkJb25TZXR1cEl0ZW06ICIiCgl9LAoJY3JlYXRlOiBmdW5jdGlvbigpIHsKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCXRoaXMuY291bnRDaGFuZ2VkKCk7Cgl9LAoJLy8qIEBwcm90ZWN0ZWQKCWluaXRDb21wb25lbnRzOiBmdW5jdGlvbigpIHsKCQl0aGlzLml0ZW1Db21wb25lbnRzID0gdGhpcy5jb21wb25lbnRzIHx8IHRoaXMua2luZENvbXBvbmVudHM7CgkJdGhpcy5jb21wb25lbnRzID0gdGhpcy5raW5kQ29tcG9uZW50cyA9IG51bGw7CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCX0sCgljb3VudENoYW5nZWQ6IGZ1bmN0aW9uKCkgewoJCXRoaXMuYnVpbGQoKTsKCX0sCglpdGVtQXRJbmRleDogZnVuY3Rpb24oaW5JbmRleCkgewoJCXJldHVybiB0aGlzLmNvbnRyb2xBdEluZGV4KGluSW5kZXgpOwoJfSwKCS8vKiBAcHVibGljCgkvKiogUmVuZGVycyB0aGUgY29sbGVjdGlvbiBvZiBpdGVtcy4gVGhpcyB3aWxsIGRlbGV0ZSBhbnkgZXhpc3RpbmcgaXRlbXMgYW5kCgkJcmVjcmVhdGUgdGhlIHJlcGVhdGVyIGlmIGNhbGxlZCBhZnRlciB0aGUgcmVwZWF0ZXIgaGFzIGJlZW4gcmVuZGVyZWQuCgkJVGhpcyBpcyBjYWxsZWQgYXV0b21hdGljYWxseSBpZiBfc2V0Q291bnRfIGlzIGNhbGxlZCwgZXZlbiBpZiB0aGUgY291bnQKCQlyZW1haW5zIHRoZSBzYW1lLgoJKi8KCWJ1aWxkOiBmdW5jdGlvbigpIHsKCQl0aGlzLmRlc3Ryb3lDbGllbnRDb250cm9scygpOwoJCWZvciAodmFyIGk9MCwgYzsgaTx0aGlzLmNvdW50OyBpKyspIHsKCQkJYyA9IHRoaXMuY3JlYXRlQ29tcG9uZW50KHtraW5kOiAiZW55by5Pd25lclByb3h5IiwgaW5kZXg6IGl9KTsKCQkJLy8gZG8gdGhpcyBhcyBhIHNlY29uZCBzdGVwIHNvICdjJyBpcyB0aGUgb3duZXIgb2YgdGhlIGNyZWF0ZWQgY29tcG9uZW50cwoJCQljLmNyZWF0ZUNvbXBvbmVudHModGhpcy5pdGVtQ29tcG9uZW50cyk7CgkJCS8vIGludm9rZSB1c2VyJ3Mgc2V0dXAgY29kZQoJCQl0aGlzLmRvU2V0dXBJdGVtKHtpbmRleDogaSwgaXRlbTogY30pOwoJCX0KCQl0aGlzLnJlbmRlcigpOwoJfSwKCS8qKgoJCVJlbmRlcnMgYSBzcGVjaWZpYyBpdGVtIGluIHRoZSBjb2xsZWN0aW9uLiBUaGlzIGRvZXMgbm90IGRlc3Ryb3kgdGhlCgkJaXRlbSwgYnV0IGp1c3QgY2FsbHMgdGhlIF9vblNldHVwSXRlbV8gZXZlbnQgaGFuZGxlciBhZ2FpbiBmb3IgaXQsIHNvCgkJYW55IHN0YXRlIHN0b3JlZCBpbgl0aGUgaXRlbSBpcyBwcmVzZXJ2ZWQuCgkqLwoJcmVuZGVyUm93OiBmdW5jdGlvbihpbkluZGV4KSB7CgkJdmFyIGMgPSB0aGlzLml0ZW1BdEluZGV4KGluSW5kZXgpOwoJCXRoaXMuZG9TZXR1cEl0ZW0oe2luZGV4OiBpbkluZGV4LCBpdGVtOiBjfSk7Cgl9Cn0pOwoKLy8gU29tZXRpbWVzIGNsaWVudCBjb250cm9scyBhcmUgaW50ZXJtZWRpYXRlZCB3aXRoIG51bGwtY29udHJvbHMuCi8vIFRoZXNlIG92ZXJyaWRlcyByZXJvdXRlIGV2ZW50cyBmcm9tIHN1Y2ggY29udHJvbHMgdG8gdGhlIG5vbWluYWwgZGVsZWdhdGUsCi8vIGFzIHdvdWxkIGhhcHBlbiBpbiB0aGUgYWJzZW5jZSBvZiBpbnRlcm1lZGlhdGlvbi4KZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLk93bmVyUHJveHkiLAoJdGFnOiBudWxsLAoJZGVjb3JhdGVFdmVudDogZnVuY3Rpb24oaW5FdmVudE5hbWUsIGluRXZlbnQsIGluU2VuZGVyKSB7CgkJaWYgKGluRXZlbnQpIHsKCQkJLy8gcHJlc2VydmUgYW4gZXhpc3RpbmcgaW5kZXggcHJvcGVydHkuCgkJCWlmIChlbnlvLmV4aXN0cyhpbkV2ZW50LmluZGV4KSkgewoJCQkJLy8gaWYgdGhlcmUgYXJlIG5lc3RlZCBpbmRpY2VzLCBzdG9yZSBhbGwgb2YgdGhlbSBpbiBhbiBhcnJheQoJCQkJLy8gYnV0IGxlYXZlIHRoZSBpbm5lcm1vc3Qgb25lIGluIHRoZSBpbmRleCBwcm9wZXJ0eQoJCQkJaW5FdmVudC5pbmRpY2VzID0gaW5FdmVudC5pbmRpY2VzIHx8IFtpbkV2ZW50LmluZGV4XTsKCQkJCWluRXZlbnQuaW5kaWNlcy5wdXNoKHRoaXMuaW5kZXgpOwoJCQl9IGVsc2UgewoJCQkJLy8gZm9yIGEgc2luZ2xlIGxldmVsLCBqdXN0IGRlY29yYXRlIHRoZSBpbmRleCBwcm9wZXJ0eQoJCQkJaW5FdmVudC5pbmRleCA9IHRoaXMuaW5kZXg7CgkJCX0KCQkJLy8gdXBkYXRlIGRlbGVnYXRlIGR1cmluZyBidWJibGluZyB0byBhY2NvdW50IGZvciBwcm94eQoJCQkvLyBieSBtb3ZpbmcgdGhlIGRlbGVnYXRlIHVwIHRvIHRoZSByZXBlYXRlciBsZXZlbAoJCQlpZiAoaW5FdmVudC5kZWxlZ2F0ZSAmJiBpbkV2ZW50LmRlbGVnYXRlLm93bmVyID09PSB0aGlzKSB7CgkJCQlpbkV2ZW50LmRlbGVnYXRlID0gdGhpcy5vd25lcjsKCQkJfQoJCX0KCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJfSwKCS8vIGV4dGVuZGluZyBlbnlvLkNvbXBvbmVudC5kZWxlZ2F0ZUV2ZW50CglkZWxlZ2F0ZUV2ZW50OiBmdW5jdGlvbihpbkRlbGVnYXRlLCBpbk5hbWUsIGluRXZlbnROYW1lLCBpbkV2ZW50LCBpblNlbmRlcikgewoJCWlmIChpbkRlbGVnYXRlID09IHRoaXMpIHsKCQkJaW5EZWxlZ2F0ZSA9IHRoaXMub3duZXIub3duZXI7CgkJfQoJCXJldHVybiB0aGlzLmluaGVyaXRlZChhcmd1bWVudHMsIFtpbkRlbGVnYXRlLCBpbk5hbWUsIGluRXZlbnROYW1lLCBpbkV2ZW50LCBpblNlbmRlcl0pOwoJfQp9KTs=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/RichText.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5SaWNoVGV4dF8gaXMgYSBtdWx0aS1saW5lIHRleHQgaW5wdXQgdGhhdCBzdXBwb3J0cyByaWNoIGZvcm1hdHRpbmcsCglzdWNoIGFzIGJvbGQsIGl0YWxpY3MsIGFuZCB1bmRlcmxpbmluZy4KCglUaGUgY29udGVudCBkaXNwbGF5ZWQgaW4gYSBSaWNoVGV4dCBtYXkgYmUgYWNjZXNzZWQgYXQgcnVudGltZSB2aWEgdGhlCglgZ2V0VmFsdWVgIGFuZCBgc2V0VmFsdWVgIG1ldGhvZHMuCgoJRm9yIG1vcmUgaW5mb3JtYXRpb24sIHNlZSB0aGUgZG9jdW1lbnRhdGlvbiBvbgoJW1RleHQgRmllbGRzXShodHRwczovL2dpdGh1Yi5jb20vZW55b2pzL2VueW8vd2lraS9UZXh0LUZpZWxkcykgaW4gdGhlIEVueW8KCURldmVsb3BlciBHdWlkZS4KKi8KZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLlJpY2hUZXh0IiwKCWNsYXNzZXM6ICJlbnlvLXJpY2h0ZXh0IGVueW8tc2VsZWN0YWJsZSIsCglwdWJsaXNoZWQ6IHsKCQkvKioKCQkJX2FsbG93SHRtbF8gaXMgZW5hYmxlZCBieSBkZWZhdWx0IGluIFJpY2hUZXh0IHRvIHRha2UgYWR2YW50YWdlIG9mCgkJCWFsbCB0aGUgcmljaCBlZGl0aW5nIHByb3BlcnRpZXMuIEhvd2V2ZXIsIHRoaXMgYWxsb3dzIGZvciAqKkFOWSoqCgkJCUhUTUwgdG8gYmUgaW5zZXJ0ZWQgaW50byB0aGUgUmljaFRleHQsIGluY2x1ZGluZyBfaWZyYW1lXyBhbmQKCQkJX3NjcmlwdF8gdGFncywgd2hpY2ggY2FuIGJlIGEgc2VjdWl0eSBjb25jZXJuIGluIHNvbWUgc2l0dWF0aW9ucy4KCQkJSWYgc2V0IHRvIGZhbHNlLCBpbnNlcnRlZCBIVE1MIHdpbGwgYmUgZXNjYXBlZC4KCQkqLwoJCWFsbG93SHRtbDogdHJ1ZSwKCQkvLyogSWYgdHJ1ZSwgdGhlIFJpY2hUZXh0IHdpbGwgbm90IGFjY2VwdCBpbnB1dCBvciBnZW5lcmF0ZSBldmVudHMKCQlkaXNhYmxlZDogZmFsc2UsCgkJLy8qIFZhbHVlIG9mIHRoZSB0ZXh0IGZpZWxkCgkJdmFsdWU6ICIiCgl9LAoJLy8qIFNldCB0byB0cnVlIHRvIGZvY3VzIHRoaXMgY29udHJvbCB3aGVuIGl0IGlzIHJlbmRlcmVkLgoJZGVmYXVsdEZvY3VzOiBmYWxzZSwKCS8vKiBAcHJvdGVjdGVkCglwcm90ZWN0ZWRTdGF0aWNzOiB7CgkJb3NJbmZvOiBbCgkJCXtvczogImFuZHJvaWQiLCB2ZXJzaW9uOiAzfSwKCQkJe29zOiAiaW9zIiwgdmVyc2lvbjogNX0KCQldLAoJCS8vKiBSZXR1cm5zIHRydWUgaWYgdGhlIHBsYXRmb3JtIGhhcyBjb250ZW50ZWRpdGFibGUgYXR0cmlidXRlLgoJCWhhc0NvbnRlbnRFZGl0YWJsZTogZnVuY3Rpb24oKSB7CgkJCWZvciAodmFyIGk9MCwgdDsgKHQ9ZW55by5SaWNoVGV4dC5vc0luZm9baV0pOyBpKyspIHsKCQkJCWlmIChlbnlvLnBsYXRmb3JtW3Qub3NdIDwgdC52ZXJzaW9uKSB7CgkJCQkJcmV0dXJuIGZhbHNlOwoJCQkJfQoJCQl9CgkJCXJldHVybiB0cnVlOwoJCX0KCX0sCglraW5kOiAiZW55by5JbnB1dCIsCglhdHRyaWJ1dGVzOiB7CgkJY29udGVudGVkaXRhYmxlOiB0cnVlCgl9LAoJaGFuZGxlcnM6IHsKCQlvbmZvY3VzOiAiZm9jdXNIYW5kbGVyIiwKCQlvbmJsdXI6ICJibHVySGFuZGxlciIsCgkJb25rZXl1cDogInVwZGF0ZVZhbHVlIiwKCQlvbmN1dDogInVwZGF0ZVZhbHVlQXN5bmMiLAoJCW9ucGFzdGU6ICJ1cGRhdGVWYWx1ZUFzeW5jIiwKCQkvLyBwcmV2ZW50IG9uaW5wdXQgaGFuZGxlciBmcm9tIGJlaW5nIGNhbGxlZCBsb3dlciBpbiB0aGUgaW5oZXJpdGFuY2UgY2hhaW4KCQlvbmlucHV0OiBudWxsCgl9LAoJLy8gY3JlYXRlIFJpY2hUZXh0IGFzIGEgZGl2IGlmIHBsYXRmb3JtIGhhcyBjb250ZW50ZWRpdGFibGUgYXR0cmlidXRlLCBvdGhlcndpc2UgY3JlYXRlIGl0IGFzIGEgdGV4dGFyZWEKCWNyZWF0ZTogZnVuY3Rpb24oKSB7CgkJdGhpcy5zZXRUYWcoZW55by5SaWNoVGV4dC5oYXNDb250ZW50RWRpdGFibGUoKT8iZGl2IjoidGV4dGFyZWEiKTsKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJfSwKCS8vIHNpbXVsYXRlIG9uY2hhbmdlIGV2ZW50IHRoYXQgaW5wdXRzIGV4cG9zZQoJZm9jdXNIYW5kbGVyOiBmdW5jdGlvbigpIHsKCQl0aGlzLl92YWx1ZSA9IHRoaXMuZ2V0KCJ2YWx1ZSIpOwoJfSwKCWJsdXJIYW5kbGVyOiBmdW5jdGlvbigpIHsKCQlpZiAodGhpcy5fdmFsdWUgIT09IHRoaXMuZ2V0KCJ2YWx1ZSIpKSB7CgkJCXRoaXMuYnViYmxlKCJvbmNoYW5nZSIpOwoJCX0KCX0sCgl2YWx1ZUNoYW5nZWQ6IGZ1bmN0aW9uKCkgewoJCXZhciB2YWwgPSB0aGlzLmdldCgidmFsdWUiKTsKCQlpZiAodGhpcy5oYXNGb2N1cygpICYmIHZhbCAhPT0gdGhpcy5ub2RlLmlubmVySFRNTCkgewoJCQl0aGlzLnNlbGVjdEFsbCgpOwoJCQl0aGlzLmluc2VydEF0Q3Vyc29yKHZhbCk7CgkJfSBlbHNlIGlmKCF0aGlzLmhhc0ZvY3VzKCkpIHsKCQkJdGhpcy5zZXQoImNvbnRlbnQiLCB2YWwpOwoJCX0KCX0sCgl1cGRhdGVWYWx1ZTogZnVuY3Rpb24oKSB7CgkJdmFyIHZhbCA9IHRoaXMubm9kZS5pbm5lckhUTUw7CgkJdGhpcy5zZXQoInZhbHVlIiwgdmFsKTsKCX0sCgl1cGRhdGVWYWx1ZUFzeW5jOiBmdW5jdGlvbigpIHsKCQllbnlvLmFzeW5jTWV0aG9kKHRoaXMuYmluZFNhZmVseSgidXBkYXRlVmFsdWUiKSk7Cgl9LAoJLy8qIEBwdWJsaWMKCS8vKiBSZXR1cm5zIHRydWUgaWYgdGhlIFJpY2hUZXh0IGlzIGZvY3VzZWQuCgloYXNGb2N1czogZnVuY3Rpb24oKSB7CgkJaWYgKHRoaXMuaGFzTm9kZSgpKSB7CgkJCXJldHVybiBkb2N1bWVudC5hY3RpdmVFbGVtZW50ID09PSB0aGlzLm5vZGU7CgkJfQoJfSwKCS8qKgoJCVJldHVybnMgdGhlIHNlbGVjdGlvbiBvYmplY3QuCgkqLwoJZ2V0U2VsZWN0aW9uOiBmdW5jdGlvbigpIHsKCQlpZiAodGhpcy5oYXNGb2N1cygpKSB7CgkJCXJldHVybiB3aW5kb3cuZ2V0U2VsZWN0aW9uKCk7CgkJfQoJfSwKCS8vKiBSZW1vdmVzIHRoZSBzZWxlY3Rpb24gb2JqZWN0LgoJcmVtb3ZlU2VsZWN0aW9uOiBmdW5jdGlvbihpblN0YXJ0KSB7CgkJdmFyIHMgPSB0aGlzLmdldFNlbGVjdGlvbigpOwoJCWlmIChzKSB7CgkJCXNbaW5TdGFydCA/ICJjb2xsYXBzZVRvU3RhcnQiIDogImNvbGxhcHNlVG9FbmQiXSgpOwoJCX0KCX0sCgkvLyogTW9kaWZpZXMgdGhlIHNlbGVjdGlvbiBvYmplY3QuCgltb2RpZnlTZWxlY3Rpb246IGZ1bmN0aW9uKGluVHlwZSwgaW5EaXJlY3Rpb24sIGluQW1vdW50KSB7CgkJdmFyIHMgPSB0aGlzLmdldFNlbGVjdGlvbigpOwoJCWlmIChzKSB7CgkJCXMubW9kaWZ5KGluVHlwZSB8fCAibW92ZSIsIGluRGlyZWN0aW9uLCBpbkFtb3VudCk7CgkJfQoJfSwKCS8vKiBNb3ZlcyB0aGUgY3Vyc29yIGFjY29yZGluZyB0byB0aGUgRWRpdGluZyBBUEkuCgltb3ZlQ3Vyc29yOiBmdW5jdGlvbihpbkRpcmVjdGlvbiwgaW5BbW91bnQpIHsKCQl0aGlzLm1vZGlmeVNlbGVjdGlvbigibW92ZSIsIGluRGlyZWN0aW9uLCBpbkFtb3VudCk7Cgl9LAoJLy8qIE1vdmVzIHRoZSBjdXJzb3IgdG8gZW5kIG9mIHRleHQgZmllbGQuCgltb3ZlQ3Vyc29yVG9FbmQ6IGZ1bmN0aW9uKCkgewoJCXRoaXMubW92ZUN1cnNvcigiZm9yd2FyZCIsICJkb2N1bWVudGJvdW5kYXJ5Iik7Cgl9LAoJLy8qIE1vdmVzIHRoZSBjdXJzb3IgdG8gc3RhcnQgb2YgdGV4dCBmaWVsZC4KCW1vdmVDdXJzb3JUb1N0YXJ0OiBmdW5jdGlvbigpIHsKCQl0aGlzLm1vdmVDdXJzb3IoImJhY2t3YXJkIiwgImRvY3VtZW50Ym91bmRhcnkiKTsKCX0sCgkvLyogU2VsZWN0cyBhbGwgY29udGVudCBpbiB0ZXh0IGZpZWxkLgoJc2VsZWN0QWxsOiBmdW5jdGlvbigpIHsKCQlpZiAodGhpcy5oYXNGb2N1cygpKSB7CgkJCWRvY3VtZW50LmV4ZWNDb21tYW5kKCJzZWxlY3RBbGwiKTsKCQl9Cgl9LAoJLy8qIEluc2VydHMgSFRNTCBhdCB0aGUgY3Vyc29yIHBvc2l0aW9uLiAgSFRNTCBpcyBlc2NhcGVkIHVubGVzcyB0aGUKCS8vKiBfYWxsb3dIVE1MXyBwcm9wZXJ0eSBpcyB0cnVlLgoJaW5zZXJ0QXRDdXJzb3I6IGZ1bmN0aW9uKGluVmFsdWUpIHsKCQlpZiAodGhpcy5oYXNGb2N1cygpKSB7CgkJCXZhciB2ID0gdGhpcy5hbGxvd0h0bWwgPyBpblZhbHVlIDogZW55by5Db250cm9sLmVzY2FwZUh0bWwoaW5WYWx1ZSkucmVwbGFjZSgvXG4vZywgIjxici8+Iik7CgkJCWRvY3VtZW50LmV4ZWNDb21tYW5kKCJpbnNlcnRIVE1MIiwgZmFsc2UsIHYpOwoJCX0KCX0KfSk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Select.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5TZWxlY3RfIGltcGxlbWVudHMgYW4gSFRNTCBzZWxlY3Rpb24gd2lkZ2V0LCB1c2luZwoJW2VueW8uT3B0aW9uXSgjZW55by5PcHRpb24pIGtpbmRzIGJ5IGRlZmF1bHQuCgoJRXhhbXBsZToKCgkJe2tpbmQ6ICJTZWxlY3QiLCBvbmNoYW5nZTogInNlbGVjdENoYW5nZWQiLCBjb21wb25lbnRzOiBbCgkJCXtjb250ZW50OiAiRGVzY2VuZGluZyIsIHZhbHVlOiAiZCJ9LAoJCQl7Y29udGVudDogIkFzY2VuZGluZyIsIHZhbHVlOiAiYSJ9CgkJXX0KCgkJc2VsZWN0Q2hhbmdlZDogZnVuY3Rpb24oaW5TZW5kZXIsIGluRXZlbnQpIHsKCQkJdmFyIHMgPSBpblNlbmRlci5nZXRWYWx1ZSgpOwoJCQlpZiAocyA9PSAiZCIpIHsKCQkJCXRoaXMuc29ydExpc3REZXNjZW5kaW5nKCk7CgkJCX0gZWxzZSB7CgkJCQl0aGlzLnNvcnRMaXN0QXNjZW5kaW5nKCk7CgkJCX0KCQl9CgoJTm90ZTogVGhpcyB1c2VzIHRoZSBgPHNlbGVjdD5gIHRhZywgd2hpY2ggaXNuJ3QgaW1wbGVtZW50ZWQKCWZvciBuYXRpdmUgd2ViT1MgYXBwbGljYXRpb25zLCBhbHRob3VnaCBpdCBkb2VzIHdvcmsgaW4gdGhlCgl3ZWJPUyBXZWIgYnJvd3Nlci4KKi8KCmVueW8ua2luZCh7CgluYW1lOiAiZW55by5TZWxlY3QiLAoJcHVibGlzaGVkOiB7CgkJLy8qIEluZGV4IG9mIHRoZSBzZWxlY3RlZCBvcHRpb24gaW4gdGhlIGxpc3QKCQlzZWxlY3RlZDogMAoJfSwKCS8vKiBAcHJvdGVjdGVkCgloYW5kbGVyczogewoJCW9uY2hhbmdlOiAiY2hhbmdlIgoJfSwKCXRhZzogInNlbGVjdCIsCglkZWZhdWx0S2luZDogImVueW8uT3B0aW9uIiwKCXJlbmRlcmVkOiBmdW5jdGlvbigpIHsKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCS8vVHJpY2sgdG8gZm9yY2UgSUU4IG9uY2hhbmdlIGV2ZW50IGJ1YmJsZQoJCWlmKGVueW8ucGxhdGZvcm0uaWUgPT0gOCl7CgkJCXRoaXMuc2V0QXR0cmlidXRlKCJvbmNoYW5nZSIsIGVueW8uYnViYmxlcik7CgkJfQoJCXRoaXMuc2VsZWN0ZWRDaGFuZ2VkKCk7Cgl9LAoJZ2V0U2VsZWN0ZWQ6IGZ1bmN0aW9uKCkgewoJCXJldHVybiBOdW1iZXIodGhpcy5nZXROb2RlUHJvcGVydHkoInNlbGVjdGVkSW5kZXgiLCB0aGlzLnNlbGVjdGVkKSk7Cgl9LAoJc2VsZWN0ZWRDaGFuZ2VkOiBmdW5jdGlvbigpIHsKCQl0aGlzLnNldE5vZGVQcm9wZXJ0eSgic2VsZWN0ZWRJbmRleCIsIHRoaXMuc2VsZWN0ZWQpOwoJfSwKCWNoYW5nZTogZnVuY3Rpb24oKSB7CgkJdGhpcy5zZWxlY3RlZCA9IHRoaXMuZ2V0U2VsZWN0ZWQoKTsKCX0sCglyZW5kZXI6IGZ1bmN0aW9uKCkgewoJCS8vIHdvcmsgYXJvdW5kIElFIGJ1ZyB3aXRoIGlubmVySFRNTCBzZXR0aW5nIG9mIDxzZWxlY3Q+LCByZXJlbmRlciBwYXJlbnQgaW5zdGVhZAoJCS8vIGh0dHA6Ly9zdXBwb3J0Lm1pY3Jvc29mdC5jb20vZGVmYXVsdC5hc3B4P3NjaWQ9a2I7ZW4tdXM7Mjc2MjI4CgkJaWYgKGVueW8ucGxhdGZvcm0uaWUpIHsKCQkJdGhpcy5wYXJlbnQucmVuZGVyKCk7CgkJfSBlbHNlIHsKCQkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCQl9Cgl9LAoJLy8qIEBwdWJsaWMKCS8vKiBSZXR1cm5zIHRoZSB2YWx1ZSBvZiB0aGUgc2VsZWN0ZWQgb3B0aW9uLgoJZ2V0VmFsdWU6IGZ1bmN0aW9uKCkgewoJCWlmICh0aGlzLmhhc05vZGUoKSkgewoJCQlyZXR1cm4gdGhpcy5ub2RlLnZhbHVlOwoJCX0KCX0KfSk7CgovKioKCV9lbnlvLk9wdGlvbl8gaW1wbGVtZW50cyB0aGUgb3B0aW9ucyBpbiBhbiBIVE1MIHNlbGVjdCB3aWRnZXQuCiovCmVueW8ua2luZCh7CgluYW1lOiAiZW55by5PcHRpb24iLAoJcHVibGlzaGVkOiB7CgkJLy8qIFZhbHVlIG9mIHRoZSBvcHRpb24KCQl2YWx1ZTogIiIKCX0sCgkvLyogQHByb3RlY3RlZAoJdGFnOiAib3B0aW9uIiwKCWNyZWF0ZTogZnVuY3Rpb24oKSB7CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCQl0aGlzLnZhbHVlQ2hhbmdlZCgpOwoJfSwKCXZhbHVlQ2hhbmdlZDogZnVuY3Rpb24oKSB7CgkJdGhpcy5zZXRBdHRyaWJ1dGUoInZhbHVlIiwgdGhpcy52YWx1ZSk7Cgl9Cn0pOwoKLyoqCglfZW55by5PcHRpb25Hcm91cF8gYWxsb3dzIGZvciB0aGUgZ3JvdXBpbmcgb2Ygb3B0aW9ucyBpbiBhIHNlbGVjdCB3aWRnZXQsCglhbmQgZm9yIHRoZSBkaXNhYmxpbmcgb2YgYmxvY2tzIG9mIG9wdGlvbnMuCiovCmVueW8ua2luZCh7CgluYW1lOiAiZW55by5PcHRpb25Hcm91cCIsCglwdWJsaXNoZWQ6IHsKCQlsYWJlbDogIiIKCX0sCgkvLyogQHByb3RlY3RlZAoJdGFnOiAib3B0Z3JvdXAiLAoJZGVmYXVsdEtpbmQ6ICJlbnlvLk9wdGlvbiIsCgljcmVhdGU6IGZ1bmN0aW9uKCkgewoJCXRoaXMuaW5oZXJpdGVkKGFyZ3VtZW50cyk7CgkJdGhpcy5sYWJlbENoYW5nZWQoKTsKCX0sCglsYWJlbENoYW5nZWQ6IGZ1bmN0aW9uKCkgewoJCXRoaXMuc2V0QXR0cmlidXRlKCJsYWJlbCIsIHRoaXMubGFiZWwpOwoJfQp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Selection.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5TZWxlY3Rpb25fIGlzIHVzZWQgdG8gbWFuYWdlIHJvdyBzZWxlY3Rpb24gc3RhdGUgZm9yIGxpc3RzLiBJdAoJcHJvdmlkZXMgc2VsZWN0aW9uIHN0YXRlIG1hbmFnZW1lbnQJZm9yIGJvdGggc2luZ2xlLXNlbGVjdCBhbmQgbXVsdGktc2VsZWN0CglsaXN0cy4KCgkJLy8gVGhlIGZvbGxvd2luZyBpcyBhbiBleGNlcnB0IGZyb20gZW55by5GbHl3ZWlnaHRSZXBlYXRlci4KCQllbnlvLmtpbmQoewoJCQluYW1lOiAiZW55by5GbHl3ZWlnaHRSZXBlYXRlciIsCgkJCS4uLgoJCQljb21wb25lbnRzOiBbCgkJCQl7a2luZDogIlNlbGVjdGlvbiIsIG9uU2VsZWN0OiAic2VsZWN0RGVzZWxlY3QiLCBvbkRlc2VsZWN0OiAic2VsZWN0RGVzZWxlY3QifSwKCQkJCS4uLgoJCQldLAoJCQl0YXA6IGZ1bmN0aW9uKGluU2VuZGVyLCBpbkV2ZW50KSB7CgkJCQkuLi4KCQkJCS8vIG1hcmsgdGhlIHRhcHBlZCByb3cgYXMgc2VsZWN0ZWQKCQkJCXRoaXMuJC5zZWxlY3Rpb24uc2VsZWN0KGluRXZlbnQuaW5kZXgpOwoJCQkJLi4uCgkJCX0sCgkJCXNlbGVjdERlc2VsZWN0OiBmdW5jdGlvbihpblNlbmRlciwgaW5FdmVudCkgewoJCQkJLy8gdGhpcyBpcyB3aGVyZSBhIHJvdyBzZWxlY3Rpb24gaGlnaGxpZ2h0IG1pZ2h0IGJlIGFwcGxpZWQKCQkJCXRoaXMucmVuZGVyUm93KGluRXZlbnQua2V5KTsKCQkJfQoJCQkuLi4KCQl9KQoqLwplbnlvLmtpbmQoewoJbmFtZTogImVueW8uU2VsZWN0aW9uIiwKCWtpbmQ6ICJlbnlvLkNvbXBvbmVudCIsCglwdWJsaXNoZWQ6IHsKCQkvLyogSWYgdHJ1ZSwgbXVsdGlwbGUgc2VsZWN0aW9ucyBhcmUgYWxsb3dlZAoJCW11bHRpOiBmYWxzZQoJfSwKCWV2ZW50czogewoJCS8qKgoJCQlGaXJlcyB3aGVuIGFuIGl0ZW0gaXMgc2VsZWN0ZWQuCgoJCQkJe2tpbmQ6ICJTZWxlY3Rpb24iLCBvblNlbGVjdDogInNlbGVjdFJvdyIuLi4KCQkJCS4uLgoJCQkJc2VsZWN0Um93OiBmdW5jdGlvbihpblNlbmRlciwgaW5FdmVudCkgewoJCQkJCS4uLgoKCQkJX2luRXZlbnQua2V5XyBpcyB3aGF0ZXZlciBrZXkgd2FzIHVzZWQgdG8gcmVnaXN0ZXIgdGhlIHNlbGVjdGlvbgoJCQkodXN1YWxseSBhIHJvdyBpbmRleCkuCgoJCQlfaW5FdmVudC5kYXRhXyByZWZlcmVuY2VzIGRhdGEgcmVnaXN0ZXJlZCB3aXRoIHRoaXMga2V5IGJ5IHRoZSBjb2RlCgkJCXRoYXQgbWFkZSB0aGUgb3JpZ2luYWwgc2VsZWN0aW9uLgoJCSovCgkJb25TZWxlY3Q6ICIiLAoJCS8qKgoJCQlGaXJlcyB3aGVuIGFuIGl0ZW0gaXMgZGVzZWxlY3RlZC4KCgkJCQl7a2luZDogIlNlbGVjdGlvbiIsIG9uU2VsZWN0OiAiZGVzZWxlY3RSb3ciLi4uCgkJCQkuLi4KCQkJCWRlc2VsZWN0Um93OiBmdW5jdGlvbihpblNlbmRlciwgaW5FdmVudCkKCQkJCQkuLi4KCgkJCV9pbkV2ZW50LmtleV8gaXMgd2hhdGV2ZXIga2V5IHdhcyB1c2VkIHRvIHJlcXVlc3QgdGhlIGRlc2VsZWN0aW9uCgkJCSh1c3VhbGx5IGEgcm93IGluZGV4KS4KCgkJCV9pbkV2ZW50LmRhdGFfIHJlZmVyZW5jZXMgZGF0YSByZWdpc3RlcmVkIHdpdGggdGhpcyBrZXkgYnkgdGhlIGNvZGUKCQkJdGhhdCBtYWRlIHRoZSBzZWxlY3Rpb24uCgkJKi8KCQlvbkRlc2VsZWN0OiAiIiwKCQkvLyogRmlyZXMgd2hlbiBzZWxlY3Rpb24gY2hhbmdlcyAoYnV0IG5vdCB3aGVuIHNlbGVjdGlvbiBpcyBjbGVhcmVkKS4KCQlvbkNoYW5nZTogIiIKCX0sCgkvLyogQHByb3RlY3RlZAoJY3JlYXRlOiBmdW5jdGlvbigpIHsKCQl0aGlzLmNsZWFyKCk7CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCX0sCgltdWx0aUNoYW5nZWQ6IGZ1bmN0aW9uKCkgewoJCWlmICghdGhpcy5tdWx0aSkgewoJCQl0aGlzLmNsZWFyKCk7CgkJfQoJCXRoaXMuZG9DaGFuZ2UoKTsKCX0sCgloaWdobGFuZGVyOiBmdW5jdGlvbihpbktleSkgewoJCWlmICghdGhpcy5tdWx0aSkgewoJCQl0aGlzLmRlc2VsZWN0KHRoaXMubGFzdFNlbGVjdGVkKTsKCQl9Cgl9LAoJLy8qIEBwdWJsaWMKCS8vKiBSZW1vdmVzIGFsbCBzZWxlY3Rpb25zLgoJY2xlYXI6IGZ1bmN0aW9uKCkgewoJCXRoaXMuc2VsZWN0ZWQgPSB7fTsKCX0sCgkvLyogUmV0dXJucyB0cnVlIGlmIHRoZSBfaW5LZXlfIHJvdyBpcyBzZWxlY3RlZC4KCWlzU2VsZWN0ZWQ6IGZ1bmN0aW9uKGluS2V5KSB7CgkJcmV0dXJuIHRoaXMuc2VsZWN0ZWRbaW5LZXldOwoJfSwKCS8qKgoJCU1hbnVhbGx5IHNldHMgYSByb3cncyBzdGF0ZSB0byBzZWxlY3RlZCBvciB1bnNlbGVjdGVkLgoKCQlfaW5EYXRhXyBpcyBhbiBvcHRpb25hbCBkYXRhIG9iamVjdAl0byBzdG9yZSBpbiB0aGUgc2VsZWN0aW9uIGZvcgoJCXRoYXQga2V5IHRoYXQgd2lsbCBiZSBzZW50IHdpdGggdGhlCV9vblNlbGVjdF8gb3IgX29uRGVzZWxlY3RfIGV2ZW50cy4KCQlJZiBub3QgdXNlZCwgdGhlIGRhdGEgd2lsbCBiZSBzZXQgdG8gJ3RydWUnLgoJKi8KCXNldEJ5S2V5OiBmdW5jdGlvbihpbktleSwgaW5TZWxlY3RlZCwgaW5EYXRhKSB7CgkJaWYgKGluU2VsZWN0ZWQpIHsKCQkJdGhpcy5zZWxlY3RlZFtpbktleV0gPSAoaW5EYXRhIHx8IHRydWUpOwoJCQl0aGlzLmxhc3RTZWxlY3RlZCA9IGluS2V5OwoJCQl0aGlzLmRvU2VsZWN0KHtrZXk6IGluS2V5LCBkYXRhOiB0aGlzLnNlbGVjdGVkW2luS2V5XX0pOwoJCX0gZWxzZSB7CgkJCXZhciB3YXMgPSB0aGlzLmlzU2VsZWN0ZWQoaW5LZXkpOwoJCQlkZWxldGUgdGhpcy5zZWxlY3RlZFtpbktleV07CgkJCXRoaXMuZG9EZXNlbGVjdCh7a2V5OiBpbktleSwgZGF0YTogd2FzfSk7CgkJfQoJCXRoaXMuZG9DaGFuZ2UoKTsKCX0sCgkvLyogRGVzZWxlY3RzIGEgcm93LgoJZGVzZWxlY3Q6IGZ1bmN0aW9uKGluS2V5KSB7CgkJaWYgKHRoaXMuaXNTZWxlY3RlZChpbktleSkpIHsKCQkJdGhpcy5zZXRCeUtleShpbktleSwgZmFsc2UpOwoJCX0KCX0sCgkvKioKCQlTZWxlY3RzIGEgcm93LiBJZiB0aGUgX211bHRpXyBwcm9wZXJ0eSBpcyBzZXQgdG8gZmFsc2UsIF9zZWxlY3RfIHdpbGwKCQlhbHNvIGRlc2VsZWN0IHRoZSBwcmV2aW91cyBzZWxlY3Rpb24uCgoJCV9pbkRhdGFfIGlzIGFuIG9wdGlvbmFsIGRhdGEgb2JqZWN0CXRvIHN0b3JlIGluIHRoZSBzZWxlY3Rpb24gZm9yCgkJdGhhdCBrZXkgdGhhdCB3aWxsIGJlIHNlbnQgd2l0aCB0aGUJX29uU2VsZWN0XyBvciBfb25EZXNlbGVjdF8gZXZlbnRzLgoJCUlmIG5vdCB1c2VkLCB0aGUgZGF0YSB3aWxsIGJlIHNldCB0byAndHJ1ZScuCgkqLwoJc2VsZWN0OiBmdW5jdGlvbihpbktleSwgaW5EYXRhKSB7CgkJaWYgKHRoaXMubXVsdGkpIHsKCQkJdGhpcy5zZXRCeUtleShpbktleSwgIXRoaXMuaXNTZWxlY3RlZChpbktleSksIGluRGF0YSk7CgkJfSBlbHNlIGlmICghdGhpcy5pc1NlbGVjdGVkKGluS2V5KSkgewoJCQl0aGlzLmhpZ2hsYW5kZXIoKTsKCQkJdGhpcy5zZXRCeUtleShpbktleSwgdHJ1ZSwgaW5EYXRhKTsKCQl9Cgl9LAoJLyoqCgkJVG9nZ2xlcyBzZWxlY3Rpb24gc3RhdGUgZm9yIGEgcm93LiBJZiB0aGUgX211bHRpXyBwcm9wZXJ0eSBpcyBzZXQgdG8KCQlmYWxzZSwgdG9nZ2xpbmcgYSBzZWxlY3Rpb24gb24gd2lsbCBkZXNlbGVjdCB0aGUgcHJldmlvdXMgc2VsZWN0aW9uLgoKCQlfaW5EYXRhXyBpcyBhbiBvcHRpb25hbCBkYXRhIG9iamVjdAl0byBzdG9yZSBpbiB0aGUgc2VsZWN0aW9uIGZvcgoJCXRoYXQga2V5IHRoYXQgd2lsbCBiZSBzZW50IHdpdGggdGhlCV9vblNlbGVjdF8gb3IgX29uRGVzZWxlY3RfIGV2ZW50cy4KCQlJZiBub3QgdXNlZCwgdGhlIGRhdGEgd2lsbCBiZSBzZXQgdG8gJ3RydWUnLgoJKi8KCXRvZ2dsZTogZnVuY3Rpb24oaW5LZXksIGluRGF0YSkgewoJCWlmICghdGhpcy5tdWx0aSAmJiB0aGlzLmxhc3RTZWxlY3RlZCAhPSBpbktleSkgewoJCQl0aGlzLmRlc2VsZWN0KHRoaXMubGFzdFNlbGVjdGVkKTsKCQl9CgkJdGhpcy5zZXRCeUtleShpbktleSwgIXRoaXMuaXNTZWxlY3RlZChpbktleSksIGluRGF0YSk7Cgl9LAoJLyoqCgkJUmV0dXJucyB0aGUgc2VsZWN0aW9uIGFzIGEgaGFzaCBpbiB3aGljaCBlYWNoIHNlbGVjdGVkIGl0ZW0gaGFzIGEgdmFsdWU7CgkJdW5zZWxlY3RlZCBpdGVtcyBhcmUgdW5kZWZpbmVkLgoJKi8KCWdldFNlbGVjdGVkOiBmdW5jdGlvbigpIHsKCQlyZXR1cm4gdGhpcy5zZWxlY3RlZDsKCX0sCgkvKioKCQlSZW1vdmUgYSByb3cgdGhhdCdzIGluY2x1ZGVkIGluIHRoZSBzZWxlY3Rpb24gc2V0LiBJZiB0aGlzIHJvdyBpcwoJCXNlbGVjdGVkLCBpdCB3aWxsIGJlIHVuc2VsZWN0ZWQuICBBbnkgcm93cyBhYm92ZSB0aGlzIHJvdyB3aWxsCgkJaGF2ZSB0aGVpciBrZXlzIHZhbHVlIHJlZHVjZWQgYnkgb25lLgoJKi8KCXJlbW92ZTogZnVuY3Rpb24oaW5LZXkpIHsKCQl2YXIgbmV3U2VsZWN0ZWQgPSB7fTsKCQlmb3IgKHZhciByb3cgaW4gdGhpcy5zZWxlY3RlZCkgewoJCQlpZiAocm93IDwgaW5LZXkpIHsKCQkJCW5ld1NlbGVjdGVkW3Jvd10gPSB0aGlzLnNlbGVjdGVkW3Jvd107CgkJCX0gZWxzZSBpZiAocm93ID4gaW5LZXkpIHsKCQkJCW5ld1NlbGVjdGVkW3JvdyAtIDFdID0gdGhpcy5zZWxlY3RlZFtyb3ddOwoJCQl9CgkJfQoJCXRoaXMuc2VsZWN0ZWQgPSBuZXdTZWxlY3RlZDsKCX0KfSk7
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/Table.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoKCVRPRE86IFdvbid0IHdvcmsgaW4gSUU4IGJlY2F1c2UJd2UgY2FuJ3Qgc2V0IGlubmVySFRNTAoJb24gdGFibGUgZWxlbWVudHMuIFdlJ2xsIG5lZWQgdG8gZmFsbCBiYWNrIHRvIGRpdnMgd2l0aAoJdGFibGUgZGlzcGxheSBzdHlsZXMgYXBwbGllZC4KCglTaG91bGQgYWxzbyBmYWNhZGUgY2VydGFpbiB1c2VmdWwgdGFibGUgZnVuY3Rpb25hbGl0eQoJKHNwZWNpZmljIHNldCBUQkQpLgoqLwoKLyoqCglfZW55by5UYWJsZV8gaW1wbGVtZW50cyBhbiBIVE1MICZsdDt0YWJsZSZndDsgZWxlbWVudC4KCVRoaXMgaXMgYSB3b3JrIGluIHByb2dyZXNzLiAKKi8KZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLlRhYmxlIiwKCXRhZzogInRhYmxlIiwKCWF0dHJpYnV0ZXM6IHtjZWxscGFkZGluZzogIjAiLCBjZWxsc3BhY2luZzogIjAifSwKCWRlZmF1bHRLaW5kOiAiZW55by5UYWJsZVJvdyIKfSk7CgovKioKCV9lbnlvLlRhYmxlUm93XyBpbXBsZW1lbnRzIGFuIEhUTUwgJmx0O3RyJmd0OyBlbGVtZW50LgoqLwplbnlvLmtpbmQoewogICAgbmFtZTogImVueW8uVGFibGVSb3ciLAogICAgdGFnOiAidHIiLAogICAgZGVmYXVsdEtpbmQ6ICJlbnlvLlRhYmxlQ2VsbCIKfSk7CgovKioKCV9lbnlvLlRhYmxlQ2VsbF8gaW1wbGVtZW50cyBhbiBIVE1MICZsdDt0ZCZndDsgZWxlbWVudC4KKi8KZW55by5raW5kKHsKICAgIG5hbWU6ICJlbnlvLlRhYmxlQ2VsbCIsCiAgICB0YWc6ICJ0ZCIKfSk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/TextArea.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5UZXh0QXJlYV8gaW1wbGVtZW50cyBhbiBIVE1MICZsdDt0ZXh0YXJlYSZndDsgZWxlbWVudCB3aXRoCgljcm9zcy1wbGF0Zm9ybSBzdXBwb3J0IGZvciBjaGFuZ2UgZXZlbnRzLgoKCUZvciBtb3JlIGluZm9ybWF0aW9uLCBzZWUgdGhlIGRvY3VtZW50YXRpb24gb24KCVtUZXh0IEZpZWxkc10oaHR0cHM6Ly9naXRodWIuY29tL2VueW9qcy9lbnlvL3dpa2kvVGV4dC1GaWVsZHMpIGluIHRoZSBFbnlvCglEZXZlbG9wZXIgR3VpZGUuCiovCmVueW8ua2luZCh7CgluYW1lOiAiZW55by5UZXh0QXJlYSIsCglraW5kOiAiZW55by5JbnB1dCIsCgkvLyogQHByb3RlY3RlZAoJdGFnOiAidGV4dGFyZWEiLAoJY2xhc3NlczogImVueW8tdGV4dGFyZWEiLAoJLy8gdGV4dGFyZWEgZG9lcyB1c2UgdmFsdWUgYXR0cmlidXRlOyBuZWVkcyB0byBiZSBraWNrZWQgd2hlbiByZW5kZXJlZC4KCXJlbmRlcmVkOiBmdW5jdGlvbigpIHsKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCXRoaXMudmFsdWVDaGFuZ2VkKCk7Cgl9Cn0pOwo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/ToolDecorator.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5Ub29sRGVjb3JhdG9yXyBsaW5lcyB1cCBjb21wb25lbnRzIGluIGEgcm93LCBjZW50ZXJlZCB2ZXJ0aWNhbGx5LgoqLwplbnlvLmtpbmQoewoJbmFtZTogImVueW8uVG9vbERlY29yYXRvciIsCgkvLyogQHByb3RlY3RlZAoJa2luZDogImVueW8uR3JvdXBJdGVtIiwKCWNsYXNzZXM6ICJlbnlvLXRvb2wtZGVjb3JhdG9yIgp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/data/DataGridList.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uIChlbnlvKSB7CgoKCS8vKkBwdWJsaWMKCS8qKgoJKi8KCWVueW8ua2luZCh7CgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIFBVQkxJQyBQUk9QRVJUSUVTCgoJCS8vKkBwdWJsaWMKCQluYW1lOiAiZW55by5EYXRhR3JpZExpc3QiLAoKCQkvLypAcHVibGljCgkJa2luZDogImVueW8uRGF0YUxpc3QiLAoKCQkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCQkvLyBQUk9URUNURUQgUFJPUEVSVElFUwoKCQkvLypAcHJvdGVjdGVkCgkJY2xhc3NlczogImVueW8tZGF0YS1ncmlkLWxpc3QiLAoKCQkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCQkvLyBDT01QVVRFRCBQUk9QRVJUSUVTCgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIFBVQkxJQyBNRVRIT0RTCgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIFBST1RFQ1RFRCBNRVRIT0RTCgoJCS8vKkBwcm90ZWN0ZWQKCQlpbml0Q29tcG9uZW50czogZnVuY3Rpb24gKCkgewoJCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCQl2YXIgJGtpbmQgPSB0aGlzLl9jaGlsZEtpbmQ7CgkJCSRraW5kLmV4dGVuZCh7CgkJCQljbGFzc2VzOiAiZW55by1kYXRhLWdyaWQtbGlzdC1pdGVtIgoJCQl9KTsKCQl9CgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIE9CU0VSVkVSUwoKCX0pOwoKfSkoZW55byk7
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/data/DataList.js"
Content-Type: application/octet-stream; x-encoding=base64
(function (enyo) {

	//*@public
	/**
	*/
	enyo.kind({

		// ...........................
		// PUBLIC PROPERTIES

		//*@public
		name: "enyo.DataList",

		//*@public
		kind: "enyo.DataRepeater",

		//*@public
		/**
			The _enyo.DataList_ kind places its rows inside of a scroller. Any
			configurable options associated with an _enyo.Scroller_ can be
			placed in this hash and will be set accordingly on the scroller
			for this list. If none are specified default _enyo.Scroller_
			settings are used.
		*/
		scrollerOptions: null,

		//*@public
		protectedStatics: {
			defaultScrollerOptions: {
				preventScrollPropagation: false
			}
		},

		//*@public
		handlers: {
			onScroll: "_didScroll",
			onpostresize: "_didResize"
		},

		//*@public
		controlParentName: "page1",

		//*@public
		classes: "enyo-data-list",

		//*@public
		containerOptions: {
			name: "scroller",
			kind: "enyo.Scroller",
			classes: "enyo-fill enyo-data-list-scroller",
			components: [
				{name: "active", classes: "enyo-data-list-active", components: [
					{name: "page1", kind: "enyo.View", classes: "enyo-data-list-page"},
					{name: "page2", kind: "enyo.View", classes: "enyo-data-list-page"}
				]},
				{name: "buffer", style: "position: relative;"}
			]
		},

		overflow: null,
		underflow: null,

		// ...........................
		// PROTECTED PROPERTIES

		_resized: false,
		_initialPosition: true,

		// ...........................
		// COMPUTED PROPERTIES

		averageBounds: enyo.computed(function () {
			// reset our flag
			this._resized = false;
			this._reset = false;
			this._rendered = false;
			var $ch = this.getClientControls(), $c;
			var $i, $l, $t = {};
			$t.height = 0;
			$t.width = 0;
			for ($i = 1, $l = $ch.length; $i < $l && $i < 10; ++$i) {
				$c = $ch[$i].getBounds();
				$t.height += !isNaN($c.height)? $c.height: 0;
				$t.width += !isNaN($c.width)? $c.width: 0;
			}
			if ($i > 1) {
				$t.height = ~~($t.height / ($i-1));
				$t.width = ~~($t.width / ($i-1));
			}
			return $t;
		}, "_resized", "_reset", "_rendered", {cached: true, defer: true}),

		// ...........................
		// PUBLIC METHODS

		//*@public
		reset: function () {
			var $d = this.get("data");
			this.resetting = true;
			this.destroyClientControls();
			this.resetBuffer();
			this.underflow = {};
			this.overflow = {};
			if ($d) {
				var $i = 0;
				for (var $k in {page1:"", page2:""}) {
					this.controlParentName = $k;
					this.discoverControlParent();
					for (; $i < $d.length && this.threshold(); ++$i) {
						this.add($d[$i], $i);
					}
				}
			}
			this.set("_reset", true);
			this.rendered();
			this.controlParentName = "page1";
			this.discoverControlParent();
			this.resetting = false;
		},

		resetBuffer: function () {
			this.$.buffer.applyStyle("height", "0");
		},

		threshold: function () {
			var $o = this.getBounds().height;
			var $h = this.controlParent.getBounds().height;
			return $h <= (2.5 * $o);
		},

		updateBuffer: function () {
			var $d = this.get("data"), $t = this.get("averageBounds");
			if ($d) {
				this.$.buffer.setBounds({height: ($t.height * $d.length)}, "px");
				this.$.scroller.resized();
			}
		},

		destroyClientControls: function () {
			for (var $i in {page1:"",page2:""}) {
				this.controlParent = this.$[$i];
				this.inherited(arguments);
			}
		},

		rendered: function () {
			this.set("_rendered", true);
			this.inherited(arguments);
			if (this._initialPosition && this.$.page1.hasNode() && this.$.page1.getBounds().height) {
				this._initialPosition = false;
				this.movePageAfter(this.$.page2, this.$.page1);
			}
		},

		reflow: function () {
			this.inherited(arguments);
			this.updateBuffer();
		},

		//*@public
		add: function (rec, idx) {
			// TODO: need to intelligently add indices to the appropriate
			// page when possible and maintain a reference to the overflow
			// indices so when a page is pushed down it can know which models
			// to use and when a page is pushed up it can know the same



			if (!this.threshold()) {
				if (this.controlParentName == "page1") {
					this.controlParentName = "page2";
					this.discoverControlParent();
					this.add(rec, idx);
					this.controlParentName = "page1";
					this.discoverControlParent();
				} else {
					var $o = this.overflow;
					if (!$o.start) {
						$o.start = idx;
					}
					$o.end = idx;
					return idx;
				}
			} else {
				var $c = this.createComponent({kind: this._childKind, model: rec, index: idx});
				$c.render();
				if (!this.batching) {
					this.reflow();
				}
			}
		},

		//*@public
		remove: function (idx) {
			var $ch = this.get("active");
			var $c = $ch[idx || (Math.abs($ch.length-1))];
			if ($c) {
				$c.destroy();
			}
		},

		//*@public
		update: function (idx) {
			var $d = this.get("data");
			var $ch = this.get("active");
			var $c = $ch[idx];
			if ($d && $c) {
				$c.set("model", $d[idx]);
			}
		},

		//*@public
		prune: function () {
			var $ch = this.get("active");
			var l = this.length;
			var $x = $ch.slice(l);
			enyo.forEach($x, function (c) {
				c.destroy();
			});
		},

		//*@public
		up: function () {
			var $c = this.controlParent;
			var $b = $c.getBounds();
			var $p = this.$.scroller.getScrollTop();
			var $n = this.$.page1 == $c? this.$.page2: this.$.page1;
			var $k = $n.getBounds();
			var $o = this.getBounds();
			if ($b.top < $k.top && $b.top !== 0) {
				if ($k.top > ($o.height + $p) && ($k.height <= $p)) {
					this.controlParentName = $n.name;
					this.controlParent = $n;
					this.pageUp();
				}
			} else {
				if ($b.top > $o.height && $b.height <= $p) {
					/* FIXME */
					this.pageUp();
					this.controlParentName = $n.name;
					this.controlParent = $n;
				}
			}
		},

		//*@public
		down: function () {
			var $p = this.$.scroller.getScrollTop();
			var $c = this.controlParent;
			var $b = $c.getBounds();
			var $n = this.$.page1 == $c? this.$.page2: this.$.page1;
			var $k = $n.getBounds();
			var $o = this.$.buffer.getBounds();
			var $f = $b.top + $b.height;
			if ($f < $p && ($o.height - $f) >= $k.height) {
				this.pageDown();
			}
		},

		movePageAfter: function (p1, p2) {
			var $p = p2.getBounds();
			p1.setBounds({top: $p.top + $p.height}, "px");
		},

		movePageBefore: function (p1, p2) {
			var $p = p2.getBounds();
			var $k = p1.getBounds();
			p1.setBounds({top: $p.top - $k.height}, "px");
		},

		pageDown: function () {
			var $p1 = this.controlParent;
			var $n = $p1.name == "page1"? "page2": "page1";
			var $p2 = this.$[$n];
			this.movePageAfter($p1, $p2);
			this.controlParentName = $n;
			this.controlParent = $p2;
			// enyo.asyncMethod(this, "updatePage", $p1, "down");
			this.updatePage($p1, "down");
		},

		updatePage: function (p, direction) {
			var $o = this.overflow;
			var $u = this.underflow;
			var $ch = p.children;
			var $d, $idx, $c, $m;
			if (direction == "down") {
				if (!isNaN($o.start)) {
					// p.teardownChildren();
					p.disconnectDom();
					$d = this.get("data");
					for ($idx = $o.start; $idx < $o.end && ($m = $d[$idx]) && ($c = $ch[$idx - $o.start]); ++$idx) {
						if ($c.index < $u.start || isNaN($u.start)) {
							$u.start = $c.index;
						} else if ($c.index > $u.end || isNaN($u.end)) {
							$u.end = $c.index;
						}
						$c.index = $idx;
						$c.set("model", $m);
					}
					p.connectDom();
					// p.render();
					p.renderReusingNode();
				}
				$o.start = ++$idx;
			}
		},

		pageUp: function () {
			var $p1 = this.controlParent;
			var $n = $p1.name == "page1"? "page2": "page1";
			var $p2 = this.$[$n];
			this.movePageBefore($p1, $p2);

		},

		batchingChanged: function (prev, val) {
			if (false === val) {
				this.rendered();
			}
		},

		initComponents: function () {
			this.inherited(arguments);
			var $k = this._childKind;
			if (!enyo.hasMixin($k, "enyo.RowSupport")) {
				enyo.applyMixin("enyo.RowSupport", $k);
			}
		},

		// ...........................
		// PROTECTED METHODS

		//*@protected
		_initContainer: function () {
			var $c = enyo.clone(this.get("containerOptions"));
			var $o = enyo.clone(this.get("scrollerOptions") || {});
			var $d = enyo.DataList.defaultScrollerOptions;
			// ultimately create a new object without modifying the prototype's
			// options, and only use default values that doesn't override other
			// set values
			this.containerOptions = enyo.mixin(enyo.mixin([$c, $o]), $d, true);
			this.inherited(arguments);
		},

		_didScroll: function (sender, event) {
			var $l = !isNaN(this._lastPos)? this._lastPost: 0;
			var $p = this.$.scroller.getScrollTop();
			var $d = $p - $l;
			// if ($d && Math.abs($d) > this.get("averageBounds").height) {
			if ($d) {
				if ($l > $p) {
					this.up();
				} else if ($l < $p) {
					this.down();
				}
			}
			this._lastPost = $p;
			return true;
		},

		_didResize: function () {
			this.set("_resized", true);
		}

	});

})(enyo);

----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/data/DataRepeater.js"
Content-Type: application/octet-stream; x-encoding=base64
KGZ1bmN0aW9uIChlbnlvKSB7CgoJLy8qQHB1YmxpYwoJLyoqCgkqLwoJZW55by5raW5kKHsKCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gUFVCTElDIFBST1BFUlRJRVMKCgkJLy8qQHB1YmxpYwoJCW5hbWU6ICJlbnlvLkRhdGFSZXBlYXRlciIsCgoJCS8vKkBwdWJsaWMKCQlraW5kOiAiZW55by5WaWV3IiwKCgkJLy8qQHB1YmxpYwoJCWNoaWxkTWl4aW5zOiBbXSwKCQkKCQkvLypAcHVibGljCgkJY29uY2F0OiBbImNoaWxkTWl4aW5zIl0sCgoJCS8vKkBwdWJsaWMKCQljb250cm9sUGFyZW50TmFtZTogImNvbnRhaW5lciIsCgkJCgkJLy8qQHB1YmxpYwoJCWNvbnRhaW5lck9wdGlvbnM6IHsKCQkJbmFtZTogImNvbnRhaW5lciIsCgkJCWtpbmQ6ICJlbnlvLlZpZXciLAoJCQljbGFzc2VzOiAiZW55by1maWxsIGVueW8tZGF0YS1yZXBlYXRlci1jb250YWluZXIiLAoJCX0sCgoJCS8vKkBwdWJsaWMKCQloYW5kbGVyczogewoJCQlvbk1vZGVsQWRkZWQ6ICJfbW9kZWxBZGRlZCIsCgkJCW9uTW9kZWxzQWRkZWQ6ICJfbW9kZWxzQWRkZWQiLAoJCQlvbk1vZGVsUmVtb3ZlZDogIl9tb2RlbFJlbW92ZWQiLAoJCQlvbk1vZGVsc1JlbW92ZWQ6ICJfbW9kZWxzUmVtb3ZlZCIKCQl9LAoKCQkvLypAcHVibGljCgkJYmluZGluZ3M6IFsKCQkJe2Zyb206ICIuY29udHJvbGxlci5sZW5ndGgiLCB0bzogIi5sZW5ndGgifSwKCQkJe2Zyb206ICIuY29udHJvbGxlci5kYXRhIiwgdG86ICIuZGF0YSJ9CgkJXSwKCQkKCgkJLy8qQHB1YmxpYwoJCWJhdGNoaW5nOiBmYWxzZSwKCgkJLy8gLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uCgkJLy8gUFJPVEVDVEVEIFBST1BFUlRJRVMKCgkJLy8qQHByb3RlY3RlZAoJCV9jaGlsZEtpbmQ6IG51bGwsCgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIFBVQkxJQyBNRVRIT0RTCgkJCgkJLy8qQHB1YmxpYwoJCWluaXRDb21wb25lbnRzOiBmdW5jdGlvbiAoKSB7CgkJCS8vIHdlIG5lZWQgdG8gZmluZCB0aGUgY2hpbGQgZGVmaW5pdGlvbiBhbmQgcHJlcGFyZSBpdCBmb3IKCQkJLy8gdXNlIGluIG91ciByZXBlYXRlciBpbmNsdWRpbmcgYWRkaW5nIGF1dG8gYmluZGluZyBzdXBwb3J0CgkJCXZhciAkY29tcG9uZW50cyA9IHRoaXMua2luZENvbXBvbmVudHMgfHwgdGhpcy5jb21wb25lbnRzIHx8IFtdOwoJCQl2YXIgJG93bmVyID0gdGhpcy5jb21wb25lbnRzPyB0aGlzLm93bmVyOiB0aGlzOwoJCQl0aGlzLl9jaGlsZEtpbmQgPSBlbnlvLmtpbmQoewoJCQkJLy8gdGFnOiBudWxsLCAvLyBubyBuZWVkIGZvciB0aGUgZXh0cmEgY29udGFpbmVyCgkJCQkvLyBUT0RPOiBpdCBzaG91bGQgYmUgcG9zc2libGUgdG8gdXNlIHRoZSBhYm92ZSBsaW5lIGJ1dCBmb3IKCQkJCS8vIG5vdyBpdCBpcyBjYXVzaW5nIGZhciB0b28gbWFueSBpc3N1ZXMKCQkJCWtpbmQ6ICJlbnlvLlZpZXciLAoJCQkJbWl4aW5zOiBbImVueW8uQXV0b0JpbmRpbmdTdXBwb3J0IiwgImVueW8uUmVwZWF0ZXJDaGlsZFN1cHBvcnQiXS5jb25jYXQodGhpcy5jaGlsZE1peGlucyB8fCBbXSksCgkJCQljb21wb25lbnRzOiAkY29tcG9uZW50cywKCQkJCWRlZmF1bHRLaW5kOiB0aGlzLmRlZmF1bHRLaW5kIHx8ICJlbnlvLlZpZXciLAoJCQkJZGVmYXVsdFByb3BzOiB7b3duZXI6ICRvd25lcn0KCQkJfSk7CgkJCXRoaXMuX2luaXRDb250YWluZXIoKTsKCQl9LAoKCQkvLypAcHVibGljCgkJY29udHJvbGxlckZpbmRBbmRJbnN0YW5jZTogZnVuY3Rpb24oY3RvciwgaW5zdCkgewoJCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCQlpZiAoaW5zdCAmJiBpbnN0Ll9pc0NvbnRyb2xsZXIpIHsKCQkJCXRoaXMucmVzZXQoKTsKCQkJfQoJCX0sCgoJCS8vIFRPRE86CgkJcmVzZXQ6IGZ1bmN0aW9uICgpIHsKCQkJdmFyICRkID0gdGhpcy5nZXQoImRhdGEiKTsKCQkJdmFyICRjID0gdGhpcy4kLnNjcm9sbGVyOwoJCQl0aGlzLmRlc3Ryb3lDbGllbnRDb250cm9scygpOwoJCQkkYy5yZXNpemVIYW5kbGVyKCk7CgkJCWlmICgkZCkgewoJCQkJZW55by5mb3JFYWNoKCRkLCB0aGlzLmFkZCwgdGhpcyk7CgkJCX0KCQl9LAoKCQlyZW5kZXI6IGZ1bmN0aW9uICgpIHsKCQkJdGhpcy5yZXNldCgpOwoJCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJCX0sCgoJCS8vKkBwdWJsaWMKCQlhZGQ6IGZ1bmN0aW9uIChyZWMpIHsKCQkJaWYgKCF0aGlzLmdlbmVyYXRlZCkgewoJCQkJcmV0dXJuOwoJCQl9CgkJCXZhciAkayA9IHRoaXMuX2NoaWxkS2luZDsKCQkJdmFyICRjID0gdGhpcy5jcmVhdGVDb21wb25lbnQoe2tpbmQ6ICRrfSk7CgkJCXZhciBiID0gdGhpcy5iYXRjaGluZzsKCQkJJGMuc2V0KCJtb2RlbCIsIHJlYyk7CgkJCWlmICghYikgewoJCQkJJGMucmVuZGVyKCk7CgkJCX0KCQl9LAoKCQkvLypAcHVibGljCgkJcmVtb3ZlOiBmdW5jdGlvbiAoaWR4KSB7CgkJCXZhciAkY2ggPSB0aGlzLmdldCgiYWN0aXZlIik7CgkJCXZhciAkYyA9ICRjaFtpZHggfHwgKE1hdGguYWJzKCRjaC5sZW5ndGgtMSkpXTsKCQkJaWYgKCRjKSB7CgkJCQkkYy5kZXN0cm95KCk7CgkJCX0KCQl9LAoKCQkvLypAcHVibGljCgkJdXBkYXRlOiBmdW5jdGlvbiAoaWR4KSB7CgkJCXZhciAkZCA9IHRoaXMuZ2V0KCJkYXRhIik7CgkJCXZhciAkY2ggPSB0aGlzLmdldCgiYWN0aXZlIik7CgkJCXZhciAkYyA9ICRjaFtpZHhdOwoJCQlpZiAoJGQgJiYgJGMpIHsKCQkJCSRjLnNldCgibW9kZWwiLCAkZFtpZHhdKTsKCQkJfQoJCX0sCgoJCS8vKkBwdWJsaWMKCQlwcnVuZTogZnVuY3Rpb24gKCkgewoJCQl2YXIgJGNoID0gdGhpcy5nZXQoImFjdGl2ZSIpOwoJCQl2YXIgbCA9IHRoaXMubGVuZ3RoOwoJCQl2YXIgJHggPSAkY2guc2xpY2UobCk7CgkJCWVueW8uZm9yRWFjaCgkeCwgZnVuY3Rpb24gKGMpIHsKCQkJCWMuZGVzdHJveSgpOwoJCQl9KTsKCQl9LAoKCQkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCQkvLyBDT01QVVRFRCBQUk9QRVJUSUVTCgoJCS8vKkBwdWJsaWMKCQlhY3RpdmU6IGVueW8uY29tcHV0ZWQoZnVuY3Rpb24gKCkgewoJCQlyZXR1cm4gdGhpcy5jb250cm9sUGFyZW50PyB0aGlzLmNvbnRyb2xQYXJlbnQuY2hpbGRyZW46IHRoaXMuY2hpbGRyZW47CgkJfSwgImNvbnRyb2xQYXJlbnQiLCB7Y2FjaGVkOiB0cnVlLCBkZWZlcjogdHJ1ZX0pLAoKCQkvLyAuLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4KCQkvLyBQUk9URUNURUQgTUVUSE9EUwoKCQkvLypAcHJvdGVjdGVkCgkJX2luaXRDb250YWluZXI6IGZ1bmN0aW9uICgpIHsKCQkJdmFyICRjb250YWluZXIgPSB0aGlzLmdldCgiY29udGFpbmVyT3B0aW9ucyIpOwoJCQl2YXIgbmFtZSA9ICRjb250YWluZXIubmFtZSB8fCAoJGNvbnRhaW5lci5uYW1lID0gInNjcm9sbGVyIik7CgkJCXRoaXMuY3JlYXRlQ2hyb21lKFskY29udGFpbmVyXSk7CgkJCXRoaXMuZGlzY292ZXJDb250cm9sUGFyZW50KCk7CgkJCWlmIChuYW1lICE9ICJzY3JvbGxlciIpIHsKCQkJCXRoaXMuJC5zY3JvbGxlciA9IHRoaXMuJFtuYW1lXTsKCQkJfQoJCX0sCgoJCS8vKkBwcm90ZWN0ZWQKCQlfbW9kZWxBZGRlZDogZnVuY3Rpb24gKHNlbmRlciwgZXZlbnQpIHsKCQkJaWYgKHNlbmRlciAhPT0gdGhpcy5jb250cm9sbGVyKSB7CgkJCQlyZXR1cm47CgkJCX0KCQkJdmFyICRtb2RlbCA9IGV2ZW50Lm1vZGVsOwoJCQl0aGlzLmFkZCgkbW9kZWwsIGV2ZW50LmluZGV4KTsKCQl9LAoKCQkvLypAcHJvdGVjdGVkCgkJX21vZGVsc0FkZGVkOiBmdW5jdGlvbiAoc2VuZGVyLCBldmVudCkgewoJCQlpZiAoc2VuZGVyICE9PSB0aGlzLmNvbnRyb2xsZXIpIHsKCQkJCXJldHVybjsKCQkJfQoJCQl0aGlzLnNldCgiYmF0Y2hpbmciLCB0cnVlKTsKCQkJZW55by5mb3JFYWNoKGV2ZW50Lm1vZGVscywgZnVuY3Rpb24gKGluZm8pIHsKCQkJCXRoaXMuYWRkKGluZm8ubW9kZWwsIGluZm8uaW5kZXgpOwoJCQl9LCB0aGlzKTsKCQkJdGhpcy5zZXQoImJhdGNoaW5nIiwgZmFsc2UpOwoJCX0sCgoJCS8vKkBwcm90ZWN0ZWQKCQlfbW9kZWxSZW1vdmVkOiBmdW5jdGlvbiAoc2VuZGVyLCBldmVudCkgewoJCQlpZiAoc2VuZGVyICE9PSB0aGlzLmNvbnRyb2xsZXIpIHsKCQkJCXJldHVybjsKCQkJfQoJCQl0aGlzLnJlbW92ZShldmVudC5pbmRleCk7CgkJfSwKCgkJLy8qQHByb3RlY3RlZAoJCV9tb2RlbHNSZW1vdmVkOiBmdW5jdGlvbiAoc2VuZGVyLCBldmVudCkgewoJCQlpZiAoc2VuZGVyICE9PSB0aGlzLmNvbnRyb2xsZXIpIHsKCQkJCXJldHVybjsKCQkJfQoJCQllbnlvLmZvckVhY2goZXZlbnQubW9kZWxzLCBmdW5jdGlvbiAoaW5mbykgewoJCQkJdGhpcy5yZW1vdmUoaW5mby5pbmRleCk7CgkJCX0sIHRoaXMpOwoJCX0sCgoJCS8vIC4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLi4uLgoJCS8vIE9CU0VSVkVSUwoKCQkvLypAcHVibGljCgkJYmF0Y2hpbmdDaGFuZ2VkOiBmdW5jdGlvbiAocHJldiwgdmFsKSB7CgkJCWlmIChmYWxzZSA9PT0gdmFsKSB7CgkJCQl0aGlzLnJlbmRlcigpOwoJCQl9CgkJfSwKCgkJLy8qQHByb3RlY3RlZAoJCV9sZW5ndGhDaGFuZ2VkOiBlbnlvLm9ic2VydmVyKGZ1bmN0aW9uIChwcm9wLCBwcmV2LCB2YWwpIHsKCQkJaWYgKCFpc05hTihwcmV2KSAmJiAhaXNOYU4odmFsKSkgewoJCQkJaWYgKHByZXYgPiB2YWwpIHsKCQkJCQl0aGlzLnBydW5lKCk7CgkJCQl9CgkJCX0KCQl9LCAibGVuZ3RoIikKCgl9KTsKCn0pKGVueW8pOwo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/data/DataTable.js"
Content-Type: application/octet-stream; x-encoding=base64
LyoqCglfZW55by5EYXRhVGFibGVfIGVuYWJsZXMgdGhlIGNyZWF0aW9uIG9mIGRhdGEtZHJpdmVuIHRhYmxlcy4KCUFsb25nIHdpdGggX2VueW8uVGFibGVfLCB0aGlzIGlzIGEgd29yayBpbiBwcm9ncmVzcy4KKi8KZW55by5raW5kKHsKCW5hbWU6ICJlbnlvLkRhdGFUYWJsZSIsCglraW5kOiAiZW55by5EYXRhUmVwZWF0ZXIiLAoJY29udHJvbGxlcjogImVueW8uQ29sbGVjdGlvbiIsCglkZWZhdWx0S2luZDogImVueW8uVGFibGVSb3ciLAoJc3R5bGU6ICJkaXNwbGF5OiB0YWJsZTsiLAoJY29udGFpbmVyT3B0aW9uczogewoJCWtpbmQ6ICJlbnlvLlRhYmxlIiwKCQluYW1lOiAiY29udGFpbmVyIiwKCQlzdHlsZTogIndpZHRoOiAxMDAlOyIKCX0sCglpbml0Q29tcG9uZW50czogZnVuY3Rpb24oKSB7CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCQl0aGlzLl9jaGlsZEtpbmQucHJvdG90eXBlLnRhZyA9IG51bGw7Cgl9Cn0pOw==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/data/RepeaterChildSupport.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5jcmVhdGVNaXhpbih7CgluYW1lOiAiZW55by5SZXBlYXRlckNoaWxkU3VwcG9ydCIsCglkZWNvcmF0ZUV2ZW50OiBmdW5jdGlvbiAoc2VuZGVyLCBldmVudCkgewoJCWV2ZW50Lm1vZGVsID0gdGhpcy5tb2RlbDsKCQl0aGlzLmluaGVyaXRlZChhcmd1bWVudHMpOwoJfQp9KTsK
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/data/RowSupport.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5jcmVhdGVNaXhpbih7CgluYW1lOiAiZW55by5Sb3dTdXBwb3J0IiwKCWRlY29yYXRlRXZlbnQ6IGZ1bmN0aW9uIChuYW1lLCBldmVudCkgewoJCWV2ZW50LnJvdyA9IHRoaXM7CgkJdGhpcy5pbmhlcml0ZWQoYXJndW1lbnRzKTsKCX0KfSk7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/data/data.css"
Content-Type: application/octet-stream; x-encoding=base64
LyogV0FSTklORzogVGhpcyBpcyBhIGdlbmVyYXRlZCBmaWxlIGZvciBiYWNrd2FyZC1jb21wYXRpYmlsaXR5LiAgTW9zdCAgICAgICovCi8qIHVzcnMgc2hvdWxkIGluc3RlYWQgbW9kaWZ5IExFU1MgZmlsZXMuICBJZiB5b3UgY2hvb3NlIHRvIGVkaXQgdGhpcyBDU1MgICAqLwovKiBkaXJlY3RseSByYXRoZXIgdGhhbiBMRVNTIGZpbGVzLCB5b3Ugc2hvdWxkIG1ha2Ugc3VyZSBsZXNzLnh4Lnl5Lm1pbi5qcyAgKi8KLyogaXMgY29tbWVudGVkIG91dCBpbiB5b3VyIGRlYnVnLmh0bWwsIGFuZCBydW4gZGVwbG95LnNoL2JhdCB1c2luZyB0aGUgICAgICovCi8qICctYycgZmxhZyB0byBkaXNhYmxlIExFU1MgY29tcGlsYXRpb24uICBUaGlzIHdpbGwgZm9yY2UgdGhlIGxvYWRlciBhbmQgICAqLwovKiBtaW5pZmllciB0byBmYWxsIGJhY2sgdG8gdXNpbmcgQ1NTIGZpbGVzIGluIHBsYWNlIG9mIHRoZSBzYW1lLW5hbWUgICAgICAgKi8KLyogTEVTUyBmaWxlLiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICovCgouZW55by1kYXRhLWxpc3QuZW55by1kYXRhLWxpc3Qtcm93cy1jb250YWluZXIuZW55by1saXN0LXJvdyB7CiAgcG9zaXRpb246IHJlbGF0aXZlOwp9Ci5lbnlvLWRhdGEtZ3JpZC1saXN0IC5lbnlvLWRhdGEtZ3JpZC1saXN0LWl0ZW0gewogIGRpc3BsYXk6IGlubGluZS1ibG9jazsKfQo=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/data/data.less"
Content-Type: application/octet-stream; x-encoding=base64
Ly8gZm9yIHRoZSBhY3RpdmUgcGFnZXMgY29udGFpbmVyCi5lbnlvLWRhdGEtbGlzdC1hY3RpdmUgewoJcG9zaXRpb246IHJlbGF0aXZlOwoJCgkvLyBmb3IgcGFnZXMKCS5lbnlvLWRhdGEtbGlzdC1wYWdlIHsKCQlwb3NpdGlvbjogYWJzb2x1dGU7Cgl9Cn0=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/data/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJImRhdGEubGVzcyIsCgkiRGF0YVJlcGVhdGVyLmpzIiwKCSJEYXRhTGlzdC5qcyIsCgkiRGF0YUdyaWRMaXN0LmpzIiwKCSJEYXRhVGFibGUuanMiLAoJIlJlcGVhdGVyQ2hpbGRTdXBwb3J0LmpzIiwKCSJSb3dTdXBwb3J0LmpzIgopOw==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/fullscreen.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5mdWxsc2NyZWVuID0gewoJCgkvLyogQHByb3RlY3RlZAoJCgkvLyogUmVmZXJlbmNlIHRvIHRoZSBjdXJyZW50IGZzIGNvbnRyb2wKCWZ1bGxzY3JlZW5Db250cm9sOiBudWxsLAoJLy8qIFJlZmVyZW5jZSB0byB0aGUgY3VycmVudCBmcyBlbGVtZW50IChmYWxsYmFjayBmb3IgcGxhdGZvcm1zIHcvbyBuYXRpdmUgc3VwcG9ydCkKCWZ1bGxzY3JlZW5FbGVtZW50OiBudWxsLAoJLy8qIFJlZmVyZW5jZSB0byBjb250cm9sIHRoYXQgcmVxdWVzdGVkIGZzCglyZXF1ZXN0b3I6IG51bGwsCgkvLyogTmF0aXZlIGFjY2Vzc29yIHVzZWQgdG8gZ2V0IHJlZmVyZW5jZSB0byB0aGUgY3VycmVudCBmcyBlbGVtZW50CgllbGVtZW50QWNjZXNzb3I6CgkJKCJmdWxsc2NyZWVuRWxlbWVudCIgaW4gZG9jdW1lbnQpID8gImZ1bGxzY3JlZW5FbGVtZW50IiA6CgkJKCJtb3pGdWxsU2NyZWVuRWxlbWVudCIgaW4gZG9jdW1lbnQpID8gIm1vekZ1bGxTY3JlZW5FbGVtZW50IiA6CgkJKCJ3ZWJraXRGdWxsc2NyZWVuRWxlbWVudCIgaW4gZG9jdW1lbnQpID8gIndlYmtpdEZ1bGxzY3JlZW5FbGVtZW50IiA6CgkJbnVsbCwKCS8vKiBOYXRpdmUgYWNjZXNzb3IgdXNlZCB0byByZXF1ZXN0IGZzCglyZXF1ZXN0QWNjZXNzb3I6CgkJKCJyZXF1ZXN0RnVsbHNjcmVlbiIgaW4gZG9jdW1lbnQuZG9jdW1lbnRFbGVtZW50KSA/ICJyZXF1ZXN0RnVsbHNjcmVlbiIgOgoJCSgibW96UmVxdWVzdEZ1bGxTY3JlZW4iIGluIGRvY3VtZW50LmRvY3VtZW50RWxlbWVudCkgPyAibW96UmVxdWVzdEZ1bGxTY3JlZW4iIDoKCQkoIndlYmtpdFJlcXVlc3RGdWxsc2NyZWVuIiBpbiBkb2N1bWVudC5kb2N1bWVudEVsZW1lbnQpID8gIndlYmtpdFJlcXVlc3RGdWxsc2NyZWVuIiA6CgkJbnVsbCwKCS8vKiBOYXRpdmUgYWNjZXNzb3IgdXNlZCB0byBjYW5jZWwgZnMKCWNhbmNlbEFjY2Vzc29yOgoJCSgiY2FuY2VsRnVsbFNjcmVlbiIgaW4gZG9jdW1lbnQpID8gImNhbmNlbEZ1bGxTY3JlZW4iIDoKCQkoIm1vekNhbmNlbEZ1bGxTY3JlZW4iIGluIGRvY3VtZW50KSA/ICJtb3pDYW5jZWxGdWxsU2NyZWVuIiA6CgkJKCJ3ZWJraXRDYW5jZWxGdWxsU2NyZWVuIiBpbiBkb2N1bWVudCkgPyAid2Via2l0Q2FuY2VsRnVsbFNjcmVlbiIgOgoJCW51bGwsCgkKCS8vKiBAcHVibGljCgkKCS8vKiBSZXR1cm4gdHJ1ZSBpZiBwbGF0Zm9ybSBzdXBwb3J0cyBhbGwgb2YgdGhlIGZ1bGxzY3JlZW4gQVBJCgluYXRpdmVTdXBwb3J0OiBmdW5jdGlvbigpIHsKCQlyZXR1cm4gKHRoaXMuZWxlbWVudEFjY2Vzc29yICE9PSBudWxsICYmIHRoaXMucmVxdWVzdEFjY2Vzc29yICE9PSBudWxsICYmIHRoaXMuY2FuY2VsQWNjZXNzb3IgIT09IG51bGwpOwoJfSwKCS8vKiBOb3JtYWxpemUgX2dldEZ1bGxzY3JlZW5FbGVtZW50KClfCglnZXRGdWxsc2NyZWVuRWxlbWVudDogZnVuY3Rpb24oKSB7CgkJcmV0dXJuICh0aGlzLm5hdGl2ZVN1cHBvcnQoKSkgPyBkb2N1bWVudFt0aGlzLmVsZW1lbnRBY2Nlc3Nvcl0gOiB0aGlzLmZ1bGxzY3JlZW5FbGVtZW50OwoJfSwKCS8vKiBSZXR1cm4gY3VycmVudCBmcyBjb250cm9sCglnZXRGdWxsc2NyZWVuQ29udHJvbDogZnVuY3Rpb24oKSB7CgkJcmV0dXJuIHRoaXMuZnVsbHNjcmVlbkNvbnRyb2w7Cgl9LAoJLy8qIE5vcm1hbGl6ZSBfcmVxdWVzdEZ1bGxzY3JlZW4oKV8KCXJlcXVlc3RGdWxsc2NyZWVuOiBmdW5jdGlvbihpbkNvbnRyb2wpIHsKCQlpZiAodGhpcy5nZXRGdWxsc2NyZWVuQ29udHJvbCgpIHx8ICEoaW5Db250cm9sLmhhc05vZGUoKSkpIHsKCQkJcmV0dXJuIGZhbHNlOwoJCX0KCgkJdGhpcy5yZXF1ZXN0b3IgPSBpbkNvbnRyb2w7CgoJCS8vIE9ubHkgdXNlIG5hdGl2ZSByZXF1ZXN0IGlmIHBsYXRmb3JtIHN1cHBvcnRzIGFsbCBvZiB0aGUgQVBJCgkJaWYgKHRoaXMubmF0aXZlU3VwcG9ydCgpKSB7CgkJCWluQ29udHJvbC5oYXNOb2RlKClbdGhpcy5yZXF1ZXN0QWNjZXNzb3JdKCk7CgkJfSBlbHNlIHsKCQkJdGhpcy5mYWxsYmFja1JlcXVlc3RGdWxsc2NyZWVuKCk7CgkJfQoKCQlyZXR1cm4gdHJ1ZTsKCX0sCgkvLyogTm9ybWFsaXplIF9jYW5jZWxGdWxsc2NyZWVuKClfCgljYW5jZWxGdWxsc2NyZWVuOiBmdW5jdGlvbigpIHsKCQlpZiAodGhpcy5uYXRpdmVTdXBwb3J0KCkpIHsKCQkJZG9jdW1lbnRbdGhpcy5jYW5jZWxBY2Nlc3Nvcl0oKTsKCQl9IGVsc2UgewoJCQl0aGlzLmZhbGxiYWNrQ2FuY2VsRnVsbHNjcmVlbigpOwoJCX0KCX0sCgkKCS8vKiBAcHJvdGVjdGVkCgkKCS8vKiBGYWxsYmFjayBzdXBwb3J0IGZvciBzZXR0aW5nIGZ1bGxzY3JlZW4gZWxlbWVudCAoZG9uZSBieSBicm93c2VyIG9uIHBsYXRmb3JtcyB3aXRoIG5hdGl2ZSBzdXBwb3J0KQoJc2V0RnVsbHNjcmVlbkVsZW1lbnQ6IGZ1bmN0aW9uKGluTm9kZSkgewoJCXRoaXMuZnVsbHNjcmVlbkVsZW1lbnQgPSBpbk5vZGU7Cgl9LAoJLy8qIFNldCBjdXJyZW50IGZzIGNvbnRyb2wKCXNldEZ1bGxzY3JlZW5Db250cm9sOiBmdW5jdGlvbihpbkNvbnRyb2wpIHsKCQl0aGlzLmZ1bGxzY3JlZW5Db250cm9sID0gaW5Db250cm9sOwoJfSwKCS8vKiBGYWxsYmFjayBmdWxsc2NyZWVuIHJlcXVlc3QgZm9yIHBsYXRmb3JtcyB3aXRob3V0IGZ1bGxzY3JlZW4gc3VwcG9ydAoJZmFsbGJhY2tSZXF1ZXN0RnVsbHNjcmVlbjogZnVuY3Rpb24oKSB7CgkJdmFyIGNvbnRyb2wgPSB0aGlzLnJlcXVlc3RvcjsKCQkKCQlpZiAoIWNvbnRyb2wpIHsKCQkJcmV0dXJuOwoJCX0KCgkJLy8gR2V0IGJlZm9yZSBub2RlIHRvIGFsbG93IHVzIHRvIGV4aXQgZmxvYXRpbmcgbGF5ZXIgdG8gdGhlIHByb3BlciBwb3NpdGlvbgoJCWNvbnRyb2wucHJldkFkZEJlZm9yZSA9IGNvbnRyb2wucGFyZW50LmNvbnRyb2xBdEluZGV4KGNvbnRyb2wuaW5kZXhJbkNvbnRhaW5lcigpICsgMSk7CgoJCS8vIFJlbmRlciBmbG9hdGluZyBsYXllciBpZiB3ZSBuZWVkIHRvCgkJaWYgKCFlbnlvLmZsb2F0aW5nTGF5ZXIuaGFzTm9kZSgpKSB7CgkJCWVueW8uZmxvYXRpbmdMYXllci5yZW5kZXIoKTsKCQl9CgoJCWNvbnRyb2wuYWRkQ2xhc3MoImVueW8tZnVsbHNjcmVlbiIpOwoJCWNvbnRyb2wuYXBwZW5kTm9kZVRvUGFyZW50KGVueW8uZmxvYXRpbmdMYXllci5oYXNOb2RlKCkpOwoJCWNvbnRyb2wucmVzaXplZCgpOwoKCQl0aGlzLnNldEZ1bGxzY3JlZW5Db250cm9sKGNvbnRyb2wpOwoJCXRoaXMuc2V0RnVsbHNjcmVlbkVsZW1lbnQoY29udHJvbC5oYXNOb2RlKCkpOwoJfSwKCS8vKiBGYWxsYmFjayBjYW5jZWwgZnVsbHNjcmVlbiBmb3IgcGxhdGZvcm1zIHdpdGhvdXQgZnVsbHNjcmVlbiBzdXBwb3J0CglmYWxsYmFja0NhbmNlbEZ1bGxzY3JlZW46IGZ1bmN0aW9uKCkgewoJCXZhciBjb250cm9sID0gdGhpcy5mdWxsc2NyZWVuQ29udHJvbCwKCQkJYmVmb3JlTm9kZSwKCQkJcGFyZW50Tm9kZQoJCTsKCgkJaWYgKCFjb250cm9sKSB7CgkJCXJldHVybjsKCQl9CgoJCS8vIEZpbmQgYmVmb3JlTm9kZSBiYXNlZCBvbiBfdGhpcy5hZGRCZWZvcmVfIGFuZCBfdGhpcy5wcmV2QWRkQmVmb3JlXwoJCWJlZm9yZU5vZGUgPSAoY29udHJvbC5wcmV2QWRkQmVmb3JlKSA/IGNvbnRyb2wucHJldkFkZEJlZm9yZS5oYXNOb2RlKCkgOiBudWxsOwoJCXBhcmVudE5vZGUgPSBjb250cm9sLnBhcmVudC5oYXNOb2RlKCk7CgkJY29udHJvbC5wcmV2QWRkQmVmb3JlID0gbnVsbDsKCgkJY29udHJvbC5yZW1vdmVDbGFzcygiZW55by1mdWxsc2NyZWVuIik7CgoJCWlmICghYmVmb3JlTm9kZSkgewoJCQljb250cm9sLmFwcGVuZE5vZGVUb1BhcmVudChwYXJlbnROb2RlKTsKCQl9IGVsc2UgewoJCQljb250cm9sLmluc2VydE5vZGVJblBhcmVudChwYXJlbnROb2RlLCBiZWZvcmVOb2RlKTsKCQl9CgoJCWNvbnRyb2wucmVzaXplZCgpOwoKCQl0aGlzLnNldEZ1bGxzY3JlZW5Db250cm9sKG51bGwpOwoJCXRoaXMuc2V0RnVsbHNjcmVlbkVsZW1lbnQobnVsbCk7Cgl9LAoJLy8qIExpc3RlbiBmb3IgZnMgY2hhbmdlIGV2ZW50IGFuZCBicm9hZGNhc3QgYXMgbm9ybWFsaXplZCBldmVudAoJZGV0ZWN0RnVsbHNjcmVlbkNoYW5nZUV2ZW50OiBmdW5jdGlvbigpIHsKCQl0aGlzLnNldEZ1bGxzY3JlZW5Db250cm9sKHRoaXMucmVxdWVzdG9yKTsKCQl0aGlzLnJlcXVlc3RvciA9IG51bGw7CgoJCS8vIEJyb2FkY2FzdCBjaGFuZ2UKCQllbnlvLlNpZ25hbHMuc2VuZCgib25GdWxsc2NyZWVuQ2hhbmdlIik7Cgl9Cn07CgovLyogTm9ybWFsaXplIHBsYXRmb3JtLXNwZWNpZmljIGZzIGNoYW5nZSBldmVudHMKZW55by5yZWFkeShmdW5jdGlvbigpIHsKCWRvY3VtZW50LmFkZEV2ZW50TGlzdGVuZXIoIndlYmtpdGZ1bGxzY3JlZW5jaGFuZ2UiLCBlbnlvLmJpbmQoZW55by5mdWxsc2NyZWVuLCAiZGV0ZWN0RnVsbHNjcmVlbkNoYW5nZUV2ZW50IiksIGZhbHNlKTsKCWRvY3VtZW50LmFkZEV2ZW50TGlzdGVuZXIoIm1vemZ1bGxzY3JlZW5jaGFuZ2UiLCAgICBlbnlvLmJpbmQoZW55by5mdWxsc2NyZWVuLCAiZGV0ZWN0RnVsbHNjcmVlbkNoYW5nZUV2ZW50IiksIGZhbHNlKTsKCWRvY3VtZW50LmFkZEV2ZW50TGlzdGVuZXIoImZ1bGxzY3JlZW5jaGFuZ2UiLCAgICAgICBlbnlvLmJpbmQoZW55by5mdWxsc2NyZWVuLCAiZGV0ZWN0RnVsbHNjcmVlbkNoYW5nZUV2ZW50IiksIGZhbHNlKTsKfSk7CgovLyogSWYgdGhpcyBwbGF0Zm9ybSBkb2Vzbid0IGhhdmUgbmF0aXZlIHN1cHBvcnQgZm9yIGZ1bGxzY3JlZW4sIGFkZCBhbiBlc2NhcGUgaGFuZGxlciB0byBtaW1pYyBuYXRpdmUgYmVoYXZpb3IKaWYoIWVueW8uZnVsbHNjcmVlbi5uYXRpdmVTdXBwb3J0KCkpIHsKCWVueW8uZGlzcGF0Y2hlci5mZWF0dXJlcy5wdXNoKAoJCWZ1bmN0aW9uKGUpIHsKCQkJaWYgKGUudHlwZSA9PT0gImtleWRvd24iICYmIGUua2V5Q29kZSA9PT0gMjcpIHsKCQkJCWVueW8uZnVsbHNjcmVlbi5jYW5jZWxGdWxsc2NyZWVuKCk7CgkJCX0KCQl9CgkpOwp9Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/package.js"
Content-Type: application/octet-stream; x-encoding=base64
ZW55by5kZXBlbmRzKAoJInVpLmNzcyIsCgkiQW5pbWF0b3IuanMiLAoJIkJhc2VMYXlvdXQuanMiLAoJIkltYWdlLmpzIiwKCSJJbnB1dC5qcyIsCgkiUmljaFRleHQuanMiLAoJIlRleHRBcmVhLmpzIiwKCSJTZWxlY3QuanMiLAoJIkdyb3VwLmpzIiwKCSJHcm91cEl0ZW0uanMiLAoJIlRvb2xEZWNvcmF0b3IuanMiLAoJIkJ1dHRvbi5qcyIsCgkiQ2hlY2tib3guanMiLAoJIlJlcGVhdGVyLmpzIiwKCSJEcmFnQXZhdGFyLmpzIiwKCSJGbG9hdGluZ0xheWVyLmpzIiwKCSJQb3B1cC5qcyIsCgkiU2VsZWN0aW9uLmpzIiwKCSJEcmF3ZXIuanMiLAoJIlRhYmxlLmpzIiwKCSJmdWxsc2NyZWVuLmpzIiwKCSJNZWRpYS5qcyIsCgkiQXVkaW8uanMiLAoJImRhdGEiCik7Cg==
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/source/ui/ui.css"
Content-Type: application/octet-stream; x-encoding=base64
LmVueW8taW5saW5lLCAuZW55by10b29sLWRlY29yYXRvciB7CglkaXNwbGF5OiBpbmxpbmUtYmxvY2s7Cn0KCi5lbnlvLWNoaWxkcmVuLWlubGluZSA+ICosIC5lbnlvLXRvb2wtZGVjb3JhdG9yID4gKiB7CglkaXNwbGF5OiBpbmxpbmUtYmxvY2s7Cn0KCi5lbnlvLWNoaWxkcmVuLW1pZGRsZSA+ICosIC5lbnlvLXRvb2wtZGVjb3JhdG9yID4gKiB7Cgl2ZXJ0aWNhbC1hbGlnbjogbWlkZGxlOwp9CgouZW55by1wb3NpdGlvbmVkIHsKCXBvc2l0aW9uOiByZWxhdGl2ZTsKfQoKLmVueW8tZmlsbCB7Cglwb3NpdGlvbjogcmVsYXRpdmU7Cgl3aWR0aDogMTAwJTsKCWhlaWdodDogMTAwJTsKfQoKLmVueW8tcG9wdXAgewoJcG9zaXRpb246IGFic29sdXRlOwoJei1pbmRleDogMTA7Cn0=
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/tools/README.md"
Content-Type: application/octet-stream; x-encoding=base64
IyBFbnlvIE1pbmlmaWVyCgpUaGUgZW55byBtaW5pZmllciB1c2VzIE5vZGUuanMsIFtVZ2xpZnlKU10oaHR0cDovL2dpdGh1Yi5jb20vbWlzaG9vL3VnbGlmeWpzKSBhbmQgdGhlIGVueW8gZGVwZW5kZW5jeSBsb2FkZXIgdG8gY29tcHJlc3MgYW55IGVueW8gcGFja2FnZSBpbnRvIGEgbWluaW1pemVkIGZvcm0uCgojIyBJbnZvY2F0aW9uCkZvciBjb252ZW5pZW5jZSwgdGhlcmUgYXJlIGJvdGggV2luZG93cyBhbmQgVW5peCB2ZXJzaW9ucyBvZiB0aGUgc2NyaXB0IHRoYXQgaW52b2tlcyB0aGUgbm9kZSB0b29sLgpUaGV5IGZvbGxvdyB0aGUgc2FtZSBpbnZvY2F0aW9uOgoKCXBhdGgvdG8vZW55by90b29scy9taW5pZnkvbWluaWZ5LnNoIHBhY2thZ2UuanMgLW91dHB1dCAvcmVsYXRpdmUvcGF0aC90by9idWlsZC9kaXIvYnVpbGRmaWxlbmFtZQoKQW4gZXhhbXBsZSBmb3IgbGliL29ueXgsIGEgVUkgd2lkZ2V0IHNldCwgcnVuIGZyb20gYGxpYi9vbnl4L21pbmlmeS9taW5pZnkuc2hgLCBidWlsZGluZyB0byBgbGliL29ueXgvYnVpbGRgCgoJLi4vLi4vLi4vZW55by90b29scy9taW5pZnkvbWluaWZ5LnNoIC1vdXRwdXQgLi4vYnVpbGQvb255eCBwYWNrYWdlLmpzCgpUaGUgbGFzdCBwYXJhbWV0ZXIgaW4gdGhlIG91dHB1dCBwYXRoLCBpZiBub3QgZW5kaW5nIGluIGEgYC9gLCB3aWxsIGJlIHVzZWQgYXMgdGhlIG5hbWUgb2YgdGhlIG91dHB1dCBidWlsZCBmaWxlcy4KVGhlIGBwYWNrYWdlLmpzYCBmaWxlIG11c3QgYmUgaW4gdGhlIHNhbWUgZGlyZWN0b3J5IGFzIHRoZSBpbnZvY2F0aW9uLgoKRm9yIGNvbnZlbmllbmNlLCBwYWNrYWdlcyBzaG91bGQgaW5jbHVkZSBhIGBtaW5pZnlgIGZvbGRlciB3aXRoIGJvdGggYSBXaW5kb3dzIGJhdGNoIGFuZCBVbml4IHNoZWxsIHNjcmlwdCB0aGF0IHJ1bnMgdGhlIG1pbmlmaWVyLgoKIyMgUnVubmluZyB0ZXN0cwoxLiBZb3UgZmlyc3QgbmVlZCB0byBpbnN0YWxsIGFuIGh0dHAgc2VydmVyIGhhbmRsaW5nIHBocCAoTUFOUCBvciBXQU1QLCAuLi4pCjIuIENvbmZpZ3VyZSB5b3VyIGh0dHAgc2VydmVyIHRvIHNlcnZlIHRoZSBmaWxlcyBvZiB0aGUgZW55byBwcm9qZWN0CjMuIFBvaW50IHlvdXIgYnJvd3NlciB0byBlbnlvL3Rvb2xzLy4uLiB0byBydW4gdGhlIHZhcmlvdXMgdGVzdGNhc2Vz
----------------------------385268511887432987672305
Content-Disposition: form-data; name="file"; filename="enyo/tools/deploy.js"
Content-Type: application/octet-stream; x-encoding=base64
#!/usr/bin/env node
/* jshint node: true */
/**
# deploy.js - portable deployment script

This portable Node.js script minifies both your application, its
libraries & the Enyo framework it is using.  The resulting application
is suitable for production usage, either hostet on a web-server or
embedded into a PhoneGap container.

The script is intended to be run from the application's root directory
(This is unlike previous deprecated incarnations of the scripts
`deploy.sh` and `deploy.bat` that shipped within the `bootplate`
application), where it expects to find a file called `package.js`
(unless the user specifies an alternate location using the `-p` flag).

This script comes along with Enyo.  It automatically uses & embeds the
Enyo version it is shipped with, unless the user specifies another
Enyo version using the `-e enyo_dir` flag.

When you application has library dependencies (for example
`lib/mylib`), this script uses the `lib/mylib/manifest.json` (if
present) or the old-fashioned deployment scripts `lib/mylib/deploy.sh`
(on Linux & Mac OSX) or `lib/mylib/deploy.bat` (on Windows).
If neither exist, then the entire library is copied (except for .git dir).

This script also expects to find the following files & folders in the
application root directory.  Each of them is copied verbatim in the
output production/deployment folder (the one optionally given using
the `-o` flag).

* `index.html`, the application startup page
* `icon.png`, the application icon
* `assets/`holds the static application assets, such as images,
  videos... etc.

 */

// Load dependencies
var nopt = require("nopt"),
    path = require('path'),
    fs = require('fs'),
    shell = require('shelljs');

var stat, script, scripts = {};

// Send message to parent node process, if any
process.on('uncaughtException', function (err) {
	var errMsg = err.toString() + err.stack;
	console.error(errMsg);
	if (process.send) {
		// only available if parent process is node
		process.send({error: errMsg});
	}
	process.exit(1);
});
// receive error messages from child node processes
process.on('message', function(msg) {
	console.dir(basename, msg);
	if (msg.error && msg.error.stack) {
		console.error(basename, msg.error.stack);
	}
	if (process.send) {
		process.send(msg);
	}
});

// Parse arguments

var node = process.argv[0],
    deploy = process.argv[1],
    sourceDir = process.cwd(),
    packageJs = path.resolve(sourceDir, "package.js"),
    enyoDir = path.resolve(__dirname, '..'),
    buildDir = path.resolve(sourceDir, "build"),
    basename = path.basename(sourceDir),
    outDir = path.resolve(sourceDir, 'deploy', basename),
    less = true, // LESS compilation, turned on by default
    verbose = false;

function printUsage() {
	// format generated using node-optimist...
	console.log('\n' +
		'Usage: ' + node + ' ' + deploy + ' [-c][-e enyo_dir][-l lib_dir][-b build_dir][-o out_dir][-p package_js][-s source_dir][-f map_from -t map_to ...]\n' +
		'\n' +
		'Options:\n' +
		'  -v  verbose operation                     [boolean]  [default: ' + verbose + ']\n' +
		'  -b  alternate build directory             [default: "' + buildDir + '"]\n' +
		'  -c  do not run the LESS compiler          [boolean]  [default: ' + less + ']\n' +
		'  -e  location of the enyo framework        [default: "' + enyoDir + '"]\n' +
		'  -l  location of the lib folder            [default: "' + enyoDir + '/../lib"]\n' +
		'  -o  alternate output directory            [default: "' + outDir + '"]\n' +
		'  -p  location of the main package.js file  [default: "' + packageJs + '"]\n' +
		'  -s  source code root directory            [default: "' + sourceDir + '"]\n' +
		'  -f  remote source mapping: from local path' +
		'  -t  remote source mapping: to remote path' +
		'\n');
}

var opt = nopt(/*knownOpts*/ {
	"build": path,
	"less": Boolean,
	"enyo": path,
	"lib": path,
	"out": path,
	"packagejs": path,
	"source": path,
	"verbose": Boolean,
	"help": Boolean,
	"mapfrom": [String, Array],
	"mapto": [String, Array]
}, /*shortHands*/ {
	"b": "--build",
	"c": "--no-less",
	"e": "--enyo",
	"l": "--lib",
	"o": "--out",
	"p": "--packagejs",
	"s": "--source",
	"v": "--verbose",
	"h": "--help",
	"f": "--mapfrom",
	"t": "--mapto",
	"?": "--help"
}, process.argv /*args*/, 2 /*slice*/);

if (opt.help) {
	printUsage();
	process.exit(1);
}

// nopt has not default values system
buildDir = opt.build || buildDir;
enyoDir = opt.enyo || enyoDir;
outDir = opt.out || outDir;
packageJs = opt.packagejs ||
	(opt.source ? path.join(opt.source, 'package.js') : undefined) ||
	packageJs;
sourceDir = opt.source ||
	(opt.packagejs ? path.dirname(opt.packagejs) : undefined) ||
	sourceDir;
less = (opt.less !== false) && less;
verbose = opt.verbose;

if ((opt.mapfrom || opt.maptop) && (!opt.mapfrom || !opt.mapto || (opt.mapfrom.length != opt.mapto.length))) {
	console.log("mapfrom:", opt.mapfrom);
	console.log("mapto:", opt.mapto);
	console.error("Error: The number of 'mapfrom' and 'mapto' arguments must match.");
	process.exit(1);
}

var minifier = path.resolve(enyoDir, 'tools', 'minifier', 'minify.js');
if (verbose) {
	console.log("Using: build_dir=" + buildDir);
	console.log("Using: enyo_dir=" + enyoDir);
	console.log("Using: out_dir=" + outDir);
	console.log("Using: packagejs=" + packageJs);
	console.log("Using: source_dir=" + sourceDir);
	console.log("Using: less=" + less);
}

// utils

function run(args) {
	var command = '"' + args.join('" "') + '"';
	var report;
	if (verbose) {
		console.log("Running: '", command, "' from '", process.cwd(), "'");
	}
	report = shell.exec(command, { silent: true });
	if (report.code !== 0) {
		throw new Error("Fail: '" + command + "'\n" + report.output);
	}
}

// Prepare target directory

shell.rm('-rf', path.resolve(outDir));
shell.rm('-rf', path.resolve(outDir));
shell.mkdir('-p', path.join(outDir));

// Build / Minify
var args;
if (!opt.mapfrom || opt.mapfrom.indexOf("enyo") < 0) {
	console.log("Minify-ing Enyo...");
	process.chdir(path.resolve(enyoDir, 'minify'));
	args = [node, minifier,
		'-no-alias',
		'-enyo', enyoDir,
		// XXX generates $buildDir/enyo.(js|css)' so this is
		// XXX rather an 'output_prefix' than an 'out_dir'...
		'-output', path.join(buildDir, 'enyo'),
		'package.js'];
	if (opt.mapfrom) {
		for (var i=0; i<opt.mapfrom.length; i++) {
			args.push("-f", opt.mapfrom[i], "-t", opt.mapto[i]);
		}
	}
	run(args);	
} else {
	console.log("Skipping Enyo minification (will be mapped to " + opt.mapto[opt.mapfrom.indexOf("enyo")] + ").");
}

console.log("Minify-ing the application...");
process.chdir(path.dirname(packageJs));
args = [node, minifier,
     '-enyo', enyoDir,
     '-output', path.join(buildDir, 'app'),
     (less ? '-less' : '-no-less'),
     'package.js'];
if (opt.mapfrom) {
	for (var i=0; i<opt.mapfrom.length; i++) {
		args.push("-f", opt.mapfrom[i], "-t", opt.mapto[i]);
	}
}
if (opt.lib) {
	args.push("-lib", opt.lib);
}
run(args);
process.chdir(sourceDir);

// Deploy / Copy

shell.mkdir('-p', path.join(outDir, 'lib'));
shell.cp(path.join(sourceDir, 'index.html'), path.join(sourceDir, 'icon.png'), outDir);
shell.cp('-r', buildDir, outDir);

var assetsSrcDir = path.join(sourceDir, 'assets');
if(shell.test('-d', assetsSrcDir)) {
	shell.cp('-r', assetsSrcDir, outDir);
}

var libSrcDir = path.join(sourceDir, 'lib');
if(shell.test('-d', libSrcDir)) {
	shell.ls(libSrcDir).forEach(deployLib);
}

function deployLib(lib) {
	if (opt.mapfrom && opt.mapfrom.indexOf("lib/" + lib) >= 0) {
		// Don't deploy libraries that are loaded from elsewhere
		return;
	}
	var libOutdir = path.join(outDir, 'lib', lib);
	// load & execute sub-'deploy.js'
	try {
		script = path.join(sourceDir, 'lib', lib, 'deploy.js');
		stat = fs.statSync(script);
		if (!stat.isFile()) {
			throw new Error("*** Not a file: '" + script + "'");
		}
		scripts[lib] = require(script);
		scripts[
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment