diff --git a/.github/workflows/docs-gh-pages.yml b/.github/workflows/docs-gh-pages.yml
new file mode 100644
index 00000000..ef4a93c9
--- /dev/null
+++ b/.github/workflows/docs-gh-pages.yml
@@ -0,0 +1,69 @@
+# Based on https://github.com/actions/starter-workflows/blob/main/pages/jekyll.yml
+name: Build Jekyll site, Deploy to Pages when on default branch
+
+env:
+ DOCS_DIR: docs
+
+on:
+ # Run workflow on any branch push.
+ # Conditionals are used to only trigger deploy on the default branch.
+ push:
+ # Uncomment to only run on specific branch pushes.
+ # branches: ["master"]
+
+ # Allows you to run this workflow manually from the Actions tab
+ workflow_dispatch:
+
+# Sets permissions of the GITHUB_TOKEN to allow deployment to GitHub Pages
+permissions:
+ contents: read
+ pages: write
+ id-token: write
+
+# Allow only one concurrent deployment per branch, skipping runs queued between the run in-progress and latest queued.
+# However, do NOT cancel in-progress runs as we want to allow the deployments to complete.
+concurrency:
+ group: "pages-${{ github.ref }}"
+ cancel-in-progress: false
+
+jobs:
+ # Build job
+ build:
+ runs-on: ubuntu-latest
+ steps:
+ - name: Checkout
+ uses: actions/checkout@v4
+ - name: Setup Ruby
+ uses: ruby/setup-ruby@8575951200e472d5f2d95c625da0c7bec8217c42 # v1.161.0
+ with:
+ ruby-version: '3.2' # Not needed with a .ruby-version file
+ working-directory: ${{ env.DOCS_DIR }}
+ bundler-cache: true # runs 'bundle install' and caches installed gems automatically
+ # cache-version: 0 # Increment this number if the cache gets corrupted and you need to force-update cached gems
+ - name: Setup Pages
+ id: pages
+ uses: actions/configure-pages@v4
+ - name: Build with Jekyll
+ # Outputs to the './_site' directory by default
+ run: cd $DOCS_DIR && bundle install && bundle exec jekyll build --baseurl "${{ steps.pages.outputs.base_path }}"
+ env:
+ JEKYLL_ENV: production
+ - name: Upload artifact
+ if: github.ref == 'refs/heads/master' # Only upload when on default branch
+ uses: actions/upload-pages-artifact@v3
+ with:
+ # Default path is './_site'.
+ path: "./${{ env.DOCS_DIR }}/_site"
+
+ # Deployment job
+ deploy:
+ if: github.ref == 'refs/heads/master' # Only deploy when on default branch
+ environment:
+ name: github-pages
+ url: ${{ steps.deployment.outputs.page_url }}
+ runs-on: ubuntu-latest
+ needs: build
+ steps:
+ - name: Deploy to GitHub Pages
+ id: deployment
+ uses: actions/deploy-pages@v4
diff --git a/docs/.gitignore b/docs/.gitignore
new file mode 100644
index 00000000..f40fbd8b
--- /dev/null
+++ b/docs/.gitignore
@@ -0,0 +1,5 @@
+_site
+.sass-cache
+.jekyll-cache
+.jekyll-metadata
+vendor
diff --git a/docs/404.html b/docs/404.html
new file mode 100644
index 00000000..086a5c9e
--- /dev/null
+++ b/docs/404.html
@@ -0,0 +1,25 @@
+---
+permalink: /404.html
+layout: default
+---
+
+
+
+
+
404
+
+
Page not found :(
+
The requested page could not be found.
+
diff --git a/docs/Gemfile b/docs/Gemfile
new file mode 100644
index 00000000..ca533039
--- /dev/null
+++ b/docs/Gemfile
@@ -0,0 +1,43 @@
+source "https://rubygems.org"
+# Hello! This is where you manage which Jekyll version is used to run.
+# When you want to use a different version, change it below, save the
+# file and run `bundle install`. Run Jekyll with `bundle exec`, like so:
+#
+# bundle exec jekyll serve
+#
+# This will help ensure the proper Jekyll version is running.
+# Happy Jekylling!
+gem "jekyll", "~> 4.3.2"
+
+# Theme
+gem "just-the-docs", "0.7.0"
+# gem "minima", "~> 2.5" # This is the default theme for new Jekyll sites.
+
+# If you want to use GitHub Pages, remove the "gem "jekyll"" above and
+# uncomment the line below. To upgrade, run `bundle update github-pages`.
+# Latest GH Pages versions are here - https://pages.github.com/versions/
+# gem "github-pages", "~> 228", group: :jekyll_plugins
+# If you have any plugins, put them here!
+group :jekyll_plugins do
+ gem "jekyll-feed", "~> 0.12"
+ # gem "jekyll-remote-theme"
+end
+
+# Windows and JRuby does not include zoneinfo files, so bundle the tzinfo-data gem
+# and associated library.
+platforms :mingw, :x64_mingw, :mswin, :jruby do
+ gem "tzinfo", ">= 1", "< 3"
+ gem "tzinfo-data"
+end
+
+# Performance-booster for watching directories on Windows
+gem "wdm", "~> 0.1.1", :platforms => [:mingw, :x64_mingw, :mswin]
+
+# Lock `http_parser.rb` gem to `v0.6.x` on JRuby builds since newer versions of the gem
+# do not have a Java counterpart.
+gem "http_parser.rb", "~> 0.6.0", :platforms => [:jruby]
+
+# Lock jekyll-sass-converter to v2.
+# This lock can be removed when upgrading from Ruby 2.7 to Ruby 3.
+# https://github.com/jekyll/jekyll/pull/9225#issuecomment-1363894633
+gem "jekyll-sass-converter", "~> 2.0"
diff --git a/docs/Gemfile.lock b/docs/Gemfile.lock
new file mode 100644
index 00000000..a5038268
--- /dev/null
+++ b/docs/Gemfile.lock
@@ -0,0 +1,88 @@
+GEM
+ remote: https://rubygems.org/
+ specs:
+ addressable (2.8.6)
+ public_suffix (>= 2.0.2, < 6.0)
+ colorator (1.1.0)
+ concurrent-ruby (1.2.2)
+ em-websocket (0.5.3)
+ eventmachine (>= 0.12.9)
+ http_parser.rb (~> 0)
+ eventmachine (1.2.7)
+ ffi (1.16.3)
+ forwardable-extended (2.6.0)
+ http_parser.rb (0.8.0)
+ i18n (1.14.1)
+ concurrent-ruby (~> 1.0)
+ jekyll (4.3.2)
+ addressable (~> 2.4)
+ colorator (~> 1.0)
+ em-websocket (~> 0.5)
+ i18n (~> 1.0)
+ jekyll-sass-converter (>= 2.0, < 4.0)
+ jekyll-watch (~> 2.0)
+ kramdown (~> 2.3, >= 2.3.1)
+ kramdown-parser-gfm (~> 1.0)
+ liquid (~> 4.0)
+ mercenary (>= 0.3.6, < 0.5)
+ pathutil (~> 0.9)
+ rouge (>= 3.0, < 5.0)
+ safe_yaml (~> 1.0)
+ terminal-table (>= 1.8, < 4.0)
+ webrick (~> 1.7)
+ jekyll-feed (0.17.0)
+ jekyll (>= 3.7, < 5.0)
+ jekyll-include-cache (0.2.1)
+ jekyll (>= 3.7, < 5.0)
+ jekyll-sass-converter (2.2.0)
+ sassc (> 2.0.1, < 3.0)
+ jekyll-seo-tag (2.8.0)
+ jekyll (>= 3.8, < 5.0)
+ jekyll-watch (2.2.1)
+ listen (~> 3.0)
+ just-the-docs (0.7.0)
+ jekyll (>= 3.8.5)
+ jekyll-include-cache
+ jekyll-seo-tag (>= 2.0)
+ rake (>= 12.3.1)
+ kramdown (2.4.0)
+ rexml
+ kramdown-parser-gfm (1.1.0)
+ kramdown (~> 2.0)
+ liquid (4.0.4)
+ listen (3.8.0)
+ rb-fsevent (~> 0.10, >= 0.10.3)
+ rb-inotify (~> 0.9, >= 0.9.10)
+ mercenary (0.4.0)
+ pathutil (0.16.2)
+ forwardable-extended (~> 2.6)
+ public_suffix (5.0.4)
+ rake (13.1.0)
+ rb-fsevent (0.11.2)
+ rb-inotify (0.10.1)
+ ffi (~> 1.0)
+ rexml (3.2.6)
+ rouge (4.2.0)
+ safe_yaml (1.0.5)
+ sassc (2.4.0)
+ ffi (~> 1.9)
+ terminal-table (3.0.2)
+ unicode-display_width (>= 1.1.1, < 3)
+ unicode-display_width (2.5.0)
+ webrick (1.8.1)
+
+PLATFORMS
+ ruby
+
+DEPENDENCIES
+ http_parser.rb (~> 0.6.0)
+ jekyll (~> 4.3.2)
+ jekyll-feed (~> 0.12)
+ jekyll-sass-converter (~> 2.0)
+ just-the-docs (= 0.7.0)
+ tzinfo (>= 1, < 3)
+ tzinfo-data
+ wdm (~> 0.1.1)
+
+BUNDLED WITH
+ 2.1.4
diff --git a/docs/README.md b/docs/README.md
new file mode 100644
index 00000000..c585e184
--- /dev/null
+++ b/docs/README.md
@@ -0,0 +1,44 @@
+These docs are built with Jekyll and served on GitHub Pages.
+
+# Locally building and viewing docs
+
+Jekyll has a dev server, which will auto-build the docs upon any changes to the source Markdown files. Only changes to `_config.yml` require a dev server restart.
+
+## With local Ruby
+
+Setup:
+
+```
+cd derby-docs && bundle install
+```
+
+Run the dev server:
+
+```
+bundle exec jekyll serve
+```
+
+The site is viewable at `http://localhost:4000/`.
+
+## With Ruby in Docker container
+
+One-time container creation:
+
+```
+docker run --name derby-docs-ruby -v "$(pwd)/docs:/derby-docs" -p 127.0.0.1:4000:4000 ruby:2.7 bash -c 'cd derby-docs && bundle install && bundle exec jekyll serve -H 0.0.0.0 -P 4000 --trace'
+```
+
+The site is viewable at `http://localhost:4000/`.
+
+Explanation of flags:
+* `-v` - Set up a Docker bind mount, mapping the host's `$PWD/docs` directory to a container directory `/derby-docs`.
+* `-p` - Map the host's local port 4000 to the container's port 4000, to allow the dev server inside the container to serve requests issued against the host.
+* `-H 0.0.0.0 -P 4000` - Have the dev server listen to connections from outside the container. This won't allow connections from outside the host.
+
+Subsequently, to run the dev server:
+
+```
+docker start -i derby-docs-ruby
+```
+
+To recreate the container with a different command or setup, run `docker rm derby-docs-ruby` to delete the container first.
diff --git a/docs/_config.yml b/docs/_config.yml
new file mode 100644
index 00000000..257e45ce
--- /dev/null
+++ b/docs/_config.yml
@@ -0,0 +1,75 @@
+# Welcome to Jekyll!
+#
+# This config file is meant for settings that affect your whole blog, values
+# which you are expected to set up once and rarely edit after that. If you find
+# yourself editing this file very often, consider using Jekyll's data files
+# feature for the data you need to update frequently.
+#
+# For technical reasons, this file is *NOT* reloaded automatically when you use
+# 'bundle exec jekyll serve'. If you change this file, please restart the server process.
+#
+# If you need help with YAML syntax, here are some quick references for you:
+# https://learn-the-web.algonquindesign.ca/topics/markdown-yaml-cheat-sheet/#yaml
+# https://learnxinyminutes.com/docs/yaml/
+#
+# Site settings
+# These are used to personalize your new site. If you look in the HTML files,
+# you will see them accessed via {{ site.title }}, {{ site.email }}, and so on.
+# You can create any custom variable you would like, and they will be accessible
+# in the templates via {{ site.myvariable }}.
+title: DerbyJS Docs
+baseurl: "/derby" # the subpath of your site, e.g. /blog
+url: "" # the base hostname & protocol for your site, e.g. http://example.com
+repository: derbyjs/derby
+
+aux_links:
+ "DerbyJS on GitHub":
+ - "//github.com/derbyjs/derby"
+
+# Footer "Edit this page on GitHub" link text
+gh_edit_link: true # show or hide edit this page link
+gh_edit_link_text: "Edit this page on GitHub"
+gh_edit_repository: "https://github.com/derbyjs/derby" # the github URL for your repo
+gh_edit_branch: "master" # the branch that your docs is served from
+gh_edit_source: "docs" # the source that your files originate from
+gh_edit_view_mode: "tree" # "tree" or "edit" if you want the user to jump into the editor immediately
+
+# Build settings
+markdown: kramdown
+# remote_theme: just-the-docs/just-the-docs
+theme: just-the-docs
+permalink: /:path/:name
+
+# Front matter defaults
+defaults:
+ -
+ scope:
+ path: "" # an empty string here means all files in the project
+ type: "pages"
+ values:
+ render_with_liquid: false
+ -
+ scope:
+ path: "assets" # an empty string here means all files in the project
+ values:
+ render_with_liquid: true
+
+# Exclude from processing.
+# The following items will not be processed, by default.
+# Any item listed under the `exclude:` key here will be automatically added to
+# the internal "default list".
+#
+# Excluded items can be processed by explicitly listing the directories or
+# their entries' file path in the `include:` list.
+#
+# exclude:
+# - .sass-cache/
+# - .jekyll-cache/
+# - gemfiles/
+# - Gemfile
+# - Gemfile.lock
+# - node_modules/
+# - vendor/bundle/
+# - vendor/cache/
+# - vendor/gems/
+# - vendor/ruby/
diff --git a/docs/_sass/custom.scss b/docs/_sass/custom.scss
new file mode 100644
index 00000000..e0783194
--- /dev/null
+++ b/docs/_sass/custom.scss
@@ -0,0 +1,37 @@
+/**
+ * Add call-out support: https://github.com/pmarsceill/just-the-docs/issues/171#issuecomment-538794741
+ */
+ $callouts: (
+ info: ($blue-300, rgba($blue-000, .2), 'Note'),
+ warn: ($yellow-300, rgba($yellow-000, .2), 'Note'),
+ danger: ($red-300, rgba($red-000, .2), 'Note')
+);
+
+@each $class, $props in $callouts {
+ .#{$class} {
+ background: nth($props, 2);
+ border-left: $border-radius solid nth($props, 1);
+ border-radius: $border-radius;
+ box-shadow: 0 1px 2px rgba(0, 0, 0, 0.12), 0 3px 10px rgba(0, 0, 0, 0.08);
+ padding: .8rem;
+
+ &::before {
+ color: nth($props, 1);
+ content: nth($props, 3);
+ display: block;
+ font-weight: bold;
+ font-size: .75em;
+ padding-bottom: .125rem;
+ }
+
+ br {
+ content: '';
+ display: block;
+ margin-top: .5rem;
+ }
+ }
+}
+
+.label-grey {
+ background: rgba($grey-dk-000, 1);
+}
diff --git a/docs/apps.md b/docs/apps.md
new file mode 100644
index 00000000..923d4347
--- /dev/null
+++ b/docs/apps.md
@@ -0,0 +1,67 @@
+---
+layout: default
+title: Apps
+---
+
+# Derby Apps
+
+Derby projects support one or more single-page apps.
+Apps have a full MVC structure, including a model provided by
+[Racer](https://github.com/derbyjs/racer), a template and styles based view, and controller
+code with application logic and routes (which map URLs to actions).
+
+On the server, apps provide a router middleware for Express. One or more app
+routers as well as server only routes can be included in the same Express
+server.
+
+Derby packages up all of an app's templates, routes, and application code when
+rendering. Regardless of which app URL the browser requests initially, the app
+is able to render any other state within the same application client-side. If
+the app cannot handle a URL, it will fall through and request from the server.
+Errors thrown during route handling also cause requests to fall through to the
+server.
+
+Derby works great with only a single app, though developers may wish to create
+separate apps if only certain sets of pages are likely to be used together. For
+example, a project might have a separate desktop web app and mobile web app. Or
+a project might have an internal administration panel app and a public content
+app.
+
+
+## Creating apps
+
+Apps are created in the file that defines the app's controller code. They are
+then associated with a server by requiring the app within the server file.
+
+> `app = derby.createApp ( name, fileName )`
+>
+> * `name`: the name of the app
+> * `fileName`: the name of the file, typically node's default __filename is used.
+>
+> * `app`: Returns an app object, typically exported as `module.exports = app`
+
+
+App names are used to automatically associate an app with template and styles files of the same
+name.
+
+The `createApp` method adds a number of methods to the app. On both the client
+and the server, these are `use`, `get`, `post`, `put`, `del`,
+and `ready`. On the server only, Derby also adds `router`,
+for use with Express.
+
+## Connecting servers to apps
+
+Because Derby shares most code between server and client, Derby server files
+can be very minimal.
+
+The server includes an app with a standard Node.js require statement. It can
+then use the `app.router()` method to create a router middleware for Express
+that handles all of the app's routes.
+
+The server also needs to create a `store` object, which is what creates models,
+coordinates data syncing, and interfaces with databases. Stores are created via
+the `derby.createStore()` method. See [Backends](models/backends).
+
+> A typical setup can be seen in the [derby-starter](https://github.com/derbyjs/derby-starter/blob/master/lib/server.js) project, which is a node module for getting started with Derby.
+>
+> The [derby-examples](https://github.com/derbyjs/derby-examples) make use of derby-starter to setup their apps.
diff --git a/docs/components.md b/docs/components.md
new file mode 100644
index 00000000..ccf175e5
--- /dev/null
+++ b/docs/components.md
@@ -0,0 +1,187 @@
+---
+layout: default
+title: Components
+has_children: true
+---
+
+# Overview
+
+Components are the building blocks of Derby applications. A component is a view associated with a controller class. The [view](views) is implemented as a Derby template and the controller is implemented as a JavaScript [class](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Classes) or [constructor function](https://developer.mozilla.org/en-US/docs/Learn/JavaScript/Objects/Object-oriented_JS). Derby creates an instance of the controller class each time it renders the component view.
+
+
+## Reuse and organization
+
+Components are reusable UI pieces, similar to custom HTML elements. In addition, they are the recommended way to structure complex applications as modular parts with clear inputs and outputs. Each significant unit of UI functionality should be its own component.
+
+Components can be rendered on the server and the client, so the same code can produce static HTML, server-rendered dynamic applications, and client-rendered applications.
+
+
+## Encapsulation
+
+Each component has a scoped model in its own namespace. Data or references to the component's parent are passed in via view attributes. If you're familiar with it, this structure is similar to the [Model-View-ViewModel (MVVM) pattern](https://en.wikipedia.org/wiki/Model%E2%80%93view%E2%80%93viewmodel)—a component's scoped model is a ViewModel.
+
+
+## Tabs Example
+
+### index.html
+```derby
+
+
+
+
Some stuff here
+
+
+
More stuff
+
+
+```
+
+### tabs.html
+```derby
+
+
+ {{each @pane as #pane, #i}}
+ {{with #i === selectedIndex as #isActive}}
+
+
+
+
+```
+
+```js
+app.component('todos-new', class TodosNew {
+ submit() {
+ const value = this.model.del('value');
+ this.emit('submit', value);
+ }
+});
+
+app.component('todos-list', class TodosList {
+ add(text) {
+ if (!text) return;
+ this.model.push('items', {text});
+ }
+ remove(index) {
+ this.model.remove('items', index);
+ }
+});
+
+app.component('todos-footer', class TodosFooter {
+ static singleton = true;
+ remaining(items) {
+ if (!items) return 0;
+ return items.filter(item => !item.done).length;
+ }
+});
+```
+
+
+
+
diff --git a/docs/components/component-class.md b/docs/components/component-class.md
new file mode 100644
index 00000000..1fb42f3c
--- /dev/null
+++ b/docs/components/component-class.md
@@ -0,0 +1,194 @@
+---
+layout: default
+title: Component class
+parent: Components
+---
+
+# Component class
+
+Derby provides a base class `Component`, from which all component classes inherit. When authoring a component, you can extend Derby's Component class with JavaScript, TypeScript, or CoffeeScript `extends` syntax.
+
+```js
+const Component = require('derby').Component;
+class MyComponent extends Component {
+ ...
+}
+app.component(MyComponent);
+```
+
+For convenience, if you register a class that does not inherit from `Component`, Derby will add `Component.prototype` to your class's prototype chain. In other words, Derby will make sure that your class inherits from `Component` at the time that you call `app.component()`.
+
+```js
+class MyComponent {
+ ...
+}
+app.component(MyComponent);
+```
+
+## Component configuration
+
+Components are configured by defining the following static properties and methods:
+
+> `MyComponent.view = '/path/to/view'` The relative file path to a template file to load. If the view file is named *index.html* and in the same directory as the controller, `__dirname` can be used
+
+> `MyComponent.is = 'my-component'` The name to use for the component's view. Often this doesn't need to be specified, because it defaults to the basename of the file or directory.
+
+> `MyComponent.DataConstructor` Constructor function for setting default values in the component's model data. Properties will be overriden by view attributes.
+
+> `MyComponent.prototype.init = function(model)` Called immediately before the view is rendered. Data and reactive functions can be initialized on the component's scoped model. This method is invovked both on the server and on the client, so the DOM and browser-only methods may not be used within init().
+
+> `MyComponent.prototype.create = function(model, dom)` Called in the browser when a component is loaded and inserted into the DOM. This method is never called on the server. DOM-related code and model event listeners should be placed in create().
+
+
+## Properties
+
+> `model`: The component's scoped model.
+
+> `dom`: An instance of Derby's wrapper around DOM methods. This should be used for adding and removing listeners to DOM events rather than native `addEventListener()`. This is important so that Derby can remove listeners when the component is destroyed.
+
+> `page`: Reference to the current page object, which is the top level controller. A new page object is created on navigation to a new URL.
+
+> `app`: Reference to the current app object. The app is persistent for the entire session.
+
+> `parent`: Reference to the containing controller.
+
+> `context`: The rendering context object.
+
+> `id`: The unique id assigned to the component.
+
+> `isDestroyed`: Initially set to `false`. Set to `true` when the component is fully destroyed.
+
+
+## Methods
+
+### Event emission
+
+Components are Node.js event emitters, so they inherit the `on`, `once`, `emit`, `removeListener`, etc. methods from [EventEmitter](https://nodejs.org/api/events.html#events_class_eventemitter).
+
+### Cleanup
+
+> `component.destroy()`
+>
+> Derby calls this method when removing a component's marker comment from the DOM. `destroy()` emits the `'destroy'` event on the component. Listen for the destroy event in order to implement custom cleanup logic. This method should not be invoked manually.
+
+```derby
+
+```
+
+```js
+class MyComponent extends Component {
+ create() {
+ this.on('destroy', function() {
+ // Custom cleanup logic
+ });
+ }
+}
+```
+
+> `boundFn = component.bind(fn)`
+> * `fn` - _Function_ - A function to be invoked with the component as its `this` value. In addition, the function will no longer be invoked once the component is destroyed
+> * `boundFn` - _Function_ - Returns a bound function, similar to JavaScript's `Function.bind()`. This function is safer to use in asynchronous code, such as with setTimeout, requestAnimationFrame, or requests to the server, because it won't call back after the component is destroyed. Internally, references to `fn` and the component are removed on `'destroy'`, which allows them to be garbage collected even if a reference to `boundFn` is held.
+
+```js
+class MyComponent extends Component {
+ load() {
+ this.set('loading', true);
+ setTimeout(this.bind(function() {
+ // This won't execute if the component has been destroyed
+ this.set('loading', false);
+ }), 200);
+ }
+}
+```
+
+### Throttling and debouncing
+
+Derby components have built-in support for common throttling and debouncing patterns. These methods are similar to those provided by general-purpose libraries like Lodash, but they also bind the `this` value to the component, provide added safety by not calling back after a component is destroyed, and release references to `fn` and the component on `'destroy'`, same as `component.bind(fn)`.
+
+> `throttledFn = component.throttle(fn, [delayArg = 0])`
+>
+> When passing in a numeric delay, calls the function at most once per that many milliseconds. Like Lodash, the function will be called on the leading and the trailing edge of the delay as appropriate. Unlike Lodash, calls are consistently called via setTimeout and are never synchronous. This should be used for reducing the frequency of ongoing updates, such as scroll events or other continuous streams of events.
+>
+> Additionally, implements an interface intended to be used with [window.requestAnimationFrame](https://developer.mozilla.org/en-US/docs/Web/API/window/requestAnimationFrame), process.nextTick, or window.setImmediate. If one of these is passed, it will be used to create a single async call following any number of synchronous calls. This mode is typically used to coalesce many synchronous events (such as multiple model events) into a single async event.
+>
+> Like `component.bind()`, will no longer call back once the component is destroyed, which avoids possible bugs and memory leaks.
+
+```js
+class MyComponent extends Component {
+ create() {
+ // Call this.update() at most once every 75 milliseconds
+ this.dom.on('scroll', window, this.throttle(this.update, 75));
+ }
+ update() {
+ // Update based on scroll location
+ }
+}
+```
+
+```js
+class MyComponent extends Component {
+ create() {
+ // Call this.update() at most once before each paint (typically 60 times / second)
+ this.dom.on('scroll', window, this.throttle(this.update, window.requestAnimationFrame));
+ }
+ update() {
+ // Update based on scroll location
+ }
+}
+```
+
+> `debouncedFn = component.debounce(fn, [delay = 0])`
+>
+> Suppresses calls until the function is no longer called for that many milliseconds. This should be used for delaying updates triggered by user input, such as window resizing, or typing text that has a live preview or client-side validation. This should not be used for inputs that trigger server requests, such as search autocomplete; use debounceAsync for those cases instead.
+>
+> Like `component.bind()`, will no longer call back once the component is destroyed, which avoids possible bugs and memory leaks.
+
+```derby
+
+
+```
+
+```js
+class MyComponent extends Component {
+ create() {
+ // Suppress calls until the user has stopped typing for 300 milliseconds
+ this.dom.on('input', this.textInput, this.debounce(this.update, 300));
+ }
+ update() {
+ // Update based on current value
+ }
+}
+```
+
+> `asyncDebouncedFn = component.debounceAsync(fn, [delay = 0])`
+>
+> Like debounce(), suppresses calls until the function is no longer called for that many milliseconds. In addition, suppresses calls while the callback function is running. In other words, the callback will not be called again until the supplied `done()` argument is called. When the debounced function is called while the callback is running, the callback will be called again immediately after `done()` is called. Thus, the callback will always receive the last value passed to the debounced function.
+>
+> This avoids the potential for multiple callbacks to execute in parallel and complete out of order. It also acts as an adaptive rate limiter. Use this method to debounce any field that triggers an async call as the user types.
+>
+> Like `component.bind()`, will no longer call back once the component is destroyed, which avoids possible bugs and memory leaks.
+
+```derby
+
+
+```
+
+```js
+class MyComponent extends Component {
+ create() {
+ // Suppress calls until the user has stopped typing for 300 milliseconds
+ // and the async function has completed
+ this.dom.on('input', this.textInput, this.debounceAsync(this.search, 300));
+ }
+ search(done) {
+ const query = this.model.get('value');
+ fetch('/api/search?q=' + query)
+ .then(response => {
+ this.model.set('response', response);
+ })
+ .catch(err => console.error(err))
+ // No additional calls to search will happen until done() is called
+ .finally(done);
+ }
+}
+```
diff --git a/docs/components/events.md b/docs/components/events.md
new file mode 100644
index 00000000..f785504a
--- /dev/null
+++ b/docs/components/events.md
@@ -0,0 +1,84 @@
+---
+layout: default
+title: Events
+parent: Components
+---
+
+# Events
+
+Functions defined on a property of a controller can be invoked from view expressions or in response to events. As a general pattern, view paths refer to the model when getting values and to the controller when calling functions.
+
+Functions are looked up on the current component's controller, the page, and the global, in that order. See the [view functions and events](../views/template-syntax/functions-and-events#controller-property-lookup) documentation for more detail.
+
+## Lifecycle events
+
+Default events are triggered during the lifecycle of a component:
+
+* `init`: Emitted before the component's `init()` function is called.
+* `create`: Emitted before the component's `create()` function is called.
+* `destroy`: Emitted before the component's `destroy()` function is called.
+
+If the functions to be called aren't defined on the component, their respective events are still triggered unconditionally.
+
+## Custom events
+
+Components support custom events. Dashes are transformed into camelCase.
+```derby
+
+```
+```js
+// Equivalent to:
+modal.on('close', function() {
+ self.reset();
+});
+modal.on('fullView', function() {
+ back.fade();
+});
+```
+
+## Emitting events
+Components can emit custom events to be handled by their parents.
+
+```derby
+
+
+```
+
+```js
+ //listen
+ modal.on('fullView', function(foo) {
+ console.log(foo);
+ })
+ //...
+ //emit
+ modal.emit("fullView", foo);
+```
+
+
+## Calling peer component methods
+
+Components and elements can be set as a property on the current controller with the `as=` HTML attribute ([more detail](../views/template-syntax/paths#controller-properties)). This paired with how controller properties are looked up on function calls makes it easy to connect events on components or elements to methods on other components.
+
+```derby
+
+
+
+```
+
+```derby
+
+
+...
+
+```
+
+## Component event arguments
+
+Component events implicitly pass through any emitted arguments. These arguments are added after any explicitly specified arguments in the expression.
+
+```derby
+
+
+
+
+```
diff --git a/docs/components/lifecycle.md b/docs/components/lifecycle.md
new file mode 100644
index 00000000..476965ff
--- /dev/null
+++ b/docs/components/lifecycle.md
@@ -0,0 +1,120 @@
+---
+layout: default
+title: Lifecycle
+parent: Components
+---
+
+# Component lifecycle
+
+
+Components are defined by writing a view template in HTML and a JavaScript controller class. Calling `app.component(MyComponent)` registers the view and controller with a Derby app. After that, an instance of the component class is created whenever its view is rendered.
+
+
+## Rendering
+
+Derby components are designed for efficient server-side HTML and client-side DOM rendering with the same code.
+
+
+### Client-side rendering
+
+*1. `new MyComponent.DataConstructor()`:* First, Derby instantiates an object that is used for the component's model data. This class should be defined to set default values that can be used in either the view or the controller of the component. Derby will set two properties, `id` and `$controller`, on this object. `id` is a string id unique to the component, and `$controller` is a reference to the component instance. These properties are set in the model data so that they can be used in the view.
+
+*2. `new MyComponent(context, data)`:* Derby calls the component's constructor with a rendering context object and the model data instance. As is idiomatic in JavaScript, it is recommended that `super(context, data)` is called at the top, and instance properties are set following that. Note that attributes passed in from the view are not yet set on the model.
+
+*3. `Component(context, data)`:* Derby's component constructor creates the component's `id` and `model`. If the custom constructor does not call super, Derby calls this method immediately after. Effectively, components behave like there was a call to super at the end of their constructor when super isn't called explicitly.
+
+*4. Attribute values from the view are set on the model:* Attribute values are passed into compoenents via the view. Derby sets and binds these values to the model after invoking the constructor.
+
+*5. `'init'` event:* Derby emits an `'init'` event on the component instance before calling the init method. This event is rarely used. However, it is provided in case containing components need to obtain a reference to a component instance before it renders.
+
+*6. `MyComponent.init(model)`:* Init is called once Derby has completed all steps to initialize the component and before rendering. All custom code that initializes data in the model prior to rendering, such as reactive functions, should be placed within the component's `init()` implementation.
+
+*7. Rendering:* Following `init()`, Derby renders the component's view and inserts it into the DOM. Hooks defined in the view, such as the `as` attribute for assigning elements and components to controller properties or `on-` attributes for adding event listeners happen at render time as well.
+
+*8. `'create'` event:* On the client only, Derby emits a `'create'` event on the component instance before calling the create method. Similar to the `'init'` event, this method is provided in case containing components need to obtain a reference to a component instance. However, the create event only happens on the client, and it is emitted after the component is rendered and inserted into the DOM.
+
+*9. `MyComponent.create(model, dom)`:* Create is called once the component is rendered and inserted into the DOM. Custom code that adds model or DOM event listeners should be placed within the component's `create()` implementation.
+
+
+### Server-side rendering
+
+Steps 1-7 of the rendering process are the same for server-side rendering. However, instead of using DOM methods to render, Derby returns a string of HTML. The key difference between client-side and server-side rendering is that `create()` is called only on the client, and server-side rendering happens within Node.js instead of the browser.
+
+There are a number of differences between Node.js and a browser environment that must be taken into account:
+
+* Server time and client time will differ. Client-time may differ by small or large amounts, and its accuracy cannot be ensured.
+
+* Servers do not have a session-appropriate timezone or locale, so JavaScript's `Date` and `Intl` features cannot be used without ensuring that the server and client are using matching implementations and configurations.
+
+* No DOM methods are available on the server.
+
+* There is no need to create bindings and event listeners on the server, because data will not be dynamically changing.
+
+* Servers are multi-tenent and long lived, so be careful to avoid global state in components. This is also a best practice in client-only applications, but it is especially important when code is executed on both the server and the client. On the server, shared state could lead to data being leaked from one session to another, and minor memory leaks in long-lived processes can build up and crash a server.
+
+
+### Server-side rendering + Client-side attachment
+
+Out of the box, Derby is optimized for server + client-side rendering. This can greatly improve perceived load time, since the browser can display the application before its scripts have loaded or executed on the client.
+
+In this type of rendering, the server renders HTML, and the browser creates DOM nodes from HTML. Then, Derby does a special kind of rendering called "attachment," where it does all of the client-side rendering steps. However, it uses the existing DOM nodes in the page rather than creating new DOM nodes.
+
+Therefore, component code must be deterministic. Pure code, where the same inputs return the same results and there are no side effects, is best. Common pitfalls:
+
+* Components should not rely on external inputs, such as `Date.now()` or the result of `Math.random()` in rendering, because their results will differ on subsequent calls. You can compute the values ahead of time and store them in the model on `_session` or `_page`, so that the values will be the same when rendered on the server and after the initial page load in the client. You may also choose to render certain values in the client only, by setting them in `create()`.
+
+* Sorting should use a stable comparison.
+
+* Rendering components should not modify persistent state or have other side effects.
+
+
+In addition, Derby requires that parsing the HTML in templates produces a matching DOM. Common pitfalls:
+
+* Templates must be valid HTML. For example, ``, is invalid HTML and will produce a DOM such as ``. This is because the [`
` element](https://html.spec.whatwg.org/multipage/grouping-content.html#the-p-element) may contain only [phrasing content](https://html.spec.whatwg.org/multipage/dom.html#phrasing-content), and the start of a `
` closes the `
`.
+
+* Templates must explicitly include all [optional tags](https://html.spec.whatwg.org/multipage/syntax.html#optional-tags). For example, `
` is valid HTML, but it will produce the DOM `
`. For simplicity, Derby does not attempt to implement these rules and requires that optional tags be written out.
+
+* All non-void elements must be explicitly closed. For example, `
One
Two
` is valid HTML, because an `
` elements' end tags are implied. Derby requires that this be written out fully as `
One
Two
`. ([Void elements](https://html.spec.whatwg.org/multipage/syntax.html#void-elements) like `` only have a start tag and the end tag must not be specified.)
+
+To test whether an HTML fragment will work in a Derby template, use an [HTML validator](https://validator.nu/) and check that setting then reading it back as `innerHTML` returns the same string.
+
+```js
+var html = '';
+var div = document.createElement('div');
+div.innerHTML = html;
+html === div.innerHTML;
+```
+
+
+## Cleanup
+
+When a binding causes a component to be removed from the DOM, Derby internally calls its `destroy()` method. (This method should not be invoked manually.) Destroying a component removes its DOM listeners, destroys its model data and model listeners, removes references created by `as` attributes in views, and removes all of Derby's internal references to the component and bindings within the component. Each of these is important for avoiding memory leaks.
+
+Using Derby's built-in features to add DOM listeners, model listeners, and bind asynchronous callbacks generally avoids the need to implement custom cleanup code. If custom cleanup code is needed, it can be implemented by listening to the component's `'destroy'` event or checking whether the component's `isDestroyed` property is `true`.
+
+
+### Singleton (stateless) components
+
+Creating a model per component, binding component attributes, and cleaning up component models and bindings can add significant overhead. However, Derby's template syntax is very expressive, and many components can be written in a stateless manner with no need for their own model.
+
+In this case, it is best to declare the component as a "singleton" component. A singleton component is also implemented with a JavaScript class for a controller, but Derby will only instantiate the class once and reuse the same instance of the class each time the component's view is rendered. Derby will not create a model or other properties on the controller, since its instance can be used in multiple places simultaneously. In addition, rendering a singleton component does not invoke `init()`, `create()`, or `destroy()`.
+
+Since singleton components do not have a model, only attribute paths may be used in views. Singleton controllers should consist of only pure functions.
+
+When a component is used many times on a page, such as a repeated item in a list or a commonly used UI element, it is best to write it statelessly for better performance. View partials are the most lightweight, singleton components allow use of custom JavaScript, and full components have their own model state.
+
+```derby
+
+
+ {{getInitials(@user.fullName)}}
+
+```
+
+```js
+app.component('user-icon', class UserIcon {
+ static singleton = true;
+ getInitials(fullName) {
+ return fullName.split(' ').map(name => name[0]).join('');
+ }
+});
+```
diff --git a/docs/components/scope.md b/docs/components/scope.md
new file mode 100644
index 00000000..a5401e81
--- /dev/null
+++ b/docs/components/scope.md
@@ -0,0 +1,52 @@
+---
+layout: default
+title: Scope
+parent: Components
+---
+
+# Scope
+
+Each component instance has its own scoped model, providing it isolation from model data for other components and remote collection data.
+
+## Attributes and data bindings
+
+The most direct way to get data into a component is to pass in a reference or a literal as a view attribute.
+
+```derby
+
+
+
+
+
+ {{each data as #user}}
+
{{#user.name}}
+ {{/each}}
+
+ {{num + 10}}
+```
+
+See [view attributes](../views/template-syntax/view-attributes) for more information.
+
+
+## Root model
+
+There are times when accessing data in the root model is desirable from within the component. This can be achieved both in the template and in the controller.
+
+```derby
+
+
+ {{#root.users[userId]}}
+```
+
+```js
+ var users = this.model.root.get("users");
+ var user = users[userId];
+ // or
+ var $users = this.model.scope("users");
+ var user = $users.get(userId);
+```
+
+
+### With block
+See the documentation for [with blocks](../views/template-syntax/blocks#with) to pass in data with an alias.
+
diff --git a/docs/components/view-partials.md b/docs/components/view-partials.md
new file mode 100644
index 00000000..1d84c8b1
--- /dev/null
+++ b/docs/components/view-partials.md
@@ -0,0 +1,69 @@
+---
+layout: default
+title: View partials
+parent: Components
+---
+
+# View partials
+
+This page goes into more detail about how view partials relate to components. For more general concepts, see the [template syntax](../views/template-syntax) documentation.
+
+While a component's controller is associated with a single view, it can contain sub-views defined as view partials. Components can also accept other views passed in as attributes.
+
+## Scope
+By default a view partial inherits the scope where it is instantiated.
+
+```derby
+
+ {{foo}}
+ {{with #root.bar as #bar}}
+
+ {{/with}}
+
+
+ i can render {{foo}} and {{#bar}}
+```
+A view partial associated with a component follows the [component scope](scope) rules. A view partial used inside a component will inherit the scope of the component.
+
+### extend
+It is possible to override another component's functionality while preserving its view. You can do this with the `extend` keyword.
+
+
+
+
+
+
+### import
+If you just want to reuse a view partial the `import` keyword is probably more appropriate. See the [namespaces and files](../views/namespaces-and-files#structuring-views-in-multiple-files) documentation for more details.
+
+
+## Programatic view management
+
+> `view = this.getView(name)`
+> * `name` the name of the view
+> * `view` a template object representing the view
+
+It is possible to access the views in a component's namespace from the controller. This may be used in conjunction with `setAttribute` to override a component's default rendering.
+An example use case would be to set a default template and then allow the user of the component to pass in a template to override the default.
+
+See the [attributes](scope#attributes-vs-model-data) documentation for more information on using `setAttribute`.
+
+
+## Component tracking
+Derby components are tracked in the DOM with an HTML comment tag. This allows components to be responsible for arbitrary DOM content, for example two table rows that otherwise cannot be wrapped by any other DOM elements.
+
+```derby
+
+```
+
+## Debugging
+
+A relatively quick way to inspect a component for debugging is to find it's comment in the browser's DOM inspector.
+In modern browsers clicking on the comment allows you to reference it in the console with `$0`.
+Once you have a reference to the comment tag you can access it's controller with `$0.$component` and it's model data with `$0.$component.model.get()`
+
+
+
+### derby-debug
+There is a plugin which makes accessing your components from the console even more accessible that is recommended for development.
+Read more about [derby-debug](https://github.com/derbyjs/derby-debug).
diff --git a/docs/index.md b/docs/index.md
new file mode 100644
index 00000000..78aca037
--- /dev/null
+++ b/docs/index.md
@@ -0,0 +1,17 @@
+---
+layout: default
+title: Introduction
+nav_order: 1
+---
+
+DerbyJS is a full-stack framework for writing modern web applications.
+
+# Realtime collaboration
+
+Effortlessly sync data across clients and servers with automatic conflict resolution powered by ShareDB's operational transformation of JSON and text.
+
+# Server and client rendering
+
+Templates can be rendered in the browser and on the server. In a browser, DerbyJS renders with fast, native DOM methods.
+
+On the server, no DOM or virtual DOM implementation is needed — the same templates return HTML as well! HTML rendering means faster page loads, full search engine support, and ability to use the same templates for all types of HTML output, such as emails.
diff --git a/docs/models.md b/docs/models.md
new file mode 100644
index 00000000..6d56dbc9
--- /dev/null
+++ b/docs/models.md
@@ -0,0 +1,60 @@
+---
+layout: default
+title: Models
+has_children: true
+---
+
+# Models
+
+DerbyJS models are provided by [Racer](https://github.com/derbyjs/racer), a realtime model synchronization engine. By building on ShareDB, Racer enables multiple users and services to interact with the same data objects with realtime conflict resolution. Racer models have a simple getter/setter and event interface for writing application logic.
+
+## Racer and ShareDB
+
+Racer provides a single interface for working with local data stored in memory and remote data synced via ShareDB. It works equally well on the server and the browser, and it is designed to be used independently from DerbyJS. This is useful when writing migrations, data-only services, or integrating with different view libraries.
+
+Remotely synced data is stored via [ShareDB](https://github.com/share/sharedb), which means that different clients can modify the same data at the same time. ShareDB uses [Operational Transformation (OT)](https://en.wikipedia.org/wiki/Operational_transformation) to automatically resolve conflicts in realtime or offline.
+
+On the server, Racer provides a `store`, which configures a connection to a database and pub/sub adapter. Every store connected to the same database and pub/sub system is synchronized in realtime.
+
+Stores create `model` objects. Models have a synchronous interface similar to interacting directly with objects. They maintain their own copy of a subset of the global state. This subset is defined via [subscriptions](backends#loading-data-into-a-model) to certain queries or documents. Models perform operations independently, and they automatically synchronize their state.
+
+Models emit events when their contents are updated, which DerbyJS uses to update the view in realtime.
+
+## Creating models
+
+Derby provides a model when calling application routes. On the server, it creates an empty model from the `store` associated with an app. When the server renders the page, the model is serialized. It is then reinitialized into the same state on the client. This model object is passed to app routes rendered on the client.
+
+Derby uses the model supplied by the store.modelMiddleware by calling `req.getModel()`. To pass data from server-side express middleware or routes, the model can be retrieved via this same method and data can be set on it before passing control to the app router.
+
+If you would like to get or set data outside of the app on the server, you can create models directly via `store.createModel()`.
+
+> `model = store.createModel(options)`
+> * `options:`
+> * `fetchOnly` Set to true to make model.subscribe calls perform a fetch instead
+> * `model` Returns a model instance associated with the given store
+
+## Store
+
+Typically, a project will have only one store, even if it has multiple apps. It is possible to have multiple stores, but a model can be associated with only a single store, and a page can have only a single model.
+
+> `store = derby.createStore(options)`
+> * `options` See the [Backends](backends) section for information on configuration
+> * `store` Returns a Racer store instance
+
+### Methods
+
+> `middleware = store.modelMiddleware()`
+> * `middleware` Returns a connect middleware function
+
+The model middleware adds a `req.getModel()` function which can be called to create or get a model (if one was already created) for a given request. It also closes this model automatically at the end of the request.
+
+Model's created from `req.getModel()` specify the option `{fetchOnly: true}`. This means that calls to `model.subscribe()` actually only fetch data and don't subscribe. This is more efficient during server-side rendering, since the model is only created for long enough to handle the route and render the page. The model then gets subscribed when it initializes in the browser.
+
+```js
+var expressApp = express();
+expressApp.use(store.modelMiddleware());
+
+expressApp.get('/', function(req, res, next) {
+ var model = req.getModel();
+});
+```
diff --git a/docs/models/backends.md b/docs/models/backends.md
new file mode 100644
index 00000000..66621262
--- /dev/null
+++ b/docs/models/backends.md
@@ -0,0 +1,122 @@
+---
+layout: default
+title: Backends and data loading
+parent: Models
+has_children: true
+---
+
+# Backends
+
+Racer stores are backed by ShareDB, which is used to persist data, perform queries, keep a journal of all operations, and pub/sub operations and changes to queries. Currently, ShareDB has [two pub/sub adapters](https://github.com/share/sharedb#database-adapters): one for in memory and one for Redis based pub/sub. ShareDB supports in memory or MongoDB storage. The database adapter [ShareDBMongo](https://github.com/share/sharedb-mongo)
+is backed by a real Mongo database and full query support. ShareDB is written with support for additional database adapters in mind.
+
+Getting started with a single-process server and MongoDB:
+
+```js
+var derby = require('derby');
+var ShareDbMongo = require('sharedb-mongo');
+
+var db = new ShareDbMongo('mongodb://localhost:27017/test');
+var backend = derby.createBackend({db: db});
+var model = backend.createModel();
+```
+
+The above examples use the in-process driver by default. In a production environment, you'll want to scale across multiple frontend servers and support updating data in other processes, such as migration scripts and additional services. For this, you should use the [ShareDB Redis pub/sub adapter](https://github.com/share/sharedb-redis-pubsub). ShareDB requires Redis 2.6 or newer, since it uses Lua scripting commands.
+
+```js
+var derby = require('derby');
+var ShareDbMongo = require('sharedb-mongo');
+var RedisPubSub = require('sharedb-redis-pubsub');
+
+var db = new ShareDbMongo('mongodb://localhost:27017/test');
+var backend = derby.createBackend({
+ db: db,
+ pubsub: new RedisPubSub()
+});
+var model = backend.createModel();
+```
+
+See [ShareDBMongo](https://github.com/share/sharedb-mongo) and [ShareDB Redis](https://github.com/share/sharedb-redis-pubsub) documentation for more information on configuration options.
+
+The Redis driver supports flushing all data from Redis or starting with an empty Redis database with journal and snapshot data in MongoDB. Thus, it is OK to start with a basic deployment using only a single process and add Redis later or to flush the Redis database if it becomes corrupt.
+
+## Mapping between database and model
+
+Racer paths are translated into database collections and documents using a natural mapping:
+
+```bash
+collection.documentId.documentProperty
+```
+
+ShareDB Mongo will add the following properties to Mongo documents for internal use:
+* `_m.ctime` - Timestamp when the ShareDB document was created
+* `_m.mtime` - Timestamp when the ShareDB document was last modified
+* `_type` - [OT type](https://github.com/share/sharedb#data-model)
+* `_v` - [Snapshot version](https://github.com/share/sharedb#data-model)
+
+In addition to `ctime` and `mtime`, custom metadata properties can be added to `_m` with middleware that modifies `snapshot.m` in apply or commit.
+
+Since these underscore-prefixed properties are for ShareDB's internal use, ShareDB Mongo will strip out these properties (`_m`, `_type`, and `_v`) as well as `_id` when it returns the document from Mongo. The `_id` is removed because Racer adds an `id` alias to all local documents. This alias references the `_id` property of the original Mongo document.
+
+If a document is an object, it will be stored as the Mongo document directly. For example,
+
+```js
+{
+ make: "Ford",
+ model: "Mustang",
+ year: 1969,
+ _m: {
+ ctime: 1494381632731,
+ mtime: 1494381635994
+ },
+ _type: "http://sharejs.org/types/JSONv0",
+ _v: 12
+}
+```
+
+If it is another type (e.g. [Plaintext OT Type](https://github.com/ottypes/text)), the value will be nested under a property on the Mongo document called `_data`.
+
+```js
+{
+ _data: "This is a text message.",
+ _m: {
+ ctime: 1494381632731,
+ mtime: 1494381635994
+ },
+ _type: "http://sharejs.org/types/text",
+ _v: 12
+}
+```
+
+It is not possible to set or delete an entire collection, or get the list of collections via the Racer API.
+
+## Loading data into a model
+
+The `subscribe`, `fetch`, `unsubscribe`, and `unfetch` methods are used to load and unload data from ShareJS. These methods don't return data directly. Rather, they load the data into a model. Once loaded, the data are then accessed via model getter methods.
+
+`subscribe` and `fetch` both return data initially, but subscribe also registers with pub/sub on the server to receive ongoing updates as the data change.
+
+> `model.subscribe(items..., callback(err))`
+> `model.fetch(items..., callback(err))`
+> `model.unsubscribe(items..., callback(err))`
+> `model.unfetch(items..., callback(err))`
+> * `items` Accepts one or more subscribe-able items, including a document path, scoped model, or query
+> * `callback` Calls back once all of the data for each query and document has been loaded or when an error is encountered
+
+Avoid subscribing or fetching queries by document id like `model.query('users', {_id: xxx})`. You can achieve the same result passing `'users.xxx'` or `model.at('users.xxx')` to subscribe or fetch, and it is much more efficient.
+
+If you only have one argument in your call to subscribe or fetch, you can also call `subscribe`, `fetch`, `unsubscribe`, and `unfetch` on the query or scoped model directly.
+
+```js
+var user = model.at('users.' + userId);
+var todosQuery = model.query('todos', {creatorId: userId});
+model.subscribe(user, todosQuery, function(err) {
+ if (err) return next(err);
+ console.log(user.get(), todosQuery.get());
+ page.render();
+});
+```
+
+Racer internally keeps track of the context in which you call subscribe or fetch, and it counts the number of times that each item is subscribed or fetched. To actually unload a document from the model, you must call the unsubscribe method the same number of times that subscribe is called and the unfetch method the same number of times that fetch is called. However, you generally don't need to worry about calling unsubscribe and unfetch manually.
+
+Instead, the `model.unload()` method can be called to unsubscribe and unfetch from all of the subscribes and fetches performed since the last call to unload. Derby calls this method on every full page render right before entering a route. By default, the actual unsubscribe and unfetch happens after a short delay, so if something gets resubscribed during routing, the item will never end up getting unsubscribed and it will callback immediately.
diff --git a/docs/models/contexts.md b/docs/models/contexts.md
new file mode 100644
index 00000000..8715041e
--- /dev/null
+++ b/docs/models/contexts.md
@@ -0,0 +1,52 @@
+---
+layout: default
+title: Data loading contexts
+parent: Backends and data loading
+grand_parent: Models
+---
+
+# Data loading contexts
+
+As data is loaded into a model with calls to fetch and subscribe, Racer tracks the number of fetches and subscribes per document path and query. Data is not removed from a model until it is released by calling unfetch and unsubscribe the matching number of times for each document or query. For example, after calling `subscribe()` on a query twice, then `unsubscribe()` once, the query would remain subscribed. It would be unsubscribed and its data would be removed from the model only after `unsubscribe()` was called once more.
+
+This behavior is helpful, since multiple parts of an application may need the same resource, but they may want perform data loading and unloading independently. For example, an edit dialog may be opened and closed while some of the same data may be displayed in a list; or a migration script may fetch data in batches in order to process a large amount of data without loading all of it into memory simultaneously.
+
+Contexts provide a way to track a group of related fetches and subscribes. In addition, they provide an `unload()` method that unfetches and unsubscribes the corresponding number of times. By default, all fetches and subscribes happen within the `'root'` context. Additional context names may be used to isolate the loading and unloading of data within the same model for independent purposes.
+
+> `childModel = model.context(name)`
+> * `name` A string uniquely identifying a context. Calling `model.context()` again with the same string will refer to the same context. By default, models have the context name `'root'`
+> * `childModel` Returns a model with a context of `name`, overriding the parent model's context name. All fetch, subscribe, and unload actions performed on this childModel will have this context
+
+> `model.unload([name])`
+> * `name` *(optional)* Unfetch and unsubscribe from all documents and queries for the corresponding number of times they were fetched and subscribed. This will end subscriptions and remove the data from the model if no remaining fetches or subscribes hold the data in the model under a different context. Defaults to the current model context name. Specifying a `name` argument overrides the default
+
+> `model.unloadAll()`
+> * Unload each context within a model. Results in all remotely loaded data being removed from a model. (Data within [local collections](paths#local-and-remote-collections) will remain.)
+
+## Usage example
+
+```js
+function openTodos(model) {
+ // Create a model with a load context inheriting from the current model
+ var dialogModel = model.context('todosDialog');
+ // Load data
+ var userId = dialogModel.scope('_session.userId').get();
+ var user = dialogModel.scope('users.' + userId);
+ var todosQuery = dialogModel.query('todos', {creatorId: userId});
+ dialogModel.subscribe(user, todosQuery, function(err) {
+ if (err) throw err;
+ // Delay display until required data is loaded
+ dialogModel.set('showTodos', true);
+ });
+}
+
+function closeTodos(model) {
+ model.set('showTodos', false);
+ // Use the same context name to unsubscribe
+ model.unload('todosDialog');
+}
+```
+
+## Automatic unloading on page navigation
+
+Derby uses Racer model contexts to unload the data for the previous page render when it performs client-side routing and a full-page render. When moving away from a page and before calling into the route for the new page, Derby calls `unloadAll()`, removing the data from all subscribes and fetches performed on the prior page.
diff --git a/docs/models/events.md b/docs/models/events.md
new file mode 100644
index 00000000..93946bbd
--- /dev/null
+++ b/docs/models/events.md
@@ -0,0 +1,157 @@
+---
+layout: default
+title: Events
+parent: Models
+---
+
+# Events
+
+Model events are based on the standard [Node.js EventEmitter](https://nodejs.org/docs/latest/api/events.html) methods, and they support the same methods: `on`, `once`, `removeListener`, `emit`, etc.
+
+## Mutation events
+
+Racer emits events whenever it mutates data via `model.set()`, `model.push()`, etc. It also emits events when data is remotely updated via a subscription. These events provide an entry point for an app to react to a specific data mutation or pattern of data mutations. The events might not be exactly the same as the methods that created them, since they can be transformed via OT.
+
+`model.on()` and `model.once()` accept a second argument for these mutation events. The second argument is a path pattern that will filter emitted events, calling the handler function only when a mutator matches the pattern. Path patterns support a single segment wildcard (`*`) anywhere in a path, and a multi-segment wildcard (`**`) at the end of the path. The multi-segment wildcard alone (`'**'`) matches all paths.
+
+> `listener = model.on(method, path, [options], eventCallback)`
+> * `method` Name of the mutator method: `'change'`, `'insert'`, `'remove'`, `'move'`, `'load'`, `'unload'`, or `'all'`
+> * `path` Pattern matching the path being mutated. For example: `'_page.user'`, `'users.*.name'`, `'users.*'`, `'users.**'` / `'users**'`, or `'**'`. `**` is valid only by itself or at the end of the path.
+> * `options` (optional)
+> * `useEventObjects` - If true, the callback is called with a structured event object instead of with a variable number of arguments. _Introduced in [racer@0.9.6](https://github.com/derbyjs/racer/releases/tag/v0.9.6)._
+> * `eventCallback` Function to call when a matching method and path are mutated
+> * Returns `listener` - the listener function subscribed to the event emitter. This is the function that should be passed to `model.removeListener`
+
+### `eventCallback` with `{useEventObjects: true}`
+
+_Introduced in [racer@0.9.6](https://github.com/derbyjs/racer/releases/tag/v0.9.6)._
+
+> `eventCallback(event, captures)`
+> * `event` - _Object_ - An instance of an Event object (see below)
+> * `captures` - _Array_ - The captured path segments, one item per wildcard in the pattern. Each `'*'` results in a string, and a `'**'` results in a sub-array of strings.
+
+Event objects:
+
+> `ChangeEvent { value, previous, passed }`
+> * `type: 'change'`
+> * `value` The current value at the path that was changed. Will be `undefined` for a deletion.
+> * `previous` The previous value at the path. Will be `undefined` if the path was previously unset.
+
+> `InsertEvent { index, values, passed }`
+> * `type: 'insert'`
+> * `index` The index at which items were inserted
+> * `values` An array of values that were inserted. Always an array, even if only one item was pushed, unshifted, or inserted.
+
+> `RemoveEvent { index, removed, passed }`
+> * `type: 'remove'`
+> * `index` The index at which items were removed
+> * `removed` An array of values that were removed. Always an array, even if only one item was popped, shifted, or removed
+
+> `MoveEvent { from, to, howMany, passed }`
+> * `type: 'move'`
+> * `from` The index from which items were moved
+> * `to` The index to which items were moved
+> * `howMany` How many items were moved
+
+> `LoadEvent { document, passed }`
+> * `type: 'load'`
+> * `document` This event fires when a document is loaded via a subscription or fetch. This the value of the newly loaded document object.
+
+> `UnloadEvent { previousDocument, passed }`
+> * `type: 'unload'`
+> * `previousDocument` This event fires when a document is removed from the model via unsubscribe or unfetch. This is the value of the document object that was unloaded.
+
+The `event.type` is useful for distinguising the actual event type when listening to `'all'`.
+
+```js
+// Matches model.push('messages', message)
+model.on('insert', 'messages', {useEventObjects: true}, function(insertEvent) {
+ console.log(insertEvent.values, 'inserted at index', insertEvent.index);
+});
+
+// Matches model.set('todos.4.completed', true), etc.
+model.on('change', 'todos.*.completed', {useEventObjects: true}, function(changeEvent, captures) {
+ console.log('todos.' + captures[0] + ' set to ' + changeEvent.value);
+});
+
+// Matches all events
+model.on('all', '**', {useEventObjects: true}, function(event, captures) {
+ var starStarSegments = captures[0];
+ console.log(event.type + ' at ' + starStarSegments.join('.') + ':', event);
+});
+```
+
+### `eventCallback` when `useEventObjects` is false or undefined
+
+The event callback receives a number of arguments based on the path pattern and method. The arguments are:
+
+> `eventCallback([captures...], [eventType], args..., passed)`
+> * `captures` The path segment or segments that is passed in only when matching wildcards in the path pattern
+> * `eventType` Only the `'all'` event adds the emitted event name after the captures and before the args
+> * `args` Event specific arguments. See below
+> * `passed` An object with properties provided via `model.pass()`. See description below
+
+Callbacks for each event type:
+
+> `changeCallback([captures...], value, previous, passed)`
+> * `value` The current value at the path that was changed. Will be `undefined` for objects that were deleted
+> * `previous` The previous value at the path. Will be `undefined` for paths set for the first time
+
+> `insertCallback([captures...], index, values, passed)`
+> * `index` The index at which items were inserted
+> * `values` An array of values that were inserted. Always an array, even if only one item was pushed, unshifted, or inserted
+
+> `removeCallback([captures...], index, removed, passed)`
+> * `index` The index at which items were removed
+> * `removed` An array of values that were removed. Always an array, even if only one item was popped, shifted, or removed
+
+> `moveCallback([captures...], from, to, howMany, passed)`
+> * `from` The index from which items were moved
+> * `to` The index to which items were moved
+> * `howMany` How many items were moved
+
+> `loadCallback([captures...], document, passed)`
+> * `document` This event fires when a document is loaded via a subscription or fetch. It emits the value of the newly loaded document object
+
+> `unloadCallback([captures...], previousDocument, passed)`
+> * `previousDocument` This event fires when a document is removed from the model via unsubscribe or unfetch. It emits the value of the document object that was unloaded
+
+```js
+// Matches model.push('messages', message)
+model.on('insert', 'messages', function(index, [message]) {
+ ...
+});
+
+// Matches model.set('todos.4.completed', true), etc.
+model.on('change', 'todos.*.completed', function(todoId, isComplete) {
+ ...
+});
+
+// Matches all events - `path` and `event` are passed in to the event callback
+model.on('all', '**', function(path, event, args...) {
+ ...
+});
+```
+
+### Passing data to event listeners
+
+> `model.pass(object)`
+> * `object` An object whose properties will each be set on the `passed` argument
+
+`model.pass()` can be chained before calling a mutator method to pass an argument to model event listeners. You must pass it an object with a property that identifies the name of the parameter.
+
+This value is only passed to local listeners, and it is not sent to the server or other clients. It is typically used to identify the originator of a particular mutation so that multiple responses to the same change and infinite loops may be avoided. Such loops could occur for listeners that respond to paths that they may modify.
+
+On a string insert or string remove mutation, a `'change`' event is emitted, since strings are immutable values, and inserting or removing from a string requires changing its entire value. However, detail on what specifically was inserted or removed is neccessary to implement view bindings properly for realtime collaborative text editing. This additional information is added to the `passed` object. On a string insert, the passed object has an additional property of `$stringInsert: {index: Number, text: String}`. On a string remove, the passed object has an additional property of `$stringRemove: {index: Number, howMany: Number}`.
+
+```js
+// Logs:
+// 'red', {}
+// 'green', {message: 'hi'}
+
+model.on('change', 'color', function(value, previous, passed) {
+ console.log(value, passed);
+});
+model.set('color', 'red');
+model.pass({message: 'hi'}).set('color', 'green');
+```
diff --git a/docs/models/filters-sorts.md b/docs/models/filters-sorts.md
new file mode 100644
index 00000000..51266dd6
--- /dev/null
+++ b/docs/models/filters-sorts.md
@@ -0,0 +1,100 @@
+---
+layout: default
+title: Filters and sorts
+parent: Models
+---
+
+# Filters and sorts
+
+Filters create a live-updating list from items in an object. The results automatically update as the input items change.
+
+> `filter = model.filter(inputPath, [additionalInputPaths...], [options], fn)`
+> * `inputPath` A path pointing to an object or array. The path's values will be retrieved from the model via `model.get()`, and then each item will be checked against the filter function
+> * `additionalInputPaths` *(optional)* Other parameters can be set in the model, and the filter function will be re-evaluated when these parameters change as well
+> * `options:`
+> * `skip` The number of first results to skip
+> * `limit` The maximum number of results. A limit of zero is equivalent to no limit
+> * `fn` A function or the name of a function defined via `model.fn()`. The function should have the arguments `function(item, key, object, additionalInputs...)`. Like functions for [`array.filter()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/filter), the function should return true for values that match the filter
+
+```js
+app.get('/search-pants', function(page, model, params, next) {
+ model.subscribe('pants', function(err) {
+ if (err) return next(err);
+ model.filter('pants', 'pricing', 'color',
+ // evaluate whether a pants item matches the search options
+ function(item, pantsId, pants, pricing, color) {
+ return item.price >= pricing.minimum
+ && item.price <= pricing.maximum
+ && item.color == color;
+ }
+ ).ref('_page.pantsList'); // bind the output of the filter
+ page.render('pants');
+ });
+});
+```
+
+If `model.filter()` is called with `null` for the function, it will create a list out of all items in the input object. This can be handy as a way to render all subscribed items in a collection, since only arrays can be used as an input to `{{each}}` template tags.
+
+> `filter = model.sort(inputPath, [options], fn)`
+> * `inputPath` A path pointing to an object or array. The path's values will be retrieved from the model via `model.get()`, and then each item will be checked against the filter function
+> * `options:`
+> * `skip` The number of first results to skip
+> * `limit` The maximum number of results. A limit of zero is equivalent to no limit
+> * `fn` A function or the name of a function defined via `model.fn()`. The function should should be a compare function with the arguments `function(a, b)`. It should return the same values as compare functions for [`array.sort()`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort)
+
+There are two default named functions defined for sorting, `'asc'` and `'desc'`. These functions compare each item with Javascript's less than and greater than operators. See MDN for more info on [sorting non-ASCII characters](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Array/sort#Sorting_non-ASCII_characters).
+
+You may define functions to be used in `model.filter()` and `model.sort()` via [`model.fn()`](functions#named-functions).
+
+A filter may have both a filter function and a sort function by chaining the two calls:
+
+```js
+app.on('model', function(model) {
+ model.fn('expensiveItem', function(item) {
+ return item.price > 100;
+ });
+ model.fn('priceSort', function(a, b) {
+ return b.price - a.price;
+ });
+});
+
+app.get('/expensive-pants', function(page, model, params, next) {
+ model.subscribe('pants', function(err) {
+ if (err) return next(err);
+ var filter = model.filter('pants', 'expensiveItem')
+ .sort('priceSort');
+ filter.ref('_page.expensivePants');
+ page.render('pants');
+ });
+});
+```
+
+## Methods
+
+The output of a filter is typically used by creating a reference from it. This sets the data in the model and keeps it updated.
+
+> `scoped = filter.ref(path)`
+> * `path` The path at which to create a refList of the filter's output
+> * `scoped` Returns a model scoped to the output path of the ref
+
+The filter's current value can also be retrieved directly via `filter.get()`.
+
+> `results = filter.get()`
+> * `results` Returns an array of results matching the filter
+
+As well as by updating its input paths, a filter can be recomputed manually by calling its `filter.update()` method. This can also be used to perform pagination, since the the `filter.skip` and `filter.limit` properties can be modified followed by calling `filter.update()`.
+
+> `filter.update()`
+
+```js
+var filter = model.sort('items', {skip: 0, limit: 10}, function(a, b) {
+ if (a && b) return a.score - b.score;
+});
+// Logs first 10 items
+console.log(filter.get());
+
+filter.skip += filter.limit;
+filter.update();
+// Logs next 10 items
+console.log(filter.get());
+```
diff --git a/docs/models/getters.md b/docs/models/getters.md
new file mode 100644
index 00000000..eead8f13
--- /dev/null
+++ b/docs/models/getters.md
@@ -0,0 +1,50 @@
+---
+layout: default
+title: Getters
+parent: Models
+---
+
+# Getters
+
+Data in the model is accessed with `model.get()`. This method returns values by reference, based on the model's scope and/or the path passed to the get method. Models allow getting undefined paths. The get method will return `undefined` and will not throw when looking up a property below another property that is undefined.
+
+Internally, model data is represented as collections of documents. Collections must be objects, and documents may be of any type, but are also typically objects. A document's data at a point in time is referred to as its snapshot. This structure allows for some documents to be remote documents synced with the server and some to be local documents only in the client. It also means that models are broken into a similar structure as database collections or tables.
+
+As model document snapshots change from local or remote mutations, the `model.root.data` object is updated. `model.get()` traverses through the properties of the model's data to lookup and return the appropriate value.
+
+```js
+model.get('_session.account') === model.root.data._session.account;
+```
+
+> `value = model.get([path])`
+> * `path` *(optional)* Path of object to get. Not supplying a path will return all data in the model starting from the current scope
+> * `value` Current value of the object at the given path. Note that objects are returned by reference and should not be modified directly
+
+> `shallowCopy = model.getCopy([path])`
+> * `path` *(optional)* Path of object to get
+> * `shallowCopy` Shallow copy of current value, going only one level deep when returning an object or array
+
+> `deepCopy = model.getDeepCopy([path])`
+> * `path` *(optional)* Path of object to get
+> * `deepCopy` Deep copy of current value
+
+## Values returned by reference
+
+`model.get()` returns values by reference. Racer will fail to function correctly if data in the model is manipulated directly instead of via its mutator methods, such as `model.set()`. You should *never* mutate objects returned from `model.get()` directly.
+
+As a convenience, Racer also provides a `model.getCopy()` method that returns a shallow copy and `model.getDeepCopy()` method that returns a deep copy. It is safe to mutate copied objects. Changes in these objects can then be updated back into the model using `model.setDiffDeep()`.
+
+```js
+// WARNING: Do NOT directly manipulate objects in the model
+var user = model.get('users.' + userId);
+user.name = 'John';
+
+// Instead, use the model setter methods
+var user = model.get('users.' + userId);
+model.set('users.' + userId + '.name', 'John');
+
+// Or, get a copy and set the difference
+var user = model.getDeepCopy('users.' + userId);
+user.name = 'John';
+model.setDiffDeep('users.' + userId, user);
+```
diff --git a/docs/models/mutators.md b/docs/models/mutators.md
new file mode 100644
index 00000000..222eef90
--- /dev/null
+++ b/docs/models/mutators.md
@@ -0,0 +1,122 @@
+---
+layout: default
+title: Mutators
+parent: Models
+---
+
+# Setters
+
+Models allow getting and setting to nested undefined paths. Setting such a path first sets each undefined or null parent to an empty object or empty array. Whether or not a path segment is a number determines whether the implied parent is created as an object or array.
+
+```js
+model.set('cars.DeLorean.DMC12.color', 'silver');
+// Logs: { cars: { DeLorean: { DMC12: { color: 'silver' }}}}
+console.log(model.get());
+```
+
+All model mutators modify data and emit events synchronously. This is only safe to do, because all remote data is synchronized with Operational Transformation, and every client will eventually see a consistent view of the same data. With a more naive approach to syncing data to the server and other clients, updates to the data would need to wait until they were confirmed successful from the server.
+
+As well as a synchronous interface, model mutators have an optional callback with an error argument `callback(err)`. This callback is called when the operation is confirmed from the server, which may be desired to confirm that data was saved before updating the UI in some rare cases. This callback should be used very rarely in practice, and data updates should be treated as sychronous, so that the UI responds immediately even if a user has a high latency connection or is currently disconnected.
+
+## General methods
+
+> `previous = model.set(path, value, [callback])`
+> * `path` Model path to set
+> * `value` Value to assign
+> * `previous` Returns the value that was set at the path previously
+> * `callback` *(optional)* Invoked upon completion of a successful or failed operation
+
+> `obj = model.del(path, [callback])`
+> * `path` Model path of object to delete
+> * `obj` Returns the deleted object
+
+> `obj = model.setNull(path, value, [callback])`
+> * `path` Model path to set
+> * `value` Value to assign only if the path is null or undefined
+> * `obj` Returns the object at the path if it is not null or undefined. Otherwise, returns the new value
+
+> `previous = model.setDiff(path, value, [callback])`
+> * `path` Model path to set
+> * `value` Value to be set if not strictly equal to the current value
+> * `previous` Returns the value that was set at the path previously
+
+> `model.setDiffDeep(path, value, [callback])`
+> `model.setArrayDiff(path, value, [callback])`
+> `model.setArrayDiffDeep(path, value, [callback])`
+> * `path` Model path to set
+> * `value` Value to be set if different. Deep methods will do a deep traversal and deep equality check. Array methods should be used when diffing two different arrays and only inserts, removes, and moves at the top level are desired. `setDiffDeep` can diff arrays as well but may produce a set on a particular property within an array. These methods may end up performing zero or many mutations.
+
+## Object methods
+
+> `id = model.add(path, object, [callback])`
+> * `path` Model path to set
+> * `object` A document to add. If the document has an `id` property, it will be set at that value underneath the path. Otherwise, an id from `model.id()` will be set on the object first
+> * `id` Returns `object.id`
+
+> `model.setEach(path, object, [callback])`
+> * `path` Model path underneath which each property will be set
+> * `object` An object whose properties are each set individually
+
+## Number methods
+
+> `number = model.increment(path, [byNumber], [callback])`
+> * `path` Model path to set
+> * `byNumber` *(optional)* Number specifying amount to increment or decrement if negative. Defaults to 1
+> * `number` Returns the new value that was set after incrementing
+
+## Array methods
+
+Array methods can only be used on paths set to arrays, null, or undefined. If the path is null or undefined, the path will first be set to an empty array before applying the method.
+
+> `length = model.push(path, value, [callback])`
+> * `path` Model path to an array
+> * `value` An item to add to the *end* of the array
+> * `length` Returns the length of the array with the new item added
+
+> `length = model.unshift(path, value, [callback])`
+> * `path` Model path to an array
+> * `value` An item to add to the *beginning* of the array
+> * `length` Returns the length of the array with the new item added
+
+> `length = model.insert(path, index, values, [callback])`
+> * `path` Model path to an array
+> * `index` Index at which to start inserting. This can also be specified by appending it to the path instead of as a separate argument
+> * `values` An array of items to insert at the index
+> * `length` Returns the length of the array with the new items added
+
+> `item = model.pop(path, [callback])`
+> * `path` Model path to an array
+> * `item` Removes the last item in the array and returns it
+
+> `item = model.shift(path, [callback])`
+> * `path` Model path to an array
+> * `item` Removes the first item in the array and returns it
+
+> `removed = model.remove(path, index, [howMany], [callback])`
+> * `path` Model path to an array
+> * `index` Index at which to start removing items
+> * `howMany` *(optional)* Number of items to remove. Defaults to 1
+> * `removed` Returns an array of removed items
+
+> `moved = model.move(path, from, to, [howMany], [callback])`
+> * `path` Model path to an array
+> * `from` Starting index of the item to move
+> * `to` New index where the item should be moved
+> * `howMany` *(optional)* Number of items to move. Defaults to 1
+> * `moved` Returns an arry of items that were moved
+
+## String methods
+
+String methods can only be used on paths set to strings, null, or undefined. If the path is null or undefined, the path will first be set to an empty string before applying the method.
+
+The string methods support collaborative text editing, and Derby uses string methods to bind changes to all text inputs and textareas by default.
+
+> `model.stringInsert(path, index, text, [callback])`
+> * `path` Model path to a string
+> * `index` Character index within the string at which to insert
+> * `text` String to insert
+
+> `model.stringRemove(path, index, howMany, [callback])`
+> * `path` Model path to a string
+> * `index` Starting character index of the string at which to remove
+> * `howMany` Number of characters to remove
diff --git a/docs/models/paths.md b/docs/models/paths.md
new file mode 100644
index 00000000..3c6c7914
--- /dev/null
+++ b/docs/models/paths.md
@@ -0,0 +1,79 @@
+---
+layout: default
+title: Paths
+parent: Models
+---
+
+# Paths
+
+All model operations happen on paths which represent nested JSON objects. These paths must be globally unique within a particular database and Redis journal.
+
+For example, the model data:
+
+```js
+{
+ title: 'Fruit store',
+ fruits: [
+ { name: 'banana', color: 'yellow' },
+ { name: 'apple', color: 'red' },
+ { name: 'lime', color: 'green' }
+ ]
+}
+```
+
+Would have paths like `title`, `fruits.1`, and `fruits.0.color`. Any path segment that is a number must be an index of an array.
+
+> **WARNING** If you want to use an id value that is a number as a path segment, be careful to prefix this with another character, such as `_` before setting it. Otherwise, you will accidentally create a gigantic array and probably run out of memory. For example, use a path like: `items._1239182389123.name` and never `items.1239182389123.name`.
+
+## Local and remote collections
+
+Collection names (i.e. the first path segment) that start with an underscore (`_`) or dollar sign (`$`) are local to a given model and are not synced. All paths that start with another character are remote, and will be synced to servers and other clients via ShareJS. Collections that begin with dollar signs are reserved for use by Racer, Derby, or extensions, and should not be used for application data.
+
+Almost all non-synced data within an application should be stored underneath the `_page` local collection. This enables Derby to automatically cleanup as the user navigates between pages. Right before rendering a new page, Derby calls `model.destroy('_page')`, which removes all data, references, event listeners, and reactive functions underneath the `_page` collection. If you have some data that you would like to be maintained between page renders, it can be stored underneath a different local collection. This is useful for setting data on the server, such as setting `_session.userId` in authentication code. However, be very careful when storing local data outside of `_page`, since bleeding state between pages is likely to be a source of unexpected bugs.
+
+## Scoped models
+
+Scoped models provide a more convenient way to interact with commonly used paths. They support the same methods, and they provide the path argument to accessors, mutators, event methods, and subscription methods. Also, wherever a path is accepted in a racer method, a scoped model can typically be used as well.
+
+> `scoped = model.at(subpath)`
+> * `subpath` The relative reference path to set. The path is appended if called on a scoped model
+> * `scoped` Returns a scoped model
+
+> `scoped = model.scope([path])`
+> * `path` *(optional)* The absolute reference path to set, or the root path by default. This will become the scope even if called on a scoped model. May be called without a path to get a model scoped to the root
+> * `scoped` Returns a scoped model
+
+> `scoped = model.parent([levels])`
+> * `levels` *(optional)* Defaults to 1. The number of path segments to remove from the end of the reference path
+> * `scoped` Returns a scoped model
+
+> `path = model.path([subpath])`
+> * `subpath` *(optional)* A relative reference path to append. Defaults to the current path
+> * `path` Returns the reference path if applicable
+
+> `isPath = model.isPath(subpath)`
+> * `subpath` A relative reference path or scoped model
+> * `isPath` Returns true if the argument can be interpreted as a path, false otherwise
+
+> `segment = model.leaf()`
+> * `segment` Returns the last segment for the reference path. Useful for getting indices, ids, or other properties set at the end of a path
+
+```js
+room = model.at('_page.room');
+
+// These are equivalent:
+room.at('name').set('Fun room');
+room.set('name', 'Fun room');
+
+// Logs: {name: 'Fun room'}
+console.log(room.get());
+// Logs: 'Fun room'
+console.log(room.get('name'));
+```
+
+## GUIDs
+
+Models provide a method to create globally unique ids. These can be used as part of a path or within mutator methods.
+
+> `guid = model.id()`
+> * `guid` Returns a globally unique identifier that can be used for model operations
diff --git a/docs/models/queries.md b/docs/models/queries.md
new file mode 100644
index 00000000..c1a7ca79
--- /dev/null
+++ b/docs/models/queries.md
@@ -0,0 +1,34 @@
+---
+layout: default
+title: Queries
+parent: Models
+---
+
+# Queries
+
+Racer can fetch or subscribe to queries based on a model value or a database-specific query. When fetching or subscribing to a query, all of the documents associated with that query are also fetched or subscribed.
+
+> `query = model.query(collectionName, path)`
+> * `collectionName` The name of a collection from which to get documents
+> * `path` A model path whose value contains a documentId or an array of documentIds
+
+> `query = model.query(collectionName, databaseQuery)`
+> * `collectionName` The name of a collection from which to get documents
+> * `databaseQuery` A query in the database native format, such as a MonogDB query
+
+# MongoDB query format
+
+The `sharedb-mongo` adapter supports most MongoDB queries that you could pass to the Mongo `find()` method. See the [Mongo DB query documentation](https://docs.mongodb.org/manual/core/read-operations/#read-operations-query-document) and the [query selectors reference](https://docs.mongodb.org/manual/reference/operator/#query-selectors). Supported MongoDB cursor methods must be passed in as part of the query. `$sort` should be used for sorting, and skips and limits should be specified as `$skip` and `$limit`. There is no `findOne()` equivalent—use `$limit: 1` instead.
+
+Note that projections, which are used to limit the fields that a query returns, may not be defined in the query. Please refer to the [guide on using projections](https://github.com/derbyparty/derby-faq/tree/master/en#i-dont-need-all-collections-fields-in-a-browser-how-to-get-only-particular-fields-collections-projection), which you can follow if you only want specific fields of a document transferred to the browser.
+
+## Query results
+
+After a query is subscribed or fetched, its results can be returned directly via `query.get()`. It is also possible to create a live-updating results set in the model via `query.ref()`.
+
+> `results = query.get()`
+> * `results` Creates and returns an array of each of the document objects matching the query
+
+> `scoped = query.ref(path)`
+> * `path` Local path at which to create an updating refList of the queries results
+> * `scoped` Returns a model scoped to the path at which results are output
diff --git a/docs/models/reactive-functions.md b/docs/models/reactive-functions.md
new file mode 100644
index 00000000..f43b09a6
--- /dev/null
+++ b/docs/models/reactive-functions.md
@@ -0,0 +1,176 @@
+---
+layout: default
+title: Reactive functions (model.start)
+parent: Models
+---
+
+# Reactive functions
+
+Reactive functions provide a simple way to update a computed value whenever one or more objects change. While model events respond to specific model methods and path patterns, reactive functions will be re-evaluated whenever any of their inputs or nested properties change in any way.
+
+Reactive functions may be run any number of times, so they should be [pure functions](https://en.wikipedia.org/wiki/Pure_function). In other words, they should always return the same results given the same input arguments, and they should be side effect free. By default, the inputs to the function are retrieved directly from the model, so be sure not to modify any object or array input arguments. For example, slice an array input before you sort it. The output of the model function is deep cloned by default.
+
+To execute a model function, you then call `model.start()` or `model.evaluate()`.
+* `evaluate()` runs a function once and returns the result.
+* `start()` also sets up event listeners that continually re-evaluate the
+* function whenever any of its input or output paths are changed.
+
+> ```
+> value = model.start(path, inputPaths, [options], fn)
+> value = model.evaluate(inputPaths, [options], fn)
+> ```
+> ```
+> // Legacy (racer <= 0.9.5)
+> value = model.start(path, inputPaths..., [options], fn)
+> value = model.evaluate(inputPaths..., [options], fn)
+> ```
+>
+> * `path` - _string | ChildModel_ - The output path at which to set the value,
+> keeping it updated as input paths change
+> * `inputPaths` - _Array_ - One or more paths whose values
+> will be retrieved from the model and passed to the function as inputs
+> * `options` - _Object_ (optional)
+> * `copy` - Controls automatic deep copying of the inputs and output of the
+> function. _Model#evaluate never deep-copies output, since the return
+> value is not set onto the model._
+> - `'output'` (default) - Deep-copy the return value of the function
+> - `'input'` - Deep-copy the inputs to the function
+> - `'both'` - Deep-copy both inputs and output
+> - `'none'` - Do not automatically copy anything
+> * `mode` - The `model.set*` method to use when setting the output. _This has
+> no effect in Model#evaluate._
+> - `'diffDeep'` (default) - Do a recursive deep-equal comparison on old
+> and new output values, attempting to issue fine-grained ops on subpaths
+> where possible.
+> - `'diff` - Do an identity comparison (`===`) on the output value, and do
+> a simple set if old and new outputs are different.
+> - `'arrayDeep'` - Compare old and new arrays item-by-item using a
+> deep-equal comparison for each item, issuing top-level array insert,
+> remove, and move ops as needed. Unlike `'diffDeep'`, this will _not_
+> issue ops deep inside array items.
+> - `'array'` - Compare old and new arrays item-by-item using identity
+> comparison (`===`) for each item, issuing top-level array insert,
+> remove, and move ops as needed.
+> * `async` - _boolean_ - If true, then upon input changes, defer evaluation
+> of the function to the next tick, instead of immediately evaluating the
+> function upon each input change. _Introduced in [racer@0.9.5](https://github.com/derbyjs/racer/releases/tag/v0.9.5)._
+> - This can improve UI performance when multiple inputs to a reactive
+> function will change in the same event loop, as `async: true` will
+> mean the function only needs be evaluated once instead of N times.
+> - _Warning:_ Avoid using `async: true` if there's any controller code
+> that does a `model.get()` on the output path or any paths downstream
+> of the output, since changes to an input path won't immediately result
+> in the output being updated.
+> * `fn` - _Function | string_ - A function or the name of a function defined
+> via `model.fn()`
+> * The function gets invoked with the values at the input paths, one input
+> per argument, and should return the computed output value.
+> * It should be a synchronous [pure function](https://en.wikipedia.org/wiki/Pure_function).
+> - One common side effect to avoid is `Array#sort` on an input array, since
+> that sorts the array in-place. If you need to do a sort, make a shallow
+> copy via `array.slice()` first, or use a sorting library that returns a
+> new array instead of sorting in-place.
+> - The function will be called both in Node and in the browser, so avoid
+> using functions whose behavior is implementation-dependent, such as the
+> one-argument form of [`String#localeCompare`](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/String/localeCompare).
+> - The function might get called with some inputs `undefined`, so be
+> defensive and check inputs' existence before using them.
+> * Return `value` - The initial value computed by the function
+
+> `model.stop(path)`
+> * `path` The path at which the output should no longer update. Note that the value is not deleted; it is just no longer updated
+
+In DerbyJS, `model.start()` functions should typically be established in the `init` method of a component. This method is called both before rendering on the server and then again before rendering in the browser. These reactive functions will be stopped as soon as the component is destroyed, which happens automatically when the component is removed from the page.
+
+```js
+MyComponent.prototype.init = function(model) {
+ model.start('total', 'first', 'second', function sum(x, y) {
+ return (x || 0) + (y || 0);
+ });
+};
+```
+
+## Two-way reactive functions
+
+Most reactive functions define a getter only. You should treat their output as read only. In addition, it is possible to define two-way reactive functions with both a setter and a getter. Note that this is a more advanced pattern and should not be used unless you are confident that it is a strong fit for your use case.
+
+```js
+// Model functions created with just a function act as getters only.
+// These functions update the output path when any input changes
+model.fn('expensiveItems', function(items) {
+ return items && items.filter(function(item) {
+ return item.price > 100;
+ });
+});
+
+// It is also possible to define both a getter and a setter function
+// if the input values may be computed from setting the output
+model.fn('fullName', {
+ // The getter function gets the current value of each of the input
+ // arguments when any input might have changed
+ get: function(firstName, lastName) {
+ return firstName + ' ' + lastName;
+ },
+ // The setter function is called with the value that was set at
+ // the output path as well as the current value of the inputs.
+ // It should return an array or object where each property is an
+ // index that corresponds to each input argument that should be set.
+ // If the function returns null, no items will be set.
+ set: function(value, firstName, lastName) {
+ return value && value.split(' ');
+ }
+});
+```
+
+## Named functions
+
+In addition to passing in a function directly, a function can be defined on a model via a name. This name can then be used in place of a function argument.
+
+> `model.fn(name, fn)`
+> * `name` A name that uniquely identifies the function
+> * `fn` A getter function or an object with the form `{get: function(), set: function()}`
+
+Reactive functions started on the server via a name are reinitialized when the page loads. In order to add functions for use in routes as well as in the client, use the `'model'` event emitted by apps, which occurs right before an app route is called on the server and once immediately upon initialization in the client. Then, you can safely start them in the appropriate route, and they will be re-established automatically on the client.
+
+In DerbyJS, this pattern is generally less preferable to initializing model functions in a component.
+
+```js
+app.on('model', function(model) {
+ // Sort the players by score and return the top X players. The
+ // function will automatically update the value of '_page.leaders'
+ // as players are added and removed, their scores change, and the
+ // cutoff value changes.
+ model.fn('topPlayers', function(players, cutoff) {
+ // Note that the input array is copied with slice before sorting
+ // it. The function should not modify the values of its inputs.
+ return players.slice().sort(function (a, b) {
+ return a.score - b.score;
+ }).slice(0, cutoff - 1);
+ });
+});
+
+app.get('/leaderboard/:gameId', function(page, model, params, next) {
+ var game = model.at('game.' + params.gameId);
+ game.subscribe(function(err) {
+ if (err) return next(err);
+ game.setNull('players', [
+ {name: 'John', score: 4000},
+ {name: 'Bill', score: 600},
+ {name: 'Kim', score: 9000},
+ {name: 'Megan', score: 3000},
+ {name: 'Sam', score: 2000}
+ ]);
+ model.set('_page.cutoff', 3);
+ model.start(
+ // Output path
+ '_page.topPlayers',
+ // Input paths
+ game.at('players'),
+ '_page.cutoff',
+ // Name of the function
+ 'topPlayers'
+ );
+ page.render();
+ });
+});
+```
diff --git a/docs/models/refs.md b/docs/models/refs.md
new file mode 100644
index 00000000..6c659688
--- /dev/null
+++ b/docs/models/refs.md
@@ -0,0 +1,94 @@
+---
+layout: default
+title: Refs
+parent: Models
+---
+
+# References
+
+References make it possible to write business logic and templates that interact with the model in a general way. They redirect model operations from a reference path to the underlying data, and they set up event listeners that emit model events on both the reference and the actual object's path.
+
+References must be declared per model, since calling `model.ref` creates a number of event listeners in addition to setting a ref object in the model. When a reference is created or removed, a `change` model event is emitted. References are not actually stored in the model data, but they can be used from getter and setter methods as if they are.
+
+> `scoped = model.ref(path, to, [options])`
+> * `path` The location at which to create a reference. This must be underneath a [local collection](paths#local-and-remote-collections) (typically `_page`), since references must be declared per model
+> * `to` The location that the reference links to. This is where the data is actually stored
+> * `options:`
+> * `updateIndices` Set true to update the ref's `to` path if it contains array indices whose parents are modified via array inserts, removes, or moves
+> * `scoped` Returns a model scoped to the output path for convenience
+
+> `model.removeRef(path)`
+> * `path` The location at which to remove the reference
+
+```js
+model.set('colors', {
+ red: {hex: '#f00'}
+, green: {hex: '#0f0'}
+, blue: {hex: '#00f'}
+});
+
+// Getting a reference returns the referenced data
+model.ref('_page.green', 'colors.green');
+// Logs {hex: '#0f0'}
+console.log(model.get('_page.green'));
+
+// Setting a property of the reference path modifies
+// the underlying data
+model.set('_page.green.rgb', [0, 255, 0]);
+// Logs {hex: '#0f0', rgb: [0, 255, 0]}
+console.log(model.get('colors.green'));
+
+// Removing the reference has no effect on the underlying data
+model.removeRef('_page.green');
+// Logs undefined
+console.log(model.get('_page.green'));
+// Logs {hex: '#0f0', rgb: [0, 255, 0]}
+console.log(model.get('colors.green'));
+```
+
+Racer also supports a special reference type created via `model.refList`. This type of reference is useful when a number of objects need to be rendered or manipulated as a list even though they are stored as properties of another object. This is also the type of reference created by a query. A reference list supports the same mutator methods as an array, so it can be bound in a view template just like an array.
+
+> `scoped = model.refList(path, collection, ids, [options])`
+> * `path` The location at which to create a reference list. This must be underneath a [local collection](paths#local-and-remote-collections) (typically `_page`), since references must be declared per model
+> * `collection` The path of an object that has properties to be mapped onto an array. Each property must be an object with a unique `id` property of the same value
+> * `ids` A path whose value is an array of ids that map the `collection` object's properties to a given order
+> * `options:`
+> * `deleteRemoved` Set true to delete objects from the `collection` path if the corresponding item is removed from the refList's output path
+> * `scoped` Returns a model scoped to the output path for convenience
+
+> `model.removeRefList(path)`
+> * `path` The location at which to remove the reference
+
+Note that if objects are inserted into a refList without an `id` property, a unique id from [`model.id()`](#guids) will be automatically added to the object.
+
+```js
+// refLists should consist of objects with an id matching
+// their property on their parent
+model.setEach('colors', {
+ red: {hex: '#f00', id: 'red'},
+ green: {hex: '#0f0', id: 'green'},
+ blue: {hex: '#00f', id: 'blue'}
+});
+model.set('_page.colorIds', ['blue', 'red']);
+model.refList('_page.myColors', 'colors', '_page.colorIds');
+
+model.push('_page.myColors', {hex: '#ff0', id: 'yellow'});
+
+// Logs: [
+// {hex: '#00f', id: 'blue'},
+// {hex: '#f00', id: 'red'},
+// {hex: '#ff0', id: 'yellow'}
+// ]
+console.log(model.get('_page.myColors'));
+```
+
+When a collection is cleaned up by `model.destroy()`, the `model.removeAllRefs()` method is invoked to remove all refs and refLists underneath the collection.
+
+> `model.removeAllRefs(from)`
+> * `from` Path underneath which to remove all refs and refLists
+
+It isn't neccessary to manually dereference model paths, but for debugging, testing, or special cases there is a `model.dereference()` method.
+
+> `resolved = model.dereference(from)`
+> * `from` Path to dereference
+> * `resolved` Returns the fully dereferenced path, possibly passing through multiple refs or refLists. Will return the input path if no references are found
diff --git a/docs/routes.md b/docs/routes.md
new file mode 100644
index 00000000..afd10dd3
--- /dev/null
+++ b/docs/routes.md
@@ -0,0 +1,101 @@
+---
+layout: default
+title: Routes
+---
+
+# Routes
+
+Routes map URL patterns to actions. Derby routes are powered by [Express](https://expressjs.com/). Within apps, routes are defined via the `get`, `post`, `put`, and `del` methods of the app created by `derby.createApp()`.
+
+> `app.get ( routePattern, callback(page, model, params, next) )`
+>
+> `app.post ( routePattern, callback(page, model, params, next) )`
+>
+> `app.put ( routePattern, callback(page, model, params, next) )`
+>
+> `app.del ( routePattern, callback(page, model, params, next) )`
+>
+> * `pattern`: A string containing a literal URL, an Express route pattern, or a regular expression. See [Express's routing documentation](https://expressjs.com/guide/routing.html) for more info.
+>
+> * `callback`: Function invoked when a request for a URL matching the appropriate HTTP method and pattern is received. Note that this function is called both on the server and the client.
+>
+> * `page`: Object with the methods [`page.render()`](#pagerender) and `page.redirect()`. All app routes should call one of these two methods or pass control by calling `next()`.
+>
+> * `model`: Derby model object
+>
+> * `params`: An object containing the matching URL parameters. The `url`, `query`, and `body` properties typically available on `req` are also added to this object.
+>
+> * `next`: A function that can be called to pass control to the next matching route. If this is called on the client, control will be passed to the next route defined in the app. If no other routes in the same app match, it will fall through to a server request.
+
+Express is used directly on the server. On the client, Derby includes Express's route matching module. When a link is clicked or a form is submitted, Derby first tries to render the new URL on the client. AJAX requests will still go directly to the server.
+
+Derby can also capture form submissions client-side. It provides support for `post`, `put`, and `del` HTTP methods using the same hidden form field [override approach](https://expressjs.com/guide.html#http-methods) as Express.
+
+## Page
+
+Unlike Express, which provides direct access to the `req` and `res` objects created by Node HTTP servers, Derby returns a `page` object. This provide the same interface on the client and the server, so that route handlers may be executed in both environments.
+
+> `page.render ( viewName )`
+>
+> * `viewName`: The name of the view to render, see [Namespaces and files](../views/namespaces-and-files) for more details.
+>
+>
+> `page.renderStatic ( statusCode, content )`
+>
+> * `statusCode`: The HTTP status code to return.
+>
+> * `content`: A string of HTML to render
+>
+> `page.redirect ( url, [status] )`
+>
+> * `url`: Destination of redirect. [Like Express][expressRedirect], may also be the string 'home' (which redirects to '/') or 'back' (which goes back to the previous URL).
+>
+> * `status`: *(optional)* Number specifying HTTP status code. Defaults to 302 on the server. Has no effect on the client.
+
+[expressRedirect]: https://expressjs.com/guide.html#res.redirect()
+
+
+### Middleware
+
+It is possible to directly use [express middleware](https://expressjs.com/guide/using-middleware.html) and get access to a [Racer model](../models#methods).
+
+
+## History
+
+For the most part, updating the URL client-side should be done with normal HTML links. The default action of requesting a new page from the server is canceled automatically if the app has a route that matches the new URL.
+
+To update the URL after an action other than clicking a link, scripts can call methods on `app.history`. For example, an app might update the URL as the user scrolls and the page loads more content from a paginated list.
+
+> `app.history.push ( url, [render], [state], [e] )`
+>
+> `app.history.replace ( url, [render], [state], [e] )`
+>
+> * `url`: New URL to set for the current window
+>
+> * `render`: *(optional)* Re-render the page after updating the URL if true. Defaults to true
+>
+> * `state`: *(optional)* A state object to pass to the `window.history.pushState` or `window.history.replaceState` method. `$render` and `$method` properties are added to this object for internal use when handling `popstate` events
+>
+> * `e`: *(optional)* An event object whose `stopPropogation` method will be called if the URL can be rendered client-side
+
+Derby's `history.push` and `history.replace` methods will update the URL via `window.history.pushState` or `window.history.replaceState`, respectively. They will fall back to setting `window.location` and server-rendering the new URL if a browser does not support these methods. The `push` method is used to update the URL and create a new entry in the browser's back/forward history. The `replace` method is used to only update the URL without creating an entry in the back/forward history.
+
+> `app.history.refresh ( )`
+>
+> Re-render the current URL client-side
+
+For convenience, the navigational methods of [`window.history`](https://developer.mozilla.org/en/DOM/window.history) can also be called on `app.history`.
+
+> `app.history.back ( )`
+>
+> * Call `window.history.back()`, which is equivalent to clicking the browser's back button
+
+> `view.history.forward ( )`
+>
+> * Call `window.history.forward()`, which is equivalent to clicking the browser's forward button
+
+> `view.history.go ( i )`
+>
+> * Call `window.history.go()`
+>
+> * `i`: An integer specifying the number of times to go back or forward. Navigates back if negative or forward if positive
diff --git a/docs/views.md b/docs/views.md
new file mode 100644
index 00000000..59944180
--- /dev/null
+++ b/docs/views.md
@@ -0,0 +1,77 @@
+---
+layout: default
+title: Views
+has_children: true
+---
+
+# Views
+
+When writing an app or new feature in Derby, you should typically start by writing its view. Derby templates can be written in HTML or Jade with [derby-jade](https://github.com/derbyparty/derby-jade). Templates define HTML/DOM output, data bindings, event listeners, and component parameters.
+
+## Creating views
+
+Views are written in HTML files. These files are parsed and added to a Derby app with the `app.loadViews()` method. This method synchronously reads template files, traverses their includes, and calls `app.views.register()` for each view.
+
+> `app.loadViews(filename)`
+> * `filename` File path to root template file at which to start loading views
+
+> `app.views.register(name, source, options)`
+> * `name` View name to add
+> * `source` Derby HTML source
+> * `options:`
+> * `tag` Name of an HTML tag that will render this view
+> * `attributes` Space separated list of HTML tags interpreted as an attribute when directly within the view instance
+> * `arrays` Space separated list of HTML tags interpreted as an array of objects attribute when directly within the view instance
+> * `unminified` Whitespace is removed from templates by default. Set true to disable
+> * `string` True if the template should be interpreted as a string instead of HTML
+
+> `view = app.views.find(name, [namespace])`
+> * `name` View name to find
+> * `namespace` *(optional)* Namespace from which to start the name lookup
+> * `view` Returns the view template object
+
+Each view is wrapped in a tag that names it. This name must end in a colon to differentiate it from a normal HTML tag. These tags can't be nested, and they need not be closed.
+
+```derby
+
+
Hello, sir.
+
+
+
Howdy!
+```
+
+is equivalent to:
+
+```js
+app.views.register('serious-title', '
Hello, sir.
');
+app.views.register('friendly-title', '
Howdy!
');
+```
+
+## Using views
+
+You can instantiate a view in a template with the `` tag, `{{view}}` expression, or by giving the view a tag name. Typically, you should use the `` tag in HTML templates. The `{{view}}` expression is useful when writing string templates or wish to include a view in an HTML attribute, script tag, or style tag. Custom tag names are global to an application. They are recommended for general purpose components, like `` or ``, but not for ordinary views.
+
+```derby
+
+
Hello, sir.
+
+
+
+
+
+
+
+ {{view 'serious-title'}}
+
+
+
+```
+
+Views may be looked up dynamically with an expression. If the view isn't found, nothing will be rendered.
+
+```derby
+
+
+
+ {{view type + '-title'}}
+```
diff --git a/docs/views/namespaces-and-files.md b/docs/views/namespaces-and-files.md
new file mode 100644
index 00000000..5ec7fa37
--- /dev/null
+++ b/docs/views/namespaces-and-files.md
@@ -0,0 +1,124 @@
+---
+layout: default
+title: Namespaces and files
+parent: Views
+---
+
+# View namespaces
+
+View names have colon (`:`) separated namespaces. Lookups of views are relative to the namespace in which they are used. Thus, sub-views within components or different sections of large applications are well encapsulated and won't cause naming conflicts.
+
+```derby
+
+ ...
+
+
+ ...
+
+
+
+
+```
+
+In addition, similar to the way that CSS allows overriding of styles by using a more specific selector, you can define views at a general namespace and then redefine them at a more specific namespace.
+
+```derby
+
+ App
+
+
+ About - App
+
+
+ Mission statement - App
+```
+
+### Custom HTML tags
+
+A view can be turned into a custom HTML tag by specifying the `tag` property in it's definition. Custom tag names are global so care should be taken in their usage.
+
+```derby
+
+
+
{{data}}
+
+
+
+
+```
+
+## Structuring views in multiple files
+
+Views should be broken into files that correspond to major pieces of functionality, different URLs, or components. Views are included from another file with the `` tag.
+
+```derby
+
+
+
+
+
+
+
+
+```
+
+Typically, view namespaces have a one-to-one correspondence with directories and files. For example, a typical structure like:
+
+#### index.html
+```derby
+
+
+
+ App
+```
+
+#### about/index.html
+```derby
+
+
+
+ About - App
+```
+
+#### about/mission.html
+```derby
+
+ Mission statement - App
+```
+
+would be equivalent to:
+
+`index.html`
+```derby
+
+ App
+
+
+ About - App
+
+
+ Mission statement - App
+```
+
+Rules for importing views work the same way as [Node.js module loading](https://nodejs.org/api/modules.html) with `require()`. The `src` attribute uses the same syntax of relative paths or paths to `node_modules`. An `index.html` file can be imported via the name of the directory that it is in, just like `index.js` files in Node.js.
+
+As well, the name `index` can be used for a view that is returned for just the name of its namespace.
+
+#### index.html
+```derby
+
+
+
+
+```
+
+#### home.html
+```derby
+
+
+
+
+
+
+ Hello!
+```
diff --git a/docs/views/template-syntax.md b/docs/views/template-syntax.md
new file mode 100644
index 00000000..c6c9b754
--- /dev/null
+++ b/docs/views/template-syntax.md
@@ -0,0 +1,81 @@
+---
+layout: default
+title: Template syntax
+parent: Views
+has_children: true
+---
+
+# Template syntax
+
+Derby’s template syntax is loosely based on [Handlebars](https://handlebarsjs.com/), a semantic templating language that extends [Mustache](https://mustache.github.io/mustache.5.html).
+
+Semantic templates encourage separation of logic from presentation. Instead of arbitrary code, there are a small set of template expressions. During rendering, data are passed to the template, and template expressions are replaced with the appropriate values. In Derby, this data comes from the model.
+
+Derby supports calling controller functions and the full JavaScript expression syntax. Expressions are parsed with [Esprima](http://esprima.org/) and do not use JavaScript's `eval()` or `new Function()`.
+
+## HTML
+
+With the exception of templates that return strings and the contents of `
+```
+
+HTML comments are removed, except for those with square brackets immediately inside of them. This syntax is used by Internet Explorer's conditional comments. It can also be used to include comments for copyright notices or the like.
+
+```derby
+
+
+
+
+```
diff --git a/docs/views/template-syntax/blocks.md b/docs/views/template-syntax/blocks.md
new file mode 100644
index 00000000..06150cec
--- /dev/null
+++ b/docs/views/template-syntax/blocks.md
@@ -0,0 +1,107 @@
+---
+layout: default
+title: Blocks
+parent: Template syntax
+grand_parent: Views
+---
+
+# Blocks
+
+Blocks are template expressions that start with special keywords. They are used to conditionally render, repeat, or control the way in which sections of a template are rendered.
+
+Similar to HTML tags, blocks end in a forward slash followed by the same keyword that started them. The closing keyword is optional but recommended for clarity. For example, both `{{with}}...{{/with}}` and `{{with}}...{{/}}` are parsed correctly.
+
+# Conditionals
+
+Conditional blocks use the `if`, `else if`, `else`, and `unless` keywords. They render the first template section that matches a condition or nothing if none match. Like in Mustache and Handlebars, zero length arrays (`[]`) are treated as falsey. Other than that, falsey values are the same as JavaScript: `false`, `undefined`, `null`, `''`, and `0`.
+
+```derby
+{{if user.name}}
+
user.name
+{{else if user}}
+
Unnamed user
+{{else}}
+ No user
+{{/if}}
+```
+
+The inverse of `if` is `unless`. For clarity, unless should only be used when there is no `else` condition. A block that has an unless and else condition can usually be writtern more clearly as an if and else.
+
+```derby
+{{unless items}}
+ Please add some items
+{{/unless}}
+```
+
+The contents of a conditional block are only re-rendered when a different condition starts to match. If the values in the conditional change, the condition expression is evaluated, but the DOM is not updated if the same section matches.
+
+# Each
+
+Each blocks repeat for each of the items in an array. They cannot iterate over objects.
+
+```derby
+{{each items}}
+
{{this.text}}
+{{else}}
+ No items
+{{/each}}
+```
+
+In addition to an alias to the array item, eaches support an alias for the index of the item. This index alias supports binding and will be updated as the array changes.
+
+```derby
+{{each items as #item, #i}}
+ {{#i + 1}}. {{#item.text}}
+{{/each}}
+```
+
+Derby has very granular model events to describe array mutations as inserts, removes, and moves. It maps these directly into efficient DOM mutations of just what changed.
+
+# With
+
+With blocks set the path context of a block, but they do not trigger re-rendering. Their primary use is to set an alias to a path inside of their contents.
+
+Aliases can be a convenient way to set a name that can be used throughout a section of a template or many nested views and/or components.
+
+```derby
+
+ {{with _session.user as #user}}
+
+ {{/with}}
+
+ {{with {name: 'Jim', age: 32} as #user}}
+
+ {{/with}}
+
+
+
{{#user.name}}
+
{{#user.age}}
+```
+
+# On
+
+To clear UI state, to optimize performance by rendering larger sections, or to work around issues with template bindings not rendering often enough, an `{{on}}` block can provide more control. Its contents will re-render whenever any of its paths change.
+
+```derby
+{{on #profile.id}}
+
{{#profile.name}}
+
+{{/on}}
+
+{{on first, second, third}}
+
+{{/on}}
+```
+
+# Unbound and bound
+
+Bindings are created by default for all template expressions. To render an initial value only and not create bindings, the `{{unbound}}` block may be wrapped around a template section. Bindings can be toggled back on with a `{{bound}}` block.
+
+```derby
+{{unbound}}
+
+ {{bound}}
+
+ {{/bound}}
+{{/unbound}}
+```
diff --git a/docs/views/template-syntax/escaping.md b/docs/views/template-syntax/escaping.md
new file mode 100644
index 00000000..4604efa8
--- /dev/null
+++ b/docs/views/template-syntax/escaping.md
@@ -0,0 +1,44 @@
+---
+layout: default
+title: Escaping
+parent: Template syntax
+grand_parent: Views
+---
+
+# Escaping
+
+Derby escapes values as required when it renders HTML. Escaping is relative to whether it is rendering inside of an HTML attribute or text. For example, Derby will escape `"` as `"` inside of an HTML attribute, and it will escape `<` as `<` inside of text.
+
+Derby's templates also follow HTML escaping rules. Derby will parse the string `{{` as the start of a template tag, so if you wish to write this value in an attribute or the text of a Derby template, you can use the HTML entity equivalent: `{{`.
+
+## Rendering unescaped HTML
+
+The `unescaped` keyword may be used to render an HTML string without escaping. It is *very unlikely* that you should use this feature. Derby has many ways of dynamically creating views. Unescaped HTML is unsafe, is typically slower, and is rarely necessary with Derby. This feature is intended only for rendering the output of a well-tested library that produces sanitized HTML, such as [Google Caja](https://developers.google.com/caja/).
+
+```derby
+
+
{{unescaped rawHtml}}
+```
+
+Instead, prefer passing in a template as an attribute or dynamically selecting a view in most cases.
+
+```derby
+
+
+
+ Custom HTML for this user!
+
+
+
+
+
+ {{@content}}
+
+```
+
+```derby
+
+
+```
+
+If you need completely dynamic generation of HTML (such as implementing an HTML or template editor in your application), it is even possible to use Derby's HTML parser and pass the returned Template object to your views. Derby will render this HTML safely without any Cross-site Scripting (XSS) concerns. You'll even be able to use Derby's template syntax! See how this is done in the [Derby render example](https://github.com/derbyjs/derby-examples/blob/master/render/index.js#L29), which powers the live template editor on the [DerbyJS home page](https://derbyjs.com/).
diff --git a/docs/views/template-syntax/functions-and-events.md b/docs/views/template-syntax/functions-and-events.md
new file mode 100644
index 00000000..607979e4
--- /dev/null
+++ b/docs/views/template-syntax/functions-and-events.md
@@ -0,0 +1,143 @@
+---
+layout: default
+title: Functions and events
+parent: Template syntax
+grand_parent: Views
+---
+
+# Functions and events
+
+Attributes beginning with `on-` add listeners to DOM events and component events. Under the hood, events on elements are added with `element.addEventListener()` and events on components are added with `component.on()`. Adding events declaritively with attributes is easier than CSS selectors and less prone to unexpectedly breaking when refactoring templates or classes for styling.
+
+```derby
+
+
+```
+
+```js
+// Equivalent to:
+input.addEventListener('mousedown', function(event) {
+ self.mousedownInput(event);
+}, false);
+input.addEventListener('blur', function(event) {
+ self.blurInput();
+ self.update();
+}, false);
+```
+
+## View functions
+
+Functions are looked up on the current component's controller, the page, and the global, in that order. The majority of functions are defined on component prototypes, generic shared utility functions are defined on the page prototype, and the global provides access to functions like `new Date()` and `console.log()`.
+
+```derby
+
+
+
+{{sum(1, 2, 4)}}
+
+{{console.log('rendering value', value)}}
+```
+
+```js
+// component prototypes are where most functions are defined
+UserList.prototype.delUser = function(userId) {
+ this.users.del(userId);
+};
+
+// app.proto is the prototype for all pages created by the app
+app.proto.sum = function() {
+ var sum = 0;
+ for (var i = 0; i < arguments.length; i++) {
+ sum += arguments[i];
+ }
+ return sum;
+};
+```
+
+### Component events
+Components support custom events. Dashes are transformed into camelCase.
+
+See the [component events](components/events) documentation for more detail on using events and component functions.
+```derby
+
+```
+
+```js
+// Equivalent to:
+modal.on('close', function() {
+ self.reset();
+});
+modal.on('fullView', function() {
+ back.fade();
+});
+```
+
+
+### Special HTML rules
+
+As a convenience, an `on-click` listener can be added to a link without an `href`. Derby will add an `href="#"` and prevent the default action automatically if no href is specified.
+
+```derby
+
+Hi
+```
+
+HTML forms have very useful behavior, but their default action on submit will navigate away from the current page. If an `on-submit` handler is added to a form with no `action` attribute, the default will be prevented.
+
+```derby
+
+```
+
+### DOM event arguments
+
+For functions invoked by DOM events only, the special arguments `$event` or `$element` may be specified. The `$event` argument is the DOM Event object passed to the listener function for `addEventListener()`. The `$element` argument is a reference to the element on which the listener attribute is specified. These arguments are only passed to functions if explicitly specified.
+
+```derby
+
+```
+
+```js
+UserList.prototype.clickRow = function(e, tr) {
+ // Ignore clicks on or in links
+ var node = e.target;
+ while (node && node !== tr) {
+ if (node.tagName === 'A') return;
+ node = node.parentNode;
+ }
+ // Cancel the original click event inside of the row
+ e.stopPropagation();
+ // Pretend like the click happened on the first link in the row
+ var event = new MouseEvent('click', e);
+ var link = tr.querySelector('a');
+ if (link) link.dispatchEvent(event);
+};
+```
+
+## Scoped model arguments
+
+Functions can be passed the value of any view path. In some cases, it can be convenient to get a [scoped model](../../models/paths#scoped-models) to the view name instead. To pass a scoped model, you can wrap the view path in `$at()`. Instead of getting the value for a view path, this will return a scoped model. It will return undefined if no scoped model can be created for a view path.
+
+```derby
+
+```
+
+```js
+app.proto.toggle = function(scoped) {
+ scoped.set(!scoped.get());
+};
+```
\ No newline at end of file
diff --git a/docs/views/template-syntax/literals.md b/docs/views/template-syntax/literals.md
new file mode 100644
index 00000000..fcf83f78
--- /dev/null
+++ b/docs/views/template-syntax/literals.md
@@ -0,0 +1,69 @@
+---
+layout: default
+title: Literals
+parent: Template syntax
+grand_parent: Views
+---
+
+# Literals
+
+Derby supports creating JavaScript literals in templates. The syntax is identical to JavaScript, except that identifiers within literals are parsed as [view paths](paths) instead of JavaScript variables. Derby parses template expressions with Esprima, so its coverage of JavaScript expression syntax is comprehensive.
+
+## Simple literals
+
+```derby
+
+{{0}}
+{{1.1e3}}
+{{0xff}}
+
+{{true}}
+{{false}}
+
+{{'Hi'}}
+{{"Hey"}}
+
+{{/([0-9])+/}}
+
+{{null}}
+
+{{undefined}}
+
+{{ [0, 1, 2] }}
+
+{{ {name: 'Jim'} }}
+```y, simple literals are instantiated at the time of parsing. Object literals created at parse time will be passed by reference to controller functions, so be careful not to modify them.
+
+```derby
+
+
+```ls in template expressions. In most cases, it makes more sense to define constants in the controller or use HTML, but this can be handy when prototyping and debugging.
+
+```derby
+
+ {{each ['A', 'B', 'C'] as #let```derby
+
+ {{each ['A', 'B', 'C'] as #letter}}
+
{{#letter}}
+ {{/each}}
+
+
+{{with {name: 'Jim', age: 23} as #user}}
+
{{#user.name}}
+
{{#user.age}}
+{{/with}}
+``` at render time and populated with the appropriate values from the model.
+
+```derby
+
+ {{each [first, 1, 2, 3] as #item}}
+
{{#item}}<```derby
+
+ {{each [first, 1, 2, 3] as #item}}
+
{{#item}}
+ {{/each}}
+
+
+
+```
\ No newline at end of file
diff --git a/docs/views/template-syntax/operators.md b/docs/views/template-syntax/operators.md
new file mode 100644
index 00000000..8602ae82
--- /dev/null
+++ b/docs/views/template-syntax/operators.md
@@ -0,0 +1,104 @@
+---
+layout: default
+title: Operators
+parent: Template syntax
+grand_parent: Views
+---
+
+# Operators
+
+All non-assigment JavaScript operators and using parentheses for grouping expressions are supported. They work exactly the same as they do in Javascript. Operators that do an assignment, such as the `++` increment operator, are not supported. This avoids rendering having side effects.
+
+```derby
+
+{{-value}}
+
+{{+value}}
+
+
+{{left + right}}
+
+{{left - right}}
+
+{{left * right}}
+
+{{left / right}}
+
+{{left % right}}
+
+
+{{!value}}
+
+{{left || right}}
+
+{{left && right}}
+
+
+{{~value}}
+
+{{left | right}}
+
+{{left & right}}
+
+{{left ^ right}}
+
+
+{{left << right}}
+
+{{left >> right}}
+
+{{left >>> right}}
+
+
+{{left === right}}
+
+{{left !== right}}
+
+{{left == right}}
+
+{{left != right}}
+
+
+{{left < right}}
+
+{{left > right}}
+
+{{left <= right}}
+
+{{left >= right}}
+
+
+{{typeof value}}
+
+{{left instanceof right}}
+
+{{left in right}}
+
+
+{{test ? consequent : alternate}}
+
+
+{{a, b, c, d}}
+```
+
+## Two-way operators
+
+In addition to getting values, operators for which there is a well defined opposite support two-way data bindings. These setters will make the relationship consistent with the value that is set.
+
+```derby
+
+
+
+
+{{each options as #option}}
+
+{{/each}}
+```
diff --git a/docs/views/template-syntax/paths.md b/docs/views/template-syntax/paths.md
new file mode 100644
index 00000000..ee6ae22d
--- /dev/null
+++ b/docs/views/template-syntax/paths.md
@@ -0,0 +1,99 @@
+---
+layout: default
+title: Paths
+parent: Template syntax
+grand_parent: Views
+---
+
+# Paths
+
+Template paths use JavaScript syntax with a few small modifications.
+
+## Model values
+
+What would be identifiers for variable names in JavaScript get a value from the model and bind to any updates. If the path returns null or undefined, nothing is rendered.
+
+Examples of rendering model values:
+
+```derby
+{{user.name}}
+
+{{user.bestFriends[0].name}}
+
+{{users[userId].name}}
+```
+
+```js
+model.get('user.name');
+
+model.get('user.bestFriends.0.name');
+
+var userId = model.get('userId');
+model.get('users.' + userId + '.name');
+```
+
+## Attributes
+
+Values are passed into views with attributes. Within the view, these values are accessed via paths that start with an at sign (`@`). In addition, there is an `@content` attribute created for any content inside of a view tag.
+
+```derby
+
+
+```
+
+See [View attributes](view-attributes) for additional detail on passing data to views.
+
+## Aliases
+
+Aliases label path expressions. They must begin with a hash (`#`) character to make it more obvious whether a path is an alias or a model value. Each of the block types support defining aliases with the `as` keyword.
+
+Aliases make it possible to refer to the scope of the current block or a parent block.
+
+```derby
+{{with user as #user}}
+
{{#user.name}}
+
{{#user.headline}}
+ {{if #user.friendList as #friendList}}
+
+
Friends of {{#user.name}}
+
+ {{each #friendList as #friend}}
+
{{#friend.name}}
+ {{/each}}
+
+ {{/if}}
+{{/with}}
+```
+
+## Relative paths - DEPRECATED
+
+Relative view paths begin with `this`. They refer to the expression in the containing block.
+
+Aliases are preferred to relative paths, as they are more clear. Relative paths came from implementing a syntax inspired by Handlebars, but Derby has been moving toward increased consistency with JavaScript, and the alternate use of the keyword `this` is confusing. Expect that this feature will be removed in a future version of Derby.
+
+```derby
+{{with user}}
+
{{this.name}}
+
{{this.headline}}
+ {{if this.friendList}}
+
Friends
+
+ {{each this}}
+
{{this.name}}
+ {{/each}}
+
+ {{/if}}
+{{/with}}
+```
diff --git a/docs/views/template-syntax/view-attributes.md b/docs/views/template-syntax/view-attributes.md
new file mode 100644
index 00000000..606346d9
--- /dev/null
+++ b/docs/views/template-syntax/view-attributes.md
@@ -0,0 +1,246 @@
+---
+layout: default
+title: View attributes
+parent: Template syntax
+grand_parent: Views
+---
+
+# View attributes
+
+A `` element or `{{view}}` expression renders a view in a particular location. In other words, it creates a "view instance." The `is` attribute specifies the name of the view to render. If a view has been declared as a component with the `app.component()` method, a view instance instantiates a component as well.
+
+In addition, to `is`, Derby treats the attributes `as`, `within`, `inherit`, `extend`, and attributes that begin with `on-` specially. These special attributes are described in below sections.
+
+Other attribute names are custom to a view or component, and they are used to pass in data or options to a given view or component instance. We'll describe use of these custom attributes first.
+
+## Custom view attributes
+
+Most view attributes are custom options particular to a view or component. They are analogous to the arguments of a function. They might provide data inputs, a two-way data binding, toggle options, or pass in a template.
+
+Custom attributes may be any [literal value](literals).
+
+```derby
+
+
+
+
+
+
+
+
+
+
+
+
+
+Hello, World.
+```
+
+They may also be any other type of expression, such as a path expression, an expression that uses an operator, or a function expression.
+
+```derby
+
+
+
+
+
+
+
+
+
+
+
+
+
+{{modalBody}}
+```
+
+Finally, an attribute value may be a template.
+
+```derby
+
+
+
+
+
+ , World.
+
+```
+
+Derby's ability to pass in templates as attribute values is very powerful and can be highly expressive. By default, any template content between the opening and closing `` tags is passed in as an attribute with the name `content`. However, a view instance may wish to specify an HTML template for another attribute as well. In these cases, an `` tag may be used immediately within a view instance.
+
+```derby
+
+
+ Introducing, {{modalTitle}}
+
+ , World.
+
+```
+
+Beyond individual attribute values, it is common to want to pass in a list of items. This can be achieved with an array attribute. `` tags produce an array of objects, where each HTML tag corresponds to an object in an array.
+
+```derby
+
+
+ Hello!
+ Greetings.
+
+```
+
+To make instantiating commonly used components more elegant, Derby supports the ability to declare a custom tag name that refers to a view as well as the attributes and array attributes that a view supports.
+
+Definition:
+
+```derby
+
+
+
+
+ {{each @panes as #pane, #i}}
+
{{#pane.title}}
+ {{/each}}
+
+ {{@panes[@selectedIndex].content}}
+```
+
+Usage:
+
+```derby
+
+
+
+ {{0}}
+ Hello!
+ Greetings.
+
+```
+
+### `within` attributes
+
+When passing values into attributes with expressions or templates, names are contextual to the location where the view is instantiated. This, too, is similar to how arguments are passed into functions: When calling a function, the name of the variable being passed in is whatever that name means at the location of the function call, not what the name means inside of the function. Inside of the function itself, the name could mean something different or be undefined.
+
+Adding `within` changes this. Rather than using the meaning of the name where the view is instantiated, Derby will use the meaning of the name wherever the attribute is ultimately rendered.
+
+This is a powerful feature for defining a component or view that has a general capability, such as a list that can be manually sorted with drag-and-drop. Then, instances can pass in a template that is rendered within that list as if it was defined there.
+
+Any attribute may be declared as `within`. Adding `within` to a view tag affect's the view's `content` attribute. To specify other attributes as within, use the expanded `` or `` form of passing in an attribute.
+
+Definition:
+
+```derby
+
+
+ {{each @options as #option, #i}}
+
{{@content}}
+ {{/each}}
+
+```
+
+Usage:
+
+```derby
+
+
+
+ {{#i + 1}}. {{#option.text}}
+
+```
+
+Within attributes are always passed in as Template objects, even if they contain only a single expression (see following section). This is because the meaning of the paths in the expression may change depending on the context in which the attribute is ultimately used. Therefore, passing a `within` attribute into a component will always set a Template object on the component's model.
+
+If a `within` attribute is passed to a component, the component's `getAttribute()` method will return the meaning of the attribute as if it were rendered immediately inside of the component's main view. Thus, model paths will refer to model paths within the component.
+
+### Relation of view attributes to a component's model
+
+View attributes are the primary way to pass values into a component's model. Right before Derby calls a component's `init` method, it gets the current value for each view attribute and sets it on the component's model with the same name. Therefore, inside of a component, the model path can be used to access the value of the attribute.
+
+How an attribute value is set on the model differs depending on the value type. The guidelines for Derby's approach are:
+
+* When model values coming from attributes are used within the component's view template, they should render what an attribute path of the same name would be able to render. This means, for example, that templates which return HTML elements cannot be set on the model as strings of HTML. Instead, they are set on the model as Template objects
+* Attributes are both an input and an output (two-way bound) when possible
+* The type of the value passed in should be as intuitive and predictable to developers as possible
+* Special syntax should be kept to a minimum. This is both to make templates easier to read and to make template authoring and editing more widely accessible to team members with HTML experience but little programming experience
+
+Keeping these goals in mind, Derby will set values on a component model with one of four approaches:
+
+**1. Literal Value – Input**
+```derby
+value="Hello"
+value="{{42}}"
+Hello
+```
+
+**2. Model Reference – Input and Output**
+```derby
+value="{{inputText}}"
+value="{{todos[#id].text}}"
+```
+
+**3. Value with Attribute Binding – Input (for now)**
+```derby
+disabled="{{!active}}"
+label="{{userLabel || 'User'}}"
+list="{{reverse(items)}}"
+options="{{ {speed: currentSpeed} }}"
+```
+
+**4. Template – Input**
+```derby
+class="{{if show}}visible{{/if}}"
+Hello
+```
+
+In all cases, the value inside the component model is updated to remain consistent with any dynamic inputs. Literal Values and Templates are set on the component model initially only. Literal Values will not change, and paths within Template objects are bound when they are later used somewhere inside the component's view. For attributes containing a single expression and no additional text or templating features, Derby either creates a Model Reference or an Attribute Binding. In both cases, the current value of the expression is set initially. In addition, when data changes for paths within the expression, the model value is updated.
+
+Derby creates a Model Reference (`model.ref()`) for attribute expressions where possible. Model References reflect changes in both directions, so this establishes a two-way binding. The expressions that can be represented with Model References are the ones that can be resolved to an equivalent path in the model: specific path expressions, aliases and attributes that refer to specific paths, and square bracket expressions.
+
+ Any dynamic expression that cannot be resolved to an equivalent model path, such as an expression using a function, an operator, or an array or object literal, establishes a Component Attribute Binding. Currently, these are only implemented as one-way input bindings—the value of such an expression is set initially, and as its dependencies change, the value is recomputed and set on the component's model again. It is worth mentioning that function expressions and some operators define both a `get` and a `set` function, enabling them to be used in two-way bindings. For example, the `!` (not) operator supports a `set` function. These two-way function and operator bindings are supported by Derby's HTML element bindings, such as ``. However, component attributes do not support use of `set` functions yet. Therefore, two-way bindings like `` do not work currently. (They do work without the `!`, assuming Derby is able to make a Model Reference.)
+
+## `as` attribute
+
+The `as` attribute may be applied to both components and DOM elements. Before a component's `create` method is called, Derby will set these items as properties of the current controller. This provides easy access to these items within the controller code of a component.
+
+```derby
+
+
+
+
+```
+
+```js
+// DOM element
+this.container.querySelectorAll('*');
+// component
+this.modal.close();
+```
+
+## `inherit` attribute
+
+Adding the `inherit` attribute changes the behavior of attribute lookup within the instantiated view. By default, attribute values are only defined in the view that they are passed into explicitly. Passing attribute values through one view into another normally requires manually repeating the attributes, such as `class="{{@class}}"`. `inherit` modifies the default behavior, making all attributes of the parent view instance implicitly available as well. Explicitly providing an attribute will take precedence over `inherit`.
+
+```derby
+
+ {{@foo}}
+
+
+
+ {{@foo}}
+```
+
+## `extend` attribute
+
+Extend is used to render the view of a component **without** instantiating the component object that would normally be created. It can be useful if one component class would like to extend from the class of another component, and it does not wish to modify the component's view.
+
+## `on-` attributes
+
+View attributes beginning with `on-` attach event listeners to component events. See [Component Events](../../components/events).
diff --git a/src/App.ts b/src/App.ts
index 7add9e7f..22c41a4a 100644
--- a/src/App.ts
+++ b/src/App.ts
@@ -183,9 +183,13 @@ export abstract class AppBase extends EventEmitter {
this.addViews(viewSource, viewName);
view = this.views.find(viewName);
- } else if (name) {
+ } else if (viewName) {
+ // Look for a previously registered view matching the component name.
view = this.views.find(viewName);
-
+ // If no match, register a new empty view, for backwards compatibility.
+ if (!view) {
+ view = this.views.register(viewName, '');
+ }
} else {
view = this.views.register(viewName, '');
}
diff --git a/test/all/ComponentHarness.mocha.js b/test/all/ComponentHarness.mocha.js
index 8db79576..6ad7c66c 100644
--- a/test/all/ComponentHarness.mocha.js
+++ b/test/all/ComponentHarness.mocha.js
@@ -258,8 +258,6 @@ describe('ComponentHarness', function() {
});
it('gets overridden without error', function() {
- function ConflictingClown() {}
- ConflictingClown.view = {is: 'clown'};
function Clown() {}
Clown.view = {
is: 'clown',
@@ -277,6 +275,8 @@ describe('ComponentHarness', function() {
'
',
dependencies: [Clown]
};
+ function ConflictingClown() {}
+ ConflictingClown.view = {is: 'clown', source: ''};
var html = new ComponentHarness(
'', Box, ConflictingClown
).renderHtml().html;
diff --git a/test/dom/components.mocha.js b/test/dom/components.mocha.js
index b7aa25ac..94913db5 100644
--- a/test/dom/components.mocha.js
+++ b/test/dom/components.mocha.js
@@ -1,6 +1,6 @@
var expect = require('chai').expect;
var pathLib = require('node:path');
-var domTestRunner = require('../../src/test-utils/domTestRunner');
+var domTestRunner = require('../../test-utils/domTestRunner');
describe('components', function() {
var runner = domTestRunner.install();
@@ -55,4 +55,4 @@ describe('components', function() {
});
});
});
-});
\ No newline at end of file
+});