From 4e4acbc97877ee076d6ff9b95369f2742f60aaf9 Mon Sep 17 00:00:00 2001 From: hextheplanet Date: Sun, 27 Jul 2025 18:42:23 -0700 Subject: [PATCH] and add the node modules lol --- node_modules/pebble-clay/LICENSE | 21 + node_modules/pebble-clay/README.md | 1267 ++++++ node_modules/pebble-clay/dist.zip | Bin 0 -> 128652 bytes .../dist/binaries/aplite/libpebble-clay.a | 1 + .../dist/binaries/basalt/libpebble-clay.a | 1 + .../dist/binaries/chalk/libpebble-clay.a | 1 + .../dist/binaries/diorite/libpebble-clay.a | 1 + .../dist/binaries/emery/libpebble-clay.a | 1 + .../aplite/src/resource_ids.auto.h | 8 + .../basalt/src/resource_ids.auto.h | 8 + .../pebble-clay/chalk/src/resource_ids.auto.h | 8 + .../diorite/src/resource_ids.auto.h | 8 + .../pebble-clay/emery/src/resource_ids.auto.h | 8 + .../include/pebble-clay/message_keys.auto.h | 8 + node_modules/pebble-clay/dist/js/index.js | 15 + node_modules/pebble-clay/index.js | 379 ++ node_modules/pebble-clay/package.json | 119 + node_modules/pebble-clay/src/config-page.html | 20 + .../pebble-clay/src/images/example.png | Bin 0 -> 138670 bytes node_modules/pebble-clay/src/js/index.js | 15 + .../src/scripts/components/button.js | 13 + .../src/scripts/components/checkboxgroup.js | 13 + .../src/scripts/components/color.js | 327 ++ .../src/scripts/components/footer.js | 7 + .../src/scripts/components/heading.js | 10 + .../src/scripts/components/index.js | 16 + .../src/scripts/components/input.js | 13 + .../src/scripts/components/radiogroup.js | 14 + .../src/scripts/components/select.js | 33 + .../src/scripts/components/slider.js | 50 + .../src/scripts/components/submit.js | 11 + .../src/scripts/components/text.js | 7 + .../src/scripts/components/toggle.js | 13 + .../pebble-clay/src/scripts/config-page.js | 39 + .../src/scripts/lib/clay-config.js | 314 ++ .../src/scripts/lib/clay-events.js | 101 + .../pebble-clay/src/scripts/lib/clay-item.js | 78 + .../src/scripts/lib/component-registry.js | 4 + .../src/scripts/lib/manipulators.js | 180 + .../pebble-clay/src/scripts/lib/utils.js | 135 + .../src/scripts/vendor/autoprefixify.js | 51 + .../src/scripts/vendor/minified.js | 3602 +++++++++++++++++ .../pebble-clay/src/styles/_clay.scss | 2 + .../pebble-clay/src/styles/clay/_base.scss | 201 + .../pebble-clay/src/styles/clay/_fonts.scss | 6 + .../pebble-clay/src/styles/clay/_mixins.scss | 43 + .../pebble-clay/src/styles/clay/_reset.scss | 58 + .../pebble-clay/src/styles/clay/_vars.scss | 39 + .../src/styles/clay/components/button.scss | 12 + .../styles/clay/components/checkboxgroup.scss | 70 + .../src/styles/clay/components/color.scss | 98 + .../src/styles/clay/components/input.scss | 49 + .../styles/clay/components/radiogroup.scss | 68 + .../src/styles/clay/components/select.scss | 43 + .../src/styles/clay/components/slider.scss | 127 + .../src/styles/clay/components/submit.scss | 3 + .../src/styles/clay/components/toggle.scss | 55 + .../src/styles/clay/elements/_button.scss | 35 + .../pebble-clay/src/styles/config-page.scss | 4 + .../src/templates/components/button.tpl | 11 + .../templates/components/checkboxgroup.tpl | 15 + .../src/templates/components/color.tpl | 20 + .../src/templates/components/footer.tpl | 1 + .../src/templates/components/heading.tpl | 3 + .../src/templates/components/input.tpl | 15 + .../src/templates/components/radiogroup.tpl | 20 + .../src/templates/components/select.tpl | 22 + .../src/templates/components/slider.tpl | 25 + .../src/templates/components/submit.tpl | 7 + .../src/templates/components/text.tpl | 3 + .../src/templates/components/toggle.tpl | 19 + 71 files changed, 7994 insertions(+) create mode 100755 node_modules/pebble-clay/LICENSE create mode 100755 node_modules/pebble-clay/README.md create mode 100644 node_modules/pebble-clay/dist.zip create mode 100644 node_modules/pebble-clay/dist/binaries/aplite/libpebble-clay.a create mode 100644 node_modules/pebble-clay/dist/binaries/basalt/libpebble-clay.a create mode 100644 node_modules/pebble-clay/dist/binaries/chalk/libpebble-clay.a create mode 100644 node_modules/pebble-clay/dist/binaries/diorite/libpebble-clay.a create mode 100644 node_modules/pebble-clay/dist/binaries/emery/libpebble-clay.a create mode 100644 node_modules/pebble-clay/dist/include/pebble-clay/aplite/src/resource_ids.auto.h create mode 100644 node_modules/pebble-clay/dist/include/pebble-clay/basalt/src/resource_ids.auto.h create mode 100644 node_modules/pebble-clay/dist/include/pebble-clay/chalk/src/resource_ids.auto.h create mode 100644 node_modules/pebble-clay/dist/include/pebble-clay/diorite/src/resource_ids.auto.h create mode 100644 node_modules/pebble-clay/dist/include/pebble-clay/emery/src/resource_ids.auto.h create mode 100644 node_modules/pebble-clay/dist/include/pebble-clay/message_keys.auto.h create mode 100644 node_modules/pebble-clay/dist/js/index.js create mode 100755 node_modules/pebble-clay/index.js create mode 100644 node_modules/pebble-clay/package.json create mode 100755 node_modules/pebble-clay/src/config-page.html create mode 100644 node_modules/pebble-clay/src/images/example.png create mode 100644 node_modules/pebble-clay/src/js/index.js create mode 100644 node_modules/pebble-clay/src/scripts/components/button.js create mode 100644 node_modules/pebble-clay/src/scripts/components/checkboxgroup.js create mode 100644 node_modules/pebble-clay/src/scripts/components/color.js create mode 100644 node_modules/pebble-clay/src/scripts/components/footer.js create mode 100644 node_modules/pebble-clay/src/scripts/components/heading.js create mode 100644 node_modules/pebble-clay/src/scripts/components/index.js create mode 100644 node_modules/pebble-clay/src/scripts/components/input.js create mode 100644 node_modules/pebble-clay/src/scripts/components/radiogroup.js create mode 100644 node_modules/pebble-clay/src/scripts/components/select.js create mode 100644 node_modules/pebble-clay/src/scripts/components/slider.js create mode 100644 node_modules/pebble-clay/src/scripts/components/submit.js create mode 100644 node_modules/pebble-clay/src/scripts/components/text.js create mode 100644 node_modules/pebble-clay/src/scripts/components/toggle.js create mode 100755 node_modules/pebble-clay/src/scripts/config-page.js create mode 100644 node_modules/pebble-clay/src/scripts/lib/clay-config.js create mode 100644 node_modules/pebble-clay/src/scripts/lib/clay-events.js create mode 100644 node_modules/pebble-clay/src/scripts/lib/clay-item.js create mode 100755 node_modules/pebble-clay/src/scripts/lib/component-registry.js create mode 100755 node_modules/pebble-clay/src/scripts/lib/manipulators.js create mode 100644 node_modules/pebble-clay/src/scripts/lib/utils.js create mode 100644 node_modules/pebble-clay/src/scripts/vendor/autoprefixify.js create mode 100644 node_modules/pebble-clay/src/scripts/vendor/minified.js create mode 100755 node_modules/pebble-clay/src/styles/_clay.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/_base.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/_fonts.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/_mixins.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/_reset.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/_vars.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/components/button.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/components/checkboxgroup.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/components/color.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/components/input.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/components/radiogroup.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/components/select.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/components/slider.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/components/submit.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/components/toggle.scss create mode 100644 node_modules/pebble-clay/src/styles/clay/elements/_button.scss create mode 100644 node_modules/pebble-clay/src/styles/config-page.scss create mode 100755 node_modules/pebble-clay/src/templates/components/button.tpl create mode 100755 node_modules/pebble-clay/src/templates/components/checkboxgroup.tpl create mode 100755 node_modules/pebble-clay/src/templates/components/color.tpl create mode 100755 node_modules/pebble-clay/src/templates/components/footer.tpl create mode 100755 node_modules/pebble-clay/src/templates/components/heading.tpl create mode 100755 node_modules/pebble-clay/src/templates/components/input.tpl create mode 100755 node_modules/pebble-clay/src/templates/components/radiogroup.tpl create mode 100755 node_modules/pebble-clay/src/templates/components/select.tpl create mode 100644 node_modules/pebble-clay/src/templates/components/slider.tpl create mode 100755 node_modules/pebble-clay/src/templates/components/submit.tpl create mode 100644 node_modules/pebble-clay/src/templates/components/text.tpl create mode 100755 node_modules/pebble-clay/src/templates/components/toggle.tpl diff --git a/node_modules/pebble-clay/LICENSE b/node_modules/pebble-clay/LICENSE new file mode 100755 index 0000000..40ebd8e --- /dev/null +++ b/node_modules/pebble-clay/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2016 Pebble Technology + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/node_modules/pebble-clay/README.md b/node_modules/pebble-clay/README.md new file mode 100755 index 0000000..d31c584 --- /dev/null +++ b/node_modules/pebble-clay/README.md @@ -0,0 +1,1267 @@ +# Clay +Clay is a JavaScript library that makes it easy to add offline configuration pages to your Pebble apps. All you need to get started is a couple lines of JavaScript and a JSON file; no servers or HTML required. + +Clay will by default automatically handle the 'showConfiguration' and 'webviewclosed' events traditionally implemented by developers to relay configuration settings to the watch side of the app. This step is not required when using Clay, since each config item is given the same `messageKey` as defined in `package.json` (or PebbleKit JS Message Keys on CloudPebble), and is automatically transmitted once the configuration page is submitted by the user. Developers can override this behavior by [handling the events manually](#handling-the-showconfiguration-and-webviewclosed-events-manually). + +Clay is distributed as a [Pebble package](https://developer.pebble.com/guides/pebble-packages/) so it is super easy to include in your project. If you are upgrading from v0.1.x of Clay you need to follow the [migration guide](#migrating-from-v01x-to-v1x) before you can get started. + +If you would like to contribute to Clay, check out the [contributing guide.](CONTRIBUTING.md) + +# Getting Started (SDK 3.13 or higher) + +1. Run `pebble package install pebble-clay` to install the package in your project +2. Create a JSON file called `config.json` and place it in your `src/js` directory. +3. In order for JSON files to work you may need to change a line in your `wscript` from `ctx.pbl_bundle(binaries=binaries, js=ctx.path.ant_glob('src/js/**/*.js'))` to `ctx.pbl_bundle(binaries=binaries, js=ctx.path.ant_glob(['src/js/**/*.js', 'src/js/**/*.json']))`. +4. Your `index.js` (`app.js` in SDK 3) file needs to `require` clay and your config file, then be initialized. Clay will by default automatically handle the 'showConfiguration' and 'webviewclosed' events. Copy and paste the following into the top of your `index.js` file: + + ```javascript + var Clay = require('pebble-clay'); + var clayConfig = require('./config.json'); + var clay = new Clay(clayConfig); + ``` +5. Ensure `pebble.enableMultiJS` is set to true in your `package.json`. +6. Next is the fun part - creating your config page. Edit your `config.json` file to build a layout of elements as described in the sections below. +7. Make sure you have defined all of your `messageKeys` in your `package.json`. More info on how that works [here.](https://developer.pebble.com/guides/communication/using-pebblekit-js/#defining-keys) + +# Getting Started (CloudPebble) + +1. Ensure `JS Handling` is set to `CommonJS-style` in your project settings. +2. Under `Dependencies` in the project navigation, enter `pebble-clay` as the `Package Name` and `^1.0.0` for the `Version`. You may use any specific version you like, however using `^1.0.0` will ensure you receive all minor version updates. +4. Create a JavaScript file called `config.js` with the following content. This will act as your config's root array element, from which the rest of the page is built up: + + ```javascript + module.exports = []; + ``` +5. Your `index.js` file needs to `require` clay and your config file, then be initialized. Clay will by default automatically handle the 'showConfiguration' and 'webviewclosed' events. Copy and paste the following into the top of your `app.js` file: + + ```javascript + var Clay = require('pebble-clay'); + var clayConfig = require('./config'); + var clay = new Clay(clayConfig); + ``` +6. Next is the fun part - creating your config page. Edit your `config.js` file to build a layout of elements as described in the sections below. +7. Make sure you have defined all of your message keys using `Automatic assignment` in your project settings. More info on how that works [here.](https://developer.pebble.com/guides/communication/using-pebblekit-js/#defining-keys) + +# Getting Started (Pebble.js) +If you are using [Pebble.js](https://developer.pebble.com/docs/pebblejs/) and would like to use Clay, The setup process is a little different. Pebble.js does not currently support message keys so you will have to use [v0.1.7](https://github.com/pebble/clay/releases/v0.1.7) of Clay. Follow the instructions in the [readme for that version.](https://github.com/pebble/clay/blob/v0.1.7/README.md) + +# Getting Started (Rocky.js) +If you are using [Rocky.js](https://developer.pebble.com/docs/rockyjs/) and would like to use Clay, please be aware that this is currently unsupported. It is possible to install the Clay package and override the 'showConfiguration' and 'webviewclosed' events and handle them manually, but Rocky.js does not currently support persistent storage, so the settings must be loaded from the phone each time. You can find an example of using Clay with Rocky.js [here](https://github.com/orviwan/rocky-leco-clay). + +# Creating Your Config File + +Clay uses JavaScript object notation (or JSON) to generate the config page for you. The structure of the page is up to you, but you do need to follow some basic rules. + +## Basic Config Structure + +Your root element **must** be an array. This represents the entire page. Inside this array you place your config items. Each config item is an object with some properties that configure how each item should be displayed. + +#### Example + +NOTE for config.js (rather than config.json) a leading `module.exports =` is required. + +```javascript +[ + { + "type": "heading", + "defaultValue": "Example Header Item" + }, + { + "type": "text", + "defaultValue": "Example text item." + } +] +``` + +## Components + + + +### Section + +Sections help divide up the page into logical groups of items. It is recommended that you place all your input-based items in at least one section. + +##### Properties + +| Property | Type | Description | +|----------|------|-------------| +| type | string | Set to `section`. | +| items | array | Array of items to include in this section. | +| capabilities | array | Array of features that the connected watch must have for this section to be present | + +##### Example + +```javascript +{ + "type": "section", + "items": [ + { + "type": "heading", + "defaultValue": "This is a section" + }, + { + "type": "input", + "messageKey": "email", + "label": "Email Address" + }, + { + "type": "toggle", + "messageKey": "enableAnimations", + "label": "Enable Animations" + } + ] +} +``` + +--- + +### Heading + +**Manipulator:** [`html`](#html) + +Headings can be used in anywhere and can have their size adjusted to suit the context. If you place a heading item at the first position of a section's `items` array then it will automatically be styled as the header for that section. + +##### Properties + +| Property | Type | Description | +|----------|------|-------------| +| type | string | Set to `heading`. | +| id | string (unique) | Set this to a unique string to allow this item to be looked up using `Clay.getItemById()` in your [custom function](#custom-function). | +| messageKey | string (unique) | The AppMessage key matching the `messageKey` item defined in your `package.json`. Set this to a unique string to allow this item to be looked up using `Clay.getItemsByMessageKey()` in your custom function. You must set this if you wish for the value of this item to be persisted after the user closes the config page. | +| defaultValue | string/HTML | The heading's text. | +| size | int | Defaults to `4`. An integer from 1 to 6 where 1 is the largest size and 6 is the smallest. (represents HTML `

`, `

`, `

`, etc). | +| capabilities | array | Array of features that the connected watch must have for this item to be present | +| group | string | Set this to allow this item, along with other items sharing the same group to be looked up using `Clay.getItemsByGroup()` in your [custom function](#custom-function) | + + +##### Example + +```javascript +{ + "type": "heading", + "id": "main-heading", + "defaultValue": "My Cool Watchface", + "size": 1 +} +``` + +--- + +### Text + +**Manipulator:** [`html`](#html) + +Text is used to provide descriptions of sections or to explain complex parts of your page. Feel free to add any extra HTML you require to the `defaultValue` + +##### Properties + +| Property | Type | Description | +|----------|------|-------------| +| type | string | Set to `text`. | +| id | string (unique) | Set this to a unique string to allow this item to be looked up using `Clay.getItemById()` in your [custom function](#custom-function). | +| messageKey | string (unique) | The AppMessage key matching the `messageKey` item defined in your `package.json`. Set this to a unique string to allow this item to be looked up using `Clay.getItemsByMessageKey()` in your custom function. You must set this if you wish for the value of this item to be persisted after the user closes the config page. | +| defaultValue | string/HTML | The content of the text element. | +| capabilities | array | Array of features that the connected watch must have for this item to be present | +| group | string | Set this to allow this item, along with other items sharing the same group to be looked up using `Clay.getItemsByGroup()` in your [custom function](#custom-function) | + + +##### Example + +```javascript +{ + "type": "text", + "defaultValue": "This will be displayed in the text element!", +} +``` + +--- + +### Input + +**Manipulator:** [`val`](#val) + +Standard text input field. + +##### Properties + +| Property | Type | Description | +|----------|------|-------------| +| type | string | Set to `input`. | +| id | string (unique) | Set this to a unique string to allow this item to be looked up using `Clay.getItemById()` in your [custom function](#custom-function). | +| messageKey | string (unique) | The AppMessage key matching the `messageKey` item defined in your `package.json`. Set this to a unique string to allow this item to be looked up using `Clay.getItemsByMessageKey()` in your custom function. You must set this if you wish for the value of this item to be persisted after the user closes the config page. | +| label | string | The label that should appear next to this item. | +| defaultValue | string | The default value of the input field. | +| description | string | Optional sub-text to include below the component | +| attributes | object | An object containing HTML attributes to set on the input field. Set `type` to values such as "email", "time", "date" etc to adjust the behavior of the component. | +| capabilities | array | Array of features that the connected watch must have for this item to be present | +| group | string | Set this to allow this item, along with other items sharing the same group to be looked up using `Clay.getItemsByGroup()` in your [custom function](#custom-function) | + + +##### Example + +```javascript +{ + "type": "input", + "messageKey": "email", + "defaultValue": "", + "label": "Email Address", + "attributes": { + "placeholder": "eg: name@domain.com", + "limit": 10, + "type": "email" + } +} +``` + +--- + +#### Toggle + +**Manipulator:** [`checked`](#checked) + +Switch for a single item. + +##### Properties + +| Property | Type | Description | +|----------|------|-------------| +| type | string | Set to `toggle`. | +| id | string (unique) | Set this to a unique string to allow this item to be looked up using `Clay.getItemById()` in your [custom function](#custom-function). | +| messageKey | string (unique) | The AppMessage key matching the `messageKey` item defined in your `package.json`. Set this to a unique string to allow this item to be looked up using `Clay.getItemsByMessageKey()` in your custom function. You must set this if you wish for the value of this item to be persisted after the user closes the config page. | +| label | string | The label that should appear next to this item. | +| defaultValue | int\|boolean | The default value of the toggle. Defaults to `false` if not specified. | +| description | string | Optional sub-text to include below the component | +| capabilities | array | Array of features that the connected watch must have for this item to be present | +| group | string | Set this to allow this item, along with other items sharing the same group to be looked up using `Clay.getItemsByGroup()` in your [custom function](#custom-function) | + + +##### Example + +```javascript +{ + "type": "toggle", + "messageKey": "invert", + "label": "Invert Colors", + "defaultValue": true +} +``` + +--- + +#### Select + +**Manipulator:** [`val`](#val) + +A dropdown menu containing multiple options. + +##### Properties + +| Property | Type | Description | +|----------|------|-------------| +| type | string | Set to `select`. | +| id | string (unique) | Set this to a unique string to allow this item to be looked up using `Clay.getItemById()` in your [custom function](#custom-function). | +| messageKey | string (unique) | The AppMessage key matching the `messageKey` item defined in your `package.json`. Set this to a unique string to allow this item to be looked up using `Clay.getItemsByMessageKey()` in your custom function. You must set this if you wish for the value of this item to be persisted after the user closes the config page. | +| label | string | The label that should appear next to this item. | +| defaultValue | string | The default value of the dropdown menu. Must match a value in the `options` array. | +| description | string | Optional sub-text to include below the component | +| options | array of objects | The options you want to appear in the dropdown menu. Each option is an object with a `label` and `value` property. | +| capabilities | array | Array of features that the connected watch must have for this item to be present | +| group | string | Set this to allow this item, along with other items sharing the same group to be looked up using `Clay.getItemsByGroup()` in your [custom function](#custom-function) | + +##### Example + +```javascript +{ + "type": "select", + "messageKey": "flavor", + "defaultValue": "grape", + "label": "Favorite Flavor", + "options": [ + { + "label": "", + "value": "" + }, + { + "label": "Berry", + "value": "berry" + }, + { + "label": "Grape", + "value": "grape" + }, + { + "label": "Banana", + "value": "banana" + } + ] +} +``` + +If you wish to use optgroups, then use the following format: + +```javascript +{ + "type": "select", + "messageKey": "flavor", + "defaultValue": "grape", + "label": "Favorite Flavor", + "options": [ + { + "label": "", + "value": "" + }, + { + "label": "Fruits", + "value": [ + { + "label": "Berry", + "value": "berry" + }, + { + "label": "Grape", + "value": "grape" + }, + { + "label": "Banana", + "value": "banana" + } + ] + }, + { + "label": "Candy", + "value": [ + { + "label": "Chocolate", + "value": "chocolate" + }, + { + "label": "Cream", + "value": "cream" + }, + { + "label": "Lollipop", + "value": "lollipop" + } + ] + } + ] +} +``` + +--- + +#### Color Picker + +**Manipulator:** [`color`](#color) + +A color picker that allows users to choose a color from the ones that are compatible with their Pebble smartwatch. +The color picker will automatically show a different layout depending on the watch connected: + + - Aplite (Firmware 2.x) - Black and white + - Aplite (Firmware 3.x) - Black and white. Will also include gray (`#AAAAAA`) if `allowGray` is set to `true` + - Basalt/chalk - The 64 colors compatible with color Pebble smartwatches. + +##### Properties + +| Property | Type | Description | +|----------|------|-------------| +| type | string | Set to `color`. | +| id | string (unique) | Set this to a unique string to allow this item to be looked up using `Clay.getItemById()` in your [custom function](#custom-function). | +| messageKey | string (unique) | The AppMessage key matching the `messageKey` item defined in your `package.json`. Set this to a unique string to allow this item to be looked up using `Clay.getItemsByMessageKey()` in your custom function. You must set this if you wish for the value of this item to be persisted after the user closes the config page. | +| label | string | The label that should appear next to this item. | +| defaultValue | string OR int | The default color. One of the [64 colors](https://developer.pebble.com/guides/tools-and-resources/color-picker/) compatible with Pebble smartwatches. Always use the uncorrected value even if `sunlight` is true. The component will do the conversion internally. | +| description | string | Optional sub-text to include below the component | +| sunlight | boolean | Use the color-corrected sunlight color palette if `true`, else the uncorrected version. Defaults to `true` if not specified. | +| layout | string OR array | Optional. Use a custom layout for the color picker. Defaults to automatically choosing the most appropriate layout for the connected watch. The layout is represented by a two dimensional array. Use `false` to insert blank spaces. You may also use one of the preset layouts by setting `layout` to: `"COLOR"`, `"GRAY"` or `"BLACK_WHITE"` | +| allowGray | boolean | Optional. Set this to `true` to include gray (`#AAAAAA`) in the color picker for aplite running on firmware 3 and above. This is optional because only a subset of the drawing operations support gray on aplite. Defaults to `false` | +| capabilities | array | Array of features that the connected watch must have for this item to be present | +| group | string | Set this to allow this item, along with other items sharing the same group to be looked up using `Clay.getItemsByGroup()` in your [custom function](#custom-function) | + +##### Example + +```javascript +{ + "type": "color", + "messageKey": "background", + "defaultValue": "ff0000", + "label": "Background Color", + "sunlight": true, + "layout": [ + [false, "00aaff", false], + ["0055ff", "0000ff", "0000aa"], + [false, "5555ff", false] + ] +} +``` + +##### Example + +```javascript +{ + "type": "color", + "messageKey": "background", + "defaultValue": "ffffff", + "label": "Background Color", + "sunlight": false, + "layout": "BLACK_WHITE" +} +``` + +##### Example + +```javascript +{ + "type": "color", + "messageKey": "background", + "defaultValue": "aaaaaa", + "label": "Background Color", + "sunlight": false, + "allowGray": true +} +``` + +--- + +#### Radio Group + +**Manipulator:** [`radiogroup`](#radiogroup) + +A list of options allowing the user can only choose one option to submit. + +##### Properties + +| Property | Type | Description | +|----------|------|-------------| +| type | string | Set to `radiogroup`. | +| id | string (unique) | Set this to a unique string to allow this item to be looked up using `Clay.getItemById()` in your [custom function](#custom-function). | +| messageKey | string (unique) | The AppMessage key matching the `messageKey` item defined in your `package.json`. Set this to a unique string to allow this item to be looked up using `Clay.getItemsByMessageKey()` in your custom function. You must set this if you wish for the value of this item to be persisted after the user closes the config page. | +| label | string | The label that should appear next to this item. | +| defaultValue | string | The default selected item. Must match a value in the `options` array. | +| description | string | Optional sub-text to include below the component | +| options | array of objects | The options you want to appear in the radio group. Each option is an object with a `label` and `value` property. | +| capabilities | array | Array of features that the connected watch must have for this item to be present | +| group | string | Set this to allow this item, along with other items sharing the same group to be looked up using `Clay.getItemsByGroup()` in your [custom function](#custom-function) | + +##### Example + +```javascript +{ + "type": "radiogroup", + "messageKey": "favorite_food", + "label": "Favorite Food", + "options": [ + { + "label": "Sushi", + "value": "sushi" + }, + { + "label": "Pizza", + "value": "pizza" + }, + { + "label": "Burgers", + "value": "burgers" + } + ] +} +``` + +--- + +#### Checkbox Group + +**Manipulator:** [`checkboxgroup`](#checkboxgroup) + +A list of options where a user may choose more than one option to submit. + +##### Properties + +| Property | Type | Description | +|----------|------|-------------| +| type | string | Set to `checkboxgroup`. | +| id | string (unique) | Set this to a unique string to allow this item to be looked up using `Clay.getItemById()` in your [custom function](#custom-function). | +| messageKey | string (unique) | The AppMessage key matching the `messageKey` item defined in your `package.json`. Set this to a unique string to allow this item to be looked up using `Clay.getItemsByMessageKey()` in your custom function. You must set this if you wish for the value of this item to be persisted after the user closes the config page. **NOTE:** The checkboxgroup component will expect you to have defined a `messageKey` in your `package.json` using array syntax. In the example below, the matching entry in the `package.json` would be `favorite_food[3]`. In CloudPebble, you must use `Automatic assignment` for message keys and the `Key Array Length` must match the number of options in the checkboxgroup | +| label | string | The label that should appear next to this item. | +| defaultValue | array of booleans | The default selected items. | +| description | string | Optional sub-text to include below the component | +| options | array of strings | The labels for each checkbox you want to appear in the checkbox group. | +| capabilities | array | Array of features that the connected watch must have for this item to be present | +| group | string | Set this to allow this item, along with other items sharing the same group to be looked up using `Clay.getItemsByGroup()` in your [custom function](#custom-function) | + +##### Example + +```javascript +{ + "type": "checkboxgroup", + "messageKey": "favorite_food", + "label": "Favorite Food", + "defaultValue": [true, false, true], + "options": ["Sushi", "Pizza", "Burgers"] +} +``` + +In the above example, Sushi and Burgers will be selected by default. + +--- + +### Generic Button + +**Manipulator:** [`button`](#button) + +##### Properties + +| Property | Type | Description | +|----------|------|-------------| +| type | string | Set to `button`. | +| id | string (unique) | Set this to a unique string to allow this item to be looked up using `Clay.getItemById()` in your [custom function](#custom-function). | +| defaultValue | string | The text displayed on the button. | +| primary | boolean | If `true` the button will be orange, if `false`, the button will be gray (defaults to `false`)| +| description | string | Optional sub-text to include below the component | +| capabilities | array | Array of features that the connected watch must have for this item to be present | +| group | string | Set this to allow this item, along with other items sharing the same group to be looked up using `Clay.getItemsByGroup()` in your [custom function](#custom-function) | + +##### Example + +```javascript +{ + "type": "button", + "primary": true, + "defaultValue": "Send" +} +``` + +--- + +### Range Slider + +**Manipulator:** [`slider`](#slider) + +The range slider is used to allow users to select numbers between two values. + +**NOTE** If you set the `step` property to anything less than `1`, +it will multiply the final value sent to the watch by 10 to the power of the number of decimal places in the value of `step`. +Eg: If you set the `step` to `0.25` and the slider has value of `34.5`, the watch will receive `3450`. +The reason why this is necessary, is because you can not send floats to the watch via `Pebble.sendAppMessage()`. +This allows your users to still be able to input values that have decimals, +you must just remember to divide the received value on the watch accordingly. + +##### Properties + +| Property | Type | Description | +|----------|------|-------------| +| type | string | Set to `slider`. | +| defaultValue | number | The value of the slider. | +| label | string | The label that should appear next to this item. | +| min | number | The minimum allowed value of the slider. Defaults to `100` | +| max | number | The maximum allowed value of the slider. Defaults to `0` | +| step | number | The multiple of the values allowed to be set on the slider. The slider will snap to these values. This value also determines the precision used when the value is sent to the watch. Defaults to 1 | +| description | string | Optional sub-text to include below the component | +| capabilities | array | Array of features that the connected watch must have for this item to be present | +| group | string | Set this to allow this item, along with other items sharing the same group to be looked up using `Clay.getItemsByGroup()` in your [custom function](#custom-function) | + +##### Example + +```javascript +{ + "type": "slider", + "messageKey": "slider", + "defaultValue": 15, + "label": "Slider", + "description": "This is the description for the slider", + "min": 10, + "max": 20, + "step": 0.25 +}, +``` + +--- + +### Submit + +**Manipulator:** [`button`](#button) + +The submit button for the page. You **MUST** include this component somewhere in the config page (traditionally at the bottom) or users will not be able to save the form. + +##### Properties + +| Property | Type | Description | +|----------|------|-------------| +| type | string | Set to `submit`. | +| defaultValue | string | The text displayed on the button. | +| id | string (unique) | Set this to a unique string to allow this item to be looked up using `Clay.getItemById()` in your [custom function](#custom-function). | +| capabilities | array | Array of features that the connected watch must have for this item to be present | +| group | string | Set this to allow this item, along with other items sharing the same group to be looked up using `Clay.getItemsByGroup()` in your [custom function](#custom-function) | + +##### Example + +```javascript +{ + "type": "submit", + "defaultValue": "Save" +} +``` + +--- + +### Message keys and array syntax + +Clay supports (and requires in some cases) the use of array syntax for message keys. Also known as `Automatic assignment` in CloudPebble. More info [here.](https://developer.pebble.com/guides/communication/using-pebblekit-js/#defining-keys) + +You can assign any component that does not return an array to a particular position in your message key. This is done by appending the position (zero indexed) wrapped in brackets to the end of the `messageKey` property. Do **NOT** include any spaces in your `messageKey`. + +##### Example + +```javascript +{ + "type": "toggle", + "messageKey": "light_switch[1]", + "label": "Invert Colors", + "defaultValue": true +} +``` +In the above example, the value of the item will be accessible with `MESSAGE_KEY_light_switch + 1` in your C code. + +Components that return an array in their `.get()` method, such as the checkboxgroup will expect the corresponding message key to already be defined with the correct length. You **CAN NOT** use the array syntax in the `messageKey` property as this would be trying to create a 2 dimensional array. + +##### Example + +```javascript +{ + "type": "checkboxgroup", + "messageKey": "favorite_food", + "label": "Favorite Food", + "defaultValue": [true, false, true], + "options": ["Sushi", "Pizza", "Burgers"] +} +``` +In the above example, the value of the item will be accessible with `MESSAGE_KEY_favorite_food`, `MESSAGE_KEY_favorite_food + 1`, and `MESSAGE_KEY_favorite_food + 2` in your C code. + +--- +### Showing items for specific platforms and features + +If you want particular items or sections to only be present in the config based on the +connected watch's capabilities, you can include the "capabilities" property in the item. +The "capabilities" property is an array of features that the connected watch must have +in order for the item or section to be included in the page. Note: all capabilities in +the array must be present for item/section to be included. + +You can also prefix the capability with `NOT_` to negate the capability. Eg: `NOT_HEALTH` +will only be included in the page if the device does **NOT** support health. + +**Warning:** Items that do not satisfy the capabilities will not be included in the page +at all. You will not be able to use methods like `clayConfig.getItemByMessageKey()` to +obtain a reference to them. However, this does mean that you will be able to have +multiple items with the same `messageKey`as long as they do not both satisfy the +same conditions. + +##### Examples + +```javascript +{ + "type": "text", + "defaultValue": "This item will only be visible for watches that are rectangular and have a microphone" + "capabilities": ["MICROPHONE", "RECT"], +} +``` + +```javascript +{ + "type": "section", + "capabilities": ["NOT_HEALTH"], + "items": [ + { + "type": "text", + "defaultValue": "Only visible for watches that do not support health" + } + ] +} +``` + +Below is the full list of capabilities + +| Capability | Description | +|------------|-------------| +| PLATFORM_APLITE | Running on Pebble/Pebble Steel.| +| PLATFORM_BASALT | Running on Pebble Time/Pebble Time Steel. | +| PLATFORM_CHALK | Running on Pebble Time Round. | +| PLATFORM_DIORITE | Running on Pebble 2 | +| PLATFORM_EMERY | Running on Time 2. | +| BW | Running on hardware that supports only black and white. | +| COLOR | Running on hardware that supports 64 colors. | +| MICROPHONE | Running on hardware that includes a microphone. | +| SMARTSTRAP | Running on hardware that includes a smartstrap connector. | +| SMARTSTRAP_POWER | Running on hardware that includes a powered smartstrap connector. | +| HEALTH | Running on hardware that supports Pebble Health and the HealthService API. | +| RECT | Running on hardware with a rectangular display. | +| ROUND | Running on hardware with a round display. | +| DISPLAY_144x168 | Running on hardware with a 144x168 pixels display. | +| DISPLAY_180x180_ROUND | Running on hardware with a 180x180 pixels round display. | +| DISPLAY_200x228 | Running on hardware with a 200x228 pixels display. | + +## Manipulators + +Each component has a **manipulator**. This is a set of methods used to talk to the item on the page. +At a minimum, manipulators must have a `.get()` and `.set(value)` method however there are also methods to assist in interactivity such as `.hide()` and `.disable()`. +**NOTE:** There is currently no way to disable or hide an entire section. You must disable/hide each item in the section to achieve this effect. + +When the config page is closed, the `.get()` method is run on all components registered with an `messageKey` to construct the object sent to the C app. + +Many of these methods fire an event when the method is called. You can listen for these events with `ClayItem.on()`. +**NOTE** These events will only be fired if the state actually changes. +Eg: If you run the `.show()` manipulator on an item that is already visible, the `show` event will not be triggered. + +#### html + +| Method | Returns | Event Fired | Description | +|--------|---------|-------------| ------------| +| `.set( [string\|HTML] value)` | `ClayItem` | `change` | Sets the content of this item. | +| `.get()` | `string` | | Gets the content of this item. | +| `.hide()` | `ClayItem` | `hide` | Hides the item | +| `.show()` | `ClayItem` | `show` | Shows the item | + +#### button + +| Method | Returns | Event Fired | Description | +|--------|---------|-------------| ------------| +| `.set( [string\|HTML] value)` | `ClayItem` | `change` | Sets the content of this item. | +| `.get()` | `string` | | Gets the content of this item. | +| `.disable()` | `ClayItem` | `disabled` | Prevents this item from being clicked by the user. | +| `.enable()` | `ClayItem` | `enabled` | Allows this item to be clicked by the user. | +| `.hide()` | `ClayItem` | `hide` | Hides the item | +| `.show()` | `ClayItem` | `show` | Shows the item | + +#### val + +| Method | Returns | Event Fired | Description | +|--------|---------|-------------| ------------| +| `.set( [string] value)` | `ClayItem` | `change` | Sets the value of this item. | +| `.get()` | `string` | | Gets the content of this item. | +| `.disable()` | `ClayItem` | `disabled` | Prevents this item from being edited by the user. | +| `.enable()` | `ClayItem` | `enabled` | Allows this item to be edited by the user. | +| `.hide()` | `ClayItem` | `hide` | Hides the item | +| `.show()` | `ClayItem` | `show` | Shows the item | + +#### checked + +| Method | Returns | Event Fired | Description | +|--------|---------|-------------| ------------| +| `.set( [boolean\|int] value)` | `ClayItem` | `change` | Check/uncheck the state of this item. | +| `.get()` | `boolean` | | `true` if checked, `false` if not. **NOTE** this will be converted to a `1` or `0` when sent to the watch. See [`Clay.getSettings()`](#methods) | +| `.disable()` | `ClayItem` | `disabled` | Prevents this item from being edited by the user. | +| `.enable()` | `ClayItem` | `enabled` | Allows this item to be edited by the user. | +| `.hide()` | `ClayItem` | `hide` | Hides the item | +| `.show()` | `ClayItem` | `show` | Shows the item | + +#### color + +| Method | Returns | Event Fired | Description | +|--------|---------|-------------| ------------| +| `.set( [string\|int] value)` | `ClayItem` | `change` | Sets the color picker to the provided color. If the value is a string, it must be provided in hex notation eg `'FF0000'`. | +| `.get()` | `int` | | Get the chosen color. This is returned as a number in order to make it easy to use on the watch side using `GColorFromHEX()`. | +| `.disable()` | `ClayItem` | `disabled` | Prevents this item from being edited by the user. | +| `.enable()` | `ClayItem` | `enabled` | Allows this item to be edited by the user. | +| `.hide()` | `ClayItem` | `hide` | Hides the item | +| `.show()` | `ClayItem` | `show` | Shows the item | + +#### radiogroup + +| Method | Returns | Event Fired | Description | +|--------|---------|-------------| ------------| +| `.set( [string] value)` | `ClayItem` | `change` | Checks the radio button that corresponds to the provided value. | +| `.get()` | `string` | | Gets the value of the checked radio button in the list. | +| `.disable()` | `ClayItem` | `disabled` | Prevents this item from being edited by the user. | +| `.enable()` | `ClayItem` | `enabled` | Allows this item to be edited by the user. | +| `.hide()` | `ClayItem` | `hide` | Hides the item | +| `.show()` | `ClayItem` | `show` | Shows the item | + +#### checkboxgroup + +| Method | Returns | Event Fired | Description | +|--------|---------|-------------| ------------| +| `.set( [array] value)` | `ClayItem` | `change` | Checks the checkboxes that corresponds to the provided list of values. | +| `.get()` | `Array.` | | Gets an array of booleans representing the list the checked items. **NOTE:** each item in the array will be converted to an `int` when sent to the watch. See [`Clay.getSettings()`](#methods) | +| `.disable()` | `ClayItem` | `disabled` | Prevents this item from being edited by the user. | +| `.enable()` | `ClayItem` | `enabled` | Allows this item to be edited by the user. | +| `.hide()` | `ClayItem` | `hide` | Hides the item | +| `.show()` | `ClayItem` | `show` | Shows the item | + +#### slider + +| Method | Returns | Event Fired | Description | +|--------|---------|-------------| ------------| +| `.set( [number] value)` | `ClayItem` | `change` | Sets the value of this item. | +| `.get()` | `number` | | Gets the value of this item. | +| `.disable()` | `ClayItem` | `disabled` | Prevents this item from being edited by the user. | +| `.enable()` | `ClayItem` | `enabled` | Allows this item to be edited by the user. | +| `.hide()` | `ClayItem` | `hide` | Hides the item | +| `.show()` | `ClayItem` | `show` | Shows the item | + +# Extending Clay + +Clay is built to allow developers to add their own basic interactivity to the config page. This can be done in a number of ways: + +## Handling The 'showConfiguration' and 'webviewclosed' Events Manually + +Clay will by default, automatically handle the 'showConfiguration' and 'webviewclosed' events. This allows the `messageKey` of each config item to be automatically delivered to the `InboxReceivedHandler` on the watch side. If you wish to override this behavior and handle the events yourself, pass an object as the 3rd parameter of the Clay constructor with `autoHandleEvents` set to `false`. + +Example: + +```javascript +var Clay = require('pebble-clay'); +var clayConfig = require('./config'); +var clayConfigAplite = require('./config-aplite'); +var clay = new Clay(clayConfig, null, { autoHandleEvents: false }); + +Pebble.addEventListener('showConfiguration', function(e) { + + // This is an example of how you might load a different config based on platform. + var platform = clay.meta.activeWatchInfo.platform || 'aplite'; + if (platform === 'aplite') { + clay.config = clayConfigAplite; + } + + Pebble.openURL(clay.generateUrl()); +}); + +Pebble.addEventListener('webviewclosed', function(e) { + if (e && !e.response) { + return; + } + + // Get the keys and values from each config item + var dict = clay.getSettings(e.response); + + // Send settings values to watch side + Pebble.sendAppMessage(dict, function(e) { + console.log('Sent config data to Pebble'); + }, function(e) { + console.log('Failed to send config data!'); + console.log(JSON.stringify(e)); + }); +}); +``` + +## Clay API (app.js) + +### `Clay([Array] config, [function] customFn, [object] options)` + +#### Constructor Parameters + +| Parameter | Type | Description | +|----------|-------|-------------| +| `config` | Array | The config that will be used to generate the configuration page | +| `customFn` | Function\|null | (Optional) The [custom function](#custom-function) to be injected into the generated configuration page. | +| `options` | Object | (Optional) See below for properties | +| `options.autoHandleEvents` | Boolean | (Optional) Defaults to `true`. If set to `false`, Clay will not [auto handle the `showConfiguration` and `webviewclosed` events](#handling-the-showconfiguration-and-webviewclosed-events-manually) | +| `options.userData` | Any | (Optional) Any arbitrary data you want to pass to your config page. It will be available in your custom function as `this.meta.userData` | + +#### Properties + +| Property | Type | Description | +|----------|------|-------------| +| `.config` | Array | Copy of the config passed to the constructor and used for generating the page. | +| `.customFn` | Function | Reference to the custom function passed to the constructor. **WARNING** this is a direct reference, not a copy of the custom function so any modification you make to it, will be reflected on the original as well | +| `.meta` | Object | Contains information about the current user and watch. **WARNING** This will only be populated in the `showConfiguration` event handler. (See example above) | +| `.meta.activeWatchInfo` | watchinfo\|null | An object containing information on the currently connected Pebble smartwatch or null if unavailable. Read more [here](https://developer.pebble.com/docs/js/Pebble/#getActiveWatchInfo). | +| `.meta.accountToken` | String | A unique account token that is associated with the Pebble account of the current user. Read more [here](https://developer.pebble.com/docs/js/Pebble/#getAccountToken). | +| `.meta.watchToken` | String | A unique token that can be used to identify a Pebble device. Read more [here](https://developer.pebble.com/docs/js/Pebble/#getWatchToken). | +| `.meta.userData` | Any | A deep copy of the arbitrary data provided in the `options.userData`. Defaults to an empty object | +| `.version` | String | The version of Clay currently being used. | + +#### Methods + +| Method | Returns | +| -------|---------| +| `Clay( [array] config, [function] customFn=null, [object] options={autoHandleEvents: true})`
`config` - an Array representing your config
`customFn` - function to be run in the context of the generated page
`options.autoHandleEvents` - set to `false` to prevent Clay from automatically handling the "showConfiguration" and "webviewclosed" events | `Clay` - a new instance of Clay. | +| `.registerComponent( [ClayComponent] component )`
Registers a custom component. | `void`. | +| `.generateUrl()` | `string` - The URL to open with `Pebble.openURL()` to use the Clay-generated config page. | +| `.getSettings( [object] response, [boolean] convert=true)`
`response` - the response object provided to the "webviewclosed" event
`convert` - Pass `false` to not convert the settings to be compatible with `Pebble.sendAppMessage()` | `Object` - object of keys and values for each config page item with an `messageKey`, where the key is the `messageKey` and the value is the chosen value of that item.

This method will do some conversions depending on the type of the setting. Arrays will use message key array syntax to make the values of each item in the array available as individual message keys. Booleans will be converted to numbers. eg `true` becomes `1` and `false` becomes `0`. If the value is a number or an array of numbers and the optional property: "precision" is the power of precision (value * 10 ^ precision) and then floored. Eg: `1.4567` with a precision set to `3` will become `1456`. Pass `false` as the second parameter to disable this behavior. See the example below for how this all works | +| `.setSettings( [string] key, [*] value )`
`key` - The key for the setting you want to set.
`value` - The value of the setting you want to set. | `void` | +| `.setSettings( [object] settings )`
`settings` - An object of key/value pairs that you would like to update the settings with. eg `{ user_name: 'Emily', show_animations: true }` | `void` | + +#### `.getSettings()` Example + +```javascript + +// webviewclosed response +{ + "favorite_food": {"value": [1, 0, 1]}, + "cool_things_enabled": {"value": true}, + "user_name": {"value": "Jane Doe"}, + "date_of_birth[0]": {"value": 1989}, + "date_of_birth[1]": {"value": 11}, + "date_of_birth[2]": {"value": 28}, + "rating": {"value": 3.5, "precision": 1}, +} + +// resulting converted values + +var messageKeys = require('message_keys'); + +messageKeys.favorite_food + 0 = 1; +messageKeys.favorite_food + 1 = 0; +messageKeys.favorite_food + 2 = 1; +messageKeys.cool_things_enabled = 1; +messageKeys.user_name = 'Jane Doe'; +messageKeys.date_of_birth + 0 = 1989; +messageKeys.date_of_birth + 1 = 11; +messageKeys.date_of_birth + 2 = 28; +messageKeys.rating = 35; + + +``` + +--- + + +## Custom Function + +When initializing Clay in your `app.js` file, you can optionally provide a function that will be copied and run on the generated config page. + +**IMPORTANT:** This function is injected by running `.toString()` on it. If you are making use of `require` or any other dynamic features, they will not work. You must make sure that everything the function needs to execute is available in the function body itself. + +This function, when injected into the config page, will be run with `ClayConfig` as its context (`this`), and [**Minified**](#minified) as its first parameter. + +Make sure to always wait for the config page to be built before manipulating items. You do this by registering a handler for the `Clay.EVENTS.AFTER_BUILD` event (see below). + +#### Example + +##### app.js + +```javascript +var Clay = require('pebble-clay'); +var clayConfig = require('./config'); +var customClay = require('./custom-clay'); +var userData = {token: 'abc123'} +var clay = new Clay(clayConfig, customClay, {userData: userData}); +``` + +##### custom-clay.js + +```javascript +module.exports = function(minified) { + var clayConfig = this; + var _ = minified._; + var $ = minified.$; + var HTML = minified.HTML; + + function toggleBackground() { + if (this.get()) { + clayConfig.getItemByMessageKey('background').enable(); + } else { + clayConfig.getItemByMessageKey('background').disable(); + } + } + + clayConfig.on(clayConfig.EVENTS.AFTER_BUILD, function() { + var coolStuffToggle = clayConfig.getItemByMessageKey('cool_stuff'); + toggleBackground.call(coolStuffToggle); + coolStuffToggle.on('change', toggleBackground); + + // Hide the color picker for aplite + if (!clayConfig.meta.activeWatchInfo || clayConfig.meta.activeWatchInfo.platform === 'aplite') { + clayConfig.getItemByMessageKey('background').hide(); + } + + // Set the value of an item based on the userData + $.request('get', 'https://some.cool/api', {token: clayConfig.meta.userData.token}) + .then(function(result) { + // Do something interesting with the data from the server + }) + .error(function(status, statusText, responseText) { + // Handle the error + }); + }); + +}; +``` + +## Clay API (Custom Function) + +### `ClayConfig([Object] settings, [Array] config, [$Minified] $rootContainer)` + +This is the main way of talking to your generated config page. An instance of this class will be passed as the context of your custom function when it runs on the generated config page. + +#### Properties + +| Property | Type | Description | +|----------|------|-------------| +| `.EVENTS.BEFORE_BUILD` | String | Dispatched prior to building the page. | +| `.EVENTS.AFTER_BUILD` | String | Dispatched after building the page. | +| `.EVENTS.BEFORE_DESTROY` | String | Dispatched prior to destroying the page. | +| `.EVENTS.AFTER_DESTROY` | String | Dispatched after destroying the page. | +| `.config` | Array | Reference to the config passed to the constructor and used for generating the page. | +| `.meta` | Object | Contains information about the current user and watch | +| `.meta.activeWatchInfo` | watchinfo\|null | An object containing information on the currently connected Pebble smartwatch or null if unavailable. Read more [here](https://developer.pebble.com/docs/js/Pebble/#getActiveWatchInfo). | +| `.meta.accountToken` | String | A unique account token that is associated with the Pebble account of the current user. Read more [here](https://developer.pebble.com/docs/js/Pebble/#getAccountToken). | +| `.meta.watchToken` | String | A unique token that can be used to identify a Pebble device. Read more [here](https://developer.pebble.com/docs/js/Pebble/#getWatchToken). | +| `.meta.userData` | Any | The data passed in the `options.userData` of the [Clay constructor.](#clayarray-config-function-customfn-object-options) | + + +#### Methods + +| Method | Returns | +|--------|---------| +| `.getAllItems()` | `Array.` - an array of all config items. | +| `.getItemByMessageKey( [string] messageKey )` | `ConfigItem\|undefined` - a single `ConfigItem` that has the provided `messageKey`, otherwise `undefined`. | +| `.getItemById( [string] id )` | `ConfigItem\|undefined` - a single `ConfigItem` that has the provided `id`, otherwise `undefined`. | +| `.getItemsByType( [string] type )` | `Array.` - an array of config items that match the provided `type`. | +| `.getItemsByGroup( [string] group )` | `Array.` - an array of config items that match the provided `group`. | +| `.serialize()` | `Object` - an object representing all items with an `messageKey` where the key is the `messageKey` and the value is an object with the `value` property set to the result of running `.get()` on the Clay item. If the Clay item has a `precision` property set, it is included in the object | +| `.build()`
Builds the config page. Will dispatch the `BEFORE_BUILD` event prior to building the page, then the `AFTER_BUILD` event once it is complete. If the config page has already been built, then the `ClayConfig.destroy()` method will be executed prior to building the page again. | `ClayConfig` | +| `.destroy()`
Destroys the config page. Will dispatch the `BEFORE_DESTROY` event prior to destroying the page, then the `AFTER_DESTROY` event once it is complete. This method wipes the config page completely, including all existing items. You will need to make sure that you re-attach your event handlers for any items that are replaced | `ClayConfig` | +| `.on( [string] events, [function] handler )`
Register an event to the provided handler. The handler will be called with this instance of `ClayConfig` as the context. If you wish to register multiple events to the same handler, then separate the events with a space | `ClayConfig` | +| `.off( [function] handler )`
Remove the given event handler. **NOTE:** This will remove the handler from all registered events. | `ClayConfig` | +| `.trigger( [string] name, [object] eventObj={} )`
Trigger the provided event and optionally pass extra data to the handler. | `ClayConfig` | +| `.registerComponent( [ClayComponent] component )`
Registers a component. You must register all components prior to calling `.build()`. This method is also available statically. | `Boolean` - `true` if the component was registered successfully, otherwise `false`. | + +--- + +### `ClayItem( [Object] config )` + +#### Properties + +| Property | Type | Description | +|----------|------|-------------| +| `.id` | String | The ID of the item if provided in the config. | +| `.messageKey` | String | The messageKey of the item if provided in the config. | +| `.config` | Object | Reference to the config passed to the constructer. | +| `$element` | $Minified | A Minified list representing the root HTML element of the config item. | +| `$manipulatorTarget` | $Minified | A Minified list representing the HTML element with **data-manipulator-target** set. This is generally pointing to the main `` element and will be used for binding events. | + + +#### Methods + +| Method | Returns | +|--------|---------| +| `.initialize( [ClayConfig] clay)`
You shouldn't ever need to run this method manually as it will automatically be called when the config is built. | `ConfigItem` | +| `.on( [string] events, [function] handler )`
Register an event to the provided handler. The handler will be called with this instance of `ClayItem` as the context. If you wish to register multiple events to the same handler, then separate the events with a space. Events will be registered against the `$manipulatorTarget` so most DOM events such as **"change"** or **"click"** can be listened for. | `ClayItem` | +| `.off( [function] handler )`
Remove the given event handler. **NOTE:** This will remove the handler from all registered events. | `ClayItem` | +| `.trigger( [string] name, [object] eventObj={} )`
Trigger the provided event and optionally pass extra data to the handler. | `ClayItem` | + +In addition to the methods above, all the methods from the item's manipulator will be attached to the `ClayItem`. This includes `.set()` and `.get()`. + +--- + +## Custom Components + +Clay is also able to be extended using custom components. This allows developers to share components with each other. + +### Component Structure + +Components are simple objects with the following properties. + +#### `ClayComponent` + +| Property | Type | Required | Description | +|----------|------|----------|-------------| +| `name` | string | yes | This is the unique way to identify the component and will be used by the config item's `type`. | +| `template` | string (HTML) | yes | This is the actual HTML content of the component. Make sure there is only **one** root node in the HTML. This HTML will be passed to Minified's `HTML()` method. Any properties provided by the config item will be made available to the template, eg: `label`. The template will also be provided with `clayId` as a unique way to set input `name` attributes. | +| `style` | string | no | Any extra CSS styles you want to inject into the page. Make sure to namespace your CSS with a class that is unique to your component in order to avoid conflicts with other components. | +| `manipulator` | string / manipulator | yes | Provide a string here to use one of the built-in manipulators, such as `val`. If an object is provided, it must have both a `.set(value)` and `.get()` method. | +| `defaults` | object | Only if your template requires it. | An object of all the defaults your template requires. | +| `initialize` | function | no | Method which will be called after the item has been added to the page. It will be called with the `ClayItem` as the context (`this`) and with [`minified`](#minified) as the first parameter and [`clayConfig`](#clayconfigobject-settings-array-config-minified-rootcontainer) as the second parameter | + +### Registering a custom component. + +Components must be registered before the config page is built. The easiest way to do this is in your `app.js` after you have initialized Clay: + +```javascript +var Clay = require('pebble-clay'); +var clayConfig = require('./config.json'); +var clay = new Clay(clayConfig); + +clay.registerComponent(require('./my-custom-component')); +``` + +## Minified + +[Minified](http://minifiedjs.com) is a super light JQuery-like library. We only bundle in a small subset of its functionality. Visit the [Minified Docs](http://minifiedjs.com/api/) for more info on how to use Minified. Below is the subset of methods available in Clay. + + - `$()` + - `$$()` + - `.get()` + - `.select()` + - `.set()` + - `.add()` + - `.ht()` + - `HTML()` + - `$.request()` + - `promise.always()` + - `promise.error()` + - `$.off()` + - `$.ready()` + - `$.wait()` + - `.on()` + - `.each()` + - `.find()` + - `_()` + - `_.copyObj()` + - `_.eachObj()` + - `_.extend()` + - `_.format()` + - `_.formatHtml()` + - `_.template()` + - `_.isObject()` + +# Migrating from v0.1.x to v1.x + +There were some changes in the 3.13 SDK that required Clay to undergo some major changes. For the majority of developers a simple find and replace over your config will do the trick. + +### appKey is now messageKey. + +You will need to update your config files and change any items that use `appKey` to `messageKey` + +##### Example + +```javascript +{ + "type": "toggle", + "appKey": "invert", + "label": "Invert Colors", + "defaultValue": true +} +``` +becomes: + +```javascript +{ + "type": "toggle", + "messageKey": "invert", + "label": "Invert Colors", + "defaultValue": true +} +``` + +### clayConfig.getItemByAppKey is now clayConfig.getItemByMessageKey. + +If you have a [custom function](#custom-function) and are using the `clayConfig.getItemByAppKey()` method you will need to change this to `clayConfig.getItemByMessageKey()` + +### clay.getSettings() will now by default, return an object where the keys are numbers. + +In the past, `clay.getSettings()` would return something that looked like: + +```javascript +{ + BACKGROUND_COLOR: 0, + NAME: 'Jane Doe', + ENABLE_TOOLS: 1 +} +``` + +It now uses the values provided in the `message_keys` module to construct this object. +The new format looks something like: + +```javascript +{ + 10000: 0, + 10001: 'Jane Doe', + 10002: 1 +} +``` + +If you wish to find out what keys are associated with what values, you must use the `message_keys` module. + +```javascript +var messageKeys = require('message_keys'); + +Pebble.addEventListener('webviewclosed', function(e) { + // Get the keys and values from each config item + var claySettings = clay.getSettings(e.response); + + // In this example messageKeys.NAME is equal to 10001 + console.log('Name is ' + claySettings[messageKeys.NAME]); // Logs: "Name is Jane Doe" +}); +``` + +**NOTE:** The above only applies to the default behavior of the method. If you pass +`false` to the second argument, a standard object will be returned with the `messageKey` +as the key. + +```javascript + +Pebble.addEventListener('webviewclosed', function(e) { + + clay.getSettings(e.response); + /* returns: + { + 10000: 0, + 10001: 'Jane Doe', + 10002: 1 + } + */ + + clay.getSettings(e.response, false); + /* returns: + { + BACKGROUND_COLOR: {value: 0}, + NAME: {value: 'Jane Doe'}, + ENABLE_TOOLS: {value: true} + } + + Notice that the value for ENABLE_TOOLS was not converted to a number from a boolean + */ +}); + +``` + +### Checkbox groups now use arrays. + +In the previous version of Clay, checkbox groups would split the values of the items with zeros. This made for clumsy usage on the C side. Checkbox groups are now much simpler to use thanks to message keys. You will however need to update you config to the new format. + +#### Changes: + - `defaultValue` and the value that is returned by the `.get()` method is now an array of booleans. + - `options` are now an array of labels and do not have their own value + - Instead of splitting the result on the C side you now use the message key syntax. In the example below `MESSAGE_KEY_favorite_food + 2` would have a value of `1` + +#### Old Format + +```javascript +{ + "type": "checkboxgroup", + "appKey": "favorite_food", + "label": "Favorite Food", + "defaultValue": ["sushi", "burgers"], + "options": [ + { + "label": "Sushi", + "value": "sushi" + }, + { + "label": "Pizza", + "value": "pizza" + }, + { + "label": "Burgers", + "value": "burgers" + } + ] +} +``` + +#### New Format + +```javascript +{ + "type": "checkboxgroup", + "messageKey": "favorite_food", + "label": "Favorite Food", + "defaultValue": [true, false, true], + "options": ["Sushi", "Pizza", "Burgers"] +} +``` + +### Clay is now a pebble package + +You no longer need to download a file into your project. Clay is now a Pebble package so you can go ahead and delete your `clay.js` and follow the getting started guide above to install clay in your project. + +Prior to SDK 3.13, `require` paths were handled in a non-standard way. When requiring modules, the name of the module was sufficient (ie. `require('config.json')`). However, with the release of SDK 3.13, the require paths changed so that you now have to require the module by using its path relative to the file it's being required in. This means requiring the config module now is done in app.js by using `require('./clay-config.json')`. An incorrect path would result in an error similar to this: + ``` + [14:16:03] javascript> JavaScript Error: + Error: Cannot find module 'clay-config.json' + at Object.loader.require (loader.js:66:11) + at _require.require (loader.js:54:48) + at Object.loader (src/js/app.js:1:1) + at _require (loader.js:57:10) + at Object.loader.require (loader.js:69:10) + ``` diff --git a/node_modules/pebble-clay/dist.zip b/node_modules/pebble-clay/dist.zip new file mode 100644 index 0000000000000000000000000000000000000000..26eec1b32f85d45f4e8948ac4a482eb7b0351d67 GIT binary patch literal 128652 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 Qw;uJr9O7p6)T$Z&7kP1A761SM literal 0 HcmV?d00001 diff --git a/node_modules/pebble-clay/dist/binaries/aplite/libpebble-clay.a b/node_modules/pebble-clay/dist/binaries/aplite/libpebble-clay.a new file mode 100644 index 0000000..8b277f0 --- /dev/null +++ b/node_modules/pebble-clay/dist/binaries/aplite/libpebble-clay.a @@ -0,0 +1 @@ +! diff --git a/node_modules/pebble-clay/dist/binaries/basalt/libpebble-clay.a b/node_modules/pebble-clay/dist/binaries/basalt/libpebble-clay.a new file mode 100644 index 0000000..8b277f0 --- /dev/null +++ b/node_modules/pebble-clay/dist/binaries/basalt/libpebble-clay.a @@ -0,0 +1 @@ +! diff --git a/node_modules/pebble-clay/dist/binaries/chalk/libpebble-clay.a b/node_modules/pebble-clay/dist/binaries/chalk/libpebble-clay.a new file mode 100644 index 0000000..8b277f0 --- /dev/null +++ b/node_modules/pebble-clay/dist/binaries/chalk/libpebble-clay.a @@ -0,0 +1 @@ +! diff --git a/node_modules/pebble-clay/dist/binaries/diorite/libpebble-clay.a b/node_modules/pebble-clay/dist/binaries/diorite/libpebble-clay.a new file mode 100644 index 0000000..8b277f0 --- /dev/null +++ b/node_modules/pebble-clay/dist/binaries/diorite/libpebble-clay.a @@ -0,0 +1 @@ +! diff --git a/node_modules/pebble-clay/dist/binaries/emery/libpebble-clay.a b/node_modules/pebble-clay/dist/binaries/emery/libpebble-clay.a new file mode 100644 index 0000000..8b277f0 --- /dev/null +++ b/node_modules/pebble-clay/dist/binaries/emery/libpebble-clay.a @@ -0,0 +1 @@ +! diff --git a/node_modules/pebble-clay/dist/include/pebble-clay/aplite/src/resource_ids.auto.h b/node_modules/pebble-clay/dist/include/pebble-clay/aplite/src/resource_ids.auto.h new file mode 100644 index 0000000..c24ede6 --- /dev/null +++ b/node_modules/pebble-clay/dist/include/pebble-clay/aplite/src/resource_ids.auto.h @@ -0,0 +1,8 @@ +#pragma once +#include + +// +// AUTOGENERATED BY BUILD +// DO NOT MODIFY - CHANGES WILL BE OVERWRITTEN +// + diff --git a/node_modules/pebble-clay/dist/include/pebble-clay/basalt/src/resource_ids.auto.h b/node_modules/pebble-clay/dist/include/pebble-clay/basalt/src/resource_ids.auto.h new file mode 100644 index 0000000..c24ede6 --- /dev/null +++ b/node_modules/pebble-clay/dist/include/pebble-clay/basalt/src/resource_ids.auto.h @@ -0,0 +1,8 @@ +#pragma once +#include + +// +// AUTOGENERATED BY BUILD +// DO NOT MODIFY - CHANGES WILL BE OVERWRITTEN +// + diff --git a/node_modules/pebble-clay/dist/include/pebble-clay/chalk/src/resource_ids.auto.h b/node_modules/pebble-clay/dist/include/pebble-clay/chalk/src/resource_ids.auto.h new file mode 100644 index 0000000..c24ede6 --- /dev/null +++ b/node_modules/pebble-clay/dist/include/pebble-clay/chalk/src/resource_ids.auto.h @@ -0,0 +1,8 @@ +#pragma once +#include + +// +// AUTOGENERATED BY BUILD +// DO NOT MODIFY - CHANGES WILL BE OVERWRITTEN +// + diff --git a/node_modules/pebble-clay/dist/include/pebble-clay/diorite/src/resource_ids.auto.h b/node_modules/pebble-clay/dist/include/pebble-clay/diorite/src/resource_ids.auto.h new file mode 100644 index 0000000..c24ede6 --- /dev/null +++ b/node_modules/pebble-clay/dist/include/pebble-clay/diorite/src/resource_ids.auto.h @@ -0,0 +1,8 @@ +#pragma once +#include + +// +// AUTOGENERATED BY BUILD +// DO NOT MODIFY - CHANGES WILL BE OVERWRITTEN +// + diff --git a/node_modules/pebble-clay/dist/include/pebble-clay/emery/src/resource_ids.auto.h b/node_modules/pebble-clay/dist/include/pebble-clay/emery/src/resource_ids.auto.h new file mode 100644 index 0000000..c24ede6 --- /dev/null +++ b/node_modules/pebble-clay/dist/include/pebble-clay/emery/src/resource_ids.auto.h @@ -0,0 +1,8 @@ +#pragma once +#include + +// +// AUTOGENERATED BY BUILD +// DO NOT MODIFY - CHANGES WILL BE OVERWRITTEN +// + diff --git a/node_modules/pebble-clay/dist/include/pebble-clay/message_keys.auto.h b/node_modules/pebble-clay/dist/include/pebble-clay/message_keys.auto.h new file mode 100644 index 0000000..c24ede6 --- /dev/null +++ b/node_modules/pebble-clay/dist/include/pebble-clay/message_keys.auto.h @@ -0,0 +1,8 @@ +#pragma once +#include + +// +// AUTOGENERATED BY BUILD +// DO NOT MODIFY - CHANGES WILL BE OVERWRITTEN +// + diff --git a/node_modules/pebble-clay/dist/js/index.js b/node_modules/pebble-clay/dist/js/index.js new file mode 100644 index 0000000..d4f2c22 --- /dev/null +++ b/node_modules/pebble-clay/dist/js/index.js @@ -0,0 +1,15 @@ +/* Clay - https://github.com/pebble/clay - Version: 1.0.4 - Build Date: 2016-11-21T20:14:28.839Z */ +!function(t){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=t();else if("function"==typeof define&&define.amd)define([],t);else{var e;e="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self?self:this,e.pebbleClay=t()}}(function(){var t;return function e(t,n,r){function o(a,s){if(!n[a]){if(!t[a]){var c="function"==typeof require&&require;if(!s&&c)return c(a,!0);if(i)return i(a,!0);var l=new Error("Cannot find module '"+a+"'");throw l.code="MODULE_NOT_FOUND",l}var u=n[a]={exports:{}};t[a][0].call(u.exports,function(e){var n=t[a][1][e];return o(n?n:e)},u,u.exports,e,t,n,r)}return n[a].exports}for(var i="function"==typeof require&&require,a=0;a0)throw new Error("Invalid string. Length must be a multiple of 4");return"="===t[e-2]?2:"="===t[e-1]?1:0}function o(t){return 3*t.length/4-r(t)}function i(t){var e,n,o,i,a,s,c=t.length;a=r(t),s=new f(3*c/4-a),o=a>0?c-4:c;var l=0;for(e=0,n=0;e>16&255,s[l++]=i>>8&255,s[l++]=255&i;return 2===a?(i=u[t.charCodeAt(e)]<<2|u[t.charCodeAt(e+1)]>>4,s[l++]=255&i):1===a&&(i=u[t.charCodeAt(e)]<<10|u[t.charCodeAt(e+1)]<<4|u[t.charCodeAt(e+2)]>>2,s[l++]=i>>8&255,s[l++]=255&i),s}function a(t){return l[t>>18&63]+l[t>>12&63]+l[t>>6&63]+l[63&t]}function s(t,e,n){for(var r,o=[],i=e;iu?u:c+a));return 1===r?(e=t[n-1],o+=l[e>>2],o+=l[e<<4&63],o+="=="):2===r&&(e=(t[n-2]<<8)+t[n-1],o+=l[e>>10],o+=l[e>>4&63],o+=l[e<<2&63],o+="="),i.push(o),i.join("")}n.byteLength=o,n.toByteArray=i,n.fromByteArray=c;for(var l=[],u=[],f="undefined"!=typeof Uint8Array?Uint8Array:Array,p="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/",d=0,h=p.length;d + * @license MIT + */ +"use strict";function r(){try{var t=new Uint8Array(1);return t.__proto__={__proto__:Uint8Array.prototype,foo:function(){return 42}},42===t.foo()&&"function"==typeof t.subarray&&0===t.subarray(1,1).byteLength}catch(e){return!1}}function o(){return a.TYPED_ARRAY_SUPPORT?2147483647:1073741823}function i(t,e){if(o()=o())throw new RangeError("Attempt to allocate Buffer larger than maximum size: 0x"+o().toString(16)+" bytes");return 0|t}function g(t){return+t!=t&&(t=0),a.alloc(+t)}function b(t,e){if(a.isBuffer(t))return t.length;if("undefined"!=typeof ArrayBuffer&&"function"==typeof ArrayBuffer.isView&&(ArrayBuffer.isView(t)||t instanceof ArrayBuffer))return t.byteLength;"string"!=typeof t&&(t=""+t);var n=t.length;if(0===n)return 0;for(var r=!1;;)switch(e){case"ascii":case"latin1":case"binary":return n;case"utf8":case"utf-8":case void 0:return W(t).length;case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":return 2*n;case"hex":return n>>>1;case"base64":return U(t).length;default:if(r)return W(t).length;e=(""+e).toLowerCase(),r=!0}}function y(t,e,n){var r=!1;if((void 0===e||e<0)&&(e=0),e>this.length)return"";if((void 0===n||n>this.length)&&(n=this.length),n<=0)return"";if(n>>>=0,e>>>=0,n<=e)return"";for(t||(t="utf8");;)switch(t){case"hex":return D(this,e,n);case"utf8":case"utf-8":return E(this,e,n);case"ascii":return B(this,e,n);case"latin1":case"binary":return S(this,e,n);case"base64":return O(this,e,n);case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":return N(this,e,n);default:if(r)throw new TypeError("Unknown encoding: "+t);t=(t+"").toLowerCase(),r=!0}}function v(t,e,n){var r=t[e];t[e]=t[n],t[n]=r}function A(t,e,n,r,o){if(0===t.length)return-1;if("string"==typeof n?(r=n,n=0):n>2147483647?n=2147483647:n<-2147483648&&(n=-2147483648),n=+n,isNaN(n)&&(n=o?0:t.length-1),n<0&&(n=t.length+n),n>=t.length){if(o)return-1;n=t.length-1}else if(n<0){if(!o)return-1;n=0}if("string"==typeof e&&(e=a.from(e,r)),a.isBuffer(e))return 0===e.length?-1:w(t,e,n,r,o);if("number"==typeof e)return e=255&e,a.TYPED_ARRAY_SUPPORT&&"function"==typeof Uint8Array.prototype.indexOf?o?Uint8Array.prototype.indexOf.call(t,e,n):Uint8Array.prototype.lastIndexOf.call(t,e,n):w(t,[e],n,r,o);throw new TypeError("val must be string, number or Buffer")}function w(t,e,n,r,o){function i(t,e){return 1===a?t[e]:t.readUInt16BE(e*a)}var a=1,s=t.length,c=e.length;if(void 0!==r&&(r=String(r).toLowerCase(),"ucs2"===r||"ucs-2"===r||"utf16le"===r||"utf-16le"===r)){if(t.length<2||e.length<2)return-1;a=2,s/=2,c/=2,n/=2}var l;if(o){var u=-1;for(l=n;ls&&(n=s-c),l=n;l>=0;l--){for(var f=!0,p=0;po&&(r=o)):r=o;var i=e.length;if(i%2!==0)throw new TypeError("Invalid hex string");r>i/2&&(r=i/2);for(var a=0;a239?4:i>223?3:i>191?2:1;if(o+s<=n){var c,l,u,f;switch(s){case 1:i<128&&(a=i);break;case 2:c=t[o+1],128===(192&c)&&(f=(31&i)<<6|63&c,f>127&&(a=f));break;case 3:c=t[o+1],l=t[o+2],128===(192&c)&&128===(192&l)&&(f=(15&i)<<12|(63&c)<<6|63&l,f>2047&&(f<55296||f>57343)&&(a=f));break;case 4:c=t[o+1],l=t[o+2],u=t[o+3],128===(192&c)&&128===(192&l)&&128===(192&u)&&(f=(15&i)<<18|(63&c)<<12|(63&l)<<6|63&u,f>65535&&f<1114112&&(a=f))}}null===a?(a=65533,s=1):a>65535&&(a-=65536,r.push(a>>>10&1023|55296),a=56320|1023&a),r.push(a),o+=s}return j(r)}function j(t){var e=t.length;if(e<=tt)return String.fromCharCode.apply(String,t);for(var n="",r=0;rr)&&(n=r);for(var o="",i=e;in)throw new RangeError("Trying to access beyond buffer length")}function F(t,e,n,r,o,i){if(!a.isBuffer(t))throw new TypeError('"buffer" argument must be a Buffer instance');if(e>o||et.length)throw new RangeError("Index out of range")}function z(t,e,n,r){e<0&&(e=65535+e+1);for(var o=0,i=Math.min(t.length-n,2);o>>8*(r?o:1-o)}function I(t,e,n,r){e<0&&(e=4294967295+e+1);for(var o=0,i=Math.min(t.length-n,4);o>>8*(r?o:3-o)&255}function L(t,e,n,r,o,i){if(n+r>t.length)throw new RangeError("Index out of range");if(n<0)throw new RangeError("Index out of range")}function K(t,e,n,r,o){return o||L(t,e,n,4,3.4028234663852886e38,-3.4028234663852886e38),$.write(t,e,n,r,23,4),n+4}function G(t,e,n,r,o){return o||L(t,e,n,8,1.7976931348623157e308,-1.7976931348623157e308),$.write(t,e,n,r,52,8),n+8}function C(t){if(t=X(t).replace(et,""),t.length<2)return"";for(;t.length%4!==0;)t+="=";return t}function X(t){return t.trim?t.trim():t.replace(/^\s+|\s+$/g,"")}function V(t){return t<16?"0"+t.toString(16):t.toString(16)}function W(t,e){e=e||1/0;for(var n,r=t.length,o=null,i=[],a=0;a55295&&n<57344){if(!o){if(n>56319){(e-=3)>-1&&i.push(239,191,189);continue}if(a+1===r){(e-=3)>-1&&i.push(239,191,189);continue}o=n;continue}if(n<56320){(e-=3)>-1&&i.push(239,191,189),o=n;continue}n=(o-55296<<10|n-56320)+65536}else o&&(e-=3)>-1&&i.push(239,191,189);if(o=null,n<128){if((e-=1)<0)break;i.push(n)}else if(n<2048){if((e-=2)<0)break;i.push(n>>6|192,63&n|128)}else if(n<65536){if((e-=3)<0)break;i.push(n>>12|224,n>>6&63|128,63&n|128)}else{if(!(n<1114112))throw new Error("Invalid code point");if((e-=4)<0)break;i.push(n>>18|240,n>>12&63|128,n>>6&63|128,63&n|128)}}return i}function Z(t){for(var e=[],n=0;n>8,o=n%256,i.push(o),i.push(r);return i}function U(t){return Q.toByteArray(C(t))}function q(t,e,n,r){for(var o=0;o=e.length||o>=t.length);++o)e[o+n]=t[o];return o}function H(t){return t!==t}var Q=t("base64-js"),$=t("ieee754"),_=t("isarray");n.Buffer=a,n.SlowBuffer=g,n.INSPECT_MAX_BYTES=50,a.TYPED_ARRAY_SUPPORT=void 0!==e.TYPED_ARRAY_SUPPORT?e.TYPED_ARRAY_SUPPORT:r(),n.kMaxLength=o(),a.poolSize=8192,a._augment=function(t){return t.__proto__=a.prototype,t},a.from=function(t,e,n){return s(null,t,e,n)},a.TYPED_ARRAY_SUPPORT&&(a.prototype.__proto__=Uint8Array.prototype,a.__proto__=Uint8Array,"undefined"!=typeof Symbol&&Symbol.species&&a[Symbol.species]===a&&Object.defineProperty(a,Symbol.species,{value:null,configurable:!0})),a.alloc=function(t,e,n){return l(null,t,e,n)},a.allocUnsafe=function(t){return u(null,t)},a.allocUnsafeSlow=function(t){return u(null,t)},a.isBuffer=function(t){return!(null==t||!t._isBuffer)},a.compare=function(t,e){if(!a.isBuffer(t)||!a.isBuffer(e))throw new TypeError("Arguments must be Buffers");if(t===e)return 0;for(var n=t.length,r=e.length,o=0,i=Math.min(n,r);o0&&(t=this.toString("hex",0,e).match(/.{2}/g).join(" "),this.length>e&&(t+=" ... ")),""},a.prototype.compare=function(t,e,n,r,o){if(!a.isBuffer(t))throw new TypeError("Argument must be a Buffer");if(void 0===e&&(e=0),void 0===n&&(n=t?t.length:0),void 0===r&&(r=0),void 0===o&&(o=this.length),e<0||n>t.length||r<0||o>this.length)throw new RangeError("out of range index");if(r>=o&&e>=n)return 0;if(r>=o)return-1;if(e>=n)return 1;if(e>>>=0,n>>>=0,r>>>=0,o>>>=0,this===t)return 0;for(var i=o-r,s=n-e,c=Math.min(i,s),l=this.slice(r,o),u=t.slice(e,n),f=0;fo)&&(n=o),t.length>0&&(n<0||e<0)||e>this.length)throw new RangeError("Attempt to write outside buffer bounds");r||(r="utf8");for(var i=!1;;)switch(r){case"hex":return k(this,t,e,n);case"utf8":case"utf-8":return x(this,t,e,n);case"ascii":return M(this,t,e,n);case"latin1":case"binary":return T(this,t,e,n);case"base64":return R(this,t,e,n);case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":return P(this,t,e,n);default:if(i)throw new TypeError("Unknown encoding: "+r);r=(""+r).toLowerCase(),i=!0}},a.prototype.toJSON=function(){return{type:"Buffer",data:Array.prototype.slice.call(this._arr||this,0)}};var tt=4096;a.prototype.slice=function(t,e){var n=this.length;t=~~t,e=void 0===e?n:~~e,t<0?(t+=n,t<0&&(t=0)):t>n&&(t=n),e<0?(e+=n,e<0&&(e=0)):e>n&&(e=n),e0&&(o*=256);)r+=this[t+--e]*o;return r},a.prototype.readUInt8=function(t,e){return e||Y(t,1,this.length),this[t]},a.prototype.readUInt16LE=function(t,e){return e||Y(t,2,this.length),this[t]|this[t+1]<<8},a.prototype.readUInt16BE=function(t,e){return e||Y(t,2,this.length),this[t]<<8|this[t+1]},a.prototype.readUInt32LE=function(t,e){return e||Y(t,4,this.length),(this[t]|this[t+1]<<8|this[t+2]<<16)+16777216*this[t+3]},a.prototype.readUInt32BE=function(t,e){return e||Y(t,4,this.length),16777216*this[t]+(this[t+1]<<16|this[t+2]<<8|this[t+3])},a.prototype.readIntLE=function(t,e,n){t=0|t,e=0|e,n||Y(t,e,this.length);for(var r=this[t],o=1,i=0;++i=o&&(r-=Math.pow(2,8*e)),r},a.prototype.readIntBE=function(t,e,n){t=0|t,e=0|e,n||Y(t,e,this.length);for(var r=e,o=1,i=this[t+--r];r>0&&(o*=256);)i+=this[t+--r]*o;return o*=128,i>=o&&(i-=Math.pow(2,8*e)),i},a.prototype.readInt8=function(t,e){return e||Y(t,1,this.length),128&this[t]?(255-this[t]+1)*-1:this[t]},a.prototype.readInt16LE=function(t,e){e||Y(t,2,this.length);var n=this[t]|this[t+1]<<8;return 32768&n?4294901760|n:n},a.prototype.readInt16BE=function(t,e){e||Y(t,2,this.length);var n=this[t+1]|this[t]<<8;return 32768&n?4294901760|n:n},a.prototype.readInt32LE=function(t,e){return e||Y(t,4,this.length),this[t]|this[t+1]<<8|this[t+2]<<16|this[t+3]<<24},a.prototype.readInt32BE=function(t,e){return e||Y(t,4,this.length),this[t]<<24|this[t+1]<<16|this[t+2]<<8|this[t+3]},a.prototype.readFloatLE=function(t,e){return e||Y(t,4,this.length),$.read(this,t,!0,23,4)},a.prototype.readFloatBE=function(t,e){return e||Y(t,4,this.length),$.read(this,t,!1,23,4)},a.prototype.readDoubleLE=function(t,e){return e||Y(t,8,this.length),$.read(this,t,!0,52,8)},a.prototype.readDoubleBE=function(t,e){return e||Y(t,8,this.length),$.read(this,t,!1,52,8)},a.prototype.writeUIntLE=function(t,e,n,r){if(t=+t,e=0|e,n=0|n,!r){var o=Math.pow(2,8*n)-1;F(this,t,e,n,o,0)}var i=1,a=0;for(this[e]=255&t;++a=0&&(a*=256);)this[e+i]=t/a&255;return e+n},a.prototype.writeUInt8=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,1,255,0),a.TYPED_ARRAY_SUPPORT||(t=Math.floor(t)),this[e]=255&t,e+1},a.prototype.writeUInt16LE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,2,65535,0),a.TYPED_ARRAY_SUPPORT?(this[e]=255&t,this[e+1]=t>>>8):z(this,t,e,!0),e+2},a.prototype.writeUInt16BE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,2,65535,0),a.TYPED_ARRAY_SUPPORT?(this[e]=t>>>8,this[e+1]=255&t):z(this,t,e,!1),e+2},a.prototype.writeUInt32LE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,4,4294967295,0),a.TYPED_ARRAY_SUPPORT?(this[e+3]=t>>>24,this[e+2]=t>>>16,this[e+1]=t>>>8,this[e]=255&t):I(this,t,e,!0),e+4},a.prototype.writeUInt32BE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,4,4294967295,0),a.TYPED_ARRAY_SUPPORT?(this[e]=t>>>24,this[e+1]=t>>>16,this[e+2]=t>>>8,this[e+3]=255&t):I(this,t,e,!1),e+4},a.prototype.writeIntLE=function(t,e,n,r){if(t=+t,e=0|e,!r){var o=Math.pow(2,8*n-1);F(this,t,e,n,o-1,-o)}var i=0,a=1,s=0;for(this[e]=255&t;++i>0)-s&255;return e+n},a.prototype.writeIntBE=function(t,e,n,r){if(t=+t,e=0|e,!r){var o=Math.pow(2,8*n-1);F(this,t,e,n,o-1,-o)}var i=n-1,a=1,s=0;for(this[e+i]=255&t;--i>=0&&(a*=256);)t<0&&0===s&&0!==this[e+i+1]&&(s=1),this[e+i]=(t/a>>0)-s&255;return e+n},a.prototype.writeInt8=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,1,127,-128),a.TYPED_ARRAY_SUPPORT||(t=Math.floor(t)),t<0&&(t=255+t+1),this[e]=255&t,e+1},a.prototype.writeInt16LE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,2,32767,-32768),a.TYPED_ARRAY_SUPPORT?(this[e]=255&t,this[e+1]=t>>>8):z(this,t,e,!0),e+2},a.prototype.writeInt16BE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,2,32767,-32768),a.TYPED_ARRAY_SUPPORT?(this[e]=t>>>8,this[e+1]=255&t):z(this,t,e,!1),e+2},a.prototype.writeInt32LE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,4,2147483647,-2147483648),a.TYPED_ARRAY_SUPPORT?(this[e]=255&t,this[e+1]=t>>>8,this[e+2]=t>>>16,this[e+3]=t>>>24):I(this,t,e,!0),e+4},a.prototype.writeInt32BE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,4,2147483647,-2147483648),t<0&&(t=4294967295+t+1),a.TYPED_ARRAY_SUPPORT?(this[e]=t>>>24,this[e+1]=t>>>16,this[e+2]=t>>>8,this[e+3]=255&t):I(this,t,e,!1),e+4},a.prototype.writeFloatLE=function(t,e,n){return K(this,t,e,!0,n)},a.prototype.writeFloatBE=function(t,e,n){return K(this,t,e,!1,n)},a.prototype.writeDoubleLE=function(t,e,n){return G(this,t,e,!0,n)},a.prototype.writeDoubleBE=function(t,e,n){return G(this,t,e,!1,n)},a.prototype.copy=function(t,e,n,r){if(n||(n=0),r||0===r||(r=this.length),e>=t.length&&(e=t.length),e||(e=0),r>0&&r=this.length)throw new RangeError("sourceStart out of bounds");if(r<0)throw new RangeError("sourceEnd out of bounds");r>this.length&&(r=this.length),t.length-e=0;--o)t[o+e]=this[o+n];else if(i<1e3||!a.TYPED_ARRAY_SUPPORT)for(o=0;o>>=0,n=void 0===n?this.length:n>>>0,t||(t=0);var i;if("number"==typeof t)for(i=e;in;++n)if(i=t[n],i===e||i!==i&&e!==e)return n;return-1}n.__esModule=!0;var o=Object.prototype.toString,i="undefined"!=typeof e?function(t){return e.isBuffer(t)}:function(){return!1},a="function"==typeof Object.keys?function(t){return Object.keys(t)}:function(t){var e=typeof t;if(null===t||"function"!==e&&"object"!==e)throw new TypeError("obj must be an Object");var n=[],r=void 0;for(r in t)Object.prototype.hasOwnProperty.call(t,r)&&n.push(r);return n},s="function"==typeof Symbol?function(t){return Object.getOwnPropertySymbols(t)}:function(){return[]};n.getKeys=a,n.getSymbols=s,n.indexOf=r,n.isBuffer=i},function(t,n,r){"use strict";function o(t,e){var n=a(t);return null!==n?n:i(t,e)}function i(t,n){if("function"!=typeof n)throw new TypeError("customizer is must be a Function");if("function"==typeof t){var r=String(t);return/^\s*function\s*\S*\([^\)]*\)\s*{\s*\[native code\]\s*}/.test(r)?t:new Function("return "+String(r))()}var o=c.call(t);if("[object Array]"===o)return[];if("[object Object]"===o&&t.constructor===Object)return{};if("[object Date]"===o)return new Date(t.getTime());if("[object RegExp]"===o){var i=String(t),a=i.lastIndexOf("/");return new RegExp(i.slice(1,a),i.slice(a+1))}if((0,s.isBuffer)(t)){var l=new e(t.length);return t.copy(l),l}var u=n(t);return void 0!==u?u:null}function a(t){var e=typeof t;return null!==t&&"object"!==e&&"function"!==e?t:null}n.__esModule=!0,n.copyValue=n.copyCollection=n.copy=void 0;var s=r(1),c=Object.prototype.toString;n.copy=o,n.copyCollection=i,n.copyValue=a},function(t,e,n){"use strict";function r(t){}function o(t){var e=arguments.length<=1||void 0===arguments[1]?r:arguments[1];if(null===t)return null;var n=(0,a.copyValue)(t);if(null!==n)return n;var o=(0,a.copyCollection)(t,e),s=null!==o?o:t,c=[t],l=[s];return i(t,e,s,c,l)}function i(t,e,n,r,o){if(null===t)return null;var c=(0,a.copyValue)(t);if(null!==c)return c;var l=(0,s.getKeys)(t).concat((0,s.getSymbols)(t)),u=void 0,f=void 0,p=void 0,d=void 0,h=void 0,m=void 0,g=void 0,b=void 0;for(u=0,f=l.length;f>u;++u)p=l[u],d=t[p],h=(0,s.indexOf)(r,d),m=void 0,g=void 0,b=void 0,-1===h?(m=(0,a.copy)(d,e),g=null!==m?m:d,null!==d&&/^(?:function|object)$/.test(typeof d)&&(r.push(d),o.push(g))):b=o[h],n[p]=b||i(d,e,g,r,o);return n}e.__esModule=!0;var a=n(2),s=n(1);e["default"]=o,t.exports=e["default"]}])})}).call(this,e("buffer").Buffer)},{buffer:2}],4:[function(t,e,n){n.read=function(t,e,n,r,o){var i,a,s=8*o-r-1,c=(1<>1,u=-7,f=n?o-1:0,p=n?-1:1,d=t[e+f];for(f+=p,i=d&(1<<-u)-1,d>>=-u,u+=s;u>0;i=256*i+t[e+f],f+=p,u-=8);for(a=i&(1<<-u)-1,i>>=-u,u+=r;u>0;a=256*a+t[e+f],f+=p,u-=8);if(0===i)i=1-l;else{if(i===c)return a?NaN:(d?-1:1)*(1/0);a+=Math.pow(2,r),i-=l}return(d?-1:1)*a*Math.pow(2,i-r)},n.write=function(t,e,n,r,o,i){var a,s,c,l=8*i-o-1,u=(1<>1,p=23===o?Math.pow(2,-24)-Math.pow(2,-77):0,d=r?0:i-1,h=r?1:-1,m=e<0||0===e&&1/e<0?1:0;for(e=Math.abs(e),isNaN(e)||e===1/0?(s=isNaN(e)?1:0,a=u):(a=Math.floor(Math.log(e)/Math.LN2),e*(c=Math.pow(2,-a))<1&&(a--,c*=2),e+=a+f>=1?p/c:p*Math.pow(2,1-f),e*c>=2&&(a++,c/=2),a+f>=u?(s=0,a=u):a+f>=1?(s=(e*c-1)*Math.pow(2,o),a+=f):(s=e*Math.pow(2,f-1)*Math.pow(2,o),a=0));o>=8;t[n+d]=255&s,d+=h,s/=256,o-=8);for(a=a<0;t[n+d]=255&a,d+=h,a/=256,l-=8);t[n+d-h]|=128*m}},{}],5:[function(t,e,n){var r={}.toString;e.exports=Array.isArray||function(t){return"[object Array]"==r.call(t)}},{}],6:[function(t,e,n){function r(t){return/^[a-z_$][0-9a-z_$]*$/gi.test(t)&&!i.test(t)}function o(t){if(a)return t.toString();var e=t.source.replace(/\//g,function(t,e,n){return 0===e||"\\"!==n[e-1]?"\\/":"/"}),n=(t.global&&"g"||"")+(t.ignoreCase&&"i"||"")+(t.multiline&&"m"||"");return"/"+e+"/"+n}/* toSource by Marcello Bastea-Forte - zlib license */ +e.exports=function(t,e,n,i){function a(t,e,n,i,s){function c(t){return n.slice(1)+t.join(","+(n&&"\n")+l)+(n?" ":"")}var l=i+n;switch(t=e?e(t):t,typeof t){case"string":return JSON.stringify(t);case"boolean":case"number":case"undefined":return""+t;case"function":return t.toString()}if(null===t)return"null";if(t instanceof RegExp)return o(t);if(t instanceof Date)return"new Date("+t.getTime()+")";var u=s.indexOf(t)+1;if(u>0)return"{$circularReference:"+u+"}";if(s.push(t),Array.isArray(t))return"["+c(t.map(function(t){return a(t,e,n,l,s.slice())}))+"]";var f=Object.keys(t);return f.length?"{"+c(f.map(function(o){return(r(o)?o:JSON.stringify(o))+":"+a(t[o],e,n,l,s.slice())}))+"}":"{}"}var s=[];return a(t,e,void 0===n?" ":n||"",i||"",s)};var i=/^(abstract|boolean|break|byte|case|catch|char|class|const|continue|debugger|default|delete|do|double|else|enum|export|extends|false|final|finally|float|for|function|goto|if|implements|import|in|instanceof|int|interface|long|native|new|null|package|private|protected|public|return|short|static|super|switch|synchronized|this|throw|throws|transient|true|try|typeof|undefined|var|void|volatile|while|with)$/,a="\\/"===new RegExp("/").source},{}],7:[function(t,e,n){e.exports={name:"pebble-clay",version:"1.0.4",description:"Pebble Config Framework",scripts:{"test-travis":"./node_modules/.bin/gulp && ./node_modules/.bin/karma start ./test/karma.conf.js --single-run --browsers chromeTravisCI && ./node_modules/.bin/eslint ./","test-debug":"(export DEBUG=true && ./node_modules/.bin/gulp && ./node_modules/.bin/karma start ./test/karma.conf.js --no-single-run)",test:"./node_modules/.bin/gulp && ./node_modules/.bin/karma start ./test/karma.conf.js --single-run",lint:"./node_modules/.bin/eslint ./",build:"gulp",dev:"gulp dev","pebble-clean":"rm -rf tmp src/js/index.js && pebble clean","pebble-publish":"npm run pebble-clean && npm run build && pebble build && pebble package publish && npm run pebble-clean","pebble-build":"npm run build && pebble build"},repository:{type:"git",url:"git+https://github.com/pebble/clay.git"},keywords:["pebble","config","configuration","pebble-package"],author:"Pebble Technology",license:"MIT",bugs:{url:"https://github.com/pebble/clay/issues"},pebble:{projectType:"package",sdkVersion:"3",targetPlatforms:["aplite","basalt","chalk","diorite","emery"],resources:{media:[]},capabilities:["configurable"]},homepage:"https://github.com/pebble/clay#readme",devDependencies:{autoprefixer:"^6.3.1",bourbon:"^4.2.6",browserify:"^13.0.0","browserify-istanbul":"^0.2.1",chai:"^3.4.1",deamdify:"^0.2.0",deepcopy:"^0.6.1",del:"^2.0.2",eslint:"^1.5.1","eslint-config-pebble":"^1.2.0","eslint-plugin-standard":"^1.3.1",gulp:"^3.9.0","gulp-autoprefixer":"^3.1.0","gulp-htmlmin":"^1.3.0","gulp-inline":"0.0.15","gulp-insert":"^0.5.0","gulp-sass":"^2.1.1","gulp-sourcemaps":"^1.6.0","gulp-uglify":"^1.5.2",joi:"^6.10.1",karma:"^0.13.19","karma-browserify":"^5.0.1","karma-chrome-launcher":"^0.2.2","karma-coverage":"^0.5.3","karma-mocha":"^0.2.1","karma-mocha-reporter":"^1.1.5","karma-source-map-support":"^1.1.0","karma-threshold-reporter":"^0.1.15",mocha:"^2.3.4",postcss:"^5.0.14","require-from-string":"^1.1.0",sassify:"^0.9.1",sinon:"^1.17.3",stringify:"^3.2.0",through:"^2.3.8",tosource:"^1.0.0","vinyl-buffer":"^1.0.0","vinyl-source-stream":"^1.1.0",watchify:"^3.7.0"},dependencies:{}}},{}],8:[function(t,e,n){"use strict";e.exports={name:"button",template:t("../../templates/components/button.tpl"),style:t("../../styles/clay/components/button.scss"),manipulator:"button",defaults:{primary:!1,attributes:{},description:""}}},{"../../styles/clay/components/button.scss":21,"../../templates/components/button.tpl":30}],9:[function(t,e,n){"use strict";e.exports={name:"checkboxgroup",template:t("../../templates/components/checkboxgroup.tpl"),style:t("../../styles/clay/components/checkboxgroup.scss"),manipulator:"checkboxgroup",defaults:{label:"",options:[],description:""}}},{"../../styles/clay/components/checkboxgroup.scss":22,"../../templates/components/checkboxgroup.tpl":31}],10:[function(t,e,n){"use strict";e.exports={name:"color",template:t("../../templates/components/color.tpl"),style:t("../../styles/clay/components/color.scss"),manipulator:"color",defaults:{label:"",description:""},initialize:function(t,e){function n(t){if("number"==typeof t)t=t.toString(16);else if(!t)return"transparent";return t=r(t),"#"+(f?p[t]:t)}function r(t){for(t=t.toLowerCase();t.length<6;)t="0"+t;return t}function o(t){switch(typeof t){case"number":return r(t.toString(16));case"string":return t.replace(/^#|^0x/,"");default:return t}}function i(t){return t.reduce(function(t,e){return t.concat(e)},[])}function a(t){t=t.replace(/^#|^0x/,"");var e=parseInt(t.slice(0,2),16)/255,n=parseInt(t.slice(2,4),16)/255,r=parseInt(t.slice(4),16)/255;e=e>.04045?Math.pow((e+.055)/1.055,2.4):e/12.92,n=n>.04045?Math.pow((n+.055)/1.055,2.4):n/12.92,r=r>.04045?Math.pow((r+.055)/1.055,2.4):r/12.92;var o=(.4124*e+.3576*n+.1805*r)/.95047,i=(.2126*e+.7152*n+.0722*r)/1,a=(.0193*e+.1192*n+.9505*r)/1.08883;return o=o>.008856?Math.pow(o,1/3):7.787*o+16/116,i=i>.008856?Math.pow(i,1/3):7.787*i+16/116,a=a>.008856?Math.pow(a,1/3):7.787*a+16/116,[116*i-16,500*(o-i),200*(i-a)]}function s(t,e){var n=t[0]-e[0],r=t[1]-e[1],o=t[2]-e[2];return Math.sqrt(Math.pow(n,2)+Math.pow(r,2)+Math.pow(o,2))}function c(){return!e.meta.activeWatchInfo||2===e.meta.activeWatchInfo.firmware.major||["aplite","diorite"].indexOf(e.meta.activeWatchInfo.platform)>-1&&!u.config.allowGray?d.BLACK_WHITE:["aplite","diorite"].indexOf(e.meta.activeWatchInfo.platform)>-1&&u.config.allowGray?d.GRAY:d.COLOR}var l=t.HTML,u=this;u.roundColorToLayout=function(t){var e=o(t);if(m.indexOf(e)===-1){var n=a(e),r=m.map(function(t){var e=a(o(t));return s(n,e)}),i=Math.min.apply(Math,r),c=r.indexOf(i);e=m[c]}return parseInt(e,16)};var f=u.config.sunlight!==!1,p={"000000":"000000","000055":"001e41","0000aa":"004387","0000ff":"0068ca","005500":"2b4a2c","005555":"27514f","0055aa":"16638d","0055ff":"007dce","00aa00":"5e9860","00aa55":"5c9b72","00aaaa":"57a5a2","00aaff":"4cb4db","00ff00":"8ee391","00ff55":"8ee69e","00ffaa":"8aebc0","00ffff":"84f5f1",550000:"4a161b",550055:"482748","5500aa":"40488a","5500ff":"2f6bcc",555500:"564e36",555555:"545454","5555aa":"4f6790","5555ff":"4180d0","55aa00":"759a64","55aa55":"759d76","55aaaa":"71a6a4","55aaff":"69b5dd","55ff00":"9ee594","55ff55":"9de7a0","55ffaa":"9becc2","55ffff":"95f6f2",aa0000:"99353f",aa0055:"983e5a",aa00aa:"955694",aa00ff:"8f74d2",aa5500:"9d5b4d",aa5555:"9d6064",aa55aa:"9a7099",aa55ff:"9587d5",aaaa00:"afa072",aaaa55:"aea382",aaaaaa:"ababab",ffffff:"ffffff",aaaaff:"a7bae2",aaff00:"c9e89d",aaff55:"c9eaa7",aaffaa:"c7f0c8",aaffff:"c3f9f7",ff0000:"e35462",ff0055:"e25874",ff00aa:"e16aa3",ff00ff:"de83dc",ff5500:"e66e6b",ff5555:"e6727c",ff55aa:"e37fa7",ff55ff:"e194df",ffaa00:"f1aa86",ffaa55:"f1ad93",ffaaaa:"efb5b8",ffaaff:"ecc3eb",ffff00:"ffeeab",ffff55:"fff1b5",ffffaa:"fff6d3"},d={COLOR:[[!1,!1,"55ff00","aaff55",!1,"ffff55","ffffaa",!1,!1],[!1,"aaffaa","55ff55","00ff00","aaff00","ffff00","ffaa55","ffaaaa",!1],["55ffaa","00ff55","00aa00","55aa00","aaaa55","aaaa00","ffaa00","ff5500","ff5555"],["aaffff","00ffaa","00aa55","55aa55","005500","555500","aa5500","ff0000","ff0055"],[!1,"55aaaa","00aaaa","005555","ffffff","000000","aa5555","aa0000",!1],["55ffff","00ffff","00aaff","0055aa","aaaaaa","555555","550000","aa0055","ff55aa"],["55aaff","0055ff","0000ff","0000aa","000055","550055","aa00aa","ff00aa","ffaaff"],[!1,"5555aa","5555ff","5500ff","5500aa","aa00ff","ff00ff","ff55ff",!1],[!1,!1,!1,"aaaaff","aa55ff","aa55aa",!1,!1,!1]],GRAY:[["000000","aaaaaa","ffffff"]],BLACK_WHITE:[["000000","ffffff"]]},h=u.config.layout||c();"string"==typeof h&&(h=d[h]),Array.isArray(h[0])||(h=[h]);var m=i(h).map(function(t){return o(t)}).filter(function(t){return t}),g="",b=h.length,y=0;h.forEach(function(t){y=t.length>y?t.length:y});for(var v=100/y,A=100/b,w=u.$element,k=0;k'}var j=0;3===y&&(j=5),2===y&&(j=8);var B=j*v/A+"%",S=j+"%";w.select(".color-box-container").add(l(g)).set("$paddingTop",B).set("$paddingRight",S).set("$paddingBottom",B).set("$paddingLeft",S),w.select(".color-box-wrap").set("$paddingBottom",v/A*100+"%");var D=w.select(".value"),N=w.select(".picker-wrap"),Y=u.$manipulatorTarget.get("disabled");w.select("label").on("click",function(){Y||N.set("show")}),u.on("change",function(){var t=u.get();D.set("$background-color",n(t)),w.select(".color-box").set("-selected"),w.select('.color-box[data-value="'+t+'"]').set("+selected")}),w.select(".color-box.selectable").on("click",function(t){u.set(parseInt(t.target.dataset.value,10)),N.set("-show")}),N.on("click",function(){N.set("-show")}),u.on("disabled",function(){Y=!0}),u.on("enabled",function(){Y=!1}),u._layout=h}}},{"../../styles/clay/components/color.scss":23,"../../templates/components/color.tpl":32}],11:[function(t,e,n){"use strict";e.exports={name:"footer",template:t("../../templates/components/footer.tpl"),manipulator:"html"}},{"../../templates/components/footer.tpl":33}],12:[function(t,e,n){"use strict";e.exports={name:"heading",template:t("../../templates/components/heading.tpl"),manipulator:"html",defaults:{size:4}}},{"../../templates/components/heading.tpl":34}],13:[function(t,e,n){"use strict";e.exports={color:t("./color"),footer:t("./footer"),heading:t("./heading"),input:t("./input"),select:t("./select"),submit:t("./submit"),text:t("./text"),toggle:t("./toggle"),radiogroup:t("./radiogroup"),checkboxgroup:t("./checkboxgroup"),button:t("./button"),slider:t("./slider")}},{"./button":8,"./checkboxgroup":9,"./color":10,"./footer":11,"./heading":12,"./input":14,"./radiogroup":15,"./select":16,"./slider":17,"./submit":18,"./text":19,"./toggle":20}],14:[function(t,e,n){"use strict";e.exports={name:"input",template:t("../../templates/components/input.tpl"),style:t("../../styles/clay/components/input.scss"),manipulator:"val",defaults:{label:"",description:"",attributes:{}}}},{"../../styles/clay/components/input.scss":24,"../../templates/components/input.tpl":35}],15:[function(t,e,n){"use strict";e.exports={name:"radiogroup",template:t("../../templates/components/radiogroup.tpl"),style:t("../../styles/clay/components/radiogroup.scss"),manipulator:"radiogroup",defaults:{label:"",options:[],description:"",attributes:{}}}},{"../../styles/clay/components/radiogroup.scss":25,"../../templates/components/radiogroup.tpl":36}],16:[function(t,e,n){"use strict";e.exports={name:"select",template:t("../../templates/components/select.tpl"),style:t("../../styles/clay/components/select.scss"),manipulator:"val",defaults:{label:"",options:[],description:"",attributes:{}},initialize:function(){function t(){var t=e.$manipulatorTarget.get("selectedIndex"),r=e.$manipulatorTarget.select("option"),o=r[t]&&r[t].innerHTML;n.set("innerHTML",o)}var e=this,n=e.$element.select(".value");t(),e.on("change",t)}}},{"../../styles/clay/components/select.scss":26,"../../templates/components/select.tpl":37}],17:[function(t,e,n){"use strict";e.exports={name:"slider",template:t("../../templates/components/slider.tpl"),style:t("../../styles/clay/components/slider.scss"),manipulator:"slider",defaults:{label:"",description:"",min:0,max:100,step:1,attributes:{}},initialize:function(){function t(){var t=e.get().toFixed(e.precision);n.set("value",t),r.set("innerHTML",t)}var e=this,n=e.$element.select(".value"),r=e.$element.select(".value-pad"),o=e.$manipulatorTarget,i=o.get("step");i=i.toString(10).split(".")[1],e.precision=i?i.length:0,e.on("change",t),o.on("|input",t),t(),n.on("|input",function(){r.set("innerHTML",this.get("value"))}),n.on("|change",function(){e.set(this.get("value")),t()})}}},{"../../styles/clay/components/slider.scss":27,"../../templates/components/slider.tpl":38}],18:[function(t,e,n){"use strict";e.exports={name:"submit",template:t("../../templates/components/submit.tpl"),style:t("../../styles/clay/components/submit.scss"),manipulator:"button",defaults:{attributes:{}}}},{"../../styles/clay/components/submit.scss":28,"../../templates/components/submit.tpl":39}],19:[function(t,e,n){"use strict";e.exports={name:"text",template:t("../../templates/components/text.tpl"),manipulator:"html"}},{"../../templates/components/text.tpl":40}],20:[function(t,e,n){"use strict";e.exports={name:"toggle",template:t("../../templates/components/toggle.tpl"),style:t("../../styles/clay/components/toggle.scss"),manipulator:"checked",defaults:{label:"",description:"",attributes:{}}}},{"../../styles/clay/components/toggle.scss":29,"../../templates/components/toggle.tpl":41}],21:[function(t,e,n){e.exports=".component-button { text-align: center; }\n\n.section .component-button { padding-bottom: 0; }\n\n.component-button .description { padding-left: 0; padding-right: 0; }\n"},{}],22:[function(t,e,n){e.exports=".component-checkbox { display: block; }\n\n.section .component-checkbox { padding-right: 0.375rem; }\n\n.component-checkbox > .label { display: block; padding-bottom: 0.35rem; }\n\n.component-checkbox .checkbox-group { padding-bottom: 0.35rem; }\n\n.component-checkbox .checkbox-group label { padding: 0.35rem 0.375rem; }\n\n.component-checkbox .checkbox-group .label { font-size: 0.9em; }\n\n.component-checkbox .checkbox-group input { opacity: 0; position: absolute; }\n\n.component-checkbox .checkbox-group i { display: block; position: relative; border-radius: 0.25rem; width: 1.4rem; height: 1.4rem; border: 0.11765rem solid #767676; -webkit-flex-shrink: 0; flex-shrink: 0; }\n\n.component-checkbox .checkbox-group input:checked + i { border-color: #ff4700; background: #ff4700; }\n\n.component-checkbox .checkbox-group input:checked + i:after { content: ''; box-sizing: border-box; -webkit-transform: rotate(45deg); transform: rotate(45deg); position: absolute; left: 0.35rem; top: -0.05rem; display: block; width: 0.5rem; height: 1rem; border: 0 solid #ffffff; border-right-width: 0.11765rem; border-bottom-width: 0.11765rem; }\n\n.component-checkbox .description { padding-left: 0; padding-right: 0; }\n"},{}],23:[function(t,e,n){e.exports=".section .component-color { padding: 0; }\n\n.component-color .value { width: 2.2652rem; height: 1.4rem; border-radius: 0.7rem; box-shadow: 0 0.1rem 0.1rem #2f2f2f; display: block; background: #000; }\n\n.component-color .picker-wrap { left: 0; top: 0; right: 0; bottom: 0; position: fixed; padding: 0.7rem 0.375rem; background: rgba(0, 0, 0, 0.65); opacity: 0; -webkit-transition: opacity 100ms ease-in 175ms; transition: opacity 100ms ease-in 175ms; pointer-events: none; z-index: 100; display: -webkit-box; display: -webkit-flex; display: flex; -webkit-box-orient: vertical; -webkit-box-direction: normal; -webkit-flex-direction: column; flex-direction: column; -webkit-box-pack: center; -webkit-justify-content: center; justify-content: center; -webkit-box-align: center; -webkit-align-items: center; align-items: center; }\n\n.component-color .picker-wrap .picker { padding: 0.7rem 0.75rem; background: #484848; box-shadow: 0 0.17647rem 0.88235rem rgba(0, 0, 0, 0.4); border-radius: 0.25rem; width: 100%; max-width: 26rem; overflow: auto; }\n\n.component-color .picker-wrap.show { -webkit-transition-delay: 0ms; transition-delay: 0ms; pointer-events: auto; opacity: 1; }\n\n.component-color .color-box-wrap { box-sizing: border-box; position: relative; height: 0; width: 100%; padding: 0 0 100% 0; }\n\n.component-color .color-box-wrap .color-box-container { position: absolute; height: 99.97%; width: 100%; left: 0; top: 0; }\n\n.component-color .color-box-wrap .color-box-container .color-box { float: left; cursor: pointer; -webkit-tap-highlight-color: transparent; }\n\n.component-color .color-box-wrap .color-box-container .color-box.rounded-tl { border-top-left-radius: 0.25rem; }\n\n.component-color .color-box-wrap .color-box-container .color-box.rounded-tr { border-top-right-radius: 0.25rem; }\n\n.component-color .color-box-wrap .color-box-container .color-box.rounded-bl { border-bottom-left-radius: 0.25rem; }\n\n.component-color .color-box-wrap .color-box-container .color-box.rounded-br { border-bottom-right-radius: 0.25rem; }\n\n.component-color .color-box-wrap .color-box-container .color-box.selected { -webkit-transform: scale(1.1); transform: scale(1.1); border-radius: 0.25rem; box-shadow: #111 0 0 0.24rem; position: relative; z-index: 100; }\n"},{}],24:[function(t,e,n){e.exports=".section .component-input { padding: 0; }\n\n.component-input label { display: block; }\n\n.component-input .label { padding-bottom: 0.7rem; }\n\n.component-input .input { position: relative; min-width: 100%; margin-top: 0.7rem; margin-left: 0; }\n\n.component-input input { display: block; width: 100%; background: #333333; border-radius: 0.25rem; padding: 0.35rem 0.375rem; border: none; vertical-align: baseline; color: #ffffff; font-size: inherit; -webkit-appearance: none; appearance: none; min-height: 2.1rem; }\n\n.component-input input::-webkit-input-placeholder { color: #858585; }\n\n.component-input input::-moz-placeholder { color: #858585; }\n\n.component-input input:-moz-placeholder { color: #858585; }\n\n.component-input input:-ms-input-placeholder { color: #858585; }\n\n.component-input input:focus { border: none; box-shadow: none; }\n\n.component-input input:focus::-webkit-input-placeholder { color: #666666; }\n\n.component-input input:focus::-moz-placeholder { color: #666666; }\n\n.component-input input:focus:-moz-placeholder { color: #666666; }\n\n.component-input input:focus:-ms-input-placeholder { color: #666666; }\n"},{}],25:[function(t,e,n){e.exports=".component-radio { display: block; }\n\n.section .component-radio { padding-right: 0.375rem; }\n\n.component-radio > .label { display: block; padding-bottom: 0.35rem; }\n\n.component-radio .radio-group { padding-bottom: 0.35rem; }\n\n.component-radio .radio-group label { padding: 0.35rem 0.375rem; }\n\n.component-radio .radio-group .label { font-size: 0.9em; }\n\n.component-radio .radio-group input { opacity: 0; position: absolute; }\n\n.component-radio .radio-group i { display: block; position: relative; border-radius: 1.4rem; width: 1.4rem; height: 1.4rem; border: 2px solid #767676; -webkit-flex-shrink: 0; flex-shrink: 0; }\n\n.component-radio .radio-group input:checked + i { border-color: #ff4700; }\n\n.component-radio .radio-group input:checked + i:after { content: ''; display: block; position: absolute; left: 15%; right: 15%; top: 15%; bottom: 15%; border-radius: 1.4rem; background: #ff4700; }\n\n.component-radio .description { padding-left: 0; padding-right: 0; }\n"},{}],26:[function(t,e,n){e.exports='.section .component-select { padding: 0; }\n\n.component-select label { position: relative; }\n\n.component-select .value { position: relative; padding-right: 1.1rem; display: block; }\n\n.component-select .value:after { content: ""; position: absolute; right: 0; top: 50%; margin-top: -0.1rem; height: 0; width: 0; border-left: 0.425rem solid transparent; border-right: 0.425rem solid transparent; border-top: 0.425rem solid #ff4700; }\n\n.component-select select { opacity: 0; position: absolute; display: block; left: 0; right: 0; top: 0; bottom: 0; width: 100%; border: none; margin: 0; padding: 0; }\n'},{}],27:[function(t,e,n){e.exports=".section .component-slider { padding: 0; }\n\n.component-slider label { display: block; }\n\n.component-slider .label-container { display: -webkit-box; display: -webkit-flex; display: flex; -webkit-box-align: center; -webkit-align-items: center; align-items: center; width: 100%; padding-bottom: 0.7rem; }\n\n.component-slider .label { -webkit-box-flex: 1; -webkit-flex: 1; flex: 1; min-width: 1rem; display: block; padding-right: 0.75rem; }\n\n.component-slider .value-wrap { display: block; position: relative; }\n\n.component-slider .value, .component-slider .value-pad { display: block; background: #333333; border-radius: 0.25rem; padding: 0.35rem 0.375rem; border: none; vertical-align: baseline; color: #ffffff; text-align: right; margin: 0; min-width: 1rem; }\n\n.component-slider .value-pad { visibility: hidden; }\n\n.component-slider .value-pad:before { content: ' '; display: inline-block; }\n\n.component-slider .value { max-width: 100%; position: absolute; left: 0; top: 0; }\n\n.component-slider .input-wrap { padding: 0 0.75rem 0.7rem; }\n\n.component-slider .input { display: block; position: relative; min-width: 100%; height: 1.4rem; overflow: hidden; margin-left: 0; }\n\n.component-slider .input:before { content: ''; display: block; position: absolute; height: 0.17647rem; background: #666666; width: 100%; top: 0.61176rem; }\n\n.component-slider .input .slider { display: block; width: 100%; -webkit-appearance: none; appearance: none; position: relative; height: 1.4rem; margin: 0; background-color: transparent; }\n\n.component-slider .input .slider:focus { outline: none; }\n\n.component-slider .input .slider::-webkit-slider-runnable-track { border: none; height: 1.4rem; width: 100%; background-color: transparent; }\n\n.component-slider .input .slider::-webkit-slider-thumb { -webkit-appearance: none; appearance: none; position: relative; height: 1.4rem; width: 1.4rem; background-color: #ff4700; border-radius: 50%; }\n\n.component-slider .input .slider::-webkit-slider-thumb:before { content: \"\"; position: absolute; left: -1000px; top: 0.61176rem; height: 0.17647rem; width: 1001px; background: #ff4700; }\n"},{}],28:[function(t,e,n){e.exports=".component-submit { text-align: center; }\n"},{}],29:[function(t,e,n){e.exports=".section .component-toggle { padding: 0; }\n\n.component-toggle input { display: none; }\n\n.component-toggle .graphic { display: inline-block; position: relative; }\n\n.component-toggle .graphic .slide { display: block; border-radius: 1.05rem; height: 1.05rem; width: 2.2652rem; background: #2f2f2f; -webkit-transition: background-color 150ms linear; transition: background-color 150ms linear; }\n\n.component-toggle .graphic .marker { background: #ececec; width: 1.4rem; height: 1.4rem; border-radius: 1.4rem; position: absolute; left: 0; display: block; top: -0.175rem; -webkit-transition: -webkit-transform 150ms linear; transition: -webkit-transform 150ms linear; transition: transform 150ms linear; transition: transform 150ms linear, -webkit-transform 150ms linear; box-shadow: 0 0.1rem 0.1rem #2f2f2f; }\n\n.component-toggle input:checked + .graphic .slide { background: #993d19; }\n\n.component-toggle input:checked + .graphic .marker { background: #ff4700; -webkit-transform: translateX(0.8652rem); transform: translateX(0.8652rem); }\n"},{}],30:[function(t,e,n){e.exports='
\n \n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],31:[function(t,e,n){e.exports='
\n {{{label}}}\n
\n {{each options}}\n \n {{/each}}\n
\n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],32:[function(t,e,n){e.exports='
\n \n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n
\n
\n
\n
\n
\n
\n
\n'},{}],33:[function(t,e,n){e.exports='
\n'},{}],34:[function(t,e,n){e.exports='
\n \n
\n'},{}],35:[function(t,e,n){e.exports='
\n \n\n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],36:[function(t,e,n){e.exports='
\n {{{label}}}\n
\n {{each options}}\n \n {{/each}}\n
\n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],37:[function(t,e,n){e.exports='
\n \n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],38:[function(t,e,n){e.exports='
\n \n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],39:[function(t,e,n){e.exports='
\n \n
\n'},{}],40:[function(t,e,n){e.exports='
\n

\n
\n'},{}],41:[function(t,e,n){e.exports='
\n \n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],42:[function(t,e,n){e.exports='
'; +},{}],"pebble-clay":[function(t,e,n){"use strict";function r(t,e,n){function r(){i.meta={activeWatchInfo:Pebble.getActiveWatchInfo&&Pebble.getActiveWatchInfo(),accountToken:Pebble.getAccountToken(),watchToken:Pebble.getWatchToken(),userData:s(n.userData||{})}}function o(t,e,n){Array.isArray(t)?t.forEach(function(t){o(t,e,n)}):"section"===t.type?o(t.items,e,n):e(t)&&n(t)}var i=this;if(!Array.isArray(t))throw new Error("config must be an Array");if(e&&"function"!=typeof e)throw new Error('customFn must be a function or "null"');n=n||{},i.config=s(t),i.customFn=e||function(){},i.components={},i.meta={activeWatchInfo:null,accountToken:"",watchToken:"",userData:{}},i.version=c,n.autoHandleEvents!==!1&&"undefined"!=typeof Pebble?(Pebble.addEventListener("showConfiguration",function(){r(),Pebble.openURL(i.generateUrl())}),Pebble.addEventListener("webviewclosed",function(t){t&&t.response&&Pebble.sendAppMessage(i.getSettings(t.response),function(){console.log("Sent config data to Pebble")},function(t){console.log("Failed to send config data!"),console.log(JSON.stringify(t))})})):"undefined"!=typeof Pebble&&Pebble.addEventListener("ready",function(){r()}),o(i.config,function(t){return a[t.type]},function(t){i.registerComponent(a[t.type])}),o(i.config,function(t){return t.appKey},function(){throw new Error("appKeys are no longer supported. Please follow the migration guide to upgrade your project")})}var o=t("./tmp/config-page.html"),i=t("tosource"),a=t("./src/scripts/components"),s=t("deepcopy/build/deepcopy.min"),c=t("./package.json").version,l=t("message_keys");r.prototype.registerComponent=function(t){this.components[t.name]=t},r.prototype.generateUrl=function(){var t={},e=!Pebble||"pypkjs"===Pebble.platform,n=e?"$$$RETURN_TO$$$":"pebblejs://close#";try{t=JSON.parse(localStorage.getItem("clay-settings"))||{}}catch(a){console.error(a.toString())}var s=o.replace("$$RETURN_TO$$",n).replace("$$CUSTOM_FN$$",i(this.customFn)).replace("$$CONFIG$$",i(this.config)).replace("$$SETTINGS$$",i(t)).replace("$$COMPONENTS$$",i(this.components)).replace("$$META$$",i(this.meta));return e?r.encodeDataUri(s,"http://clay.pebble.com.s3-website-us-west-2.amazonaws.com/#"):r.encodeDataUri(s)},r.prototype.getSettings=function(t,e){var n={};t=t.match(/^\{/)?t:decodeURIComponent(t);try{n=JSON.parse(t)}catch(o){throw new Error("The provided response was not valid JSON")}var i={};return Object.keys(n).forEach(function(t){"object"==typeof n[t]&&n[t]?i[t]=n[t].value:i[t]=n[t]}),localStorage.setItem("clay-settings",JSON.stringify(i)),e===!1?n:r.prepareSettingsForAppMessage(n)},r.prototype.setSettings=function(t,e){var n={};try{n=JSON.parse(localStorage.getItem("clay-settings"))||{}}catch(r){console.error(r.toString())}if("object"==typeof t){var o=t;Object.keys(o).forEach(function(t){n[t]=o[t]})}else n[t]=e;localStorage.setItem("clay-settings",JSON.stringify(n))},r.encodeDataUri=function(t,e){return e="undefined"!=typeof e?e:"data:text/html;charset=utf-8,",e+encodeURIComponent(t)},r.prepareForAppMessage=function(t){function e(t,e){return Math.floor(t*Math.pow(10,e||0))}var n;return Array.isArray(t)?(n=[],t.forEach(function(t,e){n[e]=r.prepareForAppMessage(t)})):n="object"==typeof t&&t?"number"==typeof t.value?e(t.value,t.precision):Array.isArray(t.value)?t.value.map(function(n){return"number"==typeof n?e(n,t.precision):n}):r.prepareForAppMessage(t.value):"boolean"==typeof t?t?1:0:t,n},r.prepareSettingsForAppMessage=function(t){var e={};Object.keys(t).forEach(function(n){var r=t[n],o=n.match(/(.+?)(?:\[(\d*)\])?$/);if(!o[2])return void(e[n]=r);var i=parseInt(o[2],10);n=o[1],"undefined"==typeof e[n]&&(e[n]=[]),e[n][i]=r});var n={};return Object.keys(e).forEach(function(t){var o=l[t],i=r.prepareForAppMessage(e[t]);i=Array.isArray(i)?i:[i],i.forEach(function(t,e){n[o+e]=t})}),Object.keys(n).forEach(function(t){if(Array.isArray(n[t]))throw new Error('Clay does not support 2 dimensional arrays for item values. Make sure you are not attempting to use array syntax (eg: "myMessageKey[2]") in the messageKey for components that return an array, such as a checkboxgroup')}),n},e.exports=r},{"./package.json":7,"./src/scripts/components":13,"./tmp/config-page.html":42,"deepcopy/build/deepcopy.min":3,message_keys:void 0,tosource:6}]},{},["pebble-clay"])("pebble-clay")}); \ No newline at end of file diff --git a/node_modules/pebble-clay/index.js b/node_modules/pebble-clay/index.js new file mode 100755 index 0000000..6b19cd0 --- /dev/null +++ b/node_modules/pebble-clay/index.js @@ -0,0 +1,379 @@ +'use strict'; + +var configPageHtml = require('./tmp/config-page.html'); +var toSource = require('tosource'); +var standardComponents = require('./src/scripts/components'); +var deepcopy = require('deepcopy/build/deepcopy.min'); +var version = require('./package.json').version; +var messageKeys = require('message_keys'); + +/** + * @param {Array} config - the Clay config + * @param {function} [customFn] - Custom code to run from the config page. Will run + * with the ClayConfig instance as context + * @param {Object} [options] - Additional options to pass to Clay + * @param {boolean} [options.autoHandleEvents=true] - If false, Clay will not + * automatically handle the 'showConfiguration' and 'webviewclosed' events + * @param {*} [options.userData={}] - Arbitrary data to pass to the config page. Will + * be available as `clayConfig.meta.userData` + * @constructor + */ +function Clay(config, customFn, options) { + var self = this; + + if (!Array.isArray(config)) { + throw new Error('config must be an Array'); + } + + if (customFn && typeof customFn !== 'function') { + throw new Error('customFn must be a function or "null"'); + } + + options = options || {}; + + self.config = deepcopy(config); + self.customFn = customFn || function() {}; + self.components = {}; + self.meta = { + activeWatchInfo: null, + accountToken: '', + watchToken: '', + userData: {} + }; + self.version = version; + + /** + * Populate the meta with data from the Pebble object. Make sure to run this inside + * either the "showConfiguration" or "ready" event handler + * @return {void} + */ + function _populateMeta() { + self.meta = { + activeWatchInfo: Pebble.getActiveWatchInfo && Pebble.getActiveWatchInfo(), + accountToken: Pebble.getAccountToken(), + watchToken: Pebble.getWatchToken(), + userData: deepcopy(options.userData || {}) + }; + } + + // Let Clay handle all the magic + if (options.autoHandleEvents !== false && typeof Pebble !== 'undefined') { + + Pebble.addEventListener('showConfiguration', function() { + _populateMeta(); + Pebble.openURL(self.generateUrl()); + }); + + Pebble.addEventListener('webviewclosed', function(e) { + + if (!e || !e.response) { return; } + + // Send settings to Pebble watchapp + Pebble.sendAppMessage(self.getSettings(e.response), function() { + console.log('Sent config data to Pebble'); + }, function(error) { + console.log('Failed to send config data!'); + console.log(JSON.stringify(error)); + }); + }); + } else if (typeof Pebble !== 'undefined') { + Pebble.addEventListener('ready', function() { + _populateMeta(); + }); + } + + /** + * If this function returns true then the callback will be executed + * @callback _scanConfig_testFn + * @param {Clay~ConfigItem} item + */ + + /** + * @callback _scanConfig_callback + * @param {Clay~ConfigItem} item + */ + + /** + * Scan over the config and run the callback if the testFn resolves to true + * @private + * @param {Clay~ConfigItem|Array} item + * @param {_scanConfig_testFn} testFn + * @param {_scanConfig_callback} callback + * @return {void} + */ + function _scanConfig(item, testFn, callback) { + if (Array.isArray(item)) { + item.forEach(function(item) { + _scanConfig(item, testFn, callback); + }); + } else if (item.type === 'section') { + _scanConfig(item.items, testFn, callback); + } else if (testFn(item)) { + callback(item); + } + } + + // register standard components + _scanConfig(self.config, function(item) { + return standardComponents[item.type]; + }, function(item) { + self.registerComponent(standardComponents[item.type]); + }); + + // validate config against teh use of appKeys + _scanConfig(self.config, function(item) { + return item.appKey; + }, function() { + throw new Error('appKeys are no longer supported. ' + + 'Please follow the migration guide to upgrade your project'); + }); +} + +/** + * Register a component to Clay. + * @param {Object} component - the clay component to register + * @param {string} component.name - the name of the component + * @param {string} component.template - HTML template to use for the component + * @param {string|Object} component.manipulator - methods to attach to the component + * @param {function} component.manipulator.set - set manipulator method + * @param {function} component.manipulator.get - get manipulator method + * @param {Object} [component.defaults] - template defaults + * @param {function} [component.initialize] - method to scaffold the component + * @return {boolean} - Returns true if component was registered correctly + */ +Clay.prototype.registerComponent = function(component) { + this.components[component.name] = component; +}; + +/** + * Generate the Data URI used by the config Page with settings injected + * @return {string} + */ +Clay.prototype.generateUrl = function() { + var settings = {}; + var emulator = !Pebble || Pebble.platform === 'pypkjs'; + var returnTo = emulator ? '$$$RETURN_TO$$$' : 'pebblejs://close#'; + + try { + settings = JSON.parse(localStorage.getItem('clay-settings')) || {}; + } catch (e) { + console.error(e.toString()); + } + + var compiledHtml = configPageHtml + .replace('$$RETURN_TO$$', returnTo) + .replace('$$CUSTOM_FN$$', toSource(this.customFn)) + .replace('$$CONFIG$$', toSource(this.config)) + .replace('$$SETTINGS$$', toSource(settings)) + .replace('$$COMPONENTS$$', toSource(this.components)) + .replace('$$META$$', toSource(this.meta)); + + // if we are in the emulator then we need to proxy the data via a webpage to + // obtain the return_to. + // @todo calculate this from the Pebble object or something + if (emulator) { + return Clay.encodeDataUri( + compiledHtml, + 'http://clay.pebble.com.s3-website-us-west-2.amazonaws.com/#' + ); + } + + return Clay.encodeDataUri(compiledHtml); +}; + +/** + * Parse the response from the webviewclosed event data + * @param {string} response + * @param {boolean} [convert=true] + * @returns {Object} + */ +Clay.prototype.getSettings = function(response, convert) { + // Decode and parse config data as JSON + var settings = {}; + response = response.match(/^\{/) ? response : decodeURIComponent(response); + + try { + settings = JSON.parse(response); + } catch (e) { + throw new Error('The provided response was not valid JSON'); + } + + // flatten the settings for localStorage + var settingsStorage = {}; + Object.keys(settings).forEach(function(key) { + if (typeof settings[key] === 'object' && settings[key]) { + settingsStorage[key] = settings[key].value; + } else { + settingsStorage[key] = settings[key]; + } + }); + + localStorage.setItem('clay-settings', JSON.stringify(settingsStorage)); + + return convert === false ? settings : Clay.prepareSettingsForAppMessage(settings); +}; + +/** + * Updates the settings with the given value(s). + * + * @signature `clay.setSettings(key, value)` + * @param {String} key - The property to set. + * @param {*} value - the value assigned to _key_. + * @return {undefined} + * + * @signature `clay.setSettings(settings)` + * @param {Object} settings - an object containing the key/value pairs to be set. + * @return {undefined} + */ +Clay.prototype.setSettings = function(key, value) { + var settingsStorage = {}; + + try { + settingsStorage = JSON.parse(localStorage.getItem('clay-settings')) || {}; + } catch (e) { + console.error(e.toString()); + } + + if (typeof key === 'object') { + var settings = key; + Object.keys(settings).forEach(function(key) { + settingsStorage[key] = settings[key]; + }); + } else { + settingsStorage[key] = value; + } + + localStorage.setItem('clay-settings', JSON.stringify(settingsStorage)); +}; + +/** + * @param {string} input + * @param {string} [prefix='data:text/html;charset=utf-8,'] + * @returns {string} + */ +Clay.encodeDataUri = function(input, prefix) { + prefix = typeof prefix !== 'undefined' ? prefix : 'data:text/html;charset=utf-8,'; + return prefix + encodeURIComponent(input); +}; + +/** + * Converts the val into a type compatible with Pebble.sendAppMessage(). + * - Strings will be returned without modification + * - Numbers will be returned without modification + * - Booleans will be converted to a 0 or 1 + * - Arrays that contain strings will be returned without modification + * eg: ['one', 'two'] becomes ['one', 'two'] + * - Arrays that contain numbers will be returned without modification + * eg: [1, 2] becomes [1, 2] + * - Arrays that contain booleans will be converted to a 0 or 1 + * eg: [true, false] becomes [1, 0] + * - Arrays must be single dimensional + * - Objects that have a "value" property will apply the above rules to the type of + * value. If the value is a number or an array of numbers and the optional + * property: "precision" is provided, then the number will be multiplied by 10 to + * the power of precision (value * 10 ^ precision) and then floored. + * Eg: 1.4567 with a precision set to 3 will become 1456 + * @param {number|string|boolean|Array|Object} val + * @param {number|string|boolean|Array} val.value + * @param {number|undefined} [val.precision=0] + * @returns {number|string|Array} + */ +Clay.prepareForAppMessage = function(val) { + + /** + * moves the decimal place of a number by precision then drop any remaining decimal + * places. + * @param {number} number + * @param {number} precision - number of decimal places to move + * @returns {number} + * @private + */ + function _normalizeToPrecision(number, precision) { + return Math.floor(number * Math.pow(10, precision || 0)); + } + + var result; + + if (Array.isArray(val)) { + result = []; + val.forEach(function(item, index) { + result[index] = Clay.prepareForAppMessage(item); + }); + } else if (typeof val === 'object' && val) { + if (typeof val.value === 'number') { + result = _normalizeToPrecision(val.value, val.precision); + } else if (Array.isArray(val.value)) { + result = val.value.map(function(item) { + if (typeof item === 'number') { + return _normalizeToPrecision(item, val.precision); + } + return item; + }); + } else { + result = Clay.prepareForAppMessage(val.value); + } + } else if (typeof val === 'boolean') { + result = val ? 1 : 0; + } else { + result = val; + } + + return result; +}; + +/** + * Converts a Clay settings dict into one that is compatible with + * Pebble.sendAppMessage(); It also uses the provided messageKeys to correctly + * assign arrays into individual keys + * @see {prepareForAppMessage} + * @param {Object} settings + * @returns {{}} + */ +Clay.prepareSettingsForAppMessage = function(settings) { + + // flatten settings + var flatSettings = {}; + Object.keys(settings).forEach(function(key) { + var val = settings[key]; + var matches = key.match(/(.+?)(?:\[(\d*)\])?$/); + + if (!matches[2]) { + flatSettings[key] = val; + return; + } + + var position = parseInt(matches[2], 10); + key = matches[1]; + + if (typeof flatSettings[key] === 'undefined') { + flatSettings[key] = []; + } + + flatSettings[key][position] = val; + }); + + var result = {}; + Object.keys(flatSettings).forEach(function(key) { + var messageKey = messageKeys[key]; + var settingArr = Clay.prepareForAppMessage(flatSettings[key]); + settingArr = Array.isArray(settingArr) ? settingArr : [settingArr]; + + settingArr.forEach(function(setting, index) { + result[messageKey + index] = setting; + }); + }); + + // validate the settings + Object.keys(result).forEach(function(key) { + if (Array.isArray(result[key])) { + throw new Error('Clay does not support 2 dimensional arrays for item ' + + 'values. Make sure you are not attempting to use array ' + + 'syntax (eg: "myMessageKey[2]") in the messageKey for ' + + 'components that return an array, such as a checkboxgroup'); + } + }); + + return result; +}; + +module.exports = Clay; diff --git a/node_modules/pebble-clay/package.json b/node_modules/pebble-clay/package.json new file mode 100644 index 0000000..e3230f0 --- /dev/null +++ b/node_modules/pebble-clay/package.json @@ -0,0 +1,119 @@ +{ + "_from": "pebble-clay", + "_id": "pebble-clay@1.0.4", + "_inBundle": false, + "_integrity": "sha512-/rXxmltdW8JyohDzXINdea+d2wnFJVNFiTXfuZsKpySURZSCFMMucX9sZPZvbHnEA4xFINM4iicyhBbvY4ALfw==", + "_location": "/pebble-clay", + "_phantomChildren": {}, + "_requested": { + "type": "tag", + "registry": true, + "raw": "pebble-clay", + "name": "pebble-clay", + "escapedName": "pebble-clay", + "rawSpec": "", + "saveSpec": null, + "fetchSpec": "latest" + }, + "_requiredBy": [ + "#USER", + "/" + ], + "_resolved": "https://registry.npmjs.org/pebble-clay/-/pebble-clay-1.0.4.tgz", + "_shasum": "fdf92f0fdc770a979c06874eaa2457cc2e762344", + "_spec": "pebble-clay", + "_where": "/pebble", + "author": { + "name": "Pebble Technology" + }, + "bugs": { + "url": "https://github.com/pebble/clay/issues" + }, + "bundleDependencies": false, + "dependencies": {}, + "deprecated": false, + "description": "Pebble Config Framework", + "devDependencies": { + "autoprefixer": "^6.3.1", + "bourbon": "^4.2.6", + "browserify": "^13.0.0", + "browserify-istanbul": "^0.2.1", + "chai": "^3.4.1", + "deamdify": "^0.2.0", + "deepcopy": "^0.6.1", + "del": "^2.0.2", + "eslint": "^1.5.1", + "eslint-config-pebble": "^1.2.0", + "eslint-plugin-standard": "^1.3.1", + "gulp": "^3.9.0", + "gulp-autoprefixer": "^3.1.0", + "gulp-htmlmin": "^1.3.0", + "gulp-inline": "0.0.15", + "gulp-insert": "^0.5.0", + "gulp-sass": "^2.1.1", + "gulp-sourcemaps": "^1.6.0", + "gulp-uglify": "^1.5.2", + "joi": "^6.10.1", + "karma": "^0.13.19", + "karma-browserify": "^5.0.1", + "karma-chrome-launcher": "^0.2.2", + "karma-coverage": "^0.5.3", + "karma-mocha": "^0.2.1", + "karma-mocha-reporter": "^1.1.5", + "karma-source-map-support": "^1.1.0", + "karma-threshold-reporter": "^0.1.15", + "mocha": "^2.3.4", + "postcss": "^5.0.14", + "require-from-string": "^1.1.0", + "sassify": "^0.9.1", + "sinon": "^1.17.3", + "stringify": "^3.2.0", + "through": "^2.3.8", + "tosource": "^1.0.0", + "vinyl-buffer": "^1.0.0", + "vinyl-source-stream": "^1.1.0", + "watchify": "^3.7.0" + }, + "homepage": "https://github.com/pebble/clay#readme", + "keywords": [ + "pebble", + "config", + "configuration", + "pebble-package" + ], + "license": "MIT", + "name": "pebble-clay", + "pebble": { + "projectType": "package", + "sdkVersion": "3", + "targetPlatforms": [ + "aplite", + "basalt", + "chalk", + "diorite", + "emery" + ], + "resources": { + "media": [] + }, + "capabilities": [ + "configurable" + ] + }, + "repository": { + "type": "git", + "url": "git+https://github.com/pebble/clay.git" + }, + "scripts": { + "build": "gulp", + "dev": "gulp dev", + "lint": "eslint ./", + "pebble-build": "npm run build && pebble build", + "pebble-clean": "rm -rf tmp src/js/index.js && pebble clean", + "pebble-publish": "npm run pebble-clean && npm run build && pebble build && pebble package publish && npm run pebble-clean", + "test": "gulp && ./node_modules/.bin/karma start ./test/karma.conf.js --single-run", + "test-debug": "(export DEBUG=true && ./node_modules/.bin/gulp && ./node_modules/.bin/karma start ./test/karma.conf.js --no-single-run)", + "test-travis": "gulp && ./node_modules/.bin/karma start ./test/karma.conf.js --single-run --browsers chromeTravisCI && ./node_modules/.bin/eslint ./" + }, + "version": "1.0.4" +} diff --git a/node_modules/pebble-clay/src/config-page.html b/node_modules/pebble-clay/src/config-page.html new file mode 100755 index 0000000..627d44e --- /dev/null +++ b/node_modules/pebble-clay/src/config-page.html @@ -0,0 +1,20 @@ + + + + + + + + + +
+ + + diff --git a/node_modules/pebble-clay/src/images/example.png b/node_modules/pebble-clay/src/images/example.png new file mode 100644 index 0000000000000000000000000000000000000000..f7246b969a29b05bac75963b8b1a2ae5dfc5a5b4 GIT binary patch literal 138670 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%0DS$OE>2%bd4jzbY`E?5f+(5{ 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

\n'},{}],31:[function(t,e,n){e.exports='
\n {{{label}}}\n
\n {{each options}}\n \n {{/each}}\n
\n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],32:[function(t,e,n){e.exports='
\n \n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n
\n
\n
\n
\n
\n
\n
\n'},{}],33:[function(t,e,n){e.exports='
\n'},{}],34:[function(t,e,n){e.exports='
\n \n
\n'},{}],35:[function(t,e,n){e.exports='
\n \n\n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],36:[function(t,e,n){e.exports='
\n {{{label}}}\n
\n {{each options}}\n \n {{/each}}\n
\n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],37:[function(t,e,n){e.exports='
\n \n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],38:[function(t,e,n){e.exports='
\n \n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],39:[function(t,e,n){e.exports='
\n \n
\n'},{}],40:[function(t,e,n){e.exports='
\n

\n
\n'},{}],41:[function(t,e,n){e.exports='
\n \n {{if description}}\n
{{{description}}}
\n {{/if}}\n
\n'},{}],42:[function(t,e,n){e.exports='
'; +},{}],"pebble-clay":[function(t,e,n){"use strict";function r(t,e,n){function r(){i.meta={activeWatchInfo:Pebble.getActiveWatchInfo&&Pebble.getActiveWatchInfo(),accountToken:Pebble.getAccountToken(),watchToken:Pebble.getWatchToken(),userData:s(n.userData||{})}}function o(t,e,n){Array.isArray(t)?t.forEach(function(t){o(t,e,n)}):"section"===t.type?o(t.items,e,n):e(t)&&n(t)}var i=this;if(!Array.isArray(t))throw new Error("config must be an Array");if(e&&"function"!=typeof e)throw new Error('customFn must be a function or "null"');n=n||{},i.config=s(t),i.customFn=e||function(){},i.components={},i.meta={activeWatchInfo:null,accountToken:"",watchToken:"",userData:{}},i.version=c,n.autoHandleEvents!==!1&&"undefined"!=typeof Pebble?(Pebble.addEventListener("showConfiguration",function(){r(),Pebble.openURL(i.generateUrl())}),Pebble.addEventListener("webviewclosed",function(t){t&&t.response&&Pebble.sendAppMessage(i.getSettings(t.response),function(){console.log("Sent config data to Pebble")},function(t){console.log("Failed to send config data!"),console.log(JSON.stringify(t))})})):"undefined"!=typeof Pebble&&Pebble.addEventListener("ready",function(){r()}),o(i.config,function(t){return a[t.type]},function(t){i.registerComponent(a[t.type])}),o(i.config,function(t){return t.appKey},function(){throw new Error("appKeys are no longer supported. Please follow the migration guide to upgrade your project")})}var o=t("./tmp/config-page.html"),i=t("tosource"),a=t("./src/scripts/components"),s=t("deepcopy/build/deepcopy.min"),c=t("./package.json").version,l=t("message_keys");r.prototype.registerComponent=function(t){this.components[t.name]=t},r.prototype.generateUrl=function(){var t={},e=!Pebble||"pypkjs"===Pebble.platform,n=e?"$$$RETURN_TO$$$":"pebblejs://close#";try{t=JSON.parse(localStorage.getItem("clay-settings"))||{}}catch(a){console.error(a.toString())}var s=o.replace("$$RETURN_TO$$",n).replace("$$CUSTOM_FN$$",i(this.customFn)).replace("$$CONFIG$$",i(this.config)).replace("$$SETTINGS$$",i(t)).replace("$$COMPONENTS$$",i(this.components)).replace("$$META$$",i(this.meta));return e?r.encodeDataUri(s,"http://clay.pebble.com.s3-website-us-west-2.amazonaws.com/#"):r.encodeDataUri(s)},r.prototype.getSettings=function(t,e){var n={};t=t.match(/^\{/)?t:decodeURIComponent(t);try{n=JSON.parse(t)}catch(o){throw new Error("The provided response was not valid JSON")}var i={};return Object.keys(n).forEach(function(t){"object"==typeof n[t]&&n[t]?i[t]=n[t].value:i[t]=n[t]}),localStorage.setItem("clay-settings",JSON.stringify(i)),e===!1?n:r.prepareSettingsForAppMessage(n)},r.prototype.setSettings=function(t,e){var n={};try{n=JSON.parse(localStorage.getItem("clay-settings"))||{}}catch(r){console.error(r.toString())}if("object"==typeof t){var o=t;Object.keys(o).forEach(function(t){n[t]=o[t]})}else n[t]=e;localStorage.setItem("clay-settings",JSON.stringify(n))},r.encodeDataUri=function(t,e){return e="undefined"!=typeof e?e:"data:text/html;charset=utf-8,",e+encodeURIComponent(t)},r.prepareForAppMessage=function(t){function e(t,e){return Math.floor(t*Math.pow(10,e||0))}var n;return Array.isArray(t)?(n=[],t.forEach(function(t,e){n[e]=r.prepareForAppMessage(t)})):n="object"==typeof t&&t?"number"==typeof t.value?e(t.value,t.precision):Array.isArray(t.value)?t.value.map(function(n){return"number"==typeof n?e(n,t.precision):n}):r.prepareForAppMessage(t.value):"boolean"==typeof t?t?1:0:t,n},r.prepareSettingsForAppMessage=function(t){var e={};Object.keys(t).forEach(function(n){var r=t[n],o=n.match(/(.+?)(?:\[(\d*)\])?$/);if(!o[2])return void(e[n]=r);var i=parseInt(o[2],10);n=o[1],"undefined"==typeof e[n]&&(e[n]=[]),e[n][i]=r});var n={};return Object.keys(e).forEach(function(t){var o=l[t],i=r.prepareForAppMessage(e[t]);i=Array.isArray(i)?i:[i],i.forEach(function(t,e){n[o+e]=t})}),Object.keys(n).forEach(function(t){if(Array.isArray(n[t]))throw new Error('Clay does not support 2 dimensional arrays for item values. Make sure you are not attempting to use array syntax (eg: "myMessageKey[2]") in the messageKey for components that return an array, such as a checkboxgroup')}),n},e.exports=r},{"./package.json":7,"./src/scripts/components":13,"./tmp/config-page.html":42,"deepcopy/build/deepcopy.min":3,message_keys:void 0,tosource:6}]},{},["pebble-clay"])("pebble-clay")}); \ No newline at end of file diff --git a/node_modules/pebble-clay/src/scripts/components/button.js b/node_modules/pebble-clay/src/scripts/components/button.js new file mode 100644 index 0000000..64fc65c --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/button.js @@ -0,0 +1,13 @@ +'use strict'; + +module.exports = { + name: 'button', + template: require('../../templates/components/button.tpl'), + style: require('../../styles/clay/components/button.scss'), + manipulator: 'button', + defaults: { + primary: false, + attributes: {}, + description: '' + } +}; diff --git a/node_modules/pebble-clay/src/scripts/components/checkboxgroup.js b/node_modules/pebble-clay/src/scripts/components/checkboxgroup.js new file mode 100644 index 0000000..8c999c4 --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/checkboxgroup.js @@ -0,0 +1,13 @@ +'use strict'; + +module.exports = { + name: 'checkboxgroup', + template: require('../../templates/components/checkboxgroup.tpl'), + style: require('../../styles/clay/components/checkboxgroup.scss'), + manipulator: 'checkboxgroup', + defaults: { + label: '', + options: [], + description: '' + } +}; diff --git a/node_modules/pebble-clay/src/scripts/components/color.js b/node_modules/pebble-clay/src/scripts/components/color.js new file mode 100644 index 0000000..fa7e20e --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/color.js @@ -0,0 +1,327 @@ +'use strict'; + +module.exports = { + name: 'color', + template: require('../../templates/components/color.tpl'), + style: require('../../styles/clay/components/color.scss'), + manipulator: 'color', + defaults: { + label: '', + description: '' + }, + initialize: function(minified, clay) { + var HTML = minified.HTML; + var self = this; + + /** + * @param {string|boolean|number} color + * @returns {string} + */ + function cssColor(color) { + if (typeof color === 'number') { + color = color.toString(16); + } else if (!color) { + return 'transparent'; + } + + color = padColorString(color); + + return '#' + (useSunlight ? sunlightColorMap[color] : color); + } + + /** + * @param {string} color + * @return {string} + */ + function padColorString(color) { + color = color.toLowerCase(); + + while (color.length < 6) { + color = '0' + color; + } + + return color; + } + + /** + * @param {number|string} value + * @returns {string|*} + */ + function normalizeColor(value) { + switch (typeof value) { + case 'number': return padColorString(value.toString(16)); + case 'string': return value.replace(/^#|^0x/, ''); + default: return value; + } + } + + /** + * @param {Array.} layout + * @returns {Array} + */ + function flattenLayout(layout) { + return layout.reduce(function(a, b) { + return a.concat(b); + }, []); + } + + /** + * Convert HEX color to LAB. + * Adapted from: https://github.com/antimatter15/rgb-lab + * @param {string} hex + * @returns {Array} - [l, a, b] + */ + function hex2lab(hex) { + hex = hex.replace(/^#|^0x/, ''); + + var r = parseInt(hex.slice(0, 2), 16) / 255; + var g = parseInt(hex.slice(2, 4), 16) / 255; + var b = parseInt(hex.slice(4), 16) / 255; + + r = (r > 0.04045) ? Math.pow((r + 0.055) / 1.055, 2.4) : r / 12.92; + g = (g > 0.04045) ? Math.pow((g + 0.055) / 1.055, 2.4) : g / 12.92; + b = (b > 0.04045) ? Math.pow((b + 0.055) / 1.055, 2.4) : b / 12.92; + + var x = (r * 0.4124 + g * 0.3576 + b * 0.1805) / 0.95047; + var y = (r * 0.2126 + g * 0.7152 + b * 0.0722) / 1.00000; + var z = (r * 0.0193 + g * 0.1192 + b * 0.9505) / 1.08883; + + x = (x > 0.008856) ? Math.pow(x, 1 / 3) : (7.787 * x) + 16 / 116; + y = (y > 0.008856) ? Math.pow(y, 1 / 3) : (7.787 * y) + 16 / 116; + z = (z > 0.008856) ? Math.pow(z, 1 / 3) : (7.787 * z) + 16 / 116; + + return [(116 * y) - 16, 500 * (x - y), 200 * (y - z)]; + } + + /** + * Find the perceptual color distance between two LAB colors + * @param {Array} labA + * @param {Array} labB + * @returns {number} + */ + function deltaE(labA, labB) { + var deltaL = labA[0] - labB[0]; + var deltaA = labA[1] - labB[1]; + var deltaB = labA[2] - labB[2]; + + return Math.sqrt(Math.pow(deltaL, 2) + + Math.pow(deltaA, 2) + + Math.pow(deltaB, 2)); + } + + /** + * Returns the layout based on the connected watch + * @returns {Array} + */ + function autoLayout() { + if (!clay.meta.activeWatchInfo || + clay.meta.activeWatchInfo.firmware.major === 2 || + ['aplite', 'diorite'].indexOf(clay.meta.activeWatchInfo.platform) > -1 && + !self.config.allowGray) { + return standardLayouts.BLACK_WHITE; + } + + if (['aplite', 'diorite'].indexOf(clay.meta.activeWatchInfo.platform) > -1 && + self.config.allowGray) { + return standardLayouts.GRAY; + } + + return standardLayouts.COLOR; + } + + /** + * @param {number|string} color + * @return {number} + */ + self.roundColorToLayout = function(color) { + var itemValue = normalizeColor(color); + + // if the color is not in the layout we will need find the closest match + if (colorList.indexOf(itemValue) === -1) { + var itemValueLAB = hex2lab(itemValue); + var differenceArr = colorList.map(function(color) { + var colorLAB = hex2lab(normalizeColor(color)); + return deltaE(itemValueLAB, colorLAB); + }); + + // Get the lowest number from the differenceArray + var lowest = Math.min.apply(Math, differenceArr); + + // Get the index for that lowest number + var index = differenceArr.indexOf(lowest); + itemValue = colorList[index]; + } + + return parseInt(itemValue, 16); + }; + + var useSunlight = self.config.sunlight !== false; + var sunlightColorMap = { + '000000': '000000', '000055': '001e41', '0000aa': '004387', '0000ff': '0068ca', + '005500': '2b4a2c', '005555': '27514f', '0055aa': '16638d', '0055ff': '007dce', + '00aa00': '5e9860', '00aa55': '5c9b72', '00aaaa': '57a5a2', '00aaff': '4cb4db', + '00ff00': '8ee391', '00ff55': '8ee69e', '00ffaa': '8aebc0', '00ffff': '84f5f1', + '550000': '4a161b', '550055': '482748', '5500aa': '40488a', '5500ff': '2f6bcc', + '555500': '564e36', '555555': '545454', '5555aa': '4f6790', '5555ff': '4180d0', + '55aa00': '759a64', '55aa55': '759d76', '55aaaa': '71a6a4', '55aaff': '69b5dd', + '55ff00': '9ee594', '55ff55': '9de7a0', '55ffaa': '9becc2', '55ffff': '95f6f2', + 'aa0000': '99353f', 'aa0055': '983e5a', 'aa00aa': '955694', 'aa00ff': '8f74d2', + 'aa5500': '9d5b4d', 'aa5555': '9d6064', 'aa55aa': '9a7099', 'aa55ff': '9587d5', + 'aaaa00': 'afa072', 'aaaa55': 'aea382', 'aaaaaa': 'ababab', 'ffffff': 'ffffff', + 'aaaaff': 'a7bae2', 'aaff00': 'c9e89d', 'aaff55': 'c9eaa7', 'aaffaa': 'c7f0c8', + 'aaffff': 'c3f9f7', 'ff0000': 'e35462', 'ff0055': 'e25874', 'ff00aa': 'e16aa3', + 'ff00ff': 'de83dc', 'ff5500': 'e66e6b', 'ff5555': 'e6727c', 'ff55aa': 'e37fa7', + 'ff55ff': 'e194df', 'ffaa00': 'f1aa86', 'ffaa55': 'f1ad93', 'ffaaaa': 'efb5b8', + 'ffaaff': 'ecc3eb', 'ffff00': 'ffeeab', 'ffff55': 'fff1b5', 'ffffaa': 'fff6d3' + }; + + /* eslint-disable comma-spacing, no-multi-spaces, max-len, + standard/array-bracket-even-spacing */ + var standardLayouts = { + COLOR: [ + [false , false , '55ff00', 'aaff55', false , 'ffff55', 'ffffaa', false , false ], + [false , 'aaffaa', '55ff55', '00ff00', 'aaff00', 'ffff00', 'ffaa55', 'ffaaaa', false ], + ['55ffaa', '00ff55', '00aa00', '55aa00', 'aaaa55', 'aaaa00', 'ffaa00', 'ff5500', 'ff5555'], + ['aaffff', '00ffaa', '00aa55', '55aa55', '005500', '555500', 'aa5500', 'ff0000', 'ff0055'], + [false , '55aaaa', '00aaaa', '005555', 'ffffff', '000000', 'aa5555', 'aa0000', false ], + ['55ffff', '00ffff', '00aaff', '0055aa', 'aaaaaa', '555555', '550000', 'aa0055', 'ff55aa'], + ['55aaff', '0055ff', '0000ff', '0000aa', '000055', '550055', 'aa00aa', 'ff00aa', 'ffaaff'], + [false , '5555aa', '5555ff', '5500ff', '5500aa', 'aa00ff', 'ff00ff', 'ff55ff', false ], + [false , false , false , 'aaaaff', 'aa55ff', 'aa55aa', false , false , false ] + ], + GRAY: [ + ['000000', 'aaaaaa', 'ffffff'] + ], + BLACK_WHITE: [ + ['000000', 'ffffff'] + ] + }; + /* eslint-enable */ + + var layout = self.config.layout || autoLayout(); + + if (typeof layout === 'string') { + layout = standardLayouts[layout]; + } + + // make sure layout is a 2D array + if (!Array.isArray(layout[0])) { + layout = [layout]; + } + + var colorList = flattenLayout(layout).map(function(item) { + return normalizeColor(item); + }).filter(function(item) { + return item; + }); + + var grid = ''; + var rows = layout.length; + var cols = 0; + layout.forEach(function(row) { + cols = row.length > cols ? row.length : cols; + }); + var itemWidth = 100 / cols; + var itemHeight = 100 / rows; + var $elem = self.$element; + + for (var i = 0; i < rows; i++) { + for (var j = 0; j < cols; j++) { + + var color = normalizeColor(layout[i][j]); + var selectable = (color ? ' selectable' : ''); + + var roundedTL = (i === 0 && j === 0) || i === 0 && !layout[i][j - 1] || + !layout[i][j - 1] && !layout[i - 1][j] ? + ' rounded-tl' : + ''; + + var roundedTR = i === 0 && !layout[i][j + 1] || + !layout[i][j + 1] && !layout[i - 1][j] ? + ' rounded-tr ' : + ''; + + var roundedBL = (i === layout.length - 1 && j === 0) || + i === layout.length - 1 && !layout[i][j - 1] || + !layout[i][j - 1] && !layout[i + 1][j] ? + ' rounded-bl' : + ''; + + var roundedBR = i === layout.length - 1 && !layout[i][j + 1] || + !layout[i][j + 1] && !layout[i + 1][j] ? + ' rounded-br' : + ''; + + grid += '' + + ''; + } + } + + // on very small layouts the boxes end up huge. The following adds extra padding + // to prevent them from being so big. + var extraPadding = 0; + if (cols === 3) { + extraPadding = 5; + } + if (cols === 2) { + extraPadding = 8; + } + var vPadding = (extraPadding * itemWidth / itemHeight) + '%'; + var hPadding = extraPadding + '%'; + $elem.select('.color-box-container') + .add(HTML(grid)) + .set('$paddingTop', vPadding) + .set('$paddingRight', hPadding) + .set('$paddingBottom', vPadding) + .set('$paddingLeft', hPadding); + + $elem.select('.color-box-wrap').set( + '$paddingBottom', + (itemWidth / itemHeight * 100) + '%' + ); + + var $valueDisplay = $elem.select('.value'); + var $picker = $elem.select('.picker-wrap'); + var disabled = self.$manipulatorTarget.get('disabled'); + + $elem.select('label').on('click', function() { + if (!disabled) { + $picker.set('show'); // toggle visibility + } + }); + + self.on('change', function() { + var value = self.get(); + $valueDisplay.set('$background-color', cssColor(value)); + $elem.select('.color-box').set('-selected'); + $elem.select('.color-box[data-value="' + value + '"]').set('+selected'); + }); + + $elem.select('.color-box.selectable').on('click', function(ev) { + self.set(parseInt(ev.target.dataset.value, 10)); + $picker.set('-show'); + }); + + $picker.on('click', function() { + $picker.set('-show'); + }); + + self.on('disabled', function() { + disabled = true; + }); + + self.on('enabled', function() { + disabled = false; + }); + + // expose layout for testing + self._layout = layout; + } +}; diff --git a/node_modules/pebble-clay/src/scripts/components/footer.js b/node_modules/pebble-clay/src/scripts/components/footer.js new file mode 100644 index 0000000..b57d9e7 --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/footer.js @@ -0,0 +1,7 @@ +'use strict'; + +module.exports = { + name: 'footer', + template: require('../../templates/components/footer.tpl'), + manipulator: 'html' +}; diff --git a/node_modules/pebble-clay/src/scripts/components/heading.js b/node_modules/pebble-clay/src/scripts/components/heading.js new file mode 100644 index 0000000..a5ee8e0 --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/heading.js @@ -0,0 +1,10 @@ +'use strict'; + +module.exports = { + name: 'heading', + template: require('../../templates/components/heading.tpl'), + manipulator: 'html', + defaults: { + size: 4 + } +}; diff --git a/node_modules/pebble-clay/src/scripts/components/index.js b/node_modules/pebble-clay/src/scripts/components/index.js new file mode 100644 index 0000000..18f8d00 --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/index.js @@ -0,0 +1,16 @@ +'use strict'; + +module.exports = { + color: require('./color'), + footer: require('./footer'), + heading: require('./heading'), + input: require('./input'), + select: require('./select'), + submit: require('./submit'), + text: require('./text'), + toggle: require('./toggle'), + radiogroup: require('./radiogroup'), + checkboxgroup: require('./checkboxgroup'), + button: require('./button'), + slider: require('./slider') +}; diff --git a/node_modules/pebble-clay/src/scripts/components/input.js b/node_modules/pebble-clay/src/scripts/components/input.js new file mode 100644 index 0000000..45fce5f --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/input.js @@ -0,0 +1,13 @@ +'use strict'; + +module.exports = { + name: 'input', + template: require('../../templates/components/input.tpl'), + style: require('../../styles/clay/components/input.scss'), + manipulator: 'val', + defaults: { + label: '', + description: '', + attributes: {} + } +}; diff --git a/node_modules/pebble-clay/src/scripts/components/radiogroup.js b/node_modules/pebble-clay/src/scripts/components/radiogroup.js new file mode 100644 index 0000000..5cd436b --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/radiogroup.js @@ -0,0 +1,14 @@ +'use strict'; + +module.exports = { + name: 'radiogroup', + template: require('../../templates/components/radiogroup.tpl'), + style: require('../../styles/clay/components/radiogroup.scss'), + manipulator: 'radiogroup', + defaults: { + label: '', + options: [], + description: '', + attributes: {} + } +}; diff --git a/node_modules/pebble-clay/src/scripts/components/select.js b/node_modules/pebble-clay/src/scripts/components/select.js new file mode 100644 index 0000000..09be191 --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/select.js @@ -0,0 +1,33 @@ +'use strict'; + +module.exports = { + name: 'select', + template: require('../../templates/components/select.tpl'), + style: require('../../styles/clay/components/select.scss'), + manipulator: 'val', + defaults: { + label: '', + options: [], + description: '', + attributes: {} + }, + initialize: function() { + var self = this; + + var $value = self.$element.select('.value'); + + /** + * Updates the HTML value of the component to match the slected option's label + * @return {void} + */ + function setValueDisplay() { + var selectedIndex = self.$manipulatorTarget.get('selectedIndex'); + var $options = self.$manipulatorTarget.select('option'); + var value = $options[selectedIndex] && $options[selectedIndex].innerHTML; + $value.set('innerHTML', value); + } + + setValueDisplay(); + self.on('change', setValueDisplay); + } +}; diff --git a/node_modules/pebble-clay/src/scripts/components/slider.js b/node_modules/pebble-clay/src/scripts/components/slider.js new file mode 100644 index 0000000..d15ad6b --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/slider.js @@ -0,0 +1,50 @@ +'use strict'; + +module.exports = { + name: 'slider', + template: require('../../templates/components/slider.tpl'), + style: require('../../styles/clay/components/slider.scss'), + manipulator: 'slider', + defaults: { + label: '', + description: '', + min: 0, + max: 100, + step: 1, + attributes: {} + }, + initialize: function() { + var self = this; + + var $value = self.$element.select('.value'); + var $valuePad = self.$element.select('.value-pad'); + var $slider = self.$manipulatorTarget; + + /** + * Sets the value display + * @return {void} + */ + function setValueDisplay() { + var value = self.get().toFixed(self.precision); + $value.set('value', value); + $valuePad.set('innerHTML', value); + } + + var step = $slider.get('step'); + step = step.toString(10).split('.')[1]; + self.precision = step ? step.length : 0; + + self.on('change', setValueDisplay); + $slider.on('|input', setValueDisplay); + setValueDisplay(); + + $value.on('|input', function() { + $valuePad.set('innerHTML', this.get('value')); + }); + + $value.on('|change', function() { + self.set(this.get('value')); + setValueDisplay(); + }); + } +}; diff --git a/node_modules/pebble-clay/src/scripts/components/submit.js b/node_modules/pebble-clay/src/scripts/components/submit.js new file mode 100644 index 0000000..85d624e --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/submit.js @@ -0,0 +1,11 @@ +'use strict'; + +module.exports = { + name: 'submit', + template: require('../../templates/components/submit.tpl'), + style: require('../../styles/clay/components/submit.scss'), + manipulator: 'button', + defaults: { + attributes: {} + } +}; diff --git a/node_modules/pebble-clay/src/scripts/components/text.js b/node_modules/pebble-clay/src/scripts/components/text.js new file mode 100644 index 0000000..ba60dba --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/text.js @@ -0,0 +1,7 @@ +'use strict'; + +module.exports = { + name: 'text', + template: require('../../templates/components/text.tpl'), + manipulator: 'html' +}; diff --git a/node_modules/pebble-clay/src/scripts/components/toggle.js b/node_modules/pebble-clay/src/scripts/components/toggle.js new file mode 100644 index 0000000..ca4eace --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/components/toggle.js @@ -0,0 +1,13 @@ +'use strict'; + +module.exports = { + name: 'toggle', + template: require('../../templates/components/toggle.tpl'), + style: require('../../styles/clay/components/toggle.scss'), + manipulator: 'checked', + defaults: { + label: '', + description: '', + attributes: {} + } +}; diff --git a/node_modules/pebble-clay/src/scripts/config-page.js b/node_modules/pebble-clay/src/scripts/config-page.js new file mode 100755 index 0000000..af4f3f5 --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/config-page.js @@ -0,0 +1,39 @@ +'use strict'; + +var minified = require('./vendor/minified'); +var ClayConfig = require('./lib/clay-config'); + +var $ = minified.$; +var _ = minified._; + +var config = _.extend([], window.clayConfig || []); +var settings = _.extend({}, window.claySettings || {}); +var returnTo = window.returnTo || 'pebblejs://close#'; +var customFn = window.customFn || function() {}; +var clayComponents = window.clayComponents || {}; +var clayMeta = window.clayMeta || {}; + +var platform = window.navigator.userAgent.match(/android/i) ? 'android' : 'ios'; +document.documentElement.classList.add('platform-' + platform); + +// Register the passed components +_.eachObj(clayComponents, function(key, component) { + ClayConfig.registerComponent(component); +}); + +var $mainForm = $('#main-form'); +var clayConfig = new ClayConfig(settings, config, $mainForm, clayMeta); + +// add listeners here +$mainForm.on('submit', function() { + // Set the return URL depending on the runtime environment + location.href = returnTo + + encodeURIComponent(JSON.stringify(clayConfig.serialize())); +}); + +// Run the custom function in the context of the ClayConfig +customFn.call(clayConfig, minified); + +// Now that we have given the dev's custom code to run and attach listeners, +// we build the config +clayConfig.build(); diff --git a/node_modules/pebble-clay/src/scripts/lib/clay-config.js b/node_modules/pebble-clay/src/scripts/lib/clay-config.js new file mode 100644 index 0000000..0a31610 --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/lib/clay-config.js @@ -0,0 +1,314 @@ +'use strict'; + +/** + * A Clay config Item + * @typedef {Object} Clay~ConfigItem + * @property {string} type + * @property {string|boolean|number} defaultValue + * @property {string} [messageKey] + * @property {string} [id] + * @property {string} [label] + * @property {Object} [attributes] + * @property {Array} [options] + * @property {Array} [items] + * @property {Array} [capabilities] + */ + +var HTML = require('../vendor/minified').HTML; +var _ = require('../vendor/minified')._; +var ClayItem = require('./clay-item'); +var utils = require('../lib/utils'); +var ClayEvents = require('./clay-events'); +var componentStore = require('./component-registry'); +var manipulators = require('./manipulators'); + +/** + * @extends ClayEvents + * @param {Object} settings - setting that were set from a previous session + * @param {Array|Object} config + * @param {M} $rootContainer + * @param {Object} meta + * @constructor + */ +function ClayConfig(settings, config, $rootContainer, meta) { + var self = this; + + var _settings = _.copyObj(settings); + var _items; + var _itemsById; + var _itemsByMessageKey; + var _isBuilt; + + /** + * Initialize the item arrays and objects + * @private + * @return {void} + */ + function _initializeItems() { + _items = []; + _itemsById = {}; + _itemsByMessageKey = {}; + _isBuilt = false; + } + + /** + * Add item(s) to the config + * @param {Clay~ConfigItem|Array} item + * @param {M} $container + * @return {void} + */ + function _addItems(item, $container) { + if (Array.isArray(item)) { + item.forEach(function(item) { + _addItems(item, $container); + }); + } else if (utils.includesCapability(meta.activeWatchInfo, item.capabilities)) { + if (item.type === 'section') { + var $wrapper = HTML('
'); + $container.add($wrapper); + _addItems(item.items, $wrapper); + } else { + var _item = _.copyObj(item); + _item.clayId = _items.length; + + var clayItem = new ClayItem(_item).initialize(self); + + if (_item.id) { + _itemsById[_item.id] = clayItem; + } + + if (_item.messageKey) { + _itemsByMessageKey[_item.messageKey] = clayItem; + } + + _items.push(clayItem); + + // set the value of the item via the manipulator to ensure consistency + var value = typeof _settings[_item.messageKey] !== 'undefined' ? + _settings[_item.messageKey] : + _item.defaultValue; + + clayItem.set(typeof value !== 'undefined' ? value : ''); + + $container.add(clayItem.$element); + } + } + } + + /** + * Throws if the config has not been built yet. + * @param {string} fnName + * @returns {boolean} + * @private + */ + function _checkBuilt(fnName) { + if (!_isBuilt) { + throw new Error( + 'ClayConfig not built. build() must be run before ' + + 'you can run ' + fnName + '()' + ); + } + return true; + } + + self.meta = meta; + self.$rootContainer = $rootContainer; + + self.EVENTS = { + /** + * Called before framework has initialized. This is when you would attach your + * custom components. + * @const + */ + BEFORE_BUILD: 'BEFORE_BUILD', + + /** + * Called after the config has been parsed and all components have their initial + * value set + * @const + */ + AFTER_BUILD: 'AFTER_BUILD', + + /** + * Called if .build() is executed after the page has already been built and + * before the existing content is destroyed + * @const + */ + BEFORE_DESTROY: 'BEFORE_DESTROY', + + /** + * Called if .build() is executed after the page has already been built and after + * the existing content is destroyed + * @const + */ + AFTER_DESTROY: 'AFTER_DESTROY' + }; + utils.updateProperties(self.EVENTS, {writable: false}); + + /** + * @returns {Array.} + */ + self.getAllItems = function() { + _checkBuilt('getAllItems'); + return _items; + }; + + /** + * @param {string} messageKey + * @returns {ClayItem} + */ + self.getItemByMessageKey = function(messageKey) { + _checkBuilt('getItemByMessageKey'); + return _itemsByMessageKey[messageKey]; + }; + + /** + * @param {string} id + * @returns {ClayItem} + */ + self.getItemById = function(id) { + _checkBuilt('getItemById'); + return _itemsById[id]; + }; + + /** + * @param {string} type + * @returns {Array.} + */ + self.getItemsByType = function(type) { + _checkBuilt('getItemsByType'); + return _items.filter(function(item) { + return item.config.type === type; + }); + }; + + /** + * @param {string} group + * @returns {Array.} + */ + self.getItemsByGroup = function(group) { + _checkBuilt('getItemsByGroup'); + return _items.filter(function(item) { + return item.config.group === group; + }); + }; + + /** + * @returns {Object} + */ + self.serialize = function() { + _checkBuilt('serialize'); + + _settings = {}; + + _.eachObj(_itemsByMessageKey, function(messageKey, item) { + _settings[messageKey] = { + value: item.get() + }; + + if (item.precision) { + _settings[messageKey].precision = item.precision; + } + }); + return _settings; + }; + + // @todo maybe don't do this and force the static method + self.registerComponent = ClayConfig.registerComponent; + + /** + * Empties the root container + * @returns {ClayConfig} + */ + self.destroy = function() { + var el = $rootContainer[0]; + self.trigger(self.EVENTS.BEFORE_DESTROY); + while (el.firstChild) { + el.removeChild(el.firstChild); + } + _initializeItems(); + self.trigger(self.EVENTS.AFTER_DESTROY); + return self; + }; + + /** + * Build the config page. This must be run before any of the get methods can be run + * If you call this method after the page has already been built, teh page will be + * destroyed and built again. + * @returns {ClayConfig} + */ + self.build = function() { + if (_isBuilt) { + self.destroy(); + } + self.trigger(self.EVENTS.BEFORE_BUILD); + _addItems(self.config, $rootContainer); + _isBuilt = true; + self.trigger(self.EVENTS.AFTER_BUILD); + return self; + }; + + _initializeItems(); + + // attach event methods + ClayEvents.call(self, $rootContainer); + + // prevent external modifications of properties + utils.updateProperties(self, { writable: false, configurable: false }); + + // expose the config to allow developers to update it before the build is run + self.config = config; +} + +/** + * Register a component to Clay. This must be called prior to .build(); + * @param {Object} component - the clay component to register + * @param {string} component.name - the name of the component + * @param {string} component.template - HTML template to use for the component + * @param {string|Object} component.manipulator - methods to attach to the component + * @param {function} component.manipulator.set - set manipulator method + * @param {function} component.manipulator.get - get manipulator method + * @param {Object} [component.defaults] - template defaults + * @param {function} [component.initialize] - method to scaffold the component + * @return {boolean} - Returns true if component was registered correctly + */ +ClayConfig.registerComponent = function(component) { + var _component = _.copyObj(component); + + if (componentStore[_component.name]) { + console.warn('Component: ' + _component.name + + ' is already registered. If you wish to override the existing' + + ' functionality, you must provide a new name'); + return false; + } + + if (typeof _component.manipulator === 'string') { + _component.manipulator = manipulators[component.manipulator]; + + if (!_component.manipulator) { + throw new Error('The manipulator: ' + component.manipulator + + ' does not exist in the built-in manipulators.'); + } + } + + if (!_component.manipulator) { + throw new Error('The manipulator must be defined'); + } + + if (typeof _component.manipulator.set !== 'function' || + typeof _component.manipulator.get !== 'function') { + throw new Error('The manipulator must have both a `get` and `set` method'); + } + + if (_component.style) { + var style = document.createElement('style'); + style.type = 'text/css'; + style.appendChild(document.createTextNode(_component.style)); + document.head.appendChild(style); + } + + componentStore[_component.name] = _component; + return true; +}; + +module.exports = ClayConfig; diff --git a/node_modules/pebble-clay/src/scripts/lib/clay-events.js b/node_modules/pebble-clay/src/scripts/lib/clay-events.js new file mode 100644 index 0000000..6eacdac --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/lib/clay-events.js @@ -0,0 +1,101 @@ +'use strict'; + +var $ = require('../vendor/minified').$; +var _ = require('../vendor/minified')._; + +/** + * Attaches event methods to the context. + * Call with ClayEvents.call(yourObject, $eventTarget) + * @param {EventEmitter|M} $eventTarget - An object that will be used as the event + * target. Must implement EventEmitter + * @constructor + */ +function ClayEvents($eventTarget) { + var self = this; + var _eventProxies = []; + + /** + * prefixes events with "|" + * @param {string} events + * @returns {string} + * @private + */ + function _transformEventNames(events) { + return events.split(' ').map(function(event) { + return '|' + event.replace(/^\|/, ''); + }).join(' '); + } + + /** + * @param {function} handler + * @param {function} proxy + * @returns {function} + * @private + */ + function _registerEventProxy(handler, proxy) { + var eventProxy = _.find(_eventProxies, function(item) { + return item.handler === handler ? item : null; + }); + + if (!eventProxy) { + eventProxy = { handler: handler, proxy: proxy }; + _eventProxies.push(eventProxy); + } + return eventProxy.proxy; + } + + /** + * @param {function} handler + * @returns {function} + * @private + */ + function _getEventProxy(handler) { + return _.find(_eventProxies, function(item) { + return item.handler === handler ? item.proxy : null; + }); + } + + /** + * Attach an event listener to the item. + * @param {string} events - a space separated list of events + * @param {function} handler + * @returns {ClayEvents} + */ + self.on = function(events, handler) { + var _events = _transformEventNames(events); + var self = this; + var _proxy = _registerEventProxy(handler, function() { + handler.apply(self, arguments); + }); + $eventTarget.on(_events, _proxy); + return self; + }; + + /** + * Remove the given event handler. NOTE: This will remove the handler from all + * registered events + * @param {function} handler + * @returns {ClayEvents} + */ + self.off = function(handler) { + var _proxy = _getEventProxy(handler); + if (_proxy) { + $.off(_proxy); + } + return self; + }; + + /** + * Trigger an event. + * @param {string} name - a single event name to trigger + * @param {Object} [eventObj] - an object to pass to the event handler, + * provided the handler does not have custom arguments. + * @returns {ClayEvents} + */ + self.trigger = function(name, eventObj) { + $eventTarget.trigger(name, eventObj); + return self; + }; +} + +module.exports = ClayEvents; diff --git a/node_modules/pebble-clay/src/scripts/lib/clay-item.js b/node_modules/pebble-clay/src/scripts/lib/clay-item.js new file mode 100644 index 0000000..ae53439 --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/lib/clay-item.js @@ -0,0 +1,78 @@ +'use strict'; + +var componentRegistry = require('./component-registry'); +var minified = require('../vendor/minified'); +var utils = require('../lib/utils'); +var ClayEvents = require('./clay-events'); + +var _ = minified._; +var HTML = minified.HTML; + +/** + * @extends ClayEvents + * @param {Clay~ConfigItem} config + * @constructor + */ +function ClayItem(config) { + var self = this; + + var _component = componentRegistry[config.type]; + + if (!_component) { + throw new Error('The component: ' + config.type + ' is not registered. ' + + 'Make sure to register it with ClayConfig.registerComponent()'); + } + + var extra = { + i18n: { + foo: 'bar' + } + }; + var _templateData = _.extend({}, _component.defaults || {}, config, extra); + var _html = _.formatHtml(_component.template.trim(), _templateData); + + /** @type {string|null} */ + self.id = config.id || null; + + /** @type {string|null} */ + self.messageKey = config.messageKey || null; + + /** @type {Object} */ + self.config = config; + /** @type {M} */ + self.$element = HTML(_html, _templateData); + + /** @type {M} */ + self.$manipulatorTarget = self.$element.select('[data-manipulator-target]'); + + // this caters for situations where the manipulator target is the root element + if (!self.$manipulatorTarget.length) { + self.$manipulatorTarget = self.$element; + } + + /** + * Run the initializer if it exists and attaches the css to the head. + * Passes minified as the first param + * @param {ClayConfig} clay + * @returns {ClayItem} + */ + self.initialize = function(clay) { + if (typeof _component.initialize === 'function') { + _component.initialize.call(self, minified, clay); + } + return self; + }; + + // attach event methods + ClayEvents.call(self, self.$manipulatorTarget); + + // attach the manipulator methods to the clayItem + _.eachObj(_component.manipulator, function(methodName, method) { + self[methodName] = method.bind(self); + }); + + // prevent external modifications of properties + utils.updateProperties(self, { writable: false, configurable: false }); +} + +module.exports = ClayItem; diff --git a/node_modules/pebble-clay/src/scripts/lib/component-registry.js b/node_modules/pebble-clay/src/scripts/lib/component-registry.js new file mode 100755 index 0000000..4a50721 --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/lib/component-registry.js @@ -0,0 +1,4 @@ +'use strict'; + +// module is blank because we dynamically add components +module.exports = {}; diff --git a/node_modules/pebble-clay/src/scripts/lib/manipulators.js b/node_modules/pebble-clay/src/scripts/lib/manipulators.js new file mode 100755 index 0000000..baf69e7 --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/lib/manipulators.js @@ -0,0 +1,180 @@ +'use strict'; + +var _ = require('../vendor/minified')._; + +/** + * @returns {ClayItem|ClayEvents} + * @extends {ClayItem} + */ +function disable() { + if (this.$manipulatorTarget.get('disabled')) { return this; } + this.$element.set('+disabled'); + this.$manipulatorTarget.set('disabled', true); + return this.trigger('disabled'); +} + +/** + * @returns {ClayItem|ClayEvents} + * @extends {ClayItem} + */ +function enable() { + if (!this.$manipulatorTarget.get('disabled')) { return this; } + this.$element.set('-disabled'); + this.$manipulatorTarget.set('disabled', false); + return this.trigger('enabled'); +} + +/** + * @returns {ClayItem|ClayEvents} + * @extends {ClayItem} + */ +function hide() { + if (this.$element[0].classList.contains('hide')) { return this; } + this.$element.set('+hide'); + return this.trigger('hide'); +} + +/** + * @returns {ClayItem|ClayEvents} + * @extends {ClayItem} + */ +function show() { + if (!this.$element[0].classList.contains('hide')) { return this; } + this.$element.set('-hide'); + return this.trigger('show'); +} + +module.exports = { + html: { + get: function() { + return this.$manipulatorTarget.get('innerHTML'); + }, + set: function(value) { + if (this.get() === value.toString(10)) { return this; } + this.$manipulatorTarget.set('innerHTML', value); + return this.trigger('change'); + }, + hide: hide, + show: show + }, + button: { + get: function() { + return this.$manipulatorTarget.get('innerHTML'); + }, + set: function(value) { + if (this.get() === value.toString(10)) { return this; } + this.$manipulatorTarget.set('innerHTML', value); + return this.trigger('change'); + }, + disable: disable, + enable: enable, + hide: hide, + show: show + }, + val: { + get: function() { + return this.$manipulatorTarget.get('value'); + }, + set: function(value) { + if (this.get() === value.toString(10)) { return this; } + this.$manipulatorTarget.set('value', value); + return this.trigger('change'); + }, + disable: disable, + enable: enable, + hide: hide, + show: show + }, + slider: { + get: function() { + return parseFloat(this.$manipulatorTarget.get('value')); + }, + set: function(value) { + var initVal = this.get(); + this.$manipulatorTarget.set('value', value); + if (this.get() === initVal) { return this; } + return this.trigger('change'); + }, + disable: disable, + enable: enable, + hide: hide, + show: show + }, + checked: { + get: function() { + return this.$manipulatorTarget.get('checked'); + }, + set: function(value) { + if (!this.get() === !value) { return this; } + this.$manipulatorTarget.set('checked', !!value); + return this.trigger('change'); + }, + disable: disable, + enable: enable, + hide: hide, + show: show + }, + radiogroup: { + get: function() { + return this.$element.select('input:checked').get('value'); + }, + set: function(value) { + if (this.get() === value.toString(10)) { return this; } + this.$element + .select('input[value="' + value.replace('"', '\\"') + '"]') + .set('checked', true); + return this.trigger('change'); + }, + disable: disable, + enable: enable, + hide: hide, + show: show + }, + checkboxgroup: { + get: function() { + var result = []; + this.$element.select('input').each(function(item) { + result.push(!!item.checked); + }); + return result; + }, + set: function(values) { + var self = this; + values = Array.isArray(values) ? values : []; + + while (values.length < this.get().length) { + values.push(false); + } + + if (_.equals(this.get(), values)) { return this; } + + self.$element.select('input') + .set('checked', false) + .each(function(item, index) { + item.checked = !!values[index]; + }); + + return self.trigger('change'); + }, + disable: disable, + enable: enable, + hide: hide, + show: show + }, + color: { + get: function() { + return parseInt(this.$manipulatorTarget.get('value'), 10) || 0; + }, + set: function(value) { + value = this.roundColorToLayout(value || 0); + + if (this.get() === value) { return this; } + this.$manipulatorTarget.set('value', value); + return this.trigger('change'); + }, + disable: disable, + enable: enable, + hide: hide, + show: show + } +}; diff --git a/node_modules/pebble-clay/src/scripts/lib/utils.js b/node_modules/pebble-clay/src/scripts/lib/utils.js new file mode 100644 index 0000000..58bfc46 --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/lib/utils.js @@ -0,0 +1,135 @@ +'use strict'; + +/** + * Batch update all the properties of an object. + * @param {Object} obj + * @param {Object} descriptor + * @param {boolean} [descriptor.configurable] + * @param {boolean} [descriptor.enumerable] + * @param {*} [descriptor.value] + * @param {boolean} [descriptor.writable] + * @param {function} [descriptor.get] + * @param {function} [descriptor.set] + * @return {void} + */ +module.exports.updateProperties = function(obj, descriptor) { + Object.getOwnPropertyNames(obj).forEach(function(prop) { + Object.defineProperty(obj, prop, descriptor); + }); +}; + +module.exports.capabilityMap = { + PLATFORM_APLITE: { + platforms: ['aplite'], + minFwMajor: 0, + minFwMinor: 0 + }, + PLATFORM_BASALT: { + platforms: ['basalt'], + minFwMajor: 0, + minFwMinor: 0 + }, + PLATFORM_CHALK: { + platforms: ['chalk'], + minFwMajor: 0, + minFwMinor: 0 + }, + PLATFORM_DIORITE: { + platforms: ['diorite'], + minFwMajor: 0, + minFwMinor: 0 + }, + PLATFORM_EMERY: { + platforms: ['emery'], + minFwMajor: 0, + minFwMinor: 0 + }, + BW: { + platforms: ['aplite', 'diorite'], + minFwMajor: 0, + minFwMinor: 0 + }, + COLOR: { + platforms: ['basalt', 'chalk', 'emery'], + minFwMajor: 0, + minFwMinor: 0 + }, + MICROPHONE: { + platforms: ['basalt', 'chalk', 'diorite', 'emery'], + minFwMajor: 0, + minFwMinor: 0 + }, + SMARTSTRAP: { + platforms: ['basalt', 'chalk', 'diorite', 'emery'], + minFwMajor: 3, + minFwMinor: 4 + }, + SMARTSTRAP_POWER: { + platforms: ['basalt', 'chalk', 'emery'], + minFwMajor: 3, + minFwMinor: 4 + }, + HEALTH: { + platforms: ['basalt', 'chalk', 'diorite', 'emery'], + minFwMajor: 3, + minFwMinor: 10 + }, + RECT: { + platforms: ['aplite', 'basalt', 'diorite', 'emery'], + minFwMajor: 0, + minFwMinor: 0 + }, + ROUND: { + platforms: ['chalk'], + minFwMajor: 0, + minFwMinor: 0 + }, + DISPLAY_144x168: { + platforms: ['aplite', 'basalt', 'diorite'], + minFwMajor: 0, + minFwMinor: 0 + }, + DISPLAY_180x180_ROUND: { + platforms: ['chalk'], + minFwMajor: 0, + minFwMinor: 0 + }, + DISPLAY_200x228: { + platforms: ['emery'], + minFwMajor: 0, + minFwMinor: 0 + } +}; + +/** + * Checks if all of the provided capabilities are compatible with the watch + * @param {Object} activeWatchInfo + * @param {Array} [capabilities] + * @return {boolean} + */ +module.exports.includesCapability = function(activeWatchInfo, capabilities) { + var notRegex = /^NOT_/; + var result = []; + + if (!capabilities || !capabilities.length) { + return true; + } + + for (var i = capabilities.length - 1; i >= 0; i--) { + var capability = capabilities[i]; + var mapping = module.exports.capabilityMap[capability.replace(notRegex, '')]; + + if (!mapping || + mapping.platforms.indexOf(activeWatchInfo.platform) === -1 || + mapping.minFwMajor > activeWatchInfo.firmware.major || + mapping.minFwMajor === activeWatchInfo.firmware.major && + mapping.minFwMinor > activeWatchInfo.firmware.minor + ) { + result.push(!!capability.match(notRegex)); + } else { + result.push(!capability.match(notRegex)); + } + } + + return result.indexOf(false) === -1; +}; diff --git a/node_modules/pebble-clay/src/scripts/vendor/autoprefixify.js b/node_modules/pebble-clay/src/scripts/vendor/autoprefixify.js new file mode 100644 index 0000000..3ab60b4 --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/vendor/autoprefixify.js @@ -0,0 +1,51 @@ +var path = require('path'); +var through = require('through'); +var postcss = require('postcss'); +var autoprefixer = require('autoprefixer'); +var requireFromString = require('require-from-string'); + +/** + * Stringifies the content + * @param {string} content + * @returns {string} + */ +function stringify (content) { + return 'module.exports = ' + JSON.stringify(content) + ';\n'; +} + +module.exports = function (file, options) { + + /** + * The function Browserify will use to transform the input. + * @param {string} file + * @returns {stream} + */ + function browserifyTransform (file) { + var extensions = ['.css', '.sass', '.scss', '.less']; + var chunks = []; + + if (extensions.indexOf(path.extname(file)) === -1) { + return through(); + } + + var write = function (buffer) { + chunks.push(buffer); + }; + + var end = function () { + var contents = requireFromString(Buffer.concat(chunks).toString('utf8')); + contents = postcss([autoprefixer(options)]).process(contents).css; + this.queue(stringify(contents)); + this.queue(null); + }; + + return through(write, end); + } + + if (typeof file !== 'string') { + options = file; + return browserifyTransform; + } else { + return browserifyTransform(file); + } +}; diff --git a/node_modules/pebble-clay/src/scripts/vendor/minified.js b/node_modules/pebble-clay/src/scripts/vendor/minified.js new file mode 100644 index 0000000..a9816fb --- /dev/null +++ b/node_modules/pebble-clay/src/scripts/vendor/minified.js @@ -0,0 +1,3602 @@ +// minified.js config start -- use this comment to re-create a configuration in the Builder +// - Only sections add, always, amdsupport, copyobj, dollardollar, +// - each, eachobj, equals, error, extend, find, format, formathtml, get, ht, +// - html, isobject, off, on, ready, request, select, set, template, trigger, +// - underscore, wait. + + +// WARNING! This file is autogenerated from minified-master.js and others. + +/* + * Minified.js - Lightweight Client-Side JavaScript Library (full package) + * Version: Version 2014 beta 5 b2 + * + * Public Domain. Use, modify and distribute it any way you like. No attribution required. + * To the extent possible under law, Tim Jansen has waived all copyright and related or neighboring rights to Minified. + * Please see http://creativecommons.org/publicdomain/zero/1.0/. + * NO WARRANTY EXPRESSED OR IMPLIED. USE AT YOUR OWN RISK. + * + * Contains code based on https://github.com/douglascrockford/JSON-js (also Public Domain). + * + * https://github.com/timjansen/minified.js + */ +// ==ClosureCompiler== +// @output_file_name minified.js +// @compilation_level ADVANCED_OPTIMIZATIONS +// ==/ClosureCompiler== + +/*$ + * @id ALL + * @doc no + * @required + * This id allows identifying whether both Web and Util are available. + */ + +///#snippet commonAmdStart + +/*$ + * @id require + * @name require() + * @syntax require(name) + * @group OPTIONS + * @module WEB, UTIL + * Returns a reference to a module. If you do not use an AMD loader to load Minified, just call require() with the + * argument 'minified' to get a reference to Minified. You can also access all modules defined using ##define(). + * + * If you do use an AMD loader, Minified will not define this function and you can use the AMD loader to obtain the + * reference to Minified. + * Minified's version of require is very simple and will only support Minified and other libraries designed + * for Minfied, but no real AMD libraries. If you need to work with libraries requiring AMD, you need a real AMD loader. + * + * @param name the name of the module to request. Minified is available as 'minified'. + * @return the reference to the module. Use the name 'minified' to get Minified. You can also access any modules defined using + * ##define(). If the name is unknown, it returns undefined. + * + * @see ##define() allows you to define modules that can be obtained using require(). + */ + +/*$ + * @id define + * @name define() + * @syntax define(name, factoryFunction) + * @group OPTIONS + * @module WEB, UTIL + * Defines a module that can be returned by ##require(), in case you don't have a AMD loader. If you have a AMD loader before you include Minified, + * define() will not be set and you can use the AMD loader's (more powerful) variant. + * + * Minified's versions of require() and define() are very simple and can not resolve things like circular references. + * Also, they are not AMD-compatible and only useful for simple modules. If you need to work with real AMD libraries that are not written + * for Minified, you need a real AMD loader. + * + * @example Creates a simple module and uses it: + *
+ * define('makeGreen', function(require) {
+ *     var MINI = require('minified'), $ = MINI.$; // obtain own ref to Minified
+ *     return function(list) {
+ *         $(list).set({$color: '#0f0', $backgroundColor: '#050'});
+ *     });
+ * });
+ *
+ * var makeGreen = require('makeGreen');
+ * makeGreen('.notGreenEnough');
+ * 
+ * + * @param name the name of the module to request. In Minified's implementation, only 'minified' is supported. + * @param factoryFunction is a function(require) will be called the first time the name is defined to obtain the module + * reference. It received a reference to ##require() (which is required for AMD backward-compatibility) and + * must return the value that is returned by ##require(). The function will only be called once, its result will + * be cached. + *
require
A reference to ##require(). While you could use require() from the global + * context, this would prevent backward compatibility with AMD.
+ *
(callback return value)
The reference to be returned by ##require().
+ * + * @see ##require() can be used to obtain references defined with ##define(). + */ + +/*$ + * @id amdsupport + * @name AMD stubs + * @configurable default + * @group OPTIONS + * @doc no + * @module WEB, UTIL + * If enabled, Minified will create stubs so you can use it without an AMD framework. + * It requires AMD's define() function. + */ +if (/^u/.test(typeof define)) { // no AMD support available ? define a minimal version + (function(def){ + var require = this['require'] = function(name) { return def[name]; }; + this['define'] = function(name, f) { def[name] = def[name] || f(require); }; + })({}); +} +/*$ + * @stop + */ + +define('minified', function() { + + ///#/snippet commonAmdStart + ///#snippet webVars + /*$ + * @id WEB + * @doc no + * @required + * This id allows identifying whether the Web module is available. + */ + + /** + * @const + */ + var _window = window; + + /** + * @const + * @type {!string} + */ + var MINIFIED_MAGIC_NODEID = 'Nia'; + + /** + * @const + * @type {!string} + */ + var MINIFIED_MAGIC_PREV = 'NiaP'; + + var setter = {}, getter = {}; + + var idSequence = 1; // used as node id to identify nodes, and as general id for other maps + + + /*$ + * @id ready_vars + * @dependency + */ + /** @type {!Array.} */ + var DOMREADY_HANDLER = /^[ic]/.test(document['readyState']) ? _null : []; // check for 'interactive' and 'complete' + /*$ + * @stop + */ + + ///#/snippet webVars + ///#snippet utilVars + /*$ + * @id UTIL + * @doc no + * @required + * This id allows identifying whether the Util module is available. + */ + + var _null = null; + + /** @const */ + var undef; + + /*$ + * @id date_constants + * @dependency + */ + function val3(v) {return v.substr(0,3);} + var MONTH_LONG_NAMES = split('January,February,March,April,May,June,July,August,September,October,November,December', /,/g); + var MONTH_SHORT_NAMES = map(MONTH_LONG_NAMES, val3); // ['Jan','Feb','Mar','Apr','May','Jun','Jul','Aug','Sep','Oct','Nov','Dec']; + var WEEK_LONG_NAMES = split('Sunday,Monday,Tuesday,Wednesday,Thursday,Friday,Saturday', /,/g); + var WEEK_SHORT_NAMES = map(WEEK_LONG_NAMES, val3); + var MERIDIAN_NAMES = split('am,pm', /,/g); + var MERIDIAN_NAMES_FULL = split('am,am,am,am,am,am,am,am,am,am,am,am,pm,pm,pm,pm,pm,pm,pm,pm,pm,pm,pm,pm', /,/g); + + var FORMAT_DATE_MAP = { + 'y': ['FullYear', nonOp], + 'Y': ['FullYear', function(d) { return d % 100; }], + 'M': ['Month', plusOne], + 'n': ['Month', MONTH_SHORT_NAMES], + 'N': ['Month', MONTH_LONG_NAMES], + 'd': ['Date', nonOp], + 'm': ['Minutes', nonOp], + 'H': ['Hours', nonOp], + 'h': ['Hours', function(d) { return (d % 12) || 12; }], + 'k': ['Hours', plusOne], + 'K': ['Hours', function(d) { return d % 12; }], + 's': ['Seconds', nonOp], + 'S': ['Milliseconds', nonOp], + 'a': ['Hours', MERIDIAN_NAMES_FULL], + 'w': ['Day', WEEK_SHORT_NAMES], + 'W': ['Day', WEEK_LONG_NAMES], + 'z': ['TimezoneOffset', function(d, dummy, timezone) { + if (timezone) + return timezone; + + var sign = d > 0 ? '-' : '+'; + var off = d < 0 ? -d : d; + return sign + pad(2, Math.floor(off/60)) + pad(2, off%60); + }] + }; + + var PARSE_DATE_MAP = { + 'y': 0, // placeholder -> ctorIndex + 'Y': [0, -2000], + 'M': [1,1], // placeholder -> [ctorIndex, offset|value array] + 'n': [1, MONTH_SHORT_NAMES], + 'N': [1, MONTH_LONG_NAMES], + 'd': 2, + 'm': 4, + 'H': 3, + 'h': 3, + 'K': [3,1], + 'k': [3,1], + 's': 5, + 'S': 6, + 'a': [3, MERIDIAN_NAMES] + }; + + /*$ + * @stop + */ + + /** @const */ + var MAX_CACHED_TEMPLATES = 99; + var templateCache={}; // template -> function + var templates = []; // list of MAX_CACHED_TEMPLATES templates + + ///#/snippet utilVars + ///#snippet commonFunctions + + /** @param s {?} */ + function toString(s) { + return s!=_null ? ''+s : ''; + } + /** + * @param s {?} + * @param o {string} + */ + function isType(s,o) { + return typeof s == o; + } + /** @param s {?} */ + function isString(s) { + return isType(s, 'string'); + } + function isObject(f) { + return !!f && isType(f, 'object'); + } + function isNode(n) { + return n && n['nodeType']; + } + function isNumber(n) { + return isType(n, 'number'); + } + function isDate(n) { + return isObject(n) && !!n['getDay']; + } + function isBool(n) { + return n === true || n === false; + } + function isValue(n) { + var type = typeof n; + return type == 'object' ? !!(n && n['getDay']) : (type == 'string' || type == 'number' || isBool(n)); + } + function nonOp(v) { + return v; + } + function plusOne(d) { + return d+1; + } + function replace(s, regexp, sub) { + return toString(s).replace(regexp, sub != _null ? sub : ''); + } + function escapeRegExp(s) { + return replace(s, /[\\\[\]\/{}()*+?.$|^-]/g, "\\$&"); + } + function trim(s) { + return replace(s, /^\s+|\s+$/g); + } + function eachObj(obj, cb, ctx) { + for (var n in obj) + if (obj.hasOwnProperty(n)) + cb.call(ctx || obj, n, obj[n]); + return obj; + } + function each(list, cb, ctx) { + if (list) + for (var i = 0; i < list.length; i++) + cb.call(ctx || list, list[i], i); + return list; + } + function filter(list, filterFuncOrObject, ctx) { + var r = []; + var f = isFunction(filterFuncOrObject) ? filterFuncOrObject : function(value) { return filterFuncOrObject != value; }; + each(list, function(value, index) { + if (f.call(ctx || list, value, index)) + r.push(value); + }); + return r; + } + function collector(iterator, obj, collectFunc, ctx) { + var result = []; + iterator(obj, function (a, b) { + if (isList(a = collectFunc.call(ctx || obj, a, b))) // extreme variable reusing: a is now the callback result + each(a, function(rr) { result.push(rr); }); + else if (a != _null) + result.push(a); + }); + return result; + } + function collectObj(obj, collectFunc, ctx) { + return collector(eachObj, obj, collectFunc, ctx); + } + function collect(list, collectFunc, ctx) { + return collector(each, list, collectFunc, ctx); + } + function keyCount(obj) { + var c = 0; + eachObj(obj, function(key) { c++; }); + return c; + } + function keys(obj) { // use Object.keys? in IE>=9 + var list = []; + eachObj(obj, function(key) { list.push(key); }); + return list; + } + function map(list, mapFunc, ctx) { + var result = []; + each(list, function(item, index) { + result.push(mapFunc.call(ctx || list, item, index)); + }); + return result; + } + function startsWith(base, start) { + if (isList(base)) { + var s2 = _(start); // convert start as we don't know whether it is a list yet + return equals(sub(base, 0, s2.length), s2); + } + else + return start != _null && base.substr(0, start.length) == start; + } + function endsWith(base, end) { + if (isList(base)) { + var e2 = _(end); + return equals(sub(base, -e2.length), e2) || !e2.length; + } + else + return end != _null && base.substr(base.length - end.length) == end; + } + function reverse(list) { + var len = list.length; + if (isList(list)) + return new M(map(list, function() { return list[--len]; })); + else + return replace(list, /[\s\S]/g, function() { return list.charAt(--len); }); + } + function toObject(list, value) { + var obj = {}; + each(list, function(item, index) { + obj[item] = value; + }); + return obj; + } + function copyObj(from, to) { + var dest = to || {}; + for (var name in from) + dest[name] = from[name]; + return dest; + } + function merge(list, target) { + var o = target; + for (var i = 0; i < list.length; i++) + o = copyObj(list[i], o); + return o; + } + function getFindFunc(findFunc) { + return isFunction(findFunc) ? findFunc : function(obj, index) { if (findFunc === obj) return index; }; + } + function getFindIndex(list, index, defaultIndex) { + return index == _null ? defaultIndex : index < 0 ? Math.max(list.length+index, 0) : Math.min(list.length, index); + } + function find(list, findFunc, startIndex, endIndex) { + var f = getFindFunc(findFunc); + var e = getFindIndex(list, endIndex, list.length); + var r; + for (var i = getFindIndex(list, startIndex, 0); i < e; i++) + if ((r = f.call(list, list[i], i)) != _null) + return r; + } + function findLast(list, findFunc, startIndex, endIndex) { + var f = getFindFunc(findFunc); + var e = getFindIndex(list, endIndex, -1); + var r; + for (var i = getFindIndex(list, startIndex, list.length-1); i > e; i--) + if ((r = f.call(list, list[i], i)) != _null) + return r; + } + function sub(list, startIndex, endIndex) { + var r = []; + if (list) { + var e = getFindIndex(list, endIndex, list.length); + for (var i = getFindIndex(list, startIndex, 0); i < e; i++) + r.push(list[i]); + } + return r; + } + function array(list) { + return map(list, nonOp); + } + function unite(list) { + return function() { + return new M(callList(list, arguments)); + }; + } + function uniq(list) { + var found = {}; + return filter(list, function(item) { + if (found[item]) + return false; + else + return found[item] = 1; + }); + } + function intersection(list, otherList) { + var keys = toObject(otherList, 1); + return filter(list, function(item) { + var r = keys[item]; + keys[item] = 0; + return r; + }); + } + function contains(list, value) { // TODO: can Array.indexOf be used in >IE8? + for (var i = 0; i < list.length; i++) + if (list[i] == value) + return true; + return false; + } + // equals if a and b have the same elements and all are equal. Supports getters. + function equals(x, y) { + var a = isFunction(x) ? x() : x; + var b = isFunction(y) ? y() : y; + var aKeys; + if (a == b) + return true; + else if (a == _null || b == _null) + return false; + else if (isValue(a) || isValue(b)) + return isDate(a) && isDate(b) && +a==+b; + else if (isList(a)) { + return (a.length == b.length) && + !find(a, function(val, index) { + if (!equals(val, b[index])) + return true; + }); + } + else { + return !isList(b) && + ((aKeys = keys(a)).length == keyCount(b)) && + !find(aKeys, function(key) { + if (!equals(a[key],b[key])) + return true; + }); + } + } + + function call(f, fThisOrArgs, args) { + if (isFunction(f)) + return f.apply(args && fThisOrArgs, map(args || fThisOrArgs, nonOp)); + } + function callList(list, fThisOrArgs, args) { + return map(list, function(f) { return call(f, fThisOrArgs, args);}); + } + function bind(f, fThis, beforeArgs, afterArgs) { + return function() { + return call(f, fThis, collect([beforeArgs, arguments, afterArgs], nonOp)); + }; + } + function partial(f, beforeArgs, afterArgs) { + return bind(f, this, beforeArgs, afterArgs); + } + function pad(digits, number) { + var signed = number < 0 ? '-' : ''; + var preDecimal = (signed?-number:number).toFixed(0); + while (preDecimal.length < digits) + preDecimal = '0' + preDecimal; + return signed + preDecimal; + } + + function processNumCharTemplate(tpl, input, fwd) { + var inHash; + var inputPos = 0; + var rInput = fwd ? input : reverse(input); + var s = (fwd ? tpl : reverse(tpl)).replace(/./g, function(tplChar) { + if (tplChar == '0') { + inHash = false; + return rInput.charAt(inputPos++) || '0'; + } + else if (tplChar == '#') { + inHash = true; + return rInput.charAt(inputPos++) || ''; + } + else + return inHash && !rInput.charAt(inputPos) ? '' : tplChar; + }); + return fwd ? s : (input.substr(0, input.length - inputPos) + reverse(s)); + } + + function getTimezone(match, idx, refDate) { // internal helper, see below + if (idx == _null || !match) + return 0; + return parseFloat(match[idx]+match[idx+1])*60 + parseFloat(match[idx]+match[idx+2]) + refDate.getTimezoneOffset(); + } + + // formats number with format string (e.g. "#.000", "#,#", "00000", "000.00", "000.000.000,00", "000,000,000.##") + // choice syntax: :|:|... + // e.g. 0:no item|1:one item|>=2:# items + // ="null" used to compare with nulls. + // choice also works with strings or bools, e.g. ERR:error|WAR:warning|FAT:fatal|ok + function formatValue(fmt, value) { + var format = replace(fmt, /^\?/); + if (isDate(value)) { + var timezone, match; + + if (match = /^\[(([+-])(\d\d)(\d\d))\]\s*(.*)/.exec(format)) { + timezone = match[1]; + value = dateAdd(value, 'minutes', getTimezone(match, 2, value)); + format = match[5]; + } + + return replace(format, /(\w)(\1*)(?:\[([^\]]+)\])?/g, function(s, placeholderChar, placeholderDigits, params) { + var val = FORMAT_DATE_MAP[placeholderChar]; + if (val) { + var d = value['get' + val[0]](); + var optionArray = (params && params.split(',')); + + if (isList(val[1])) + d = (optionArray || val[1])[d]; + else + d = val[1](d, optionArray, timezone); + if (d != _null && !isString(d)) + d = pad(placeholderDigits.length+1, d); + return d; + } + else + return s; + }); + + } + else + return find(format.split(/\s*\|\s*/), function(fmtPart) { + var match, numFmtOrResult; + if (match = /^([<>]?)(=?)([^:]*?)\s*:\s*(.*)$/.exec(fmtPart)) { + var cmpVal1 = value, cmpVal2 = +(match[3]); + if (isNaN(cmpVal2) || !isNumber(cmpVal1)) { + cmpVal1 = (cmpVal1==_null) ? "null" : toString(cmpVal1); // not ""+value, because undefined is treated as null here + cmpVal2 = match[3]; + } + if (match[1]) { + if ((!match[2] && cmpVal1 == cmpVal2 ) || + (match[1] == '<' && cmpVal1 > cmpVal2) || + (match[1] == '>' && cmpVal1 < cmpVal2)) + return _null; + } + else if (cmpVal1 != cmpVal2) + return _null; + numFmtOrResult = match[4]; + } + else + numFmtOrResult = fmtPart; + + if (isNumber(value)) + return numFmtOrResult.replace(/[0#](.*[0#])?/, function(numFmt) { + var decimalFmt = /^([^.]+)(\.)([^.]+)$/.exec(numFmt) || /^([^,]+)(,)([^,]+)$/.exec(numFmt); + var signed = value < 0 ? '-' : ''; + var numData = /(\d+)(\.(\d+))?/.exec((signed?-value:value).toFixed(decimalFmt ? decimalFmt[3].length:0)); + var preDecimalFmt = decimalFmt ? decimalFmt[1] : numFmt; + var postDecimal = decimalFmt ? processNumCharTemplate(decimalFmt[3], replace(numData[3], /0+$/), true) : ''; + + return (signed ? '-' : '') + + (preDecimalFmt == '#' ? numData[1] : processNumCharTemplate(preDecimalFmt, numData[1])) + + (postDecimal.length ? decimalFmt[2] : '') + + postDecimal; + }); + else + return numFmtOrResult; + }); + } + // returns date; null if optional and not set; undefined if parsing failed + function parseDate(fmt, date) { + var indexMap = {}; // contains reGroupPosition -> typeLetter or [typeLetter, value array] + var reIndex = 1; + var timezoneOffsetMatch; + var timezoneIndex; + var match; + + var format = replace(fmt, /^\?/); + if (format!=fmt && !trim(date)) + return _null; + + if (match = /^\[([+-])(\d\d)(\d\d)\]\s*(.*)/.exec(format)) { + timezoneOffsetMatch = match; + format = match[4]; + } + + var parser = new RegExp(format.replace(/(.)(\1*)(?:\[([^\]]*)\])?/g, function(wholeMatch, placeholderChar, placeholderDigits, param) { + if (/[dmhkyhs]/i.test(placeholderChar)) { + indexMap[reIndex++] = placeholderChar; + var plen = placeholderDigits.length+1; + return "(\\d"+(plen<2?"+":("{1,"+plen+"}"))+")"; + } + else if (placeholderChar == 'z') { + timezoneIndex = reIndex; + reIndex += 3; + return "([+-])(\\d\\d)(\\d\\d)"; + } + else if (/[Nna]/.test(placeholderChar)) { + indexMap[reIndex++] = [placeholderChar, param && param.split(',')]; + return "([a-zA-Z\\u0080-\\u1fff]+)"; + } + else if (/w/i.test(placeholderChar)) + return "[a-zA-Z\\u0080-\\u1fff]+"; + else if (/\s/.test(placeholderChar)) + return "\\s+"; + else + return escapeRegExp(wholeMatch); + })); + + if (!(match = parser.exec(date))) + return undef; + + var ctorArgs = [0, 0, 0, 0, 0, 0, 0]; + for (var i = 1; i < reIndex; i++) { + var matchVal = match[i]; + var indexEntry = indexMap[i]; + if (isList(indexEntry)) { // for a, n or N + var placeholderChar = indexEntry[0]; + var mapEntry = PARSE_DATE_MAP[placeholderChar]; + var ctorIndex = mapEntry[0]; + var valList = indexEntry[1] || mapEntry[1]; + var listValue = find(valList, function(v, index) { if (startsWith(matchVal.toLowerCase(), v.toLowerCase())) return index; }); + if (listValue == _null) + return undef; + if (placeholderChar == 'a') + ctorArgs[ctorIndex] += listValue * 12; + else + ctorArgs[ctorIndex] = listValue; + } + else if (indexEntry) { // for numeric values (yHmMs) + var value = parseFloat(matchVal); + var mapEntry = PARSE_DATE_MAP[indexEntry]; + if (isList(mapEntry)) + ctorArgs[mapEntry[0]] += value - mapEntry[1]; + else + ctorArgs[mapEntry] += value; + } + } + var d = new Date(ctorArgs[0], ctorArgs[1], ctorArgs[2], ctorArgs[3], ctorArgs[4], ctorArgs[5], ctorArgs[6]); + return dateAdd(d, 'minutes', -getTimezone(timezoneOffsetMatch, 1, d) - getTimezone(match, timezoneIndex, d)); + } + // format ?##00,00## + // returns number; null if optional and not set; undefined if parsing failed + function parseNumber(fmt, value) { + var format = replace(fmt, /^\?/); + if (format!=fmt && !trim(value)) + return _null; + var decSep = (/(^|[^0#.,])(,|[0#.]*,[0#]+|[0#]+\.[0#]+\.[0#.,]*)($|[^0#.,])/.test(format)) ? ',' : '.'; + var r = parseFloat(replace(replace(replace(value, decSep == ',' ? /\./g : /,/g), decSep, '.'), /^[^\d-]*(-?\d)/, '$1')); + return isNaN(r) ? undef : r; + } + function now() { + return new Date(); + } + function dateClone(date) { + return new Date(+date); + } + function capWord(w) { + return w.charAt(0).toUpperCase() + w.substr(1); + } + function dateAddInline(d, cProp, value) { + d['set'+cProp](d['get'+cProp]() + value); + return d; + } + function dateAdd(date, property, value) { + if (value == _null) + return dateAdd(now(), date, property); + return dateAddInline(dateClone(date), capWord(property), value); + } + function dateMidnight(date) { + var od = date || now(); + return new Date(od.getFullYear(), od.getMonth(), od.getDate()); + } + function dateDiff(property, date1, date2) { + var d1t = +date1; + var d2t = +date2; + var dt = d2t - d1t; + if (dt < 0) + return -dateDiff(property, date2, date1); + + var propValues = {'milliseconds': 1, 'seconds': 1000, 'minutes': 60000, 'hours': 3600000}; + var ft = propValues[property]; + if (ft) + return dt / ft; + + var cProp = capWord(property); + var calApproxValues = {'fullYear': 8.64e7*365, 'month': 8.64e7*365/12, 'date': 8.64e7}; // minimum values, a little bit below avg values + var minimumResult = Math.floor((dt / calApproxValues[property])-2); // -2 to remove the imperfections caused by the values above + + var d = dateAddInline(new Date(d1t), cProp, minimumResult); + for (var i = minimumResult; i < minimumResult*1.2+4; i++) { // try out 20% more than needed, just to be sure + if (+dateAddInline(d, cProp, 1) > d2t) + return i; + } + // should never ever be reached + } + + function ucode(a) { + return '\\u' + ('0000' + a.charCodeAt(0).toString(16)).slice(-4); + } + + function escapeJavaScriptString(s) { + return replace(s, /[\x00-\x1f'"\u2028\u2029]/g, ucode); + } + + // reimplemented split for IE8 + function split(str, regexp) { + + return str.split(regexp); + } + + function template(template, escapeFunction) { + if (templateCache[template]) + return templateCache[template]; + else { + var funcBody = 'with(_.isObject(obj)?obj:{}){'+ + map(split(template, /{{|}}}?/g), function(chunk, index) { + var match, c1 = trim(chunk), c2 = replace(c1, /^{/), escapeSnippet = (c1==c2) ? 'esc(' : ''; + if (index%2) { // odd means JS code + if (match = /^each\b(\s+([\w_]+(\s*,\s*[\w_]+)?)\s*:)?(.*)/.exec(c2)) + return 'each('+(trim(match[4])?match[4]:'this')+', function('+match[2]+'){'; + else if (match = /^if\b(.*)/.exec(c2)) + return 'if('+match[1]+'){'; + else if (match = /^else\b\s*(if\b(.*))?/.exec(c2)) + return '}else ' + (match[1] ? 'if('+match[2] +')' : '')+'{'; + else if (match = /^\/(if)?/.exec(c2)) + return match[1] ? '}\n' : '});\n'; + else if (match = /^(var\s.*)/.exec(c2)) + return match[1]+';'; + else if (match = /^#(.*)/.exec(c2)) + return match[1]; + else if (match = /(.*)::\s*(.*)/.exec(c2)) + return 'print('+escapeSnippet+'_.formatValue("'+escapeJavaScriptString(match[2])+'",'+(trim(match[1])?match[1]:'this')+(escapeSnippet&&')')+'));\n'; + else + return 'print('+escapeSnippet+(trim(c2)?c2:'this')+(escapeSnippet&&')')+');\n'; + } + else if (chunk){ + return 'print("'+escapeJavaScriptString(chunk)+'");\n'; + } + }).join('')+'}'; + var f = (new Function('obj', 'each', 'esc', 'print', '_', funcBody)); + var t = function(obj, thisContext) { + var result = []; + f.call(thisContext || obj, obj, function(obj, func) { + if (isList(obj)) + each(obj, function(value, index) { func.call(value, value, index); }); + else + eachObj(obj, function(key, value) { func.call(value, key, value); }); + }, escapeFunction || nonOp, function() {call(result['push'], result, arguments);}, _); + return result.join(''); + }; + if (templates.push(t) > MAX_CACHED_TEMPLATES) + delete templateCache[templates.shift()]; + return templateCache[template] = t; + } + } + + function escapeHtml(s) { + return replace(s, /[<>'"&]/g, function(s) { + return '&#'+s.charCodeAt(0)+';'; + }); + } + + function formatHtml(tpl, obj) { + return template(tpl, escapeHtml)(obj); + } + + function listBindArray(func) { + return function(arg1, arg2) { + return new M(func(this, arg1, arg2)); + }; + } + function listBind(func) { + return function(arg1, arg2, arg3) { + return func(this, arg1, arg2, arg3); + }; + } + function funcArrayBind(func) { + return function(arg1, arg2, arg3) { + return new M(func(arg1, arg2, arg3)); + }; + } + + ///#/snippet commonFunctions + ///#snippet webFunctions + + // note: only the web version has the f.item check + function isFunction(f) { + return typeof f == 'function' && !f['item']; // item check as work-around for webkit bug 14547 + } + + function isList(v) { + return v && v.length != _null && !isString(v) && !isNode(v) && !isFunction(v) && v !== _window; + } + + // used by IE impl of on() only + function push(obj, prop, value) { + (obj[prop] = (obj[prop] || [])).push(value); + } + // used by IE impl of on()/off() only + function removeFromArray(array, value) { + for (var i = 0; array && i < array.length; i++) + if (array[i] === value) + array['splice'](i--, 1); + } + + function extractNumber(v) { + return parseFloat(replace(v, /^[^\d-]+/)); + } + + // retrieves the node id of the element, create one if needed. + function getNodeId(el) { + return (el[MINIFIED_MAGIC_NODEID] = (el[MINIFIED_MAGIC_NODEID] || ++idSequence)); + } + + // collect variant that filters out duplicate nodes from the given list, returns a new array + function collectUniqNodes(list, func) { + var result = []; + var nodeIds = {}; + var currentNodeId; + + flexiEach(list, function(value) { + flexiEach(func(value), function(node) { + if (!nodeIds[currentNodeId = getNodeId(node)]) { + result.push(node); + nodeIds[currentNodeId] = true; + } + }); + }); + return result; + } + + // finds out the 'natural' height of the first element, the one if $$slide=1 + function getNaturalHeight(elementList, factor) { + var q = {'$position': 'absolute', '$visibility': 'hidden', '$display': 'block', '$height': _null}; + var oldStyles = elementList['get'](q); + var h = elementList['set'](q)['get']('clientHeight'); + elementList['set'](oldStyles); + return h*factor + 'px'; + } + + + + + // @condblock !ie8compatibility + function on(subSelector, eventSpec, handler, args, bubbleSelector) { + if (isFunction(eventSpec)) + return this['on'](_null, subSelector, eventSpec, handler, args); + else if (isString(args)) + return this['on'](subSelector, eventSpec, handler, _null, args); + else + return this['each'](function(baseElement, index) { + flexiEach(subSelector ? dollarRaw(subSelector, baseElement) : baseElement, function(registeredOn) { + flexiEach(toString(eventSpec).split(/\s/), function(namePrefixed) { + var name = replace(namePrefixed, /[?|]/g); + var prefix = replace(namePrefixed, /[^?|]/g); + var capture = (name == 'blur' || name == 'focus') && !!bubbleSelector; // bubble selectors for 'blur' and 'focus' registered as capuring! + var triggerId = idSequence++; + + // returns true if processing should be continued + function triggerHandler(eventName, event, target) { + var match = !bubbleSelector; + var el = bubbleSelector ? target : registeredOn; + if (bubbleSelector) { + var selectorFilter = getFilterFunc(bubbleSelector, registeredOn); + while (el && el != registeredOn && !(match = selectorFilter(el))) + el = el['parentNode']; + } + return (!match) || (name != eventName) || ((handler.apply($(el), args || [event, index]) && prefix=='?') || prefix == '|'); + }; + + function eventHandler(event) { + if (!triggerHandler(name, event, event['target'])) { + event['preventDefault'](); + event['stopPropagation'](); + } + }; + + registeredOn.addEventListener(name, eventHandler, capture); + + if (!registeredOn['M']) + registeredOn['M'] = {}; + registeredOn['M'][triggerId] = triggerHandler; // to be called by trigger() + + handler['M'] = collector(flexiEach, [handler['M'], function () { // this function will be called by off() + registeredOn.removeEventListener(name, eventHandler, capture); + delete registeredOn['M'][triggerId]; + }], nonOp); + + }); + }); + }); + } + // @condend !ie8compatibility + + + // @condblock !ie8compatibility + function off(handler) { + callList(handler['M']); + handler['M'] = _null; + } + // @condend !ie8compatibility + + // for remove & window.unload, IE only + function detachHandlerList(dummy, handlerList) { + flexiEach(handlerList, function(h) { + h.element.detachEvent('on'+h.eventType, h.handlerFunc); + }); + } + + function ready(handler) { + if (DOMREADY_HANDLER) + DOMREADY_HANDLER.push(handler); + else + setTimeout(handler, 0); + } + + function $$(selector, context, childrenOnly) { + return dollarRaw(selector, context, childrenOnly)[0]; + } + + function EE(elementName, attributes, children) { + var e = $(document.createElement(elementName)); + // @condblock UTIL + // this attributes != null check is only required with Util's isObject() implementation. Web's isObject() is simpler. + return (isList(attributes) || (attributes != _null && !isObject(attributes)) ) ? e['add'](attributes) : e['set'](attributes)['add'](children); + // @condend UTIL + // @cond !UTIL return (isList(attributes) || (!isObject(attributes)) ) ? e['add'](attributes) : e['set'](attributes)['add'](children); + } + + function clone(listOrNode) { + return collector(flexiEach, listOrNode, function(e) { + var c; + if (isList(e)) + return clone(e); + else if (isNode(e)) { + c = e['cloneNode'](true); + c['removeAttribute'] && c['removeAttribute']('id'); + return c; + } + else + return e; + }); + } + + /*$ + * @stop + */ + + function $(selector, context, childOnly) { + // @condblock ready + return isFunction(selector) ? ready(selector) : new M(dollarRaw(selector, context, childOnly)); + // @condend + // @cond !ready return new M(dollarRaw(selector, context)); + } + + // implementation of $ that does not produce a Minified list, but just an array + + + + + + + + + + + // @condblock !ie7compatibility + function dollarRaw(selector, context, childOnly) { + function flatten(a) { // flatten list, keep non-lists, remove nulls + return isList(a) ? collector(flexiEach, a, flatten) : a; + } + function filterElements(list) { // converts into array, makes sure context is respected + return filter(collector(flexiEach, list, flatten), function(node) { + var a = node; + while (a = a['parentNode']) + if (a == context[0] || childOnly) + return a == context[0]; + // fall through to return undef + }); + } + + if (context) { + if ((context = dollarRaw(context)).length != 1) + return collectUniqNodes(context, function(ci) { return dollarRaw(selector, ci, childOnly);}); + else if (isString(selector)) { + if (isNode(context[0]) != 1) + return []; + else + return childOnly ? filterElements(context[0].querySelectorAll(selector)) : context[0].querySelectorAll(selector); + } + else + return filterElements(selector); + + } + else if (isString(selector)) + return document.querySelectorAll(selector); + else + return collector(flexiEach, selector, flatten); + }; + // @condend !ie7compatibility + + // If context is set, live updates will be possible. + // Please note that the context is not evaluated for the '*' and 'tagname.classname' patterns, because context is used only + // by on(), and in on() only nodes in the right context will be checked + function getFilterFunc(selector, context) { + function wordRegExpTester(name, prop) { + var re = RegExp('(^|\\s+)' + name + '(?=$|\\s)', 'i'); + return function(obj) {return name ? re.test(obj[prop]) : true;}; + } + + var nodeSet = {}; + var dotPos = nodeSet; + if (isFunction(selector)) + return selector; + else if (isNumber(selector)) + return function(v, index) { return index == selector; }; + else if (!selector || selector == '*' || + (isString(selector) && (dotPos = /^([\w-]*)\.?([\w-]*)$/.exec(selector)))) { + var nodeNameFilter = wordRegExpTester(dotPos[1], 'tagName'); + var classNameFilter = wordRegExpTester(dotPos[2], 'className'); + return function(v) { + return isNode(v) == 1 && nodeNameFilter(v) && classNameFilter(v); + }; + } + else if (context) + return function(v) { + return $(selector, context)['find'](v)!=_null; // live search instead of node set, for on() + }; + else { + $(selector)['each'](function(node) { + nodeSet[getNodeId(node)] = true; + }); + return function(v) { + return nodeSet[getNodeId(v)]; + }; + } + } + + function getInverseFilterFunc(selector) { + var f = getFilterFunc(selector); + return function(v) {return f(v) ? _null : true;}; + } + ///#/snippet webFunctions + + ///#snippet extrasFunctions + function flexiEach(list, cb) { + if (isList(list)) + each(list, cb); + else if (list != _null) + cb(list, 0); + return list; + } + + function Promise() { + this['state'] = null; + this['values'] = []; + this['parent'] = null; + } + + /*$ + * @id promise + * @name _.promise() + * @syntax _.promise() + * @syntax _.promise(otherPromises...) + * @module WEB+UTIL + * + * Creates a new ##promiseClass#Promise##, optionally assimilating other promises. If no other promise is given, + * a fresh new promise is returned. + * + * The returned promise provides the methods ##fulfill() and ##reject() that can be called directly to change the promise's state, + * as well as the more powerful ##fire(). + * + * If one promise is given as parameter, the new promise assimilates the given promise as-is, and just forwards + * fulfillment and rejection with the original values. + * + * If more than one promise are given, it will assimilate all of them with slightly different rules: + *
  • the new promise is fulfilled if all assimilated promises have been fulfilled. The fulfillment values + * of all assimilated promises are given to the handler as arguments. Note that the fulfillment values themselves are always + * arrays, as a promise can have several fulfillment values in Minified's implementation.
  • + *
  • when one of the promises is rejected, the new promise is rejected immediately. The rejection handler gets the + * promises rejection value (first argument if it got several) as first argument, an array of the result values + * of all promises as a second (that means one array of arguments for each promise), and the index of the failed + * promise as third. + *
+ * + * @example A simple promise that is fulfilled after 1 second, using Minified's invocation syntax: + *
var p = _.promise();
+   * setTimeout(function() {
+   *     p.fire(true);
+   * }, 1000);
+   * 
+ * + * @example Request three files in parallel. When all three have been downloaded, concatenate them into a single string. + *
+   * var files = _('fileA.txt', 'fileA.txt', 'fileC.txt');
+   * var content;
+   * _.promise(files.map(function(file) {
+   *      return $.request('get', '/txts/' + file);
+   * })).then(function(fileRslt1, fileRslt2, fileRslt3) {
+   *      content = _(fileRslt1, fileRslt2, fileRslt3).map( function(result) { return result[0]; }).join('');
+   * }).error(function(status, response, xhr, url) {
+   *    alert('failed to load file '+url);
+   * });
+   * 
+ * + * @param otherPromises one or more promises to assimilate (varargs). You can also pass lists of promises. + * @return the new promise. + */ + function promise() { + var deferred = []; // this function calls the functions supplied by then() + + var assimilatedPromises = arguments; + var assimilatedNum = assimilatedPromises.length; + var numCompleted = 0; // number of completed, assimilated promises + var rejectionHandlerNum = 0; + + var obj = new Promise(); + + obj['errHandled'] = function() { + rejectionHandlerNum++; + if (obj['parent']) + obj['parent']['errHandled'](); + }; + + /*$ + * @id fire + * @name promise.fire() + * @syntax _.fire(newState) + * @syntax _.fire(newState, values) + * @module WEB+UTIL + * + * Changes the state of the promise into either fulfilled or rejected. This will also notify all ##then() handlers. If the promise + * already has a state, the call will be ignored. + * + * fire() can be invoked as a function without context ('this'). Every promise has its own instance. + * + * @example A simple promise that is fulfilled after 1 second, using Minified's invocation syntax: + *
var p = _.promise();
+     * setTimeout(function() {
+     *     p.fire(true, []);
+     * }, 1000);
+     * 
+ * + * @example Call fire() without a context: + *
var p = _.promise(function(resolve, reject) {
+     *     setTimeout(resolve.fire, 1000);
+     * });
+     * 
+ * + * @param newState true to set the Promise to fulfilled, false to set the state as rejected. If you pass null or + * undefined, the promise's state does not change. + * @param values optional an array of values to pass to ##then() handlers as arguments. You can also pass a non-list argument, which will then + * be passed as only argument. + * @return the promise + */ + var fire = obj['fire'] = function(newState, newValues) { + if (obj['state'] == null && newState != null) { + obj['state'] = !!newState; + obj['values'] = isList(newValues) ? newValues : [newValues]; + setTimeout(function() { + each(deferred, function(f) {f();}); + }, 0); + } + return obj; + }; + + // use promise varargs + each(assimilatedPromises, function assimilate(promise, index) { + try { + if (promise['then']) + promise['then'](function(v) { + var then; + if ((isObject(v) || isFunction(v)) && isFunction(then = v['then'])) + assimilate(v, index); + else { + obj['values'][index] = array(arguments); + if (++numCompleted == assimilatedNum) + fire(true, assimilatedNum < 2 ? obj['values'][index] : obj['values']); + } + }, + function(e) { + obj['values'][index] = array(arguments); + fire(false, assimilatedNum < 2 ? obj['values'][index] : [obj['values'][index][0], obj['values'], index]); + }); + else + promise(function() {fire(true, array(arguments));}, function() {fire(false, array(arguments)); }); + } + catch (e) { + fire(false, [e, obj['values'], index]); + } + }); + + /*$ + * @id stop + * @name promise.stop() + * @syntax promise.stop() + * @module WEB+UTIL + * Stops an ongoing operation, if supported. Currently the only promises supporting this are those returned by ##request(), ##animate(), ##wait() and + * ##asyncEach(). + * stop() invocation will be propagated over promises returned by ##then() and promises assimilated by ##promise(). You only need to invoke stop + * with the last promise, and all dependent promises will automatically stop as well. + * + * stop() can be invoked as a function without context ('this'). Every promise has its own instance. + * + * @return In some cases, the stop() can return a value. This is currently only done by ##animate() and ##wait(), which will return the actual duration. + * ##asyncEach()'s promise will also return any value it got from the promise that it stopped. + * + * @example Animation chain that can be stopped. + *
+     * var div = $('#myMovingDiv').set({$left: '0px', $top: '0px'});
+     * var prom = div.animate({$left: '200px', $top: '0px'}, 600, 0)
+     *    .then(function() {
+     *           return _.promise(div.animate({$left: '200px', $top: '200px'}, 800, 0),
+     *                            div.animate({$backgroundColor: '#f00'}, 200));
+     *    }).then(function() {
+     *           return div.animate({$left: '100px', $top: '100px'}, 400);
+     *    });
+     *
+     * $('#stopButton').on('click', prom.stop);
+     * 
+ */ + obj['stop'] = function() { + each(assimilatedPromises, function(promise) { + if (promise['stop']) + promise['stop'](); + }); + + return obj['stop0'] && call(obj['stop0']); + }; + + /*$ + * @id then + * @name promise.then() + * @syntax promise.then() + * @syntax promise.then(onSuccess) + * @syntax promise.then(onSuccess, onError) + * + * @module WEB + * Registers two callbacks that will be invoked when the ##promise#Promise##'s asynchronous operation finished + * successfully (onSuccess) or an error occurred (onError). The callbacks will be called after + * then() returned, from the browser's event loop. + * You can chain then() invocations, as then() returns another Promise object that you can attach to. + * + * The full distribution of Minified implements the Promises/A+ specification, allowing interoperability with other Promises frameworks. + * + * Note: If you use the Web module, you will get a simplified Promises implementation that cuts some corners. The most notable + * difference is that when a then() handler throws an exception, this will not be caught and the promise returned by + * then will not be automatically rejected. + * + * @example Simple handler for an HTTP request. Handles only success and ignores errors. + *
+     * $.request('get', '/weather.html')
+     *     .then(function(txt) {
+     *        alert('Got response!');
+     *     });
+     * 
+ * + * @example Including an error handler. + *
+     * $.request('get', '/weather.html')
+     *     .then(function(txt) {
+     *        alert('Got response!');
+     *     }, function(err) {
+     *        alert('Error!');
+     *     }));
+     * 
+ * + * @example Chained handler. + *
+     * $.request('get', '/weather.do')
+     *     .then(function(txt) {
+     *        showWeather(txt);
+     *     }
+     *     .then(function() {
+     *        return $.request('get', '/traffic.do');
+     *     }
+     *     .then(function(txt) {
+     *        showTraffic(txt);
+     *     }
+     *     .then(function() {
+     *        alert('All result displayed');
+     *     }, function() {
+     *        alert('An error occurred');
+     *     });
+     * 
+ * + * @param onSuccess optional a callback function to be called when the operation has been completed successfully. The exact arguments it receives depend on the operation. + * If the function returns a ##promise#Promise##, that Promise will be evaluated to determine the state of the promise returned by then(). If it returns any other value, the + * returned Promise will also succeed. If the function throws an error, the returned Promise will be in error state. + * Pass null or undefined if you do not need the success handler. + * @param onError optional a callback function to be called when the operation failed. The exact arguments it receives depend on the operation. If the function returns a ##promise#Promise##, that promise will + * be evaluated to determine the state of the Promise returned by then(). If it returns anything else, the returned Promise will + * have success status. If the function throws an error, the returned Promise will be in the error state. + * You can pass null or undefined if you do not need the error handler. + * @return a new ##promise#Promise## object. If you specified a callback for success or error, the new Promises's state will be determined by that callback if it is called. + * If no callback has been provided and the original Promise changes to that state, the new Promise will change to that state as well. + */ + var then = obj['then'] = function (onFulfilled, onRejected) { + var promise2 = promise(); + var callCallbacks = function() { + try { + var f = (obj['state'] ? onFulfilled : onRejected); + if (isFunction(f)) { + (function resolve(x) { + try { + var then, cbCalled = 0; + if ((isObject(x) || isFunction(x)) && isFunction(then = x['then'])) { + if (x === promise2) + throw new TypeError(); + then.call(x, function(x) { if (!cbCalled++) resolve(x); }, function(value) { if (!cbCalled++) promise2['fire'](false, [value]);}); + promise2['stop0'] = x['stop']; + } + else + promise2['fire'](true, [x]); + } + catch(e) { + if (!(cbCalled++)) { + promise2['fire'](false, [e]); + if (!rejectionHandlerNum) + throw e; + } + } + })(call(f, undef, obj['values'])); + } + else + promise2['fire'](obj['state'], obj['values']); + } + catch (e) { + promise2['fire'](false, [e]); + if (!rejectionHandlerNum) + throw e; + } + }; + if (isFunction(onRejected)) + obj['errHandled'](); + promise2['stop0'] = obj['stop']; + promise2['parent'] = obj; + if (obj['state'] != null) + setTimeout(callCallbacks, 0); + else + deferred.push(callCallbacks); + return promise2; + }; + + /*$ + * @id always + * @group REQUEST + * @name promise.always() + * @syntax promise.always(callback) + * @configurable default + * @module WEB+UTIL + * Registers a callback that will always be called when the ##promise#Promise##'s operation ended, no matter whether the operation succeeded or not. + * This is a convenience function that will call ##then() with the same function for both arguments. It shares all of its semantics. + * + * @example Simple handler for a HTTP request. + *
+     * $.request('get', '/weather.html')
+     *     .always(function() {
+     *        alert('Got response or error!');
+     *     });
+     * 
+ * + * @param callback a function to be called when the operation has been finished, no matter what its result was. The exact arguments depend on the operation and may + * vary depending on whether it succeeded or not. If the function returns a ##promise#Promise##, that Promise will + * be evaluated to determine the state of the returned Promise. If provided and it returns regularly, the returned promise will + * have success status. If it throws an error, the returned Promise will be in the error state. + * @return a new ##promise#Promise## object. Its state is determined by the callback. + */ + obj['always'] = function(func) { return then(func, func); }; + + /*$ + * @id error + * @group REQUEST + * @name promise.error() + * @syntax promise.error(callback) + * @configurable default + * @module WEB, UTIL + * Registers a callback that will be called when the operation failed. + * This is a convenience function that will invoke ##then() with only the second argument set. It shares all of its semantics. + * + * @example Simple handler for a HTTP request. + *
+     * $.request('get', '/weather.html')
+     *     .error(function() {
+     *        alert('Got error!');
+     *     });
+     * 
+ * + * @param callback a function to be called when the operation has failed. The exact arguments depend on the operation. If the function returns a ##promise#Promise##, that Promise will + * be evaluated to determine the state of the returned Promise. If it returns regularly, the returned Promise will + * have success status. If it throws an error, the returned Promise will be in error state. + * @return a new ##promise#Promise## object. Its state is determined by the callback. + */ + obj['error'] = function(func) { return then(0, func); }; + + return obj; + } + + ///#/snippet extrasFunctions + ///#snippet extrasDocs + /*$ + * @id length + * @group SELECTORS + * @requires dollar + * @name list.length + * @syntax length + * @module WEB, UTIL + * + * Contains the number of elements in the ##list#Minified list##. + * + * @example With Web module: + *
+   * var list = $('input');
+   * var myValues = {};
+   * for (var i = 0; i < list.length; i++)
+   *    myValues[list[i].name] = list[i].value;
+   * 
+ * + * @example With Util module: + *
+   * var list = _(1, 2, 3);
+   * var sum = 0;
+   * for (var i = 0; i < list.length; i++)
+   *    sum += list[i];
+   * 
+ */ + /*$ + * @stop + */ + ///#/snippet extrasDocs + + ///#snippet utilM + + /* + * syntax: M(list, assimilateSublists) + * M(null, singleElement) + * + * + */ + /** @constructor */ + function M(list, assimilateSublists) { + var self = this, idx = 0; + if (list) + for (var i = 0, len = list.length; i < len; i++) { + var item = list[i]; + if (assimilateSublists && isList(item)) + for (var j = 0, len2 = item.length; j < len2; j++) + self[idx++] = item[j]; + else + self[idx++] = item; + } + else + self[idx++] = assimilateSublists; + + self['length'] = idx; + self['_'] = true; + } + + function _() { + return new M(arguments, true); + } + + ///#/snippet utilM + + //// LIST FUNCTIONS //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + + copyObj({ + ///#snippet utilListFuncs + /*$ + * @id each + * @group LIST + * @requires + * @configurable default + * @name .each() + * @altname _.each() + * @syntax list.each(callback) + * @syntax list.each(callback, ctx) + * @syntax _.each(list, callback) + * @syntax _.each(list, callback, ctx) + * @module UTIL, WEB + * Invokes the given function once for each item in the list. The function will be called with the item as first parameter and + * the zero-based index as second. Unlike JavaScript's built-in forEach() it will be invoked for each item in the list, + * even if it is undefined. + * + * @example Creates the sum of all list entries. + *
+     * var sum = 0;
+     * _(17, 4, 22).each(function(item, index) {
+     *     sum += item;
+     * });
+     * 
+ * + * @example The previous example with a native array: + *
+     * var sum = 0;
+     * _.each([17, 4, 22], function(item, index) {
+     *     sum += item;
+     * });
+     * 
+ * + * @example This goes through all h2 elements of the class 'section' on a web page and changes their content: + *
+     * $('h2.section').each(function(item, index) {
+     *     item.innerHTML = 'Section ' + index + ': ' + item.innerHTML;
+     * });
+     * 
+ * + * @param list a list to iterate. Can be an array, a ##list#Minified list## or any other array-like structure with + * length property. + * @param callback The callback function(item, index) to invoke for each list element. + *
item
The current list element.
+ *
index
The second the zero-based index of the current element.
+ *
this
The given context if not null. Otherwise the list.
+ * The callback's return value will be ignored. + * @param ctx optional a context to pass to the callback as 'this'. Only supported in UTIL module. + * @return the list + * + * @see ##per() works like each(), but wraps the list elements in a list. + * @see ##find() can be used instead of each() if you need to abort the loop. + * @see ##eachObj() iterates through the properties of an object. + */ + 'each': listBind(each), + + /*$ + * @id equals + * @group LIST + * @requires + * @configurable default + * @name .equals() + * @altname _.equals() + * @syntax list.equals(otherObject) + * @syntax _.equals(thisObject, otherObject) + * @module UTIL + * Checks whether two values, lists or objects are equal in a deep comparison. + * + * First equals() checks whether it got a function as parameter. + * If yes, it will be invoked without arguments and equals() calls itself recursively with the function's result. + * + * Once both values are no functions anymore, the values will be evaluated, If the first value is... + *
  • ...null or undefined, they are only equal if the other one is also either null or undefined.
  • + *
  • ...a value as defined by ##_.isValue(), but not a Date, they are equal if the other value is the same type and is equal according to the '==' operator.
  • + *
  • ...a Date, they are equal if the other value is a Date representing the same time.
  • + *
  • ...a list or array, they are equal if the other value is also either a list or an array, has the same number of items and all items equal the items of the other + * list at the same position. The equality of list items is determined recursively using the same rules, so you can also nest lists.
  • + *
  • ...a function, it will be invoked without arguments and its return value is evaluated using these rules as if the value has been passed.
  • + *
  • ...any other object, they are equal if they contain exactly the same keys (as defined by ##_.eachObj()) and all values are equal as determined using these rules + * recursively.
  • + *
+ * + * Please note that, according to the rules, a ##list#Minified list## is equal to an array, as long as their content is equal. equals does not + * differentiate between null and undefined. + * + * equals is commutative. If you swap the parameters, the result is the same as long as no functions are involved. + * + * @example Compare a list and an array: + *
+     *  _.equals([1, 2, 3], _(1, 2, 3));  // returns true
+     * 
+ * + * @example Same result, but with a list method: + *
+     *  _(1, 2, 3).equals([1, 2, 3]);  // returns true
+     * 
+ * + * @param thisObject The first reference to evaluate. + * @param otherObject The second reference to evaluate. + * @return true if both references are equal. False otherwise. + */ + 'equals': listBind(equals), + + /*$ + * @id find + * @group LIST + * @requires + * @configurable default + * @name .find() + * @altname _.find() + * @syntax list.find(findFunc) + * @syntax list.find(element) + * @syntax list.find(findFunc, startIndex) + * @syntax list.find(element, startIndex) + * @syntax _.find(list, findFunc) + * @syntax _.find(list, element) + * @syntax _.find(list, findFunc, startIndex) + * @syntax _.find(list, element, startIndex) + * @module WEB, UTIL + * Finds a specific value in the list. There are two ways of calling find(): + *
    + *
  1. With a value as argument. Then find() will search for the first occurrence of an identical value in the list, + * using the '===' operator for comparisons, and return the index. If it is not found, + * find() returns undefined.
  2. + *
  3. With a callback function. find() will then call the given function for each list element until the function + * returns a value that is not null or undefined. This value will be returned.
  4. + *
+ * + * find() can also be used as an alternative to ##each() if you need to abort the loop. + * + * @example Finds the first negative number in the list: + *
+     * var i = _(1, 2, -4, 5, 2, -1).find(function(value, index) { if (value < 0) return index; }); // returns 2
+     * 
+ + * @example Finds the index of the first 5 in the array: + *
+     * var i = _.find([3, 6, 7, 6, 5, 4, 5], 5); // returns 4 (index of first 5)
+     * 
+ * + * @example Determines the position of the element with the id '#wanted' among all li elements: + *
+     * var elementIndex = $('li').find($$('#wanted'));
+     * 
+ * + * @example Goes through the elements to find the first div that has the class 'myClass', and returns this element: + *
+     * var myClassElement = $('div').find(function(e) { if ($(e).is('.myClass')) return e; });
+     * 
+ * + * @param list A list to use as input. Can be an array, a ##list#Minified list## or any other array-like structure with + * length property. + * @param findFunc The callback function(item, index) that will be invoked for every list item until it returns a non-null value: + *
item
The current list element.
index
The second the zero-based index of the current element.
+ *
this
This list.
+ *
(callback return value)
If the callback returns something other than null or + * undefined, find() will return it directly. Otherwise it will continue.
+ * @param element the element to search for + * @param startIndex optional the 0-based index of the first element to search. + * @return if called with an element, either the element's index in the list or undefined if not found. If called with a callback function, + * it returns either the value returned by the callback or undefined. + * + * @see ##findLast() is the equivalent to find() for the list's end. + */ + 'find': listBind(find), + + /*$ + * @stop + */ + dummySort:0 + , + ///#/snippet utilListFuncs + ///#snippet webListFuncs + + /*$ + * @id select + * @group SELECTORS + * @requires dollar + * @configurable default + * @name .select() + * @syntax list.select(selector) + * @syntax list.select(selector, childrenOnly) + * @module WEB + * Executes a selector with the list as context. list.select(selector, childrenOnly) is equivalent + * to $(selector, list, childrenOnly). + * + * @example Returns a list of all list elements: + *
+     * var parents = $('ol.myList').select('li', true);
+     * 
+ * + * @example Returns a list of all child elements: + *
+     * var children = $('.myElements').select('*', true);
+     * 
+ * + * @param selector a selector or any other valid first argument for #dollar#$(). + * @param childrenOnly optional if set, only direct children of the context nodes are included in the list. Children of children will be filtered out. If omitted or not + * true, all descendants of the context will be included. + * @return the new list containing the selected descendants. + * + * @see ##only() executes a selector on the list elements, instead of their descendants. + */ + 'select': function(selector, childOnly) { + return $(selector, this, childOnly); + }, + + /*$ + * @id get + * @group SELECTORS + * @requires dollar + * @configurable default + * @name .get() + * @syntax list.get(name) + * @syntax list.get(name, toNumber) + * @syntax list.get(list) + * @syntax list.get(list, toNumber) + * @syntax list.get(map) + * @syntax list.get(map, toNumber) + * @module WEB + * Retrieves properties, attributes and styles from the list's first element. The syntax to request those values is mostly identical with ##set(). You can either + * get a single value if you specify only one name, or get an object map when you specify several names using an array or an object map. + * + * The name parameter defines what kind of data you are reading. The following name schemes are supported: + * + * + * + * + * + * + * + * + * + * + * + * + *
Name SchemaExampleSets what?Description
nameinnerHTMLPropertyA name without prefix of '$' or '@' gets a property of the object.
@name@hrefAttributeGets the HTML attribute using getAttribute().
%name%phoneData-AttributeGets a data attribute using getAttribute(). Data attributes are + * attributes whose names start with 'data-'. '%myattr' and '@data-myattr' are equivalent.
$name$fontSizeCSS PropertyGets a style using the element's style object. + * The syntax for the CSS styles is camel-case (e.g. "$backgroundColor", not "$background-color"). Shorthand properties like "border" or "margin" are + * not supported. You must use the full name, e.g. "$marginTop". Minified will try to determine the effective style + * and thus will return the value set in style sheets if not overwritten using a regular style.
$$CSS ClassesA simple $ returns the CSS classes of the element and is identical with "className".
$$$$StyleReads the element's style attribute in a browser-independent way. On legacy IEs it uses + * style.cssText, and on everything else just the "style" attribute.
$$show$$showShow/HideReturns 1 if the element is visible and 0 if it is not visible. An element counts as + * visible if '$visibility' is not 'hidden' and '$display' is not 'none'. Other properties will be ignored, even if they can also be used to hide the element.
$$fade$$fadeFade EffectThe name '$$fade' returns the opacity of the element as a value between 0 and 1. + * '$$fade' will also automatically evaluate the element's 'visibility' and 'display' styles to find out whether the element is actually visible.
$$slide$$slideSlide Effect'$$slide' returns the height of the element in pixels with a 'px' suffix and is equivalent to '$height'. + * Please note that you can pass that 'px' value to '$$slide' in ##set(), which will then set the according '$height'.
$$scrollX, $$scrollY$$scrollYScroll CoordinatesThe names '$$scrollX' and + * '$$scrollY' can be used on $(window) to retrieve the scroll coordinates of the document. + * The coordinates are specified in pixels without a 'px' unit postfix.
+ * + * @example Retrieves the id, title attribute and the background color of the element '#myElement': + *
+     * var id = $('#myElement).get('id');
+     * var title = $('#myElement).get('@title');
+     * var bgColor = $('#myElement).get('$backgroundColor');
+     * 
+ * + * @example Retrieves the id, title attribute and the background color of the element '#myElement' as a map: + *
+     * var m = $('#myElement).get(['id', '@title', '$backgroundColor']);
+     * var id = m.id;
+     * var title = m['@title'];
+     * var bgColor = m.$backgroundColor;
+     * 
+ * + * @example Uses ##get() and ##set() to reposition an element: + *
+     * var coords = $('#myElement').get({$top: 0, $left: 0}, true);
+     * coords.$top = coords.$top + 10 + 'px';
+     * coords.$left = coords.$left + 20 + 'px';
+     * $('#myElement').set(coords);
+     * 
+ * Please note that the values of $top and $left in the get() invocation do not matter and will be ignored! + * + * @param name the name of a single property or attribute to modify. Unprefixed names set properties, a '$' prefix sets CSS styles and + * '@' sets attributes. Please see the table above for special properties and other options. + * @param list in order to retrieve more than one value, you can specify several names in an array or list. get() will then return an object map + * containing the values. + * @param map if you specify an object that is neither list nor string, get() will use it as a map of property names. Each property name will be requested. + * The values of the properties in the map will be ignored. get() will then return a new object map containing of results. + * @param toNumber if 'true', get() converts all returned values into numbers. If they are strings, + * get() removes any non-numeric characters before the conversion. This is useful when you request + * a CSS property such as '$marginTop' that returns a value with a unit suffix, like "21px". get() will convert it + * into a number and return 21. If the returned value is not parsable as a number, NaN will be returned. + * @return if get() was called with a single name, it returns the corresponding value. + * If a list or map was given, get() returns a new object map with the names as keys and the values as values. + * It returns undefined if the list is empty. + * + * @see ##set() sets values using the same property syntax. + */ + 'get': function(spec, toNumber) { + var self = this; + var element = self[0]; + + if (element) { + if (isString(spec)) { + var match = /^(\W*)(.*)/.exec(replace(spec, /^%/,'@data-')); + var prefix = match[1]; + var s; + + if (getter[prefix]) + s = getter[prefix](this, match[2]); + else if (spec == '$') + s = self['get']('className'); + else if (spec == '$$') { + s = self['get']('@style'); + } + else if (spec == '$$slide') + s = self['get']('$height'); + else if (spec == '$$fade' || spec == '$$show') { + if (self['get']('$visibility') == 'hidden' || self['get']('$display') == 'none') + s = 0; + else if (spec == '$$fade') { + s = + isNaN(self['get']('$opacity', true)) ? 1 : self['get']('$opacity', true); + } + else // $$show + s = 1; + } + else if (prefix == '$') { + s = _window['getComputedStyle'](element, _null)['getPropertyValue'](replace(match[2], /[A-Z]/g, function (match2) { return '-' + match2.toLowerCase(); })); + } + else if (prefix == '@') + s = element.getAttribute(match[2]); + else + s = element[match[2]]; + return toNumber ? extractNumber(s) : s; + } + else { + var r = {}; + (isList(spec) ? flexiEach : eachObj)(spec, function(name) { + r[name] = self['get'](name, toNumber); + }); + return r; + } + } + }, + + /*$ + * @id set + * @group SELECTORS + * @requires dollar get + * @configurable default + * @name .set() + * @syntax list.set(name, value) + * @syntax list.set(properties) + * @syntax list.set(cssClasses) + * @module WEB + * + * Modifies the list's elements by setting their properties, attributes, CSS styles and/or CSS classes. You can either supply a + * single name and value to set only one property, or you can provide an object that contains name/value pairs to describe more than one property. + * More complex operations can be accomplished by supplying functions as values. They will then be called for each element that will + * be set. + * + * The name parameter defines what kind of data you are setting. The following name schemes are supported: + * + * + * + * + * + * + * + * + * + * + * + * + * + *
Name SchemaExampleSets what?Description
nameinnerHTMLPropertyA name without prefix of '$' or '@' sets a property of the object.
@name@hrefAttributeSets the HTML attribute using setAttribute(). In order to stay compatible with Internet Explorer 7 and earlier, + * you should not set the attributes '@class' and '@style'. Instead use '$' and '$$' as shown below.
%name%phoneData-AttributeSets a data attribute using setAttribute(). Data attributes are + * attributes whose names start with 'data-'. '%myattr' and '@data-myattr' are equivalent.
$name$fontSizeCSS PropertySets a style using the element's style object. + * The syntax for the CSS styles is camel-case (e.g. "$backgroundColor", not "$background-color").
$$CSS ClassesA simple $ modifies the element's CSS classes using the object's className property. The value is a + * space-separated list of class names. If prefixed with '-' the class is removed, a '+' prefix adds the class and a class name without prefix toggles the class. + * The name '$' can also be omitted if set is called with class names as only argument.
$$$$StyleSets the element's style attribute in a browser-independent way.
$$show$$showShow/HideIf true or a number not 0, it will make sure the element is visible by + * making sure '$display' is not 'none' and by setting '$visibility' to 'visible'. Please see ##show() for details. If the value is false or 0, it + * will be hidden by setting '$display' to 'none'.
$$fade$$fadeFade EffectThe name '$$fade' sets the opacity of the element in a browser-independent way. The value must be a number + * between 0 and 1. '$$fade' will also automatically control the element's 'visibility' style. If the value is 0, + * the element's visibility will automatically be set to 'hidden'. If the value is larger, the visibility will be set to + * 'visible'. '$$fade' only works with block elements.
$$slide$$slideSlide EffectThe name '$$slide' allows a vertical slide-out or slide-in effect. The value must be a number + * between 0 and 1 and will be used to set the element's '$height'. '$$slide' will also automatically control the element's 'visibility' + * style. If the value is 0, the element's visibility will automatically be set to 'hidden'. If the value is larger, + * the visibility will be set to 'visible'. '$$slide' only works with block elements and will not set the + * element's margin or padding. If you need a margin or padding, you should wrap the elements in a simple <div>.
$$scrollX, $$scrollY$$scrollYScroll CoordinatesThe names '$$scrollX' and + * '$$scrollY' can be used on $(window) to set the scroll coordinates of the document. + * The coordinates are specified in pixels, but must not use a 'px' unit postfix.
+ * + * @example Unchecking checkboxes: + *
+     * $('input.checkbox').set('checked', false);
+     * 
+ * + * @example Changing the innerHTML property of an element: + *
+     * $('#toc').set('innerHTML', 'Content');
+     * 
+ * + * @example Changing attributes: + *
+     * $('a.someLinks').set('@href', 'http://www.example.com/');
+     * 
+ * + * @example Removing attributes: + *
+     * $('a.someLinks').set('@title', null);
+     * 
+ * + * @example Changing styles: + *
+     * $('.bigText').set('$fontSize', 'x-large');
+     * 
+ * + * @example Adding and removing CSS classes: + *
+     * $('.myElem').set('$', '+myClass -otherClass');
+     * 
+ * + * @example Toggling a CSS class: + *
+     * $('.myElem').set('$', 'on');
+     * 
+ * + * @example Shortcut for CSS manipulation: + *
+     * $('.myElem').set('+myClass -otherClass on');
+     * 
+ * + * @example Making an element transparent: + *
+     * $('.seeThrough').set('$$fade', 0.5);
+     * 
+ * + * @example Making an element visible. Note that $$fade will set the element's display style to 'block' and visibility style to 'visible'. + *
+     * $('.myElem').set('$$fade', 1);
+     * 
+ * + * @example Using a map to change several properties: + *
+     * $('input.checkbox').set({checked: false,
+     *                          '@title': 'Check this'});
+     * 
+ * + * @example Changing CSS with a map: + *
+     * $('.importantText').set({$fontSize: 'x-large',
+     *                          $color: 'black',
+     *                          $backgroundColor: 'red',
+     *                          $: '+selected -default'});
+     * 
+ * + * @example You can specify a function as value to modify a value instead of just setting it: + *
+     * $('h2').set('innerHTML', function(oldValue, index) {
+     *     return 'Chapter ' + index + ': ' + oldValue.toUpperCase();
+     * });
+     * 
+ * + * @param name the name of a single property or attribute to modify. Unprefixed names set properties, a '$' prefix sets CSS styles and + * '@' sets attributes. Please see the table above for special properties and other options. + * @param value the value to set. If value is null and name specified an attribute, the attribute will be removed. + * If dollar ('$') has been passed as name, the value can contain space-separated CSS class names. If prefixed with a '+' the class will be added, + * with a '-' prefix the class will be removed. Without prefix, the class will be toggled. + * If value is a function, the function(oldValue, index, obj) will be invoked for each list element + * to evaluate the new value: + *
oldValue
The old value of the property to be changed, as returned by ##get(). + * For the CSS style names, this is the computed style of the property
+ *
index
The list index of the object owning the property
+ *
obj
The list element owning the property.
+ *
(callback return value)
The value to be set.
+ * Functions are not supported by '$'. + * @param properties a Object as map containing names as keys and the values to set as map values. See above for the name and value syntax. + * @param cssClasses if set() is invoked with a string as single argument, the name "$" (CSS classes) is assumed and the argument is the + * value. See above for CSS syntax. + * Instead of a string, you can also specify a function(oldValue, index, obj) to modify the existing classes. + * @return the list + * + * @see ##get() retrieves values using the same property syntax. + * @see ##animate() animates values using the same property syntax. + * @see ##toggle() can toggle between two sets of values. + * @see ##dial() allows smooth transitions between two sets of values. + */ + 'set': function (name, value) { + var self = this; + if (value !== undef) { + var match = /^(\W*)(.*)/.exec(replace(replace(name, /^\$float$/, 'cssFloat'), /^%/,'@data-')); + var prefix = match[1]; + + if (setter[prefix]) + setter[prefix](this, match[2], value); + else if (name == '$$fade') { + this['set']({'$visibility': value ? 'visible' : 'hidden', '$opacity': value}); + } + else if (name == '$$slide') { + self['set']({'$visibility': value ? 'visible' : 'hidden', '$overflow': 'hidden', + '$height': /px/.test(value) ? value : function(oldValue, idx, element) { return getNaturalHeight($(element), value);} + }); + } + else if (name == '$$show') { + if (value) + self['set']({'$visibility': value ? 'visible' : 'hidden', '$display': ''}) // that value? part is only for gzip + ['set']({'$display': function(oldVal) { // set for 2nd time: now we get the stylesheet's $display + return oldVal == 'none' ? 'block' : oldVal; + }}); + else + self['set']({'$display': 'none'}); + } + else if (name == '$$') { + self['set']('@style', value); + } + else + flexiEach(this, function(obj, c) { + var newValue = isFunction(value) ? value($(obj)['get'](name), c, obj) : value; + if (prefix == '$') { + if (match[2]) + obj['style'][match[2]] = newValue; + else { + flexiEach(newValue && newValue.split(/\s+/), function(clzz) { + var cName = replace(clzz, /^[+-]/); + + if (/^\+/.test(clzz)) + obj['classList'].add(cName); + else if (/^-/.test(clzz)) + obj['classList'].remove(cName); + else + obj['classList'].toggle(cName); + }); + } + } + else if (name == '$$scrollX') + obj['scroll'](newValue, $(obj)['get']('$$scrollY')); + else if (name == '$$scrollY') + obj['scroll']($(obj)['get']('$$scrollX'), newValue); + else if (prefix == '@') { + if (newValue == _null) + obj.removeAttribute(match[2]); + else + obj.setAttribute(match[2], newValue); + } + else + obj[match[2]] = newValue; + }); + } + else if (isString(name) || isFunction(name)) + self['set']('$', name); + else + eachObj(name, function(n,v) { self['set'](n, v); }); + return self; + }, + + /*$ + * @id add + * @group ELEMENT + * @requires dollar each + * @configurable default + * @name .add() + * @syntax list.add(text) + * @syntax list.add(node) + * @syntax list.add(list) + * @syntax list.add(factoryFunction) + * @module WEB + * Adds the given node(s) as children to the list's HTML elements. If a string has been given, it will be added as text node. + * DOM nodes will be added directly. If you pass a list, all its elements will be added using the rules above. + * + * When you pass a DOM node and the target list has more than one element, the original node will be added to the first list element, + * and ##clone#clones## to all following list elements. + * + * ##EE(), ##HTML() and ##clone() are compatible with add() and can help you create new HTML nodes. + * + * @example Using the following HTML: + *
+     * <div id="comments">Here is some text.<br/></div>
+     * 
+ * The next line appends a text node to the div: + *
+     * $('#comments').add('Some additional text.');
+     * 
+ * This results in: + *
+     * <div id="comments">Here is some text.<br/>Some additional text.</div>
+     * 
+ * + * @example Using the following HTML: + *
+     * <ul id="myList">
+     *   <li>First list entry</li>
+     *   <li>Second list entry</li>
+     * </ul>
+     * 
+ * The following Javascript adds an element to the list: + *
+     * $('#myList').add(EE('li', 'My extra point');
+     * 
+ * This results in + *
+     * <ul id="myList">
+     *   <li>First list entry</li>
+     *   <li>Second list entry</li>
+     *   <li>My extra point</li>
+     * </ul>
+     * 
+ * + * @example Use a list to add several elements at once: + *
+     * $('#comments').add([
+     *      EE('br'),
+     *     'Some text',
+     *     EE('span', {'className': 'highlight'}, 'Some highlighted text')
+     * ]);
+     * 
+ * + * @example If you need to customize the content, you can write a factory function: + *
+     * $('.chapter').add(function(parent, index) { return EE('h2', 'Chapter number ' + index); });
+     * 
+ * + * @param text a string or number to add as text node + * @param node a DOM node to add to the list. If the list has more than one element, the given node will be added to the first element. + * For all additional elements, the node will be cloned using ##clone(). + * @param list a list containing text and/or nodes. May also contain nested lists with nodes or text.. + * @param factoryFunction a function(listItem, listIndex) that will be invoked for each list element to create the nodes: + *
listItem
The list element that will receive the new children.
+ *
listIndex
The index of the list element that will receive the new children.
+ *
(callback return value)
The node(s) to be added to the list element. + * Can be either a string for a text node, an HTML element or a list containing strings and/or DOM nodes. + * If a function is returned, it will be invoked recursively with the same arguments.
+ * @return the current list + * + * @see ##fill() works like add(), but deletes all children before adding the new nodes. + * @see ##addFront() adds nodes as first child, not as last. + * @see ##addAfter() adds nodes not as children but as siblings. + * @see ##addBefore() also adds nodes not as children but as siblings. + * @see ##replace() replaces existing nodes. + */ + 'add': function (children, addFunction) { + return this['each'](function(e, index) { + var lastAdded; + function appendChildren(c) { + if (isList(c)) + flexiEach(c, appendChildren); + else if (isFunction(c)) + appendChildren(c(e, index)); + else if (c != _null) { // must check null, as 0 is a valid parameter + var n = isNode(c) ? c : document.createTextNode(c); + if (lastAdded) + lastAdded['parentNode']['insertBefore'](n, lastAdded['nextSibling']); + else if (addFunction) + addFunction(n, e, e['parentNode']); + else + e.appendChild(n); + lastAdded = n; + } + } + appendChildren(index &&!isFunction(children) ? clone(children) : children); + }); + }, + + /*$ + * @id on + * @group EVENTS + * @requires dollar each + * @configurable default + * @name .on() + * @syntax list.on(names, eventHandler) + * @syntax list.on(selector, names, eventHandler) + * @syntax list.on(names, customFunc, args) + * @syntax list.on(selector, names, customFunc, args) + * @syntax list.on(names, eventHandler, bubbleSelector) + * @syntax list.on(names, customFunc, args, bubbleSelector) + * @module WEB + * Registers the function as event handler for all items in the list. + * + * By default, Minified cancels event propagation and disables element's default behavior for all elements that have an event handler. + * You can override this, either by prefixing the event name with a '|', or by prefixing them with '?' and returning a true + * in the handler. Both will reinstate the original JavaScript behavior. + * + * Handlers are called with the original event object as first argument, the index of the source element in the + * list as second argument and 'this' set to the source element of the event (e.g. the button that has been clicked). + * + * Instead of the event objects, you can also pass an array of arguments that will be passed instead of event object and index. + * + * Optionally you can specify two a selector strings to qualify only certain events. The first one is a selector + * that allows you to select only specific children of the list elements. This is mostly useful for adding events to DOM trees + * generated using ##HTML() or ##EE(). + * + * The second type of selector is the bubble selector that allows you to receive only events that bubbled up from + * elements matching the selector. The selector is executed in the context of the element you registered on to identify whether the + * original target of the event qualifies. If not, the handler is not called. + * + * Minified always registers event handlers with event bubbling enabled. Event capture is not supported. + * + * Event handlers can be unregistered using #off#$.off(). + * + * @example Adds a handler to all divs which paints the div background color to red when clicked. + *
+     * $('div').on('click', function() {
+     *    this.style.backgroundColor = 'red';    // 'this' contains the element that caused the event
+     * });
+     * 
+ * + * @example Registers a handler to call a method setStatus('running') using an inline function: + *
+     * $('#myButton').on('click', function() {
+     *    setStatus('running');
+     * });
+     * 
+ * The previous example can bere written like this, using on()'s args parameter: + *
+     * $('#myButton').on('click', setStatus, ['running']);
+     * 
+ * + * @example Adds two handlers on an input field. The event names are prefixed with '|' and thus keep their original behavior: + *
+     * $('#myInput').on('|keypress |keydown', function() {
+     *    // do something
+     * });
+     * 
+ * + * @example Adds a click handler that will abort the operation by returning false, unless the user confirms it: + *
+     * $('#myLink').on('?click', function() {
+     *    return window.confirm('Really leave?');
+     * });
+     * 
+ * + * @example Adds a button and registers a click handler for it using a sub-selector. + *
+     * $('#myForm').add(HTML("<li><button>click me</button></li>").on('button', 'click', myClickHandler));
+     * 
+ * + * @example Adds listeners for all clicks on a table's rows using the bubble selector 'tr'. + *
+     * $('#table').on('change', 'tr', function(event, index, selectedIndex) {
+     *    alert("Click on table row number: " + selectedIndex);
+     * }, 'tr');
+     * 
+ * Please note that bubble selectors will even listen to events for + * table rows that have been added after you registered for the events. + * + * @param selector optional a selector string for ##dollar#$()## to register the event only on those children of the list elements that + * match the selector. + * Supports all valid parameters for $() except functions. + * @param names the space-separated names of the events to register for, e.g. 'click'. Case-sensitive. The 'on' prefix in front of + * the name must not used. You can register the handler for more than one event by specifying several + * space-separated event names. If the name is prefixed + * with '|' (pipe), the event will be passed through and the event's default actions will be executed by the browser. + * If the name is prefixed with '?', the event will only be passed through if the handler returns true. + * @param eventHandler the callback function(event, index) to invoke when the event has been triggered: + *
+ *
event
The original DOM event object.
+ *
index
The index of the target object in the ##list#Minified list## .
+ *
this
A ##list#Minified list## containing the target element as only item (same as event.target).
+ *
(callback return value)
The return value will only be used if the event name prefix was '?'. + * Then, a return value false will stop all further processing of the event and disable event bubbling. + * true will keep the event alive.
+ *
+ * @param customFunc a function to be called instead of a regular event handler with the arguments given in args. + * 'this' will be a ##list#Minified list## containing the target element as only item (same element as event.target). + * @param args optional an array of arguments to pass to the custom callback function instead of the event objects. If omitted, the function is + * called as event handler with the event object as argument. + * @param bubbleSelector optional a selector string for ##dollar#$()## to receive only events that bubbled up from an + * element that matches this selector. + * Supports all valid parameters for $() except functions. Analog to ##is(), + * the selector is optimized for the simple patterns '.classname', 'tagname' and 'tagname.classname'. + * @return the list + * @see ##off() allows you to unregister an event handler. + * @see ##onClick() as a shortcut for 'click' events. + * @see ##onOver() to simplify mouseover/mouseout events. + * @see ##onFocus() as convenient way to register for focus events. + * @see ##onChange() to get notified when an input's content changes. + */ + 'on': on, + + /*$ + * @id trigger + * @group EVENTS + * @requires on each + * @configurable default + * @name .trigger() + * @syntax list.trigger(name) + * @syntax list.trigger(name, eventObject) + * @module WEB + * + * Triggers event handlers registered with ##on(). + * Any event that has been previously registered using ##on() can be invoked with trigger(). Please note that + * it will not simulate the default behavior on the elements, such as a form submit when you click on a submit button. Event bubbling + * is supported, thus unless there's an event handler that cancels the event, the event will be triggered on all parent elements. + * + * + * @example Simulates a 'click' event on the button. + *
+     * $('#myButton').trigger('click');
+     * 
+ * + * @param name a single event name to trigger + * @param eventObj optional an object to pass to the event handler, provided the handler does not have custom arguments. + * Anything you pass here will be directly given to event handlers as event object, so you need to know what + * they expect. + * @return the list + * @see ##on() registers events that can be triggered. + */ + 'trigger': function (eventName, eventObj) { + return this['each'](function(element, index) { + var bubbleOn = true, el = element; + while(el && bubbleOn) { + eachObj(el['M'], function(id, f) { + bubbleOn = bubbleOn && f(eventName, eventObj, element); + }); + el = el['parentNode']; + } + }); + } + + /*$ + * @stop + */ + // @cond !trigger dummyTrigger:0 + , + ///#/snippet webListFuncs + ///#snippet extrasListFuncs + + /*$ + * @id ht + * @group ELEMENT + * @requires set template + * @configurable default + * @name .ht() + * @syntax list.ht(templateString, object...) + * @syntax list.ht(templateFunction, object...) + * @syntax list.ht(idSelector, object...) + * @module WEB+UTIL + * Replaces the content of the list elements with the HTML generated using the given template. The template uses + * ##template() syntax and HTML-escaped its output using ##escapeHtml(). + * + * @example When you have a HTML snippet like this: + *
+     * <div id="price"></div>
+     * 
+ * Then you can format the price value like this: + *
+     * var price = 14.9;
+     * $('#price').ht('<b>${{::0.00}}</b>', price);
+     * 
+ * Results in: + *
+     * <div id="price"><b>$14.90</b></div>
+     * 
+ * + * @example Render a list of names: + *
+     * var names = [ {first: 'James', last: 'Sullivan'},
+     *               {first: 'Michael', last: 'Wazowski'} ];
+     * $('#list').ht('<h2>{{listName}}</h2>'+
+     *               '<ul>{{each n: names}}<li>{{n.first}} {{n.last}}</li>{{/each}}</ul>',
+     *               {listName: 'Guys', names: names});
+     * 
+ * The code creates this: + *
+     * <h2>Guys</h2>
+     * <ul><li>James Sullivan<li><li>Michael Wazowski</li></ul>
+     * 
+ * + * @example You can store templates in <script> tags. First you need to create a <script> tag with a type not + * supported by the browser and put your template in there, like this: + *
<script id="myTimeTpl" type="minified-template">The time is {{HH:mm:ss}}.</script>
+ * Then you can specify the tag's id directly to access it: + *
$('#timeDisplay').ht('#myTimeTpl', new Date());
+ * + * @param templateString the template using ##template() syntax. Please note, because this is a template, you should + * avoid creating the template itself dynamically, as compiling templates is expensive and + * Minified will cache only a limited number of templates. Exception: If the template string does not use + * any template functionality (no {{}}), it does not need to be compiled and won't be cached.
+ * The template will use ##escapeHtml() as escape function, so all template substitutions will be HTML-escaped, + * unless you use triple curly-braces. + * @param templateFunction instead of a HTML template, ht() can also use a template function, e.g. one + * created by ##template(). It will be invoked with the object as only argument. + * @param idSelector if you pass an ID CSS selector in the form "#myScript", Minified will recognize this and use the content + * of the specified <script> element as template. This allows you to put your template into + * a <script> tag with a non-JavaScript type (see example). Any string that starts with '#' and does not + * contain any spaces is used as selector. + * @param object optional one or more objects to pass to the template. If object is not set, the template is called with undefined + * as object. If exactly one object is given, it is passed directly to the template. If you specify more than one + * object, they are ##merge#merged##. + * @return the current list + * + * @see ##HTML() creates only the nodes and can be used with ##add() and other methods to add the nodes to the DOM, giving you more flexibility than ht(). + */ + 'ht': function(htmlTemplate, object) { + var o = arguments.length > 2 ? merge(sub(arguments, 1)) : object; + return this['set']('innerHTML', isFunction(htmlTemplate) ? htmlTemplate(o) : + /{{/.test(htmlTemplate) ? formatHtml(htmlTemplate, o) : + /^#\S+$/.test(htmlTemplate) ? formatHtml($$(htmlTemplate)['text'], o) : htmlTemplate); + } + /*$ + * @stop + */ + // @cond !ht dummyHt:0 + ///#/snippet extrasListFuncs + }, M.prototype); + + //// DOLLAR FUNCTIONS //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + copyObj({ + ///#snippet webDollarFuncs + /*$ + * @id request + * @group REQUEST + * @requires + * @configurable default + * @name $.request() + * @syntax $.request(method, url) + * @syntax $.request(method, url, data) + * @syntax $.request(method, url, data, settings) + * @module WEB + * Initiates a HTTP request to the given URL, using XMLHttpRequest. It returns a ##promiseClass#Promise## object that allows you to obtain the result. + * + * @example Invokes a REST web service and parses the resulting document using JSON: + *
+     * $.request('get', 'http://service.example.com/weather', {zipcode: 90210})
+     *    .then(function(txt) {
+     *         var json = $.parseJSON(txt);
+     *         $('#weatherResult').fill('Today's forecast is is: ' + json.today.forecast);
+     *    })
+     *    .error(function(status, statusText, responseText) {
+     *         $('#weatherResult').fill('The weather service was not available.');
+     *    });
+     * 
+ * + * @example Sending a JSON object to a REST web service: + *
+     * var myRequest = {         // create a request object that can be serialized via JSON
+     *      request: 'register',
+     *      entries: [
+     *        {name: 'Joe',
+     *      	    job: 'Plumber'
+     *        }
+     *      ]};
+     *
+     * function failureHandler() {
+     *   $('#registrationResult').fill('Registration failed');
+     * }
+     *
+     * $.request('post', 'http://service.example.com/directory', $.toJSON(myRequest))
+     *     .then(function(txt) {
+     *        if (txt == 'OK')
+     *             $('#registrationResult').fill('Registration succeeded');
+     *        else
+     *              failureHandler();
+     *        })
+     *     .error(failureHandler);
+     * 
+ * + * @example Using HTTP authentication and a custom XMLHttpRequest property. + *
var handler = $.request('get', 'http://service.example.com/userinfo', null, {xhr: {withCredentials: true}, user: 'me', pass: 'secret'});
+ * + * + * @param method the HTTP method, e.g. 'get', 'post' or 'head' (rule of thumb: use 'post' for requests that change data + * on the server, and 'get' to request data). Not case sensitive. + * @param url the server URL to request. May be a relative URL (relative to the document) or an absolute URL. Note that unless you do something + * fancy on the server (keyword to google: Access-Control-Allow-Origin), you can only call URLs on the server your script originates from. + * @param data optional data to send in the request, either as POST body or as URL parameters. It can be either a plain object as map of + * parameters (for all HTTP methods), a string (for all HTTP methods), a DOM document ('post' only) or a FormData object ('post' only). + * If the method is 'post', it will be sent as body, otherwise parameters are appended to the URL. In order to send several parameters with the + * same name, use an array of values in the map. Use null as value for a parameter without value. + * @param settings optional a map of additional parameters. Supports the following properties (all optional): + *
headers
a map of HTTP headers to add to the request. Note that you should use the proper capitalization for the + * header 'Content-Type', if you set it, because otherwise it may be overwritten.
+ *
xhr
a map of properties to set in the XMLHttpRequest object before the request is sent, for example {withCredentials: true}.
+ *
user
username for HTTP authentication, together with the pass parameter
+ *
pass
password for HTTP authentication, together with the user parameter
+ *
+ * @return a ##promiseClass#Promise## containing the request's status. If the request has successfully completed with a HTTP status 2xx, + * the promise's completion handler will be called as function(text, xhr): + *
text
The response sent by the server as text.
+ *
xhr
The XMLHttpRequest used for the request. This allows you to retrieve the response in different + * formats (e.g. responseXml for an XML document), to retrieve headers and more.
+ * The failure handler will be called as function(statusCode, statusText, text): + *
statusCode
The HTTP status (never 200; 0 if no HTTP request took place).
+ *
text
The response's body text, if there was any, or the exception as string if the browser threw one.
+ *
xhr
The XMLHttpRequest used for the request. This allows you to retrieve the response in different + * formats (e.g. responseXml for an XML document), to retrieve headers and more..
+ * The returned promise supports ##stop(). Calling stop() will invoke the XHR's abort() method. + * The underlying XmlHttpRequest can also be obtained from the promise's xhr property. + * + * @see ##values() serializes an HTML form in a format ready to be sent by $.request. + * @see ##$.parseJSON() can be used to parse JSON responses. + * @see ##$.toJSON() can create JSON messages. + * @see ##_.format() can be useful for creating REST-like URLs, if you use JavaScript's built-in escape() function. + */ + 'request': function (method, url, data, settings0) { + var settings = settings0 || {}; + var xhr, callbackCalled = 0, prom = promise(), dataIsMap = data && (data['constructor'] == settings['constructor']); + try { + prom['xhr'] = xhr = new XMLHttpRequest(); + + prom['stop0'] = function() { xhr['abort'](); }; + // @condend + + if (dataIsMap) { // if data is parameter map... + data = collector(eachObj, data, function processParam(paramName, paramValue) { + return collector(flexiEach, paramValue, function(v) { + return encodeURIComponent(paramName) + ((v != _null) ? '=' + encodeURIComponent(v) : ''); + }); + }).join('&'); + } + + if (data != _null && !/post/i.test(method)) { + url += '?' + data; + data = _null; + } + + xhr['open'](method, url, true, settings['user'], settings['pass']); + if (dataIsMap && /post/i.test(method)) + xhr['setRequestHeader']('Content-Type', 'application/x-www-form-urlencoded'); + + eachObj(settings['headers'], function(hdrName, hdrValue) { + xhr['setRequestHeader'](hdrName, hdrValue); + }); + eachObj(settings['xhr'], function(name, value) { + xhr[name] = value; + }); + + xhr['onreadystatechange'] = function() { + if (xhr['readyState'] == 4 && !callbackCalled++) { + if (xhr['status'] >= 200 && xhr['status'] < 300) + prom['fire'](true, [xhr['responseText'], xhr]); + else + prom['fire'](false, [xhr['status'], xhr['responseText'], xhr]); + } + }; + + xhr['send'](data); + } + catch (e) { + if (!callbackCalled) + prom['fire'](false, [0, _null, toString(e)]); + } + + return prom; + }, + + /* + * JSON Module. Uses browser built-ins or json.org implementation if available. Otherwise its own implementation, + * originally based on public domain implementation http://www.JSON.org/json2.js / http://www.JSON.org/js.html. + * Extremely simplified code, made variables local, removed all side-effects (especially new properties for String, Date and Number). + */ + + /*$ + * @id ready + * @group EVENTS + * @requires ready_vars ready_init + * @configurable default + * @name $.ready() + * @syntax $.ready(handler) + * @module WEB + * Registers a handler to be called as soon as the HTML has been fully loaded in the browser. Does not necessarily wait for images and other elements, + * only the main HTML document needs to be complete. On older browsers it is the same as window.onload. + * + * If you call ready() after the page is completed, the handler is scheduled for invocation in the event loop as soon as possible. + * + * A shortcut for ready() is to call ##dollar#$()## with the handler function. It does the same with fewer characters. + * + * @example Registers a handler that sets some text in an element: + *
+     * $.ready(function() {
+     *   $('#someElement').fill('ready() called');
+     * });
+     * 
+ * + * @param handler the function() to be called when the HTML is ready. + * @see ##dollar#$()## calls ready() when invoked with a function, offering a more convenient syntax. + */ + 'ready': ready, + + /*$ + * @id off + * @group EVENTS + * @requires on + * @configurable default + * @name $.off() + * @syntax $.off(handler) + * @module WEB + * Removes the given event handler. The call will be ignored if the given handler has not been registered using ##on(). + * If the handler has been registered for more than one element or event, it will be removed from all instances. + * + * Please note that you can not unregister event handlers registered using ##onOver() or ##onChange(). + * + * @example Adds a handler to an element: + *
+     * function myEventHandler() {
+     *    this.style.backgroundColor = 'red';        // 'this' contains the element that caused the event
+     * }
+     * $('#myElement').on('click', myEventHandler);  // add event handler
+     *
+     * window.setInterval(function() {               // after 5s, remove event handler
+     *    $.off(myEventHandler);
+     * }, 5000);
+     * 
+ * + * @param handler the handler to unregister, as given to ##on(). It must be a handler that has previously been registered using ##on(). + * If the handler is not registered as event handler, the function does nothing. + * + * @see ##on() registers an event handler. + */ + 'off': off + + /*$ + * @stop + */ + // @cond !off dummyOff:null + , + ///#/snippet webDollarFuncs + ///#snippet extrasDollarFuncs + + /*$ + * @id wait + * @group EVENTS + * @configurable default + * @requires promise + * @name $.wait() + * @syntax $.wait() + * @syntax $.wait(durationMs) + * @syntax $.wait(durationMs, args) + * @module WEB+UTIL + * + * Creates a new ##promise#Promise## that will be fulfilled as soon as the specified number of milliseconds have passed. This is mainly useful for animation, + * because it allows you to chain delays into your animation chain. + * + * The operation can be interrupted by calling the promise's ##stop() function. + * + * @example Chained animation using Promise callbacks. The element is first moved to the position 200/0, then to 200/200, waits for 50ms + * and finally moves to 100/100. + *
+     * var div = $('#myMovingDiv').set({$left: '0px', $top: '0px'});
+     * div.animate({$left: '200px', $top: '0px'}, 600, 0)
+     *    .then(function() {
+     *           div.animate({$left: '200px', $top: '200px'}, 800, 0);
+     *    }).then(function() {
+     *    	     return _.wait(50);
+     *    }).then(function() {
+     *           div.animate({$left: '100px', $top: '100px'}, 400);
+     *    });
+     * });
+     * 
+ * + * + * @param durationMs optional the number of milliseconds to wait. If omitted, the promise will be fulfilled as soon as the browser can run it + * from the event loop. + * @param args optional an array or list of arguments to pass to the promise handler + * @return a ##promise#Promise## object that will be fulfilled when the time is over, or fail when the promise's ##stop() has been called. + * The promise argument of a fulfilled promise is the args parameter as given to wait(). The returned promise supports ##stop() + * to interrupt the promise. + */ + 'wait': function(durationMs, args) { + var p = promise(); + var id = setTimeout(function() { + p['fire'](true, args); + }, durationMs); + p['stop0'] = function() { p['fire'](false); clearTimeout(id); }; + return p; + } + + /*$ + * @stop + */ + // @cond !wait dummyWait:0 + + ///#/snippet extrasDollarFuncs + }, $); + + //// UNDERSCORE FUNCTIONS //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + + copyObj({ + ///#snippet utilUnderscoreFuncs + // @condblock each + 'each': each, + // @condend + // @condblock each + 'toObject': toObject, + // @condend + // @condblock find + 'find': find, + // @condend + // @condblock equals + 'equals': equals, + // @condend + + /*$ + * @id copyobj + * @group OBJECT + * @requires + * @configurable default + * @name _.copyObj() + * @syntax _.copyObj(from) + * @syntax _.copyObj(from, to) + * @module UTIL + * Copies every property of the first object into the second object. The properties are copied as shallow-copies. + * + * @example Copying properties: + *
var target = {a:3, c: 3};
+     * _.copyObj({a: 1, b: 2}, target); // target is now {a: 1, b: 2, c: 3}
+ * + * @example Inline property merge: + *
var target = _.copyObj({a: 1, b: 2}, {a:3, c: 3}); // target is now {a: 1, b: 2, c: 3}
+ * + * @example Duplicating an object: + *
var target = _.copyObj({a: 1, b: 2}); // target is now {a: 1, b: 2}
+ * + * @param from the object to copy from + * @param to optional the object to copy to. If not given, a new object will be created. + * @return the object that has been copied to + * + * @see ##_.extend() is very similar to copyObj(), but with a slightly different syntax. + * @see ##_.merge() copies a list of objects into a new object. + */ + 'copyObj': copyObj, + + /*$ + * @id extend + * @group OBJECT + * @requires + * @configurable default + * @name _.extend() + * @syntax _.extend(target, src...) + * @module UTIL + * Copies every property of the source objects into the first object. The source objects are specified using variable arguments. + * There can be more than one. + * The properties are copied as shallow-copies. + * + * Please note: Unlike jQuery, extend does not directly add a function to extend Minified, although + * you can use it to for this. To add a function to ##list#Minified lists##, add a property to + * ##M#MINI.M##. If you want to extend $ or _, just assign the new function(s) as property. + * + * @example Copying properties: + *
var target = {a:3, c: 3};
+     * _.extend(target, {a: 1, b: 2}); // target is now {a: 1, b: 2, c: 3}
+ * + * @example Using several source values: + *
var extend = _.extend({a: 1, b: 2}, {a:3, c: 3}, {d: 5}); // target is now {a: 1, b: 2, c: 3, d: 5}
+ * + * @param target the object to copy to + * @param src the object(s) to copy from. Variable argument, there can be any number of sources. Nulls and undefined + * parameters will be ignored. + * @return the target + * + * @see ##_.copyObj() is very similar to extend(), but with a slightly different and more straightforward syntax. + * @see ##_.merge() copies a list of objects into a new object. + */ + 'extend': function(target) { + return merge(sub(arguments, 1), target); + }, + + /*$ + * @id eachobj + * @group OBJECT + * @requires + * @configurable default + * @name _.eachObj() + * @syntax _.eachObj(obj, callback) + * @syntax _.eachObj(obj, callback, ctx) + * @module UTIL + * Invokes the given function once for each property of the given object. The callback is not invoked for inherited properties. + * + * @example Dumps all properties of an object. + *
+     * var s = '';
+     * _.eachObj({a: 1, b: 5, c: 2}, function(key, value) {
+     *     s += 'key=' + key + ' value=' + value + '\n';
+     * });
+     * 
+ * + * @param obj the object to use + * @param callback The callback function(key, value) to invoke for each property. + *
key
The name of the current property.
+ *
value
The value of the current property.
+ *
this
The given context. If not set, the object itself.
+ * The callback's return value will be ignored. + * @param ctx optional a context to pass to the callback as 'this'. + * @return the object + * + * @see ##_.each() iterates through a list. + */ + 'eachObj': eachObj, + + /*$ + * @id isobject + * @group TYPE + * @requires + * @configurable default + * @name _.isObject() + * @syntax _.isObject(obj) + * @module UTIL + * Checks whether the given reference is an object as defined by typeof. + * + * @param obj the object to test + * @return true if the object is an object, false otherwise. + */ + 'isObject': isObject, + + /*$ + * @id format + * @group FORMAT + * @requires template + * @configurable default + * @name _.format() + * @syntax _.format() + * @syntax _.format(template, object) + * @syntax _.format(template, object, escapeFunction) + * @module UTIL + * Formats an object using a ##template#template##. The template syntax is shared with ##_.template(). The only difference is that + * format() frees you from the extra step of creating the template. In any case, whether you use + * format() or ##_.template(), the template will be cached. Be careful when you create templates dynamically, as + * every template is cached and consumes memory.
+ * If you only want to format a single value, use ##_.formatValue(). + * + * @example Format a name: + *
var s = _.formatHtml("{{first}} {{last}}", {first: 'Tim', last: 'Taylor'});
+ * + * @example Format a list of dates: + *
var s = _.format("{{each}}{{this :: yyyy-MM-dd}}{{/each}}", dateList);
+ * + * @param template The ##template#template## as a string. The template, once created, will be cached. + * @param object the object to format + * @param escapeFunction optional The callback function(inputString) that will be used + * to escape all output: + *
inputString
The string to escape.
+ *
(callback return value)
The escaped string.
+ * If no escapeFunction has been given, the output will not be escaped. + * ##_.escapeHtml() can be used as an escape function for HTML, and ##_.escapeRegExp() for regular expressions. + * JavaScript's built-in escape() function can escape URL components. + * See ##_.htmlFormat() for a version of format() that already includes HTML escaping. + * @return the string created by the template + * + * @see ##_.template() creates a template function, using the same syntax. + * @see ##_.formatHtml() is a variant of format() with HTML-escpaping built it. + * @see ##_.formatValue() formats a single number or date. + * @see ##_.escapeRegExp() can be used by format() to escape regular expressions. + */ + 'format': function(tpl, object, escapeFunction) { + return template(tpl, escapeFunction)(object); + }, + + /*$ + * @id template + * @group FORMAT + * @requires date_constants + * @configurable default + * @name _.template() + * @syntax _.template(template) + * @syntax _.template(template, escapeFunction) + * @module UTIL + * Parses a Handlebars-like template to create a reusable template function. + * + * The syntax of the template uses a syntax that superficially looks like + *
Handlebars. Unlike Handlebars, it is based on raw JavaScript expressions and thus gives you + * complete freedom, but also offers you shortcuts for formatting, iteration and conditionals. + * + * Every template can receive exactly one object as input. If you need more than one value as input, put all required values + * into an object. + * + * Use double curly braces to embed a JavaScript expression and insert its result: + *
{{a}} plus {{b}} is {{a+b}}
+ * + * To use such a template, create it with template() and then execute the resulting function: + *
var myTemplate = _.template('{{a}} plus {{b}} is {{a+b}}');
+     * var result = myTemplate({a: 5, b: 7});
+ * If you pass an object as input, its properties will be mapped using JavaScript's with + * statement and are available as variables throughout the template. + * + * If you have only a simple value to render, you can pass it directly and access it through the pre-defined + * variable obj: + *
var myTemplate = _.template('The result is {{obj}}.');
+     * var result = myTemplate(17);
+ * Alternatively, you could also access the input as this, but be aware that JavaScript wraps simples types + * such as Number and Boolean. this is the default, so you can omit it to get the same result: + *
var myTemplate = _.template('The result is {{ }}.');
+     * var result = myTemplate(17);
+ * + * Minified templates can use ##_.formatValue() formats directly. Just separate them from the expression by + * a double-colon: + *
The price is {{obj::#.00}}.
+ * + * Conditions can be expressed using if and else: + *
Hello {{if visits==0}}New{{else if visits<10}}Returning{{else}}Regular{{/if}} Customer.
+ * You can use any JavaScript expression as condition. + * + * Use each to iterate through a list: + *
var myTemplate = _.template(
+     * 	   '{{each names}}{{this.firstName}} {{this.lastName}}{{/each}}');
+     * var result = myTemplate({names: [{firstName: 'Joe', lastName: 'Jones'},
+     *                                  {firstName: 'Marc', lastName: 'Meyer'}]});
+ * each will iterate through the members of the given object. It + * calls its body for each item and put a reference to the item into this. + * Optionally, you can specify up to two variables to store the value in and + * the zero-based index of the current item: + *
var myTemplate = _.template(
+     * 	   '{{each value, index: names}}{{index}}. {{value.firstName}} {{value.lastName}}{{/each}}');
+     * 
+ * + * If you do not pass an expression to each, it will take the list from this: + *
var myTemplate = _.template('{{each value:}}{{value}};{{/each}}');
+     * var result = myTemplate([1, 2, 3]);
+ * + * Beside lists, you can also iterate through the properties of an object. The property name will be stored + * in the first given parameter and the value in this and the second parameter: + *
var myTemplate = _.template('{{each key, value: nicknames}}{{key}}: {{value}}{{/each}}');
+     * var result = myTemplate({nicknames: {Matt: 'Matthew', John: 'Jonathan'} });
+ * + * Shorter version of the previous example that uses this for the value: + *
var myTemplate = _.template('{{each key: nicknames}}{{key}}: {{this}}{{/each}}');
+ * + * If you do not need the key, you can omit the variable specification: + *
var myTemplate = _.template('{{each nicknames}}{{this}}{{/each}}');
+ * + * You can define your own variables, using the regular JavaScript syntax, with 'var': + *
var myTemplate = _.template('{{var s=very.long.name, sum=a+b;}}{{s.desc}}, {{sum}}');
+ * + * In some situations, it may be inevitable to embed raw JavaScript in the template. + * To embed JavaScript code, prefix the code with a '#': + *
var myTemplate = _.template(
+     *     '{{each}}{{#var sum = 0; for (var i = 0; i < 3; i++) sum += this.numbers[i]; }}{{sum}}{{/each}}');
+     * var result = myTemplate([['Foreword', 'Intro'], ['Something', 'Something else']]);
+ * + * + * By default, all output will be escaped. You can prevent this by using triple-curly-braces: + *
Here's the original: {{{rawText}}}
. + * + * The template's JavaScript code is executed in a sandbox without access to global variables. Minified defines the + * following variables for you: + * + * + * + * + * + * + * + *
NameDesciption
thisThe template object outside of each. Inside eachs, the current value.
objThe parameter given to the template function.
_A reference to Minified Util.
escThe escape function given when the template has been defined. If no function has been given, + * a default function that returns the input unmodified.
printA function(text,...) that appends one or more strings to the template result.
eachA function(listOrObject, eachCallback) that can iterate over lists or object properties. + * The eachCallback is a function(key, value) for objects or function(value, index) + * for arrays that will be invoked for each item. + *
+ * + * Every template you create is already cached, so it not an expensive operation to call ##_.template() a second + * time with the same template. However, because of caching, you should be careful when creating templates + * dynamically, as this will fill the cache up quickly. + * + * @param template The template as a string using the syntax described below. + * @param escapeFunction optional The callback function(inputString) that will be used + * to escape all output: + *
inputString
The string to escape.
+ *
(callback return value)
The escaped string.
+ * If no escapeFunction has been given, the output will not be escaped. + * ##_.escapeHtml() can be used as an escape function for HTML, and ##_.escapeRegExp() for regular expressions. + * JavaScript's built-in escape() function can escape URL components. + * @return the value returned by the last invocation of func + * + * @see ##_.format() shares template()'s syntax but returns the result directly. + * @see ##_.formatHtml() is a variant of format() with HTML escaping. + * @see ##_.escapeHtml() can be used by template() to escape HTML. + * @see ##_.escapeRegExp() can be used by template() to escape regular expressions. + * @see ##HTML() creates a HTML element tree from a template. + */ + 'template': template, + + /*$ + * @id formathtml + * @group FORMAT + * @requires template + * @configurable default + * @name _.formatHtml() + * @syntax _.formatHtml() + * @syntax _.formatHtml(template, object) + * @module UTIL + * Formats an object using a ##template#template## with HTML escaping for the output. + * The template syntax is shared with ##_.template(). Output in double curly braces is automatically escaped using ##_.escapeHtml(). + * formatHtml() just creates a new template with HTML escaping and invokes it immediately. + * The template will be cached. Be careful when you create templates dynamically, as + * every template is cached and consumes memory.
+ * If you only want to format a single value, use ##_.formatValue(). + * + * @example Format a name: + *
var s = _.formatHtml("{{first}} {{last}}", {first: 'Tim', last: 'Taylor'});
+ * + * @example Format a list of dates: + *
var s = _.formatHtml("{{each}}{{::yyyy-MM-dd}}{{/each}}", dateList);
+ * + * @param template The ##template#template## as a string. The template, once created, will be cached. + * @param object the object to format + * @return the string created by the template + * + * @see ##ht() works uses formatHtml to set element's innerHTML. + * @see ##HTML() create HTML nodes using formatHtml. + * @see ##_.template() creates a template function, using the same syntax. + * @see ##_.format() allows you to specify alternative escape mechanisms. + */ + 'formatHtml': formatHtml + /*$ + * @stop + */ + + // @cond !format dummyFormatHtml:0 + , + + ///#/snippet utilUnderscoreFuncs + ///#snippet extrasUnderscoreFuncs + // @condblock promise + 'promise': promise + // @condend promise + + /*$ + * @stop + */ + // @cond !promise dummyPromise:0 + + ///#/snippet extrasUnderscoreFuncs + }, _); + + ////INITIALIZATION //////////////////////////////////////////////////////////////////////////////////////////////////////////////////// + ///#snippet webInit + /*$ + * @id ready_init + * @dependency + */ + document.addEventListener("DOMContentLoaded", function() { + callList(DOMREADY_HANDLER); + DOMREADY_HANDLER = _null; + }, false); + /*$ + @stop + */ + + + ///#/snippet webInit + + return { + ///#snippet extrasExports + + /*$ + * @id html + * @group ELEMENT + * @requires template ht + * @configurable default + * @name HTML() + * @syntax HTML(templateString, object...) + * @syntax HTML(templateFunction, object...) + * @syntax HTML(idSelector, object...) + * @module WEB + * Creates a ##list#list## of HTML nodes from the given HTML template. The list is compatible with ##add(), ##fill() and related methods. + * The template uses the ##template() syntax with ##escapeHtml() escaping for values. + * + * Please note that the function HTML will not be automatically exported by Minified. You should always import it + * using the recommended import statement: + *
+     * var MINI = require('minified'), $ = MINI.$, $$ = MINI.$$, EE = MINI.EE, HTML = MINI.HTML;
+     * 
+ * + * @example Creating a HTML element showing a number: + *
+     * <div id="price">-</div>
+     * 
+ * Then the price can be set like this: + *
+     * var price = 14.9;
+     * $('#price').fill(HTML('<b>${{::0.99}}</b>', price));
+     * 
+ * Results in: + *
+     * <div id="price"><b>$14.90</b></div>
+     * 
+ * + * @example Adding elements to an existing list: + *
+     * var names = [ {first: 'James', last: 'Sullivan'},
+     *               {first: 'Michael', last: 'Wazowski'} ];
+     * $('#list').add(HTML('{{each}}<li>{{this.first}} {{this.last}}</li>{{/each}}', names);
+     * 
+ * The code adds this to #list: + *
+     * <li>James Sullivan<li><li>Michael Wazowski</li>
+     * 
+ * + * @example You can store templates in <script> tags. First you need to create a <script> tag with a type not + * supported by the browser and put your template in there, like this: + *
<script id="myTimeTpl" type="minified-template">The time is {{HH:mm:ss}}.</script>
+ * Then you can specify the tag's id directly to access it: + *
$('#timeDisplay').fill(HTML('#myTimeTpl', new Date()));
+ * + * @param templateString the template using ##template() syntax. Please note, because this is a template, you should + * avoid creating the template itself dynamically, as compiling templates is expensive and + * Minified will cache only a limited number of templates. Exception: If the template string does not use + * any template functionality (no {{}}), it does not need to be compiled and won't be cached. + * The template will use ##escapeHtml() as escape function, so all template substitutions will be HTML-escaped, + * unless you use triple curly-braces. + * @param templateFunction instead of a HTML template HTML() also accepts a template function, e.g. one + * created by ##template(). It will be invoked with the object as only argument. + * @param idSelector if you pass an ID CSS selector in the form "#myScript", Minified will recognize this and use the content + * of the specified <script> element as template. This allows you to put your template into + * a <script> tag with a non-JavaScript type (see example). Any string that starts with '#' and does not + * contain any spaces is used as selector. + * @param object optional one or more objects to pass to the template. If object is not set, the template is called with undefined + * as object. If exactly one object is given, it is passed directly to the template. If you specify more than one + * object, they are ##merge#merged##. + * @return the list containing the new HTML nodes + * + * @see ##ht() is a shortcut for fill(HTML()). + * @see ##EE() is a different way of creating HTML nodes. + */ + 'HTML': function () { + var div = EE('div'); + return _(call(div['ht'], div, arguments)[0].childNodes); + }, + /*$ + * @stop + */ + + ///#/snippet extrasExports + ///#snippet utilExports + /*$ + * @id underscore + * @group LIST + * @name _() + * @syntax _(item...) + * @configurable default + * @module UTIL + * Creates a new Minified list. Supports variable arguments so you can add items directly to the list. For arguments that are lists + * (as defined by ##_.isList()), the list content will be added to the new list. Unlike #dollar#$()#, this is not done recursively + * and thus you can create a list of lists by wrapping arguments in a list. Another difference between _() and $() + * is that $() will automatically remove null values while _() will keep them. + * + * @example Creating an empty list: + *
_()
+ * + * @example Creating a list with three items: + *
_(1, 2, 3)
+ * + * @example Creating the same list, but by passing an array. One array level will be flattened: + *
_([1, 2, 3])
+ * + * @example Creating a list containing the arrays [1, 2] and [3, 4]. + *
_([[1, 2], [3, 4]])
+ * + * @example Merging two lists: + *
var a = _("a", "b", "c");
+     * var b = _("x", "y", "z");
+     * var merged = _(a, b);    // contains _("a", "b", "c", "x", "y", "z")
+     * 
+ * + * @example Adding two elements to a list: + *
var a = _(1, 2, 3);
+     * var a4 = _(a, 4);       // contains _(1, 2, 3, 4)
+     * 
+ * + * @example Mixing different list types and single elements: + *
_(1, [], [2, 3], _(), _(4, 5)); // same content as _(1, 2, 3, 4, 5)
+ * + * @param item an item to add to the new list. If it is a list (as defined by ##_.isList()), its content will be to the new + * ##Minified list#list## (but NOT recursively). + */ + '_': _, + /*$ + * @stop + */ + ///#/snippet utilExports + ///#snippet webExports + + /*$ + * @id dollar + * @group SELECTORS + * @requires + * @dependency yes + * @name $() + * @syntax $() + * @syntax $(selector) + * @syntax $(selector, context) + * @syntax $(selector, context, childOnly) + * @syntax $(list) + * @syntax $(list, context) + * @syntax $(list, context, childOnly) + * @syntax $(object) + * @syntax $(object, context) + * @syntax $(object, context, childOnly) + * @syntax $(domreadyFunction) + * @module WEB + * Creates a new ##list#Minified list##, or register a DOMReady-handler. + * The most common usage is with a CSS-like selector. $() will then create a list containing all elements of the current HTML + * document that fulfill the filter conditions. Alternatively you can also specify a list of objects or a single object. + * Nested lists will automatically be flattened, and nulls will automatically be removed from the resulting list. + * If you call $() without any arguments, it will return an empty list. + * + * Additionally, you can specify a second argument to provide a context. Contexts only make sense if you selected + * HTML nodes with the first parameter. Then the context limits the resulting list to include only those nodes + * that are descendants of the context nodes. The context can be either a selector, a list or a single HTML node, and will be + * processed like the first argument. A third arguments allows you to limit the list to + * only those elements that are direct children of the context nodes (so a child of a child would be filtered out). + * + * The lists created by $() are the same type as the ##list#Minified lists## created by Util's #underscore#_() constructor and other + * Util methods. All Util methods work on lists created by $(). If you want to add your own methods to those lists, + * use ##M#MINI.M##. + * + * As a special shortcut, if you pass a function to $(), it will be registered using #ready#$.ready() to be executed + * when the DOM model is complete. + * + * @example A simple selector to find an element by id. + *
+     * var l0 = $('#myElementId');
+     * 
+ * + * @example You can pass an object reference to create a list containing only this element: + *
+     * var l1 = $(document.getElementById('myElementId'));
+     * 
+ * + * @example Lists and arrays will be copied: + *
+     * var l2 = $([elementA, elementB, elementC]);
+     * 
+ * + * @example Lists will be automatically flattened and nulls removed. So this list l3 has the same content as l2: + *
+     * var l3 = $([elementA, [elementB, null, elementC], null]);
+     * 
+ * + * @example This is a simple selector to find all elements with the given class. + *
+     * var l4 = $('.myClass');
+     * 
+ * + * @example A selector to find all elements of the given type. + *
+     * var l5 = $('input'); // finds all input elements
+     * 
+ * + * @example A selector to find all elements with the given type and class. + *
+     * var l6 = $('input.myRadio'); // finds all input elements with class 'myRadio'
+     * 
+ * + * @example A selector to find all elements that are descendants of the given element. + *
+     * var l7 = $('#myForm input'); // finds all input elements contained in the element myForm
+     * 
+ * + * @example A selector to find all elements that have either a CSS class 'a' or class 'b': + *
+     * var l8 = $('.a, .b'); // finds all elements that have class a or class b
+     * 
+ * + * @example A selector that finds all elements that are descendants of the element myDivision, are inside an element with the + * class .myForm and are input elements: + *
+     * var l9 = $('#myDivision .myForm input');
+     * 
+ * + * @example Contexts can make it easier to specify ancestors: + *
+     * var l10 = $('.myRadio', '#formA, #formB, #formC');
+     * 
+ * The result is identical to: + *
+     * var l10 = $('#formA .myRadio, #formB .myRadio, #formC .myRadio');
+     * 
+ * + * @example Using one of the list functions, ##set(), on the list, and setting the element's text color. '$' at the beginning of the property name sets a CSS value. + *
+     * $('#myElementId').set('$color', 'red');
+     * 
+ * + * @example Most list methods return the list you invoked them on, allowing you to chain them: + *
+     * $('#myForm .myRadio').addClass('uncheckedRadio')
+     *                      .set('checked', true)
+     *                      .on('click', function() {
+     *                             $(this).set({@: 'uncheckedRadio');
+     *                      });
+     * 
+ * + * @example Using $() as a #ready#$.ready() shortcut: + *
+     * $(function() {
+     *   // in here you can safely work with the HTML document
+     * });
+     * 
+ * + * @param selector a simple, CSS-like selector for HTML elements. It supports '#id' (lookup by id), '.class' (lookup by class), + * 'element' (lookup by elements) and 'element.class' (combined class and element). Use commas to combine several selectors. + * You can also join two or more selectors by space to find elements which are descendants of the previous selectors. + * For example, use 'div' to find all div elements, '.header' to find all elements containing a class name called 'header', and + * 'a.popup' for all a elements with the class 'popup'. To find all elements with 'header' or 'footer' class names, + * write '.header, .footer'. To find all divs elements below the element with the id 'main', use '#main div'. + * The selector "*" will return all elements. + * @param list a list to copy. It can be an array, another Minified list, a DOM nodelist or anything else that has a length property and + * allows read access by index. A shallow copy of the list will be returned. Nulls will be automatically removed from the copy. Nested lists + * will be flattened, so the result only contains nodes. + * @param object an object to create a single-element list containing only the object. If the argument is null, an empty list will be returned. + * @param domreadyFunction a function to be registered using #ready#$.ready(). + * @param context optional an optional selector, node or list of nodes which specifies one or more common ancestor nodes for the selection. The context can be specified as + * a selector, a list or using a single object, just like the first argument. + * The returned list will contain only descendants of the context nodes. All others will be filtered out. + * @param childOnly optional if set, only direct children of the context nodes are included in the list. Children of children will be filtered out. If omitted or not + * true, all descendants of the context will be included. + * @return the array-like ##list#Minified list## object containing the content specified by the selector. + * Please note that if the first argument was a list, the existing order will be kept. If the first argument was a simple selector, the nodes are in document order. + * If you combined several selectors using commas, only the individual results of the selectors will keep the document order, + * but will then be joined to form a single list. This list will + * not be in document order anymore, unless you use a build without legacy IE support. + * Duplicate nodes will be removed from selectors, but not from lists. + * + * @see #underscore#_() is Util's alternative constructor for ##list#Minified lists## + * @see ##dollardollar#$$()## works like $(), but returns the resulting list's first element. + */ + '$': $, + + /*$ + * @id dollardollar + * @group SELECTORS + * @requires + * @configurable default + * @name $$() + * @syntax $(selector) + * @syntax $(selector, context) + * @syntax $(selector, context, childOnly) + * @shortcut $$() - It is recommended that you assign MINI.$$ to a variable $$. + * @module WEB + * Returns a DOM object containing the first match of the given selector, or undefined if no match was found. + * $$ allows you to easily access an element directly. It is the equivalent to writing $(selector)[0]. + * + * Please note that the function $$ will not be automatically exported by Minified. You should always import it + * using the recommended import statement: + *
+     * var MINI = require('minified'), $ = MINI.$, $$ = MINI.$$, EE = MINI.EE;
+     * 
+ * + * @example Select the checkbox 'myCheckbox': + *
+     * $$('#myCheckbox').checked = true;
+     * 
+ * + * @param selector a simple, CSS-like selector for the element. Uses the same syntax as #dollar#$(). The most common + * parameter for this function is the id selector with the syntax "#id". + * @param context optional an optional selector, node or list of nodes which specifies one or more common ancestor nodes for the selection. The context can be specified as + * a selector, a list or using a single object, just like the first argument. + * The returned list will contain only descendants of the context nodes. All others will be filtered out. + * @param childOnly optional if set, only direct children of the context nodes are included in the list. Children of children will be filtered out. If omitted or not + * true, all descendants of the context will be included. + * @return a DOM object of the first match, or undefined if the selector did not return at least one match + * + * @see ##dollar#$()## creates a list using the selector, instead of returning only the first result. + */ + '$$': $$, + + /*$ + * @id M + * @name M + * @syntax MINI.M + * @module WEB, UTIL + * + * Exposes the internal class used by all ##list#Minified lists##. This is mainly intended to allow you adding your + * own functions. + * + * @example Adding a function printLength() to M: + *
+     * MINI.M.prototype.printLength = function() { console.log(this.length); };
+     * 
+ */ + 'M': M, + + /*$ + * @id getter + * @requires get + * @name MINI.getter + * @syntax MINI.getter + * @module WEB + * + * Exposes a map of prefix handlers used by ##get(). You can add support for a new prefix in get() + * by adding a function to this map. The prefix can be any string consisting solely of non-alphanumeric characters + * that's not already used by Minified. + * + * You must not replace getters by a new map, but must always modify the existing map. + * + * The function's signature is function(list, name) where + *
list
Is the Minified list to get the value from. By convention you should always use only the first element. The list is + * non-empty and the first elememt can't be null or undefined (get() automatically returns undefined in + * all other case).
+ *
name
The name of the property. That's the part AFTER the prefix.
+ *
(callback return value)
The value to return to the user.
+ * + * @example Adding a shortcut '||' for accessing border style properties: + *
+     * MINI.getter['||'] = function(list, name) {
+     * 	return list.get('$border' + name.replace(/^[a-z]/, function(a) { return a.toUpperCase()});
+     * };
+     *
+     * var borderColor = $('#box').get('||color'); // same as '$borderColor'
+     * var borderLeftRadius = $('#box').get('||leftRadius'); // same as '$borderLeftRadius'
+     * 
+ * + * @example Adding XLink attribute support to get(). This is useful if you work with SVG. The prefix is '>'. + *
+     * MINI.getter['>'] = function(list, name) {
+     * 	return list[0].getAttributeNS('http://www.w3.org/1999/xlink', name);
+     * };
+     *
+     * var xlinkHref = $('#svgLink').get('>href');
+     * 
+ */ + 'getter': getter, + + /*$ + * @id setter + * @requires set + * @name MINI.setter + * @syntax MINI.setter + * @module WEB + * + * Exposes a map of prefix handlers used by ##set(). You can add support for a new prefix in set() + * by adding a function to this map. The prefix can be any string consisting solely of non-alphanumeric characters + * that's not already used by Minified. + * + * You must not replace setters by a new map, but must always modify the existing map. + * + * The function's signature is function(list, name, value) where + *
list
Is the Minified list to use.
+ *
name
The name of the property. That's the part AFTER the prefix.
+ *
value
Either the value to set, or a callback function to create the value that you must call for each + * value (see ##set() ).
+ *
+ * + * If you provide complete ##get() and ##set() support for a prefix, you are also able to use it in other Minified + * function such as ##animate() and ##toggle(). + * + * @example Adding a shortcut '||' for accessing border style properties. As it's just calling ##set() for an existing + * property, it is not required to extra code for the callback. + *
+     * MINI.setter['||'] = function(list, name, value) {
+     * 	list.set('$border' + name.replace(/^[a-z]/, function(a) { return a.toUpperCase()}, value);
+     * };
+     *
+     * $('#box').set('||color', 'red');   // same as set('$borderColor', 'red')
+     * $('#box').set('||leftRadius', 4);  // same as set('$borderLeftRadius', 4)
+     * 
+ * + * @example Adding XLink attribute support to set(). This is useful if you work with SVG. The prefix is '>'. + *
+     * MINI.setter['>'] = function(list, name, value) {
+     * 	list.each(function(obj, index) {
+     * 		var v;
+     * 		if (_.isFunction(value))
+     * 			v = value(obj.getAttributeNS('http://www.w3.org/1999/xlink', name), index, obj);
+     * 		else
+     * 			v = value;
+     *
+     *		if (v == null)
+     *			obj.removeAttributeNS('http://www.w3.org/1999/xlink', name);
+     *		else
+     *			obj.setAttributeNS('http://www.w3.org/1999/xlink', name, v);
+     *	});
+     * };
+     *
+     * $('#svgLink').set('>href', 'http://minifiedjs.com/');
+     * 
+ */ + 'setter': setter + /*$ + * @stop + */ + ///#/snippet webExports + }; + + ///#snippet commonAmdEnd +}); +///#/snippet commonAmdEnd +///#snippet webDocs + +/*$ + * @id list + * @name Minified Lists + * @module WEB, UTIL + * + * Minified lists are Array-like objects provided by Minified. Like a regular JavaScript array, + * they provide a length property and you can access their content using the index operator (a[5]). + * However, they do not provide the same methods as JavaScript's native array and are designed to be immutable, so + * there is no direct way to add something to a Minified list. Instead Minified provides a number of functions and methods + * that take a list and create a modified copy which, for example, may contain additional elements. + * + * Minified lists are typically created either using the Web module's #dollar#$() function or with the Util module's + * #underscore#_() function, but many functions in the Util module also return a Minified list. + * + * The Util module provides a function ##_.array() that converts a Minified list to a regular JavaScript array. + */ + +/*$ + * @id promiseClass + * @name Promise + * @module WEB, UTIL + * + * Promises are objects that represent the future result of an asynchronous operation. When you start such an operation, using #request#$.request(), + * ##animate(), or ##wait(), you will get a Promise object that allows you to get the result as soon as the operation is finished. + * + * Minified's full distribution ships with a Promises/A+-compliant implementation of Promises that should + * be able to interoperate with most other Promises implementations. Minified's Web module in stand-alone distribution comes with a limited implementation. + * See below for details. + * + * What may be somewhat surprising about this Promises specification is that the only standard-compliant way to access the result is to + * register callbacks. They will be invoked as soon as the operation is finished. + * If the operation already ended when you register the callbacks, the callback will then just be called from the event loop as soon + * as possible (but never while the ##then() you register them with is still running).
+ * This design forces you to handle the operation result asynchronously and disencourages 'bad' techniques such as polling. + * + * The central method of a Promise, and indeed the only required function in Promises/A+, is ##then(). It allows you to register + * two callback methods, one for success (called 'fulfillment' in Promises/A+ terminology) and one for failures (called 'rejection' in Promises/A+). + * + * This example shows you how to use then(): + *
+ * $.request('get', 'http://example.com/weather?zip=90210')
+ *  .then(function success(result) {
+ *      alert('The weather is ' + result);
+ *  }, function error(exception) {
+ *  	alert('Something went wrong');
+ *  });
+ * 
+ * + * What makes Promises so special is that ##then() itself returns a new Promise, which is based on the Promise then() was called on, but can be + * modified by the outcome of callbacks. Both arguments to then() are optional, and you can also write the code like this: + *
+ * $.request('get', 'http://example.com/weather?zip=90210')
+ *  .then(function success(result) {
+ *      alert('The weather is ' + result);
+ *  })
+ *  .then(null, function error(exception) {
+ *  	alert('Something went wrong');
+ *  });
+ * 
+ * + * Because the first ##then() returns a new Promise based on the original Promise, the second then() will handle errors of the request just like + * the first one did. There is only one subtle difference in the second example: the error handler will not only be called if the request failed, + * but also when the request succeded but the success handler threw an exception. That's one of the two differences between the original Promise and + * the Promise returned by then(). Any exception thrown in a callback causes the new Promise to be in error state. + * + * Before I show you the second difference between the original Promise and the new Promise, let me make the example a bit more readable + * by using ##error(), which is not part of Promises/A+, but a simple extension by Minified. It just registers the failure callback without + * forcing you to specify null as first argument: + *
+ * $.request('get', 'http://example.com/weather?zip=90210')
+ *  .then(function success(result) {
+ *      alert('The weather is ' + result);
+ *  })
+ *  .error(function error(exception) {  // error(callback) is equivalent to then(null, callback)
+ *  	alert('Something went wrong');
+ *  });
+ * 
+ * + * A very powerful capability of Promises is that you can easily chain them. If a ##then() callback returns a value, the new Promise returned + * by then() will be marked as success (fulfilled) and this value is the result of the operation. If a callback returns a Promise, + * the new Promise will assume the state of the returned Promise. You can use the latter to create chains of asynchronous operations, + * but you still need only a single error handler for all of them and you do not need to nest functions to achieve this: + *
+ * $.request('get', 'http://example.com/zipcode?location=Beverly+Hills,+CA')
+ *  .then(function(resultZip) {
+ *      return $.request('get', 'http://example.com/weather', {zip: resultZip});
+ *  })
+ *  .then(function(resultWeather) {
+ *      alert('The weather in Beverly Hills is ' + resultWeather);
+ *  })
+ *  .error(function(exception) {
+ *  	alert('Something went wrong');
+ *  });
+ * 
+ * + * Only the full Minified distribution allows you to create promises yourself, using the ##promise() function. The Promises/A+ + * specification does not specify how to fulfill a promise, but in Minified's implementation every Promise object has a function fire() + * that needs to be called when the promise result is ready. It requires two arguments. + * The first is a boolean, true for a successful operation and false for a failure. The second is an array or list containing the + * arguments to call the corresponding ##then() handler with. + * + * The following example is a function, similar to ##wait(), that returns a Promise which succeeds after the given amount + * of milliseconds has passed. + * It then fulfills the promise with the number of milliseconds as argument. + * + *
+ * function timeout(durationMs) {
+ *		var p = _.promise();
+ *		setTimeout(function() { p.fire(true, [durationMs]); }, durationMs);
+ *		return p;
+ * }
+ * 
+ * Call it like this: + *
+ * timeout(1000).then(function(ms) { window.alert(ms+ ' milliseconds have passed.'); });
+ * 
+ * + *

Limited Promises Implementation in Web module

+ * If you use only the Web module, instead of the full implementation, the promises implementation is not fully Promises/A+ compliant. + * One major difference is that it does not allow you create promises yourself. The only way to get a promise in the Web module + * is from functions like ##animate() and ##request(). The other difference is that the interoperability with other promises frameworks + * is limited, even though it should be good enough most of the time. + * + * There are two things you may run into when you use Web's simplified implementation with a complete implementation: + *
  1. The simplified implementation does not support recursive thenables. So when you register callbacks with ##then(), + * you can return a promise or a thenable, but only if that promise is not also returning a promise.
  2. + *
  3. Many corner cases required by the Promises/A+ specification are not handled. When interoperating using + * reasonable implementations, you may never run into this, but Promises/A+ has detailed rules for things like ##then() + * methods implemented as dynamic getter and returning a new value on each invocation or throwing exceptions. If you need + * a water-proof implementation, you need to use the complete implementation in Minified's full package.
+ */ +/*$ + * @stop + */ + +///#/snippet webDocs + diff --git a/node_modules/pebble-clay/src/styles/_clay.scss b/node_modules/pebble-clay/src/styles/_clay.scss new file mode 100755 index 0000000..400f93c --- /dev/null +++ b/node_modules/pebble-clay/src/styles/_clay.scss @@ -0,0 +1,2 @@ +@import "clay/vars"; +@import "clay/mixins"; diff --git a/node_modules/pebble-clay/src/styles/clay/_base.scss b/node_modules/pebble-clay/src/styles/clay/_base.scss new file mode 100644 index 0000000..9e76657 --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/_base.scss @@ -0,0 +1,201 @@ +@import "bourbon"; +@import "clay"; + +html, body { + @include font-pfdin(regular); + -webkit-font-smoothing: antialiased; + font-size: $em-base; + line-height: $base-line-height; + height:100%; + color: $color-white; + + &.platform-ios { + font-size: 16px; + } + +} + +body { + background-color: $color-gray-2; + padding: 0 $item-spacing-h $item-spacing-v; +} + +em { + font-style: italic; +} + +strong { + @include font-pfdin(medium); + color: $color-orange; +} + +a { + color: $color-gray-8; + + &:hover { + color: inherit; + } +} + +h1, h2, h3, h4 { + text-transform: uppercase; + @include font-pfdin(medium); + @include pfdin-uppercase(); +} + +h1 { + @include font-size(2); +} + +h2 { + @include font-size(1.8); +} + +h3 { + @include font-size(1.5); +} + +h4 { + @include font-size(1.2); +} + +h5 { + @include font-size(1); +} + +h6 { + @include font-size(0.8); +} + +input { + font-family: inherit; + font-size: inherit; + line-height: inherit; +} + +label { + display: flex; + justify-content: space-between; + align-items: center; + padding: $item-spacing-v $item-spacing-h; + + .input { + white-space: nowrap; + display: flex; + max-width: 50%; + margin-left: $item-spacing-h; + } + + &.invalid .input:after { + $size: 1.1rem; + + content: "!"; + display: inline-block; + color: $color-white; + background: $color-orange; + border-radius: $size / 2; + width: $size; + text-align: center; + height: $size; + font-size: $size * 0.75; + vertical-align: middle; + line-height: $size; + box-shadow: $box-shadow-small-components; + @include font-pfdin(medium); + flex: 0 0 $size; + margin-left: 0.3rem; + } +} + +.hide { + display: none !important; +} + +.tap-highlight { + @include tap-highlight(); + border-radius: $border-radius; +} + +.component { + padding-top: $item-spacing-v; + + &.disabled { + pointer-events: none; + > * { + opacity: 0.25; + } + } +} + +.section { + background: $color-gray-4; + border-radius: 0.25rem; + box-shadow: $color-gray-1 0 0.15rem 0.25rem; + + > .component { + padding-bottom: $item-spacing-v; + padding-right: $item-spacing-h; + padding-left: $item-spacing-h; + position: relative; + margin-top: 1rem; + + &:not(.hide) ~ .component { + margin-top: 0; + } + + &:first-child:after { + display: none; + } + + &:after { + content: ""; + background: $color-gray-6; + display: block; + position: absolute; + top: 0; + left: $item-spacing-h / 2; + right: $item-spacing-h / 2; + height: 1px; + pointer-events: none; + } + } + + // don't show the separator for the first non-hidden item + > .component:not(.hide):after { + display: none; + } + + > .component:not(.hide) ~ .component:not(.hide):after { + display: block; + } + + > .component-heading:first-child { + background: $color-gray-3; + border-radius: $border-radius $border-radius 0 0; + + // don't show the separator for the first non-hidden item after the heading + &:after, + ~ .component:not(.hide):after { + display: none; + } + + ~ .component:not(.hide) ~ .component:not(.hide):after { + display: block; + } + } + +} + +.description { + padding: 0 $item-spacing-h $item-spacing-v ; + @include font-size(0.9); + color: $color-gray-9; + text-align: left; +} + +.inputs { + display: block; + width:100%; + border-collapse: collapse; +} + diff --git a/node_modules/pebble-clay/src/styles/clay/_fonts.scss b/node_modules/pebble-clay/src/styles/clay/_fonts.scss new file mode 100644 index 0000000..673393e --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/_fonts.scss @@ -0,0 +1,6 @@ +@font-face { + font-family: PFDinDisplayProRegularWebfont; + src: url(data:application/font-woff;charset=utf-8;base64,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) format('woff'); + font-weight: normal; + font-style: normal; +} diff --git a/node_modules/pebble-clay/src/styles/clay/_mixins.scss b/node_modules/pebble-clay/src/styles/clay/_mixins.scss new file mode 100644 index 0000000..c1c0e6f --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/_mixins.scss @@ -0,0 +1,43 @@ +@import "vars"; + +@mixin font-pfdin($weight: normal) { + + font-weight: normal; + + @if $weight == normal or $weight == regular { + font-family: 'PFDinDisplayPro-Regular', PFDinDisplayProRegularWebfont, sans-serif; + } + + @if $weight == medium { + // iOS includes the PFDinDisplayPro-Medium in the webview but Android does not, + // so we apply a faux bold on the regular weight. + font-family: 'PFDinDisplayPro-Medium', PFDinDisplayProRegularWebfont, sans-serif; + + .platform-android & { + font-family: PFDinDisplayProRegularWebfont, sans-serif; + font-weight: bold; + letter-spacing: 0.025em; + } + } +} + +@mixin pfdin-uppercase { + text-transform: uppercase; + position: relative; + top: 0.05rem; + line-height: 0.9; +} + +@mixin font-size($size: 1) { + $font-size: $size * 1rem; + font-size:$font-size; + line-height: $base-line-height * ceil($font-size / $base-line-height); +} + +@mixin tap-highlight($color: rgba(255, 255, 255, 0.1)) { + -webkit-tap-highlight-color: $color; + + &:active { + background-color: $color; + } +} diff --git a/node_modules/pebble-clay/src/styles/clay/_reset.scss b/node_modules/pebble-clay/src/styles/clay/_reset.scss new file mode 100644 index 0000000..680694b --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/_reset.scss @@ -0,0 +1,58 @@ +/* http://meyerweb.com/eric/tools/css/reset/ + v2.0b1 | 201101 + NOTE: WORK IN PROGRESS + USE WITH CAUTION AND TEST WITH ABANDON */ + +html, body, div, span, applet, object, iframe, +h1, h2, h3, h4, h5, h6, p, blockquote, pre, +a, abbr, acronym, address, big, cite, code, +del, dfn, em, img, ins, kbd, q, s, samp, +small, strike, strong, sub, sup, tt, var, +b, u, i, center, +dl, dt, dd, ol, ul, li, +fieldset, form, label, legend, +table, caption, tbody, tfoot, thead, tr, th, td, +article, aside, canvas, details, figcaption, figure, +footer, header, hgroup, menu, nav, section, summary, +time, mark, audio, video { + margin: 0; + padding: 0; + border: 0; + outline: 0; + font-size: 100%; + font: inherit; + vertical-align: baseline; +} + +input, textarea, button { + outline: none; +} + +/* HTML5 display-role reset for older browsers */ +article, aside, details, figcaption, figure, +footer, header, hgroup, menu, nav, section { + display: block; +} + +body { + line-height: 1; +} + +ol, ul { + list-style: none; +} + +blockquote:before, blockquote:after, +q:before, q:after { + content: ''; + content: none; +} + +/* apply a natural box layout model to all elements, but allowing components to change */ +html { + box-sizing: border-box; +} + +*, *:before, *:after { + box-sizing: inherit; +} diff --git a/node_modules/pebble-clay/src/styles/clay/_vars.scss b/node_modules/pebble-clay/src/styles/clay/_vars.scss new file mode 100644 index 0000000..a6a11ed --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/_vars.scss @@ -0,0 +1,39 @@ +@import "bourbon"; + +// Bourbon settings +$em-base: 17px; + +// custom vars +$base-line-height: 1.4; +$base-height: rem($base-line-height * $em-base); +$border-radius: 0.25rem; +$item-spacing-v: $base-height / 2; +$item-spacing-h: 0.75rem; +$small-component-width: $base-height * $golden; + +// ---- Colors ------ +$color-orange: #ff4700; +$color-orange-dark: #993d19; +$color-red: #ff0000; +$color-white: #ffffff; +$color-black: #000000; + +// background color +$color-gray-0: #111111; +$color-gray-1: #2f2f2f; +$color-gray-2: #333333; +$color-gray-3: #414141; +$color-gray-4: #484848; // divider +$color-gray-5: #5b5b5b; +$color-gray-6: #666666; +$color-gray-7: #767676; +$color-gray-8: #858585; +$color-gray-9: #a4a4a4 ; +$color-gray-10: #ececec; +$color-gray-11: #f2f2f2; + +$button-padding: 0.6rem; +$button-padding-ios: 0.5rem; + +$box-shadow-small-components: 0 0.1rem 0.1rem $color-gray-1; +$box-shadow-large-components: 0 rem(3px) rem(15px) rgba(0,0,0,0.4); diff --git a/node_modules/pebble-clay/src/styles/clay/components/button.scss b/node_modules/pebble-clay/src/styles/clay/components/button.scss new file mode 100644 index 0000000..b6c6e13 --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/components/button.scss @@ -0,0 +1,12 @@ +.component-button { + text-align: center; + + .section & { + padding-bottom: 0; + } + + .description { + padding-left: 0; + padding-right: 0; + } +} diff --git a/node_modules/pebble-clay/src/styles/clay/components/checkboxgroup.scss b/node_modules/pebble-clay/src/styles/clay/components/checkboxgroup.scss new file mode 100644 index 0000000..651a470 --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/components/checkboxgroup.scss @@ -0,0 +1,70 @@ +@import "bourbon"; +@import "clay"; + +.component-checkbox { + + display: block; + + .section & { + padding-right: $item-spacing-h / 2; + } + + > .label { + display: block; + padding-bottom: $item-spacing-v / 2; + } + + .checkbox-group { + + padding-bottom: $item-spacing-v / 2; + + label { + padding: $item-spacing-v / 2 $item-spacing-h / 2; + } + + .label { + font-size: 0.9em; + } + + input { + opacity: 0; + position: absolute; + } + + i { + display: block; + position: relative; + border-radius: $border-radius; + width: $base-height; + height: $base-height; + border: rem(2px) solid $color-gray-7; + flex-shrink: 0; + } + + input:checked + i { + border-color: $color-orange; + background: $color-orange; + + &:after { + content: ''; + box-sizing: border-box; + transform: rotate(45deg); + position: absolute; + left: 0.35rem; + top: -0.05rem; + display: block; + width: 0.5rem; + height: 1rem; + border: 0 solid $color-white; + border-right-width: rem(2px); + border-bottom-width: rem(2px); + + } + } + } + + .description { + padding-left: 0; + padding-right: 0; + } +} diff --git a/node_modules/pebble-clay/src/styles/clay/components/color.scss b/node_modules/pebble-clay/src/styles/clay/components/color.scss new file mode 100644 index 0000000..c1b680c --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/components/color.scss @@ -0,0 +1,98 @@ +@import "clay"; + +.component-color { + + .section & { + padding: 0; + } + + .value { + width: $small-component-width; + height: $base-height; + border-radius: $base-height / 2; + box-shadow: $box-shadow-small-components; + display: block; + background: #000; + } + + .picker-wrap { + left: 0; + top: 0; + right: 0; + bottom: 0; + position: fixed; + padding: $item-spacing-v $item-spacing-h /2; + background: rgba(0, 0, 0, 0.65); + opacity: 0; + transition: opacity 100ms ease-in 175ms; + pointer-events: none; + z-index: 100; + display: flex; + flex-direction: column; + justify-content: center; + align-items: center; + + .picker { + padding: $item-spacing-v $item-spacing-h; + background: $color-gray-4; + box-shadow: $box-shadow-large-components; + border-radius: $border-radius; + width: 100%; + max-width: 26rem; + overflow: auto; + } + + &.show { + transition-delay: 0ms; + pointer-events: auto; + opacity: 1; + } + } + + .color-box-wrap { + box-sizing: border-box; + position: relative; + height: 0; + width: 100%; + padding: 0 0 100% 0; // overridden with inline style + + .color-box-container { + position: absolute; + height: 99.97%; + width: 100%; + left: 0; + top: 0; + + .color-box { + float:left; + cursor: pointer; + -webkit-tap-highlight-color: rgba(0,0,0,0); + + &.rounded-tl { + border-top-left-radius: $border-radius; + } + + &.rounded-tr { + border-top-right-radius: $border-radius; + } + + &.rounded-bl { + border-bottom-left-radius: $border-radius; + } + + &.rounded-br { + border-bottom-right-radius: $border-radius; + } + + &.selected { + transform: scale(1.1); + border-radius: $border-radius; + box-shadow: #111 0 0 0.24rem; + position: relative; + z-index: 100; + } + } + + } + } +} diff --git a/node_modules/pebble-clay/src/styles/clay/components/input.scss b/node_modules/pebble-clay/src/styles/clay/components/input.scss new file mode 100644 index 0000000..5a252bf --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/components/input.scss @@ -0,0 +1,49 @@ +@import "clay"; + +.component-input { + + .section & { + padding: 0; + } + + label { + display: block; + } + + .label { + padding-bottom: $item-spacing-v; + } + + .input { + position: relative; + min-width: 100%; + margin-top: $item-spacing-v; + margin-left: 0; + } + + input { + display:block; + width: 100%; + background: $color-gray-2; + border-radius: $border-radius; + padding: $item-spacing-v / 2 $item-spacing-h / 2; + border: none; + vertical-align: baseline; + color: $color-white; + font-size: inherit; + appearance: none; + min-height: $item-spacing-v + ($base-height + 0rem) ; + + @include placeholder { + color: $color-gray-8; + } + + &:focus { + @include placeholder { + color: $color-gray-6; + } + border: none; + box-shadow: none ; + } + } +} diff --git a/node_modules/pebble-clay/src/styles/clay/components/radiogroup.scss b/node_modules/pebble-clay/src/styles/clay/components/radiogroup.scss new file mode 100644 index 0000000..a1578e0 --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/components/radiogroup.scss @@ -0,0 +1,68 @@ +@import "bourbon"; +@import "clay"; + +.component-radio { + + display: block; + + .section & { + padding-right: $item-spacing-h / 2; + } + + > .label { + display: block; + padding-bottom: $item-spacing-v / 2; + } + + .radio-group { + + padding-bottom: $item-spacing-v / 2; + + label { + padding: $item-spacing-v / 2 $item-spacing-h / 2; + } + + .label { + font-size: 0.9em; + } + + input { + opacity: 0; + position: absolute; + } + + i { + display: block; + position: relative; + border-radius: $base-height; + width: $base-height; + height: $base-height; + border: 2px solid $color-gray-7; + flex-shrink: 0; + } + + input:checked + i { + border-color: $color-orange; + + &:after { + $padding: 15%; + content: ''; + display: block; + position: absolute; + left: $padding; + right: $padding; + top: $padding; + bottom: $padding; + border-radius: $base-height; + background: $color-orange; + + } + } + + } + + .description { + padding-left: 0; + padding-right: 0; + } +} diff --git a/node_modules/pebble-clay/src/styles/clay/components/select.scss b/node_modules/pebble-clay/src/styles/clay/components/select.scss new file mode 100644 index 0000000..c369451 --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/components/select.scss @@ -0,0 +1,43 @@ +@import "bourbon"; +@import "clay"; + +.component-select { + + .section & { + padding: 0; + } + + label { + position: relative; + } + + .value { + $triangle-size: 0.85rem; + position: relative; + padding-right: $triangle-size + 0.25rem; + display: block; + + &:after { + content: ""; + position: absolute; + right: 0; + top: 50%; + margin-top: -0.1rem; + @include triangle($triangle-size, $color-orange, down); + } + } + + select { + opacity: 0; + position: absolute; + display: block; + left: 0; + right: 0; + top: 0; + bottom: 0; + width: 100%; + border: none; + margin:0; + padding:0; + } +} diff --git a/node_modules/pebble-clay/src/styles/clay/components/slider.scss b/node_modules/pebble-clay/src/styles/clay/components/slider.scss new file mode 100644 index 0000000..3ca800b --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/components/slider.scss @@ -0,0 +1,127 @@ +@import "clay"; + +.component-slider { + + .section & { + padding: 0; + } + + label { + display: block; + } + + .label-container { + display: flex; + align-items: center; + width: 100%; + padding-bottom: $item-spacing-v; + } + + .label { + flex: 1; + min-width: 1rem; + display: block; + padding-right: $item-spacing-h; + } + + .value-wrap { + display: block; + position: relative; + } + + .value, + .value-pad { + display: block; + background: $color-gray-2; + border-radius: $border-radius; + padding: $item-spacing-v / 2 $item-spacing-h / 2; + border: none; + vertical-align: baseline; + color: $color-white; + text-align: right; + margin: 0; + min-width: 1rem; + } + + .value-pad { + visibility: hidden; + + &:before { + content: ' '; + display: inline-block; + } + } + + .value { + max-width: 100%; + position: absolute; + left:0; + top:0 + } + + .input-wrap { + padding: 0 $item-spacing-h $item-spacing-v; + } + + .input { + $track-height: rem(3px); + + display: block; + position: relative; + min-width: 100%; + height: $base-height; + overflow: hidden; + margin-left: 0; + + &:before { + content: ''; + display: block; + position: absolute; + height: $track-height; + background: $color-gray-6; + width: 100%; + top: $base-height / 2 - $track-height / 2; + } + + .slider { + display: block; + width: 100%; + appearance: none; + position: relative; + height: $base-height; + margin: 0; + background-color: transparent; + + &:focus { + outline: none; + } + + &::-webkit-slider-runnable-track { + border: none; + height: $base-height; + width: 100%; + background-color: transparent; + } + + &::-webkit-slider-thumb { + appearance: none; + position: relative; + height: $base-height; + width: $base-height; + background-color: $color-orange; + border-radius: 50%; + + &:before { + content: ""; + position: absolute; + left: -1000px; + top: $base-height / 2 - $track-height / 2; + height: $track-height; + width: 1001px; + background: $color-orange; + } + } + } + } + +} diff --git a/node_modules/pebble-clay/src/styles/clay/components/submit.scss b/node_modules/pebble-clay/src/styles/clay/components/submit.scss new file mode 100644 index 0000000..74b11f4 --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/components/submit.scss @@ -0,0 +1,3 @@ +.component-submit { + text-align: center; +} diff --git a/node_modules/pebble-clay/src/styles/clay/components/toggle.scss b/node_modules/pebble-clay/src/styles/clay/components/toggle.scss new file mode 100644 index 0000000..92c9e68 --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/components/toggle.scss @@ -0,0 +1,55 @@ +@import "clay"; + +.component-toggle { + + $slide-height: $base-height * 0.75; + $slide-width: $small-component-width; + $marker-diameter: $base-height; + + .section & { + padding: 0; + } + + input { + display: none; // @todo make sure this doesn't mess up interactivity on iOS + } + + .graphic { + display: inline-block; + position: relative; + + .slide { + display:block; + border-radius: $slide-height; + height: $slide-height; + width: $slide-width; + background: $color-gray-1; + transition: background-color 150ms linear; + } + + .marker { + background: $color-gray-10; + width: $marker-diameter; + height: $marker-diameter; + border-radius: $marker-diameter; + position: absolute; + left: 0; + display: block; + top: ($marker-diameter - $slide-height) / 2 * -1; + transition: transform 150ms linear; + box-shadow: $box-shadow-small-components; + } + } + + input:checked + .graphic { + .slide { + background: $color-orange-dark; + } + + .marker { + background: $color-orange; + transform: translateX($slide-width - $marker-diameter); + } + } + +} diff --git a/node_modules/pebble-clay/src/styles/clay/elements/_button.scss b/node_modules/pebble-clay/src/styles/clay/elements/_button.scss new file mode 100644 index 0000000..794ff54 --- /dev/null +++ b/node_modules/pebble-clay/src/styles/clay/elements/_button.scss @@ -0,0 +1,35 @@ +@import "clay"; + +button, +.button { + @include font-pfdin(medium); + @include font-size(1); + text-transform: uppercase; + background-color: $color-gray-7; + border-radius: $border-radius; + border: none; + display: inline-block; + + color: $color-white; + min-width: 12rem; + text-align: center; + margin: 0 auto $item-spacing-v; + padding: $button-padding; + + @include tap-highlight($color-gray-8); + + .platform-ios & { + padding: $button-padding-ios; + } + + &.primary, &[type="submit"] { + background-color: $color-orange; + + @include tap-highlight($color-red); + } +} + +a.button { + text-decoration: none; + color: $color-white; +} diff --git a/node_modules/pebble-clay/src/styles/config-page.scss b/node_modules/pebble-clay/src/styles/config-page.scss new file mode 100644 index 0000000..1481598 --- /dev/null +++ b/node_modules/pebble-clay/src/styles/config-page.scss @@ -0,0 +1,4 @@ +@import "clay/fonts"; +@import "clay/reset"; +@import "clay/base"; +@import "clay/elements/button"; diff --git a/node_modules/pebble-clay/src/templates/components/button.tpl b/node_modules/pebble-clay/src/templates/components/button.tpl new file mode 100755 index 0000000..187a22c --- /dev/null +++ b/node_modules/pebble-clay/src/templates/components/button.tpl @@ -0,0 +1,11 @@ +
+ + {{if description}} +
{{{description}}}
+ {{/if}} +
diff --git a/node_modules/pebble-clay/src/templates/components/checkboxgroup.tpl b/node_modules/pebble-clay/src/templates/components/checkboxgroup.tpl new file mode 100755 index 0000000..153c58d --- /dev/null +++ b/node_modules/pebble-clay/src/templates/components/checkboxgroup.tpl @@ -0,0 +1,15 @@ +
+ {{{label}}} +
+ {{each options}} + + {{/each}} +
+ {{if description}} +
{{{description}}}
+ {{/if}} +
diff --git a/node_modules/pebble-clay/src/templates/components/color.tpl b/node_modules/pebble-clay/src/templates/components/color.tpl new file mode 100755 index 0000000..043b3be --- /dev/null +++ b/node_modules/pebble-clay/src/templates/components/color.tpl @@ -0,0 +1,20 @@ +
+ + {{if description}} +
{{{description}}}
+ {{/if}} +
+
+
+
+
+
+
+
diff --git a/node_modules/pebble-clay/src/templates/components/footer.tpl b/node_modules/pebble-clay/src/templates/components/footer.tpl new file mode 100755 index 0000000..d867721 --- /dev/null +++ b/node_modules/pebble-clay/src/templates/components/footer.tpl @@ -0,0 +1 @@ +
diff --git a/node_modules/pebble-clay/src/templates/components/heading.tpl b/node_modules/pebble-clay/src/templates/components/heading.tpl new file mode 100755 index 0000000..ba6b0d7 --- /dev/null +++ b/node_modules/pebble-clay/src/templates/components/heading.tpl @@ -0,0 +1,3 @@ +
+ +
diff --git a/node_modules/pebble-clay/src/templates/components/input.tpl b/node_modules/pebble-clay/src/templates/components/input.tpl new file mode 100755 index 0000000..323a9b5 --- /dev/null +++ b/node_modules/pebble-clay/src/templates/components/input.tpl @@ -0,0 +1,15 @@ +
+ + + {{if description}} +
{{{description}}}
+ {{/if}} +
diff --git a/node_modules/pebble-clay/src/templates/components/radiogroup.tpl b/node_modules/pebble-clay/src/templates/components/radiogroup.tpl new file mode 100755 index 0000000..f2e4ba8 --- /dev/null +++ b/node_modules/pebble-clay/src/templates/components/radiogroup.tpl @@ -0,0 +1,20 @@ +
+ {{{label}}} +
+ {{each options}} + + {{/each}} +
+ {{if description}} +
{{{description}}}
+ {{/if}} +
diff --git a/node_modules/pebble-clay/src/templates/components/select.tpl b/node_modules/pebble-clay/src/templates/components/select.tpl new file mode 100755 index 0000000..60fe0fb --- /dev/null +++ b/node_modules/pebble-clay/src/templates/components/select.tpl @@ -0,0 +1,22 @@ +
+ + {{if description}} +
{{{description}}}
+ {{/if}} +
diff --git a/node_modules/pebble-clay/src/templates/components/slider.tpl b/node_modules/pebble-clay/src/templates/components/slider.tpl new file mode 100644 index 0000000..f5a309f --- /dev/null +++ b/node_modules/pebble-clay/src/templates/components/slider.tpl @@ -0,0 +1,25 @@ +
+ + {{if description}} +
{{{description}}}
+ {{/if}} +
diff --git a/node_modules/pebble-clay/src/templates/components/submit.tpl b/node_modules/pebble-clay/src/templates/components/submit.tpl new file mode 100755 index 0000000..872a80d --- /dev/null +++ b/node_modules/pebble-clay/src/templates/components/submit.tpl @@ -0,0 +1,7 @@ +
+ +
diff --git a/node_modules/pebble-clay/src/templates/components/text.tpl b/node_modules/pebble-clay/src/templates/components/text.tpl new file mode 100644 index 0000000..ad8d675 --- /dev/null +++ b/node_modules/pebble-clay/src/templates/components/text.tpl @@ -0,0 +1,3 @@ +
+

+
diff --git a/node_modules/pebble-clay/src/templates/components/toggle.tpl b/node_modules/pebble-clay/src/templates/components/toggle.tpl new file mode 100755 index 0000000..4ed2365 --- /dev/null +++ b/node_modules/pebble-clay/src/templates/components/toggle.tpl @@ -0,0 +1,19 @@ +
+ + {{if description}} +
{{{description}}}
+ {{/if}} +
-- 2.30.2

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 literal 0 HcmV?d00001 diff --git a/node_modules/pebble-clay/src/js/index.js b/node_modules/pebble-clay/src/js/index.js new file mode 100644 index 0000000..d4f2c22 --- /dev/null +++ b/node_modules/pebble-clay/src/js/index.js @@ -0,0 +1,15 @@ +/* Clay - https://github.com/pebble/clay - Version: 1.0.4 - Build Date: 2016-11-21T20:14:28.839Z */ +!function(t){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=t();else if("function"==typeof define&&define.amd)define([],t);else{var e;e="undefined"!=typeof window?window:"undefined"!=typeof global?global:"undefined"!=typeof self?self:this,e.pebbleClay=t()}}(function(){var t;return function e(t,n,r){function o(a,s){if(!n[a]){if(!t[a]){var c="function"==typeof require&&require;if(!s&&c)return c(a,!0);if(i)return i(a,!0);var l=new Error("Cannot find module '"+a+"'");throw l.code="MODULE_NOT_FOUND",l}var u=n[a]={exports:{}};t[a][0].call(u.exports,function(e){var n=t[a][1][e];return o(n?n:e)},u,u.exports,e,t,n,r)}return n[a].exports}for(var i="function"==typeof require&&require,a=0;a0)throw new Error("Invalid string. Length must be a multiple of 4");return"="===t[e-2]?2:"="===t[e-1]?1:0}function o(t){return 3*t.length/4-r(t)}function i(t){var e,n,o,i,a,s,c=t.length;a=r(t),s=new f(3*c/4-a),o=a>0?c-4:c;var l=0;for(e=0,n=0;e>16&255,s[l++]=i>>8&255,s[l++]=255&i;return 2===a?(i=u[t.charCodeAt(e)]<<2|u[t.charCodeAt(e+1)]>>4,s[l++]=255&i):1===a&&(i=u[t.charCodeAt(e)]<<10|u[t.charCodeAt(e+1)]<<4|u[t.charCodeAt(e+2)]>>2,s[l++]=i>>8&255,s[l++]=255&i),s}function a(t){return l[t>>18&63]+l[t>>12&63]+l[t>>6&63]+l[63&t]}function s(t,e,n){for(var r,o=[],i=e;iu?u:c+a));return 1===r?(e=t[n-1],o+=l[e>>2],o+=l[e<<4&63],o+="=="):2===r&&(e=(t[n-2]<<8)+t[n-1],o+=l[e>>10],o+=l[e>>4&63],o+=l[e<<2&63],o+="="),i.push(o),i.join("")}n.byteLength=o,n.toByteArray=i,n.fromByteArray=c;for(var l=[],u=[],f="undefined"!=typeof Uint8Array?Uint8Array:Array,p="ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/",d=0,h=p.length;d + * @license MIT + */ +"use strict";function r(){try{var t=new Uint8Array(1);return t.__proto__={__proto__:Uint8Array.prototype,foo:function(){return 42}},42===t.foo()&&"function"==typeof t.subarray&&0===t.subarray(1,1).byteLength}catch(e){return!1}}function o(){return a.TYPED_ARRAY_SUPPORT?2147483647:1073741823}function i(t,e){if(o()=o())throw new RangeError("Attempt to allocate Buffer larger than maximum size: 0x"+o().toString(16)+" bytes");return 0|t}function g(t){return+t!=t&&(t=0),a.alloc(+t)}function b(t,e){if(a.isBuffer(t))return t.length;if("undefined"!=typeof ArrayBuffer&&"function"==typeof ArrayBuffer.isView&&(ArrayBuffer.isView(t)||t instanceof ArrayBuffer))return t.byteLength;"string"!=typeof t&&(t=""+t);var n=t.length;if(0===n)return 0;for(var r=!1;;)switch(e){case"ascii":case"latin1":case"binary":return n;case"utf8":case"utf-8":case void 0:return W(t).length;case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":return 2*n;case"hex":return n>>>1;case"base64":return U(t).length;default:if(r)return W(t).length;e=(""+e).toLowerCase(),r=!0}}function y(t,e,n){var r=!1;if((void 0===e||e<0)&&(e=0),e>this.length)return"";if((void 0===n||n>this.length)&&(n=this.length),n<=0)return"";if(n>>>=0,e>>>=0,n<=e)return"";for(t||(t="utf8");;)switch(t){case"hex":return D(this,e,n);case"utf8":case"utf-8":return E(this,e,n);case"ascii":return B(this,e,n);case"latin1":case"binary":return S(this,e,n);case"base64":return O(this,e,n);case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":return N(this,e,n);default:if(r)throw new TypeError("Unknown encoding: "+t);t=(t+"").toLowerCase(),r=!0}}function v(t,e,n){var r=t[e];t[e]=t[n],t[n]=r}function A(t,e,n,r,o){if(0===t.length)return-1;if("string"==typeof n?(r=n,n=0):n>2147483647?n=2147483647:n<-2147483648&&(n=-2147483648),n=+n,isNaN(n)&&(n=o?0:t.length-1),n<0&&(n=t.length+n),n>=t.length){if(o)return-1;n=t.length-1}else if(n<0){if(!o)return-1;n=0}if("string"==typeof e&&(e=a.from(e,r)),a.isBuffer(e))return 0===e.length?-1:w(t,e,n,r,o);if("number"==typeof e)return e=255&e,a.TYPED_ARRAY_SUPPORT&&"function"==typeof Uint8Array.prototype.indexOf?o?Uint8Array.prototype.indexOf.call(t,e,n):Uint8Array.prototype.lastIndexOf.call(t,e,n):w(t,[e],n,r,o);throw new TypeError("val must be string, number or Buffer")}function w(t,e,n,r,o){function i(t,e){return 1===a?t[e]:t.readUInt16BE(e*a)}var a=1,s=t.length,c=e.length;if(void 0!==r&&(r=String(r).toLowerCase(),"ucs2"===r||"ucs-2"===r||"utf16le"===r||"utf-16le"===r)){if(t.length<2||e.length<2)return-1;a=2,s/=2,c/=2,n/=2}var l;if(o){var u=-1;for(l=n;ls&&(n=s-c),l=n;l>=0;l--){for(var f=!0,p=0;po&&(r=o)):r=o;var i=e.length;if(i%2!==0)throw new TypeError("Invalid hex string");r>i/2&&(r=i/2);for(var a=0;a239?4:i>223?3:i>191?2:1;if(o+s<=n){var c,l,u,f;switch(s){case 1:i<128&&(a=i);break;case 2:c=t[o+1],128===(192&c)&&(f=(31&i)<<6|63&c,f>127&&(a=f));break;case 3:c=t[o+1],l=t[o+2],128===(192&c)&&128===(192&l)&&(f=(15&i)<<12|(63&c)<<6|63&l,f>2047&&(f<55296||f>57343)&&(a=f));break;case 4:c=t[o+1],l=t[o+2],u=t[o+3],128===(192&c)&&128===(192&l)&&128===(192&u)&&(f=(15&i)<<18|(63&c)<<12|(63&l)<<6|63&u,f>65535&&f<1114112&&(a=f))}}null===a?(a=65533,s=1):a>65535&&(a-=65536,r.push(a>>>10&1023|55296),a=56320|1023&a),r.push(a),o+=s}return j(r)}function j(t){var e=t.length;if(e<=tt)return String.fromCharCode.apply(String,t);for(var n="",r=0;rr)&&(n=r);for(var o="",i=e;in)throw new RangeError("Trying to access beyond buffer length")}function F(t,e,n,r,o,i){if(!a.isBuffer(t))throw new TypeError('"buffer" argument must be a Buffer instance');if(e>o||et.length)throw new RangeError("Index out of range")}function z(t,e,n,r){e<0&&(e=65535+e+1);for(var o=0,i=Math.min(t.length-n,2);o>>8*(r?o:1-o)}function I(t,e,n,r){e<0&&(e=4294967295+e+1);for(var o=0,i=Math.min(t.length-n,4);o>>8*(r?o:3-o)&255}function L(t,e,n,r,o,i){if(n+r>t.length)throw new RangeError("Index out of range");if(n<0)throw new RangeError("Index out of range")}function K(t,e,n,r,o){return o||L(t,e,n,4,3.4028234663852886e38,-3.4028234663852886e38),$.write(t,e,n,r,23,4),n+4}function G(t,e,n,r,o){return o||L(t,e,n,8,1.7976931348623157e308,-1.7976931348623157e308),$.write(t,e,n,r,52,8),n+8}function C(t){if(t=X(t).replace(et,""),t.length<2)return"";for(;t.length%4!==0;)t+="=";return t}function X(t){return t.trim?t.trim():t.replace(/^\s+|\s+$/g,"")}function V(t){return t<16?"0"+t.toString(16):t.toString(16)}function W(t,e){e=e||1/0;for(var n,r=t.length,o=null,i=[],a=0;a55295&&n<57344){if(!o){if(n>56319){(e-=3)>-1&&i.push(239,191,189);continue}if(a+1===r){(e-=3)>-1&&i.push(239,191,189);continue}o=n;continue}if(n<56320){(e-=3)>-1&&i.push(239,191,189),o=n;continue}n=(o-55296<<10|n-56320)+65536}else o&&(e-=3)>-1&&i.push(239,191,189);if(o=null,n<128){if((e-=1)<0)break;i.push(n)}else if(n<2048){if((e-=2)<0)break;i.push(n>>6|192,63&n|128)}else if(n<65536){if((e-=3)<0)break;i.push(n>>12|224,n>>6&63|128,63&n|128)}else{if(!(n<1114112))throw new Error("Invalid code point");if((e-=4)<0)break;i.push(n>>18|240,n>>12&63|128,n>>6&63|128,63&n|128)}}return i}function Z(t){for(var e=[],n=0;n>8,o=n%256,i.push(o),i.push(r);return i}function U(t){return Q.toByteArray(C(t))}function q(t,e,n,r){for(var o=0;o=e.length||o>=t.length);++o)e[o+n]=t[o];return o}function H(t){return t!==t}var Q=t("base64-js"),$=t("ieee754"),_=t("isarray");n.Buffer=a,n.SlowBuffer=g,n.INSPECT_MAX_BYTES=50,a.TYPED_ARRAY_SUPPORT=void 0!==e.TYPED_ARRAY_SUPPORT?e.TYPED_ARRAY_SUPPORT:r(),n.kMaxLength=o(),a.poolSize=8192,a._augment=function(t){return t.__proto__=a.prototype,t},a.from=function(t,e,n){return s(null,t,e,n)},a.TYPED_ARRAY_SUPPORT&&(a.prototype.__proto__=Uint8Array.prototype,a.__proto__=Uint8Array,"undefined"!=typeof Symbol&&Symbol.species&&a[Symbol.species]===a&&Object.defineProperty(a,Symbol.species,{value:null,configurable:!0})),a.alloc=function(t,e,n){return l(null,t,e,n)},a.allocUnsafe=function(t){return u(null,t)},a.allocUnsafeSlow=function(t){return u(null,t)},a.isBuffer=function(t){return!(null==t||!t._isBuffer)},a.compare=function(t,e){if(!a.isBuffer(t)||!a.isBuffer(e))throw new TypeError("Arguments must be Buffers");if(t===e)return 0;for(var n=t.length,r=e.length,o=0,i=Math.min(n,r);o0&&(t=this.toString("hex",0,e).match(/.{2}/g).join(" "),this.length>e&&(t+=" ... ")),""},a.prototype.compare=function(t,e,n,r,o){if(!a.isBuffer(t))throw new TypeError("Argument must be a Buffer");if(void 0===e&&(e=0),void 0===n&&(n=t?t.length:0),void 0===r&&(r=0),void 0===o&&(o=this.length),e<0||n>t.length||r<0||o>this.length)throw new RangeError("out of range index");if(r>=o&&e>=n)return 0;if(r>=o)return-1;if(e>=n)return 1;if(e>>>=0,n>>>=0,r>>>=0,o>>>=0,this===t)return 0;for(var i=o-r,s=n-e,c=Math.min(i,s),l=this.slice(r,o),u=t.slice(e,n),f=0;fo)&&(n=o),t.length>0&&(n<0||e<0)||e>this.length)throw new RangeError("Attempt to write outside buffer bounds");r||(r="utf8");for(var i=!1;;)switch(r){case"hex":return k(this,t,e,n);case"utf8":case"utf-8":return x(this,t,e,n);case"ascii":return M(this,t,e,n);case"latin1":case"binary":return T(this,t,e,n);case"base64":return R(this,t,e,n);case"ucs2":case"ucs-2":case"utf16le":case"utf-16le":return P(this,t,e,n);default:if(i)throw new TypeError("Unknown encoding: "+r);r=(""+r).toLowerCase(),i=!0}},a.prototype.toJSON=function(){return{type:"Buffer",data:Array.prototype.slice.call(this._arr||this,0)}};var tt=4096;a.prototype.slice=function(t,e){var n=this.length;t=~~t,e=void 0===e?n:~~e,t<0?(t+=n,t<0&&(t=0)):t>n&&(t=n),e<0?(e+=n,e<0&&(e=0)):e>n&&(e=n),e0&&(o*=256);)r+=this[t+--e]*o;return r},a.prototype.readUInt8=function(t,e){return e||Y(t,1,this.length),this[t]},a.prototype.readUInt16LE=function(t,e){return e||Y(t,2,this.length),this[t]|this[t+1]<<8},a.prototype.readUInt16BE=function(t,e){return e||Y(t,2,this.length),this[t]<<8|this[t+1]},a.prototype.readUInt32LE=function(t,e){return e||Y(t,4,this.length),(this[t]|this[t+1]<<8|this[t+2]<<16)+16777216*this[t+3]},a.prototype.readUInt32BE=function(t,e){return e||Y(t,4,this.length),16777216*this[t]+(this[t+1]<<16|this[t+2]<<8|this[t+3])},a.prototype.readIntLE=function(t,e,n){t=0|t,e=0|e,n||Y(t,e,this.length);for(var r=this[t],o=1,i=0;++i=o&&(r-=Math.pow(2,8*e)),r},a.prototype.readIntBE=function(t,e,n){t=0|t,e=0|e,n||Y(t,e,this.length);for(var r=e,o=1,i=this[t+--r];r>0&&(o*=256);)i+=this[t+--r]*o;return o*=128,i>=o&&(i-=Math.pow(2,8*e)),i},a.prototype.readInt8=function(t,e){return e||Y(t,1,this.length),128&this[t]?(255-this[t]+1)*-1:this[t]},a.prototype.readInt16LE=function(t,e){e||Y(t,2,this.length);var n=this[t]|this[t+1]<<8;return 32768&n?4294901760|n:n},a.prototype.readInt16BE=function(t,e){e||Y(t,2,this.length);var n=this[t+1]|this[t]<<8;return 32768&n?4294901760|n:n},a.prototype.readInt32LE=function(t,e){return e||Y(t,4,this.length),this[t]|this[t+1]<<8|this[t+2]<<16|this[t+3]<<24},a.prototype.readInt32BE=function(t,e){return e||Y(t,4,this.length),this[t]<<24|this[t+1]<<16|this[t+2]<<8|this[t+3]},a.prototype.readFloatLE=function(t,e){return e||Y(t,4,this.length),$.read(this,t,!0,23,4)},a.prototype.readFloatBE=function(t,e){return e||Y(t,4,this.length),$.read(this,t,!1,23,4)},a.prototype.readDoubleLE=function(t,e){return e||Y(t,8,this.length),$.read(this,t,!0,52,8)},a.prototype.readDoubleBE=function(t,e){return e||Y(t,8,this.length),$.read(this,t,!1,52,8)},a.prototype.writeUIntLE=function(t,e,n,r){if(t=+t,e=0|e,n=0|n,!r){var o=Math.pow(2,8*n)-1;F(this,t,e,n,o,0)}var i=1,a=0;for(this[e]=255&t;++a=0&&(a*=256);)this[e+i]=t/a&255;return e+n},a.prototype.writeUInt8=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,1,255,0),a.TYPED_ARRAY_SUPPORT||(t=Math.floor(t)),this[e]=255&t,e+1},a.prototype.writeUInt16LE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,2,65535,0),a.TYPED_ARRAY_SUPPORT?(this[e]=255&t,this[e+1]=t>>>8):z(this,t,e,!0),e+2},a.prototype.writeUInt16BE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,2,65535,0),a.TYPED_ARRAY_SUPPORT?(this[e]=t>>>8,this[e+1]=255&t):z(this,t,e,!1),e+2},a.prototype.writeUInt32LE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,4,4294967295,0),a.TYPED_ARRAY_SUPPORT?(this[e+3]=t>>>24,this[e+2]=t>>>16,this[e+1]=t>>>8,this[e]=255&t):I(this,t,e,!0),e+4},a.prototype.writeUInt32BE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,4,4294967295,0),a.TYPED_ARRAY_SUPPORT?(this[e]=t>>>24,this[e+1]=t>>>16,this[e+2]=t>>>8,this[e+3]=255&t):I(this,t,e,!1),e+4},a.prototype.writeIntLE=function(t,e,n,r){if(t=+t,e=0|e,!r){var o=Math.pow(2,8*n-1);F(this,t,e,n,o-1,-o)}var i=0,a=1,s=0;for(this[e]=255&t;++i>0)-s&255;return e+n},a.prototype.writeIntBE=function(t,e,n,r){if(t=+t,e=0|e,!r){var o=Math.pow(2,8*n-1);F(this,t,e,n,o-1,-o)}var i=n-1,a=1,s=0;for(this[e+i]=255&t;--i>=0&&(a*=256);)t<0&&0===s&&0!==this[e+i+1]&&(s=1),this[e+i]=(t/a>>0)-s&255;return e+n},a.prototype.writeInt8=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,1,127,-128),a.TYPED_ARRAY_SUPPORT||(t=Math.floor(t)),t<0&&(t=255+t+1),this[e]=255&t,e+1},a.prototype.writeInt16LE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,2,32767,-32768),a.TYPED_ARRAY_SUPPORT?(this[e]=255&t,this[e+1]=t>>>8):z(this,t,e,!0),e+2},a.prototype.writeInt16BE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,2,32767,-32768),a.TYPED_ARRAY_SUPPORT?(this[e]=t>>>8,this[e+1]=255&t):z(this,t,e,!1),e+2},a.prototype.writeInt32LE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,4,2147483647,-2147483648),a.TYPED_ARRAY_SUPPORT?(this[e]=255&t,this[e+1]=t>>>8,this[e+2]=t>>>16,this[e+3]=t>>>24):I(this,t,e,!0),e+4},a.prototype.writeInt32BE=function(t,e,n){return t=+t,e=0|e,n||F(this,t,e,4,2147483647,-2147483648),t<0&&(t=4294967295+t+1),a.TYPED_ARRAY_SUPPORT?(this[e]=t>>>24,this[e+1]=t>>>16,this[e+2]=t>>>8,this[e+3]=255&t):I(this,t,e,!1),e+4},a.prototype.writeFloatLE=function(t,e,n){return K(this,t,e,!0,n)},a.prototype.writeFloatBE=function(t,e,n){return K(this,t,e,!1,n)},a.prototype.writeDoubleLE=function(t,e,n){return G(this,t,e,!0,n)},a.prototype.writeDoubleBE=function(t,e,n){return G(this,t,e,!1,n)},a.prototype.copy=function(t,e,n,r){if(n||(n=0),r||0===r||(r=this.length),e>=t.length&&(e=t.length),e||(e=0),r>0&&r=this.length)throw new RangeError("sourceStart out of bounds");if(r<0)throw new RangeError("sourceEnd out of bounds");r>this.length&&(r=this.length),t.length-e=0;--o)t[o+e]=this[o+n];else if(i<1e3||!a.TYPED_ARRAY_SUPPORT)for(o=0;o>>=0,n=void 0===n?this.length:n>>>0,t||(t=0);var i;if("number"==typeof t)for(i=e;in;++n)if(i=t[n],i===e||i!==i&&e!==e)return n;return-1}n.__esModule=!0;var o=Object.prototype.toString,i="undefined"!=typeof e?function(t){return e.isBuffer(t)}:function(){return!1},a="function"==typeof Object.keys?function(t){return Object.keys(t)}:function(t){var e=typeof t;if(null===t||"function"!==e&&"object"!==e)throw new TypeError("obj must be an Object");var n=[],r=void 0;for(r in t)Object.prototype.hasOwnProperty.call(t,r)&&n.push(r);return n},s="function"==typeof Symbol?function(t){return Object.getOwnPropertySymbols(t)}:function(){return[]};n.getKeys=a,n.getSymbols=s,n.indexOf=r,n.isBuffer=i},function(t,n,r){"use strict";function o(t,e){var n=a(t);return null!==n?n:i(t,e)}function i(t,n){if("function"!=typeof n)throw new TypeError("customizer is must be a Function");if("function"==typeof t){var r=String(t);return/^\s*function\s*\S*\([^\)]*\)\s*{\s*\[native code\]\s*}/.test(r)?t:new Function("return "+String(r))()}var o=c.call(t);if("[object Array]"===o)return[];if("[object Object]"===o&&t.constructor===Object)return{};if("[object Date]"===o)return new Date(t.getTime());if("[object RegExp]"===o){var i=String(t),a=i.lastIndexOf("/");return new RegExp(i.slice(1,a),i.slice(a+1))}if((0,s.isBuffer)(t)){var l=new e(t.length);return t.copy(l),l}var u=n(t);return void 0!==u?u:null}function a(t){var e=typeof t;return null!==t&&"object"!==e&&"function"!==e?t:null}n.__esModule=!0,n.copyValue=n.copyCollection=n.copy=void 0;var s=r(1),c=Object.prototype.toString;n.copy=o,n.copyCollection=i,n.copyValue=a},function(t,e,n){"use strict";function r(t){}function o(t){var e=arguments.length<=1||void 0===arguments[1]?r:arguments[1];if(null===t)return null;var n=(0,a.copyValue)(t);if(null!==n)return n;var o=(0,a.copyCollection)(t,e),s=null!==o?o:t,c=[t],l=[s];return i(t,e,s,c,l)}function i(t,e,n,r,o){if(null===t)return null;var c=(0,a.copyValue)(t);if(null!==c)return c;var l=(0,s.getKeys)(t).concat((0,s.getSymbols)(t)),u=void 0,f=void 0,p=void 0,d=void 0,h=void 0,m=void 0,g=void 0,b=void 0;for(u=0,f=l.length;f>u;++u)p=l[u],d=t[p],h=(0,s.indexOf)(r,d),m=void 0,g=void 0,b=void 0,-1===h?(m=(0,a.copy)(d,e),g=null!==m?m:d,null!==d&&/^(?:function|object)$/.test(typeof d)&&(r.push(d),o.push(g))):b=o[h],n[p]=b||i(d,e,g,r,o);return n}e.__esModule=!0;var a=n(2),s=n(1);e["default"]=o,t.exports=e["default"]}])})}).call(this,e("buffer").Buffer)},{buffer:2}],4:[function(t,e,n){n.read=function(t,e,n,r,o){var i,a,s=8*o-r-1,c=(1<>1,u=-7,f=n?o-1:0,p=n?-1:1,d=t[e+f];for(f+=p,i=d&(1<<-u)-1,d>>=-u,u+=s;u>0;i=256*i+t[e+f],f+=p,u-=8);for(a=i&(1<<-u)-1,i>>=-u,u+=r;u>0;a=256*a+t[e+f],f+=p,u-=8);if(0===i)i=1-l;else{if(i===c)return a?NaN:(d?-1:1)*(1/0);a+=Math.pow(2,r),i-=l}return(d?-1:1)*a*Math.pow(2,i-r)},n.write=function(t,e,n,r,o,i){var a,s,c,l=8*i-o-1,u=(1<>1,p=23===o?Math.pow(2,-24)-Math.pow(2,-77):0,d=r?0:i-1,h=r?1:-1,m=e<0||0===e&&1/e<0?1:0;for(e=Math.abs(e),isNaN(e)||e===1/0?(s=isNaN(e)?1:0,a=u):(a=Math.floor(Math.log(e)/Math.LN2),e*(c=Math.pow(2,-a))<1&&(a--,c*=2),e+=a+f>=1?p/c:p*Math.pow(2,1-f),e*c>=2&&(a++,c/=2),a+f>=u?(s=0,a=u):a+f>=1?(s=(e*c-1)*Math.pow(2,o),a+=f):(s=e*Math.pow(2,f-1)*Math.pow(2,o),a=0));o>=8;t[n+d]=255&s,d+=h,s/=256,o-=8);for(a=a<0;t[n+d]=255&a,d+=h,a/=256,l-=8);t[n+d-h]|=128*m}},{}],5:[function(t,e,n){var r={}.toString;e.exports=Array.isArray||function(t){return"[object Array]"==r.call(t)}},{}],6:[function(t,e,n){function r(t){return/^[a-z_$][0-9a-z_$]*$/gi.test(t)&&!i.test(t)}function o(t){if(a)return t.toString();var e=t.source.replace(/\//g,function(t,e,n){return 0===e||"\\"!==n[e-1]?"\\/":"/"}),n=(t.global&&"g"||"")+(t.ignoreCase&&"i"||"")+(t.multiline&&"m"||"");return"/"+e+"/"+n}/* toSource by Marcello Bastea-Forte - zlib license */ +e.exports=function(t,e,n,i){function a(t,e,n,i,s){function c(t){return n.slice(1)+t.join(","+(n&&"\n")+l)+(n?" ":"")}var l=i+n;switch(t=e?e(t):t,typeof t){case"string":return JSON.stringify(t);case"boolean":case"number":case"undefined":return""+t;case"function":return t.toString()}if(null===t)return"null";if(t instanceof RegExp)return o(t);if(t instanceof Date)return"new Date("+t.getTime()+")";var u=s.indexOf(t)+1;if(u>0)return"{$circularReference:"+u+"}";if(s.push(t),Array.isArray(t))return"["+c(t.map(function(t){return a(t,e,n,l,s.slice())}))+"]";var f=Object.keys(t);return f.length?"{"+c(f.map(function(o){return(r(o)?o:JSON.stringify(o))+":"+a(t[o],e,n,l,s.slice())}))+"}":"{}"}var s=[];return a(t,e,void 0===n?" ":n||"",i||"",s)};var i=/^(abstract|boolean|break|byte|case|catch|char|class|const|continue|debugger|default|delete|do|double|else|enum|export|extends|false|final|finally|float|for|function|goto|if|implements|import|in|instanceof|int|interface|long|native|new|null|package|private|protected|public|return|short|static|super|switch|synchronized|this|throw|throws|transient|true|try|typeof|undefined|var|void|volatile|while|with)$/,a="\\/"===new RegExp("/").source},{}],7:[function(t,e,n){e.exports={name:"pebble-clay",version:"1.0.4",description:"Pebble Config Framework",scripts:{"test-travis":"./node_modules/.bin/gulp && ./node_modules/.bin/karma start ./test/karma.conf.js --single-run --browsers chromeTravisCI && ./node_modules/.bin/eslint ./","test-debug":"(export DEBUG=true && ./node_modules/.bin/gulp && ./node_modules/.bin/karma start ./test/karma.conf.js --no-single-run)",test:"./node_modules/.bin/gulp && ./node_modules/.bin/karma start ./test/karma.conf.js --single-run",lint:"./node_modules/.bin/eslint ./",build:"gulp",dev:"gulp dev","pebble-clean":"rm -rf tmp src/js/index.js && pebble clean","pebble-publish":"npm run pebble-clean && npm run build && pebble build && pebble package publish && npm run pebble-clean","pebble-build":"npm run build && pebble build"},repository:{type:"git",url:"git+https://github.com/pebble/clay.git"},keywords:["pebble","config","configuration","pebble-package"],author:"Pebble Technology",license:"MIT",bugs:{url:"https://github.com/pebble/clay/issues"},pebble:{projectType:"package",sdkVersion:"3",targetPlatforms:["aplite","basalt","chalk","diorite","emery"],resources:{media:[]},capabilities:["configurable"]},homepage:"https://github.com/pebble/clay#readme",devDependencies:{autoprefixer:"^6.3.1",bourbon:"^4.2.6",browserify:"^13.0.0","browserify-istanbul":"^0.2.1",chai:"^3.4.1",deamdify:"^0.2.0",deepcopy:"^0.6.1",del:"^2.0.2",eslint:"^1.5.1","eslint-config-pebble":"^1.2.0","eslint-plugin-standard":"^1.3.1",gulp:"^3.9.0","gulp-autoprefixer":"^3.1.0","gulp-htmlmin":"^1.3.0","gulp-inline":"0.0.15","gulp-insert":"^0.5.0","gulp-sass":"^2.1.1","gulp-sourcemaps":"^1.6.0","gulp-uglify":"^1.5.2",joi:"^6.10.1",karma:"^0.13.19","karma-browserify":"^5.0.1","karma-chrome-launcher":"^0.2.2","karma-coverage":"^0.5.3","karma-mocha":"^0.2.1","karma-mocha-reporter":"^1.1.5","karma-source-map-support":"^1.1.0","karma-threshold-reporter":"^0.1.15",mocha:"^2.3.4",postcss:"^5.0.14","require-from-string":"^1.1.0",sassify:"^0.9.1",sinon:"^1.17.3",stringify:"^3.2.0",through:"^2.3.8",tosource:"^1.0.0","vinyl-buffer":"^1.0.0","vinyl-source-stream":"^1.1.0",watchify:"^3.7.0"},dependencies:{}}},{}],8:[function(t,e,n){"use strict";e.exports={name:"button",template:t("../../templates/components/button.tpl"),style:t("../../styles/clay/components/button.scss"),manipulator:"button",defaults:{primary:!1,attributes:{},description:""}}},{"../../styles/clay/components/button.scss":21,"../../templates/components/button.tpl":30}],9:[function(t,e,n){"use strict";e.exports={name:"checkboxgroup",template:t("../../templates/components/checkboxgroup.tpl"),style:t("../../styles/clay/components/checkboxgroup.scss"),manipulator:"checkboxgroup",defaults:{label:"",options:[],description:""}}},{"../../styles/clay/components/checkboxgroup.scss":22,"../../templates/components/checkboxgroup.tpl":31}],10:[function(t,e,n){"use strict";e.exports={name:"color",template:t("../../templates/components/color.tpl"),style:t("../../styles/clay/components/color.scss"),manipulator:"color",defaults:{label:"",description:""},initialize:function(t,e){function n(t){if("number"==typeof t)t=t.toString(16);else if(!t)return"transparent";return t=r(t),"#"+(f?p[t]:t)}function r(t){for(t=t.toLowerCase();t.length<6;)t="0"+t;return t}function o(t){switch(typeof t){case"number":return r(t.toString(16));case"string":return t.replace(/^#|^0x/,"");default:return t}}function i(t){return t.reduce(function(t,e){return t.concat(e)},[])}function a(t){t=t.replace(/^#|^0x/,"");var e=parseInt(t.slice(0,2),16)/255,n=parseInt(t.slice(2,4),16)/255,r=parseInt(t.slice(4),16)/255;e=e>.04045?Math.pow((e+.055)/1.055,2.4):e/12.92,n=n>.04045?Math.pow((n+.055)/1.055,2.4):n/12.92,r=r>.04045?Math.pow((r+.055)/1.055,2.4):r/12.92;var o=(.4124*e+.3576*n+.1805*r)/.95047,i=(.2126*e+.7152*n+.0722*r)/1,a=(.0193*e+.1192*n+.9505*r)/1.08883;return o=o>.008856?Math.pow(o,1/3):7.787*o+16/116,i=i>.008856?Math.pow(i,1/3):7.787*i+16/116,a=a>.008856?Math.pow(a,1/3):7.787*a+16/116,[116*i-16,500*(o-i),200*(i-a)]}function s(t,e){var n=t[0]-e[0],r=t[1]-e[1],o=t[2]-e[2];return Math.sqrt(Math.pow(n,2)+Math.pow(r,2)+Math.pow(o,2))}function c(){return!e.meta.activeWatchInfo||2===e.meta.activeWatchInfo.firmware.major||["aplite","diorite"].indexOf(e.meta.activeWatchInfo.platform)>-1&&!u.config.allowGray?d.BLACK_WHITE:["aplite","diorite"].indexOf(e.meta.activeWatchInfo.platform)>-1&&u.config.allowGray?d.GRAY:d.COLOR}var l=t.HTML,u=this;u.roundColorToLayout=function(t){var e=o(t);if(m.indexOf(e)===-1){var n=a(e),r=m.map(function(t){var e=a(o(t));return s(n,e)}),i=Math.min.apply(Math,r),c=r.indexOf(i);e=m[c]}return parseInt(e,16)};var f=u.config.sunlight!==!1,p={"000000":"000000","000055":"001e41","0000aa":"004387","0000ff":"0068ca","005500":"2b4a2c","005555":"27514f","0055aa":"16638d","0055ff":"007dce","00aa00":"5e9860","00aa55":"5c9b72","00aaaa":"57a5a2","00aaff":"4cb4db","00ff00":"8ee391","00ff55":"8ee69e","00ffaa":"8aebc0","00ffff":"84f5f1",550000:"4a161b",550055:"482748","5500aa":"40488a","5500ff":"2f6bcc",555500:"564e36",555555:"545454","5555aa":"4f6790","5555ff":"4180d0","55aa00":"759a64","55aa55":"759d76","55aaaa":"71a6a4","55aaff":"69b5dd","55ff00":"9ee594","55ff55":"9de7a0","55ffaa":"9becc2","55ffff":"95f6f2",aa0000:"99353f",aa0055:"983e5a",aa00aa:"955694",aa00ff:"8f74d2",aa5500:"9d5b4d",aa5555:"9d6064",aa55aa:"9a7099",aa55ff:"9587d5",aaaa00:"afa072",aaaa55:"aea382",aaaaaa:"ababab",ffffff:"ffffff",aaaaff:"a7bae2",aaff00:"c9e89d",aaff55:"c9eaa7",aaffaa:"c7f0c8",aaffff:"c3f9f7",ff0000:"e35462",ff0055:"e25874",ff00aa:"e16aa3",ff00ff:"de83dc",ff5500:"e66e6b",ff5555:"e6727c",ff55aa:"e37fa7",ff55ff:"e194df",ffaa00:"f1aa86",ffaa55:"f1ad93",ffaaaa:"efb5b8",ffaaff:"ecc3eb",ffff00:"ffeeab",ffff55:"fff1b5",ffffaa:"fff6d3"},d={COLOR:[[!1,!1,"55ff00","aaff55",!1,"ffff55","ffffaa",!1,!1],[!1,"aaffaa","55ff55","00ff00","aaff00","ffff00","ffaa55","ffaaaa",!1],["55ffaa","00ff55","00aa00","55aa00","aaaa55","aaaa00","ffaa00","ff5500","ff5555"],["aaffff","00ffaa","00aa55","55aa55","005500","555500","aa5500","ff0000","ff0055"],[!1,"55aaaa","00aaaa","005555","ffffff","000000","aa5555","aa0000",!1],["55ffff","00ffff","00aaff","0055aa","aaaaaa","555555","550000","aa0055","ff55aa"],["55aaff","0055ff","0000ff","0000aa","000055","550055","aa00aa","ff00aa","ffaaff"],[!1,"5555aa","5555ff","5500ff","5500aa","aa00ff","ff00ff","ff55ff",!1],[!1,!1,!1,"aaaaff","aa55ff","aa55aa",!1,!1,!1]],GRAY:[["000000","aaaaaa","ffffff"]],BLACK_WHITE:[["000000","ffffff"]]},h=u.config.layout||c();"string"==typeof h&&(h=d[h]),Array.isArray(h[0])||(h=[h]);var m=i(h).map(function(t){return o(t)}).filter(function(t){return t}),g="",b=h.length,y=0;h.forEach(function(t){y=t.length>y?t.length:y});for(var v=100/y,A=100/b,w=u.$element,k=0;k'}var j=0;3===y&&(j=5),2===y&&(j=8);var B=j*v/A+"%",S=j+"%";w.select(".color-box-container").add(l(g)).set("$paddingTop",B).set("$paddingRight",S).set("$paddingBottom",B).set("$paddingLeft",S),w.select(".color-box-wrap").set("$paddingBottom",v/A*100+"%");var D=w.select(".value"),N=w.select(".picker-wrap"),Y=u.$manipulatorTarget.get("disabled");w.select("label").on("click",function(){Y||N.set("show")}),u.on("change",function(){var t=u.get();D.set("$background-color",n(t)),w.select(".color-box").set("-selected"),w.select('.color-box[data-value="'+t+'"]').set("+selected")}),w.select(".color-box.selectable").on("click",function(t){u.set(parseInt(t.target.dataset.value,10)),N.set("-show")}),N.on("click",function(){N.set("-show")}),u.on("disabled",function(){Y=!0}),u.on("enabled",function(){Y=!1}),u._layout=h}}},{"../../styles/clay/components/color.scss":23,"../../templates/components/color.tpl":32}],11:[function(t,e,n){"use strict";e.exports={name:"footer",template:t("../../templates/components/footer.tpl"),manipulator:"html"}},{"../../templates/components/footer.tpl":33}],12:[function(t,e,n){"use strict";e.exports={name:"heading",template:t("../../templates/components/heading.tpl"),manipulator:"html",defaults:{size:4}}},{"../../templates/components/heading.tpl":34}],13:[function(t,e,n){"use strict";e.exports={color:t("./color"),footer:t("./footer"),heading:t("./heading"),input:t("./input"),select:t("./select"),submit:t("./submit"),text:t("./text"),toggle:t("./toggle"),radiogroup:t("./radiogroup"),checkboxgroup:t("./checkboxgroup"),button:t("./button"),slider:t("./slider")}},{"./button":8,"./checkboxgroup":9,"./color":10,"./footer":11,"./heading":12,"./input":14,"./radiogroup":15,"./select":16,"./slider":17,"./submit":18,"./text":19,"./toggle":20}],14:[function(t,e,n){"use strict";e.exports={name:"input",template:t("../../templates/components/input.tpl"),style:t("../../styles/clay/components/input.scss"),manipulator:"val",defaults:{label:"",description:"",attributes:{}}}},{"../../styles/clay/components/input.scss":24,"../../templates/components/input.tpl":35}],15:[function(t,e,n){"use strict";e.exports={name:"radiogroup",template:t("../../templates/components/radiogroup.tpl"),style:t("../../styles/clay/components/radiogroup.scss"),manipulator:"radiogroup",defaults:{label:"",options:[],description:"",attributes:{}}}},{"../../styles/clay/components/radiogroup.scss":25,"../../templates/components/radiogroup.tpl":36}],16:[function(t,e,n){"use strict";e.exports={name:"select",template:t("../../templates/components/select.tpl"),style:t("../../styles/clay/components/select.scss"),manipulator:"val",defaults:{label:"",options:[],description:"",attributes:{}},initialize:function(){function t(){var t=e.$manipulatorTarget.get("selectedIndex"),r=e.$manipulatorTarget.select("option"),o=r[t]&&r[t].innerHTML;n.set("innerHTML",o)}var e=this,n=e.$element.select(".value");t(),e.on("change",t)}}},{"../../styles/clay/components/select.scss":26,"../../templates/components/select.tpl":37}],17:[function(t,e,n){"use strict";e.exports={name:"slider",template:t("../../templates/components/slider.tpl"),style:t("../../styles/clay/components/slider.scss"),manipulator:"slider",defaults:{label:"",description:"",min:0,max:100,step:1,attributes:{}},initialize:function(){function t(){var t=e.get().toFixed(e.precision);n.set("value",t),r.set("innerHTML",t)}var e=this,n=e.$element.select(".value"),r=e.$element.select(".value-pad"),o=e.$manipulatorTarget,i=o.get("step");i=i.toString(10).split(".")[1],e.precision=i?i.length:0,e.on("change",t),o.on("|input",t),t(),n.on("|input",function(){r.set("innerHTML",this.get("value"))}),n.on("|change",function(){e.set(this.get("value")),t()})}}},{"../../styles/clay/components/slider.scss":27,"../../templates/components/slider.tpl":38}],18:[function(t,e,n){"use strict";e.exports={name:"submit",template:t("../../templates/components/submit.tpl"),style:t("../../styles/clay/components/submit.scss"),manipulator:"button",defaults:{attributes:{}}}},{"../../styles/clay/components/submit.scss":28,"../../templates/components/submit.tpl":39}],19:[function(t,e,n){"use strict";e.exports={name:"text",template:t("../../templates/components/text.tpl"),manipulator:"html"}},{"../../templates/components/text.tpl":40}],20:[function(t,e,n){"use strict";e.exports={name:"toggle",template:t("../../templates/components/toggle.tpl"),style:t("../../styles/clay/components/toggle.scss"),manipulator:"checked",defaults:{label:"",description:"",attributes:{}}}},{"../../styles/clay/components/toggle.scss":29,"../../templates/components/toggle.tpl":41}],21:[function(t,e,n){e.exports=".component-button { text-align: center; }\n\n.section .component-button { padding-bottom: 0; }\n\n.component-button .description { padding-left: 0; padding-right: 0; }\n"},{}],22:[function(t,e,n){e.exports=".component-checkbox { display: block; }\n\n.section .component-checkbox { padding-right: 0.375rem; }\n\n.component-checkbox > .label { display: block; padding-bottom: 0.35rem; }\n\n.component-checkbox .checkbox-group { padding-bottom: 0.35rem; }\n\n.component-checkbox .checkbox-group label { padding: 0.35rem 0.375rem; }\n\n.component-checkbox .checkbox-group .label { font-size: 0.9em; }\n\n.component-checkbox .checkbox-group input { opacity: 0; position: absolute; }\n\n.component-checkbox .checkbox-group i { display: block; position: relative; border-radius: 0.25rem; width: 1.4rem; height: 1.4rem; border: 0.11765rem solid #767676; -webkit-flex-shrink: 0; flex-shrink: 0; }\n\n.component-checkbox .checkbox-group input:checked + i { border-color: #ff4700; background: #ff4700; }\n\n.component-checkbox .checkbox-group input:checked + i:after { content: ''; box-sizing: border-box; -webkit-transform: rotate(45deg); transform: rotate(45deg); position: absolute; left: 0.35rem; top: -0.05rem; display: block; width: 0.5rem; height: 1rem; border: 0 solid #ffffff; border-right-width: 0.11765rem; border-bottom-width: 0.11765rem; }\n\n.component-checkbox .description { padding-left: 0; padding-right: 0; }\n"},{}],23:[function(t,e,n){e.exports=".section .component-color { padding: 0; }\n\n.component-color .value { width: 2.2652rem; height: 1.4rem; border-radius: 0.7rem; box-shadow: 0 0.1rem 0.1rem #2f2f2f; display: block; background: #000; }\n\n.component-color .picker-wrap { left: 0; top: 0; right: 0; bottom: 0; position: fixed; padding: 0.7rem 0.375rem; background: rgba(0, 0, 0, 0.65); opacity: 0; -webkit-transition: opacity 100ms ease-in 175ms; transition: opacity 100ms ease-in 175ms; pointer-events: none; z-index: 100; display: -webkit-box; display: -webkit-flex; display: flex; -webkit-box-orient: vertical; -webkit-box-direction: normal; -webkit-flex-direction: column; flex-direction: column; -webkit-box-pack: center; -webkit-justify-content: center; justify-content: center; -webkit-box-align: center; -webkit-align-items: center; align-items: center; }\n\n.component-color .picker-wrap .picker { padding: 0.7rem 0.75rem; background: #484848; box-shadow: 0 0.17647rem 0.88235rem rgba(0, 0, 0, 0.4); border-radius: 0.25rem; width: 100%; max-width: 26rem; overflow: auto; }\n\n.component-color .picker-wrap.show { -webkit-transition-delay: 0ms; transition-delay: 0ms; pointer-events: auto; opacity: 1; }\n\n.component-color .color-box-wrap { box-sizing: border-box; position: relative; height: 0; width: 100%; padding: 0 0 100% 0; }\n\n.component-color .color-box-wrap .color-box-container { position: absolute; height: 99.97%; width: 100%; left: 0; top: 0; }\n\n.component-color .color-box-wrap .color-box-container .color-box { float: left; cursor: pointer; -webkit-tap-highlight-color: transparent; }\n\n.component-color .color-box-wrap .color-box-container .color-box.rounded-tl { border-top-left-radius: 0.25rem; }\n\n.component-color .color-box-wrap .color-box-container .color-box.rounded-tr { border-top-right-radius: 0.25rem; }\n\n.component-color .color-box-wrap .color-box-container .color-box.rounded-bl { border-bottom-left-radius: 0.25rem; }\n\n.component-color .color-box-wrap .color-box-container .color-box.rounded-br { border-bottom-right-radius: 0.25rem; }\n\n.component-color .color-box-wrap .color-box-container .color-box.selected { -webkit-transform: scale(1.1); transform: scale(1.1); border-radius: 0.25rem; box-shadow: #111 0 0 0.24rem; position: relative; z-index: 100; }\n"},{}],24:[function(t,e,n){e.exports=".section .component-input { padding: 0; }\n\n.component-input label { display: block; }\n\n.component-input .label { padding-bottom: 0.7rem; }\n\n.component-input .input { position: relative; min-width: 100%; margin-top: 0.7rem; margin-left: 0; }\n\n.component-input input { display: block; width: 100%; background: #333333; border-radius: 0.25rem; padding: 0.35rem 0.375rem; border: none; vertical-align: baseline; color: #ffffff; font-size: inherit; -webkit-appearance: none; appearance: none; min-height: 2.1rem; }\n\n.component-input input::-webkit-input-placeholder { color: #858585; }\n\n.component-input input::-moz-placeholder { color: #858585; }\n\n.component-input input:-moz-placeholder { color: #858585; }\n\n.component-input input:-ms-input-placeholder { color: #858585; }\n\n.component-input input:focus { border: none; box-shadow: none; }\n\n.component-input input:focus::-webkit-input-placeholder { color: #666666; }\n\n.component-input input:focus::-moz-placeholder { color: #666666; }\n\n.component-input input:focus:-moz-placeholder { color: #666666; }\n\n.component-input input:focus:-ms-input-placeholder { color: #666666; }\n"},{}],25:[function(t,e,n){e.exports=".component-radio { display: block; }\n\n.section .component-radio { padding-right: 0.375rem; }\n\n.component-radio > .label { display: block; padding-bottom: 0.35rem; }\n\n.component-radio .radio-group { padding-bottom: 0.35rem; }\n\n.component-radio .radio-group label { padding: 0.35rem 0.375rem; }\n\n.component-radio .radio-group .label { font-size: 0.9em; }\n\n.component-radio .radio-group input { opacity: 0; position: absolute; }\n\n.component-radio .radio-group i { display: block; position: relative; border-radius: 1.4rem; width: 1.4rem; height: 1.4rem; border: 2px solid #767676; -webkit-flex-shrink: 0; flex-shrink: 0; }\n\n.component-radio .radio-group input:checked + i { border-color: #ff4700; }\n\n.component-radio .radio-group input:checked + i:after { content: ''; display: block; position: absolute; left: 15%; right: 15%; top: 15%; bottom: 15%; border-radius: 1.4rem; background: #ff4700; }\n\n.component-radio .description { padding-left: 0; padding-right: 0; }\n"},{}],26:[function(t,e,n){e.exports='.section .component-select { padding: 0; }\n\n.component-select label { position: relative; }\n\n.component-select .value { position: relative; padding-right: 1.1rem; display: block; }\n\n.component-select .value:after { content: ""; position: absolute; right: 0; top: 50%; margin-top: -0.1rem; height: 0; width: 0; border-left: 0.425rem solid transparent; border-right: 0.425rem solid transparent; border-top: 0.425rem solid #ff4700; }\n\n.component-select select { opacity: 0; position: absolute; display: block; left: 0; right: 0; top: 0; bottom: 0; width: 100%; border: none; margin: 0; padding: 0; }\n'},{}],27:[function(t,e,n){e.exports=".section .component-slider { padding: 0; }\n\n.component-slider label { display: block; }\n\n.component-slider .label-container { display: -webkit-box; display: -webkit-flex; display: flex; -webkit-box-align: center; -webkit-align-items: center; align-items: center; width: 100%; padding-bottom: 0.7rem; }\n\n.component-slider .label { -webkit-box-flex: 1; -webkit-flex: 1; flex: 1; min-width: 1rem; display: block; padding-right: 0.75rem; }\n\n.component-slider .value-wrap { display: block; position: relative; }\n\n.component-slider .value, .component-slider .value-pad { display: block; background: #333333; border-radius: 0.25rem; padding: 0.35rem 0.375rem; border: none; vertical-align: baseline; color: #ffffff; text-align: right; margin: 0; min-width: 1rem; }\n\n.component-slider .value-pad { visibility: hidden; }\n\n.component-slider .value-pad:before { content: ' '; display: inline-block; }\n\n.component-slider .value { max-width: 100%; position: absolute; left: 0; top: 0; }\n\n.component-slider .input-wrap { padding: 0 0.75rem 0.7rem; }\n\n.component-slider .input { display: block; position: relative; min-width: 100%; height: 1.4rem; overflow: hidden; margin-left: 0; }\n\n.component-slider .input:before { content: ''; display: block; position: absolute; height: 0.17647rem; background: #666666; width: 100%; top: 0.61176rem; }\n\n.component-slider .input .slider { display: block; width: 100%; -webkit-appearance: none; appearance: none; position: relative; height: 1.4rem; margin: 0; background-color: transparent; }\n\n.component-slider .input .slider:focus { outline: none; }\n\n.component-slider .input .slider::-webkit-slider-runnable-track { border: none; height: 1.4rem; width: 100%; background-color: transparent; }\n\n.component-slider .input .slider::-webkit-slider-thumb { -webkit-appearance: none; appearance: none; position: relative; height: 1.4rem; width: 1.4rem; background-color: #ff4700; border-radius: 50%; }\n\n.component-slider .input .slider::-webkit-slider-thumb:before { content: \"\"; position: absolute; left: -1000px; top: 0.61176rem; height: 0.17647rem; width: 1001px; background: #ff4700; }\n"},{}],28:[function(t,e,n){e.exports=".component-submit { text-align: center; }\n"},{}],29:[function(t,e,n){e.exports=".section .component-toggle { padding: 0; }\n\n.component-toggle input { display: none; }\n\n.component-toggle .graphic { display: inline-block; position: relative; }\n\n.component-toggle .graphic .slide { display: block; border-radius: 1.05rem; height: 1.05rem; width: 2.2652rem; background: #2f2f2f; -webkit-transition: background-color 150ms linear; transition: background-color 150ms linear; }\n\n.component-toggle .graphic .marker { background: #ececec; width: 1.4rem; height: 1.4rem; border-radius: 1.4rem; position: absolute; left: 0; display: block; top: -0.175rem; -webkit-transition: -webkit-transform 150ms linear; transition: -webkit-transform 150ms linear; transition: transform 150ms linear; transition: transform 150ms linear, -webkit-transform 150ms linear; box-shadow: 0 0.1rem 0.1rem #2f2f2f; }\n\n.component-toggle input:checked + .graphic .slide { background: #993d19; }\n\n.component-toggle input:checked + .graphic .marker { background: #ff4700; -webkit-transform: translateX(0.8652rem); transform: translateX(0.8652rem); }\n"},{}],30:[function(t,e,n){e.exports='