This branch contains a preview of the Polymer 2.0 library. The codebase is under active development, features may not be fully implemented, and APIs may change prior to the final 2.0 release.
🚧 To evaluate Polymer 2.0, please load the webcomponentsjs/webcomponents-lite.js
or webcomponentsjs/webcomponents-loader.js
polyfills from the v1
branch of webcomponentsjs
-
Take advantage of native "v1" Web Components implementations across browsers.
The primary goal of the Polymer 2.0 release is to take advantage of native, cross-browser support for Web Components.
Polymer 1.x is built on top of the so-called "v0" Web Components specs, which are supported natively only in Google Chrome; using Polymer in other browsers has always required the use of polyfills.
Beginning this fall, multiple browsers will be shipping native implementations of the new "v1" specs for Shadow DOM and Custom Elements, yielding better web components performance and reducing the need for polyfills.
Polymer 2.0 features full support for the v1 specs, taking advantage of native browser implementations where they are available and depending on updated v1 polyfills from webcomponentsjs where necessary.
Polymer 2.0 also embraces the new ES-class-based mechanism for defining custom elements, bringing idiomatic Polymer style closer to "vanilla" custom element authoring.
-
Provide a smooth migration path from Polymer 1.x.
Our second major goal is to provide as easy a transition as possible for developers who have built elements and apps with Polymer 1.x, making Polymer 2.0 a sturdy bridge to the future.
To upgrade, you will need to make some changes to your 1.x-based elements and apps. These changes are necessitated by both the v0-to-v1 spec transition and a handful of key improvements in Polymer itself (see our remaining goals, below).
However, we've taken care to limit the number of changes that are strictly required and to ease the process of upgrading:
-
Polymer 2.0 introduces a new ES6 class-based syntax, but we've provided a lightweight compatibility layer allowing you to upgrade your 1.x code with minimal modifications. Depending on your needs, you can either take advantage of the compatibility layer or jump straight to idiomatic 2.0 style.
-
Before releasing Polymer 2.0, we'll also provide an upgrade tool to automate as many of the changes (both required and recommended) as possible.
-
Finally, we're working on guidelines for building and testing "hybrid" elements that will run in both Polymer 1.x and Polymer 2.0. We plan to ship hybrid versions of all of the elements that we provide, easing the transition for developers who use them. Third-party element providers may also choose to ship hybrid elements.
-
If you have an especially large app or constraints that don't allow for an all-at-once upgrade, you can also use hybrid elements to migrate your app from 1.x to 2.0 in piecewise fashion: update your elements to hybrid form, individually or in batches, while running against Polymer 1.x; then cut over to Polymer 2.0 when all of your elements have been updated.
-
-
Eliminate leaky abstractions.
Seamless interoperability is one of Web Components' major selling points. Generally speaking, web components "just work" anywhere you use HTML elements. To use them, you need only be aware of their public attributes, properties, methods and events; you don't need to know anything about their inner workings. This means you can easily mix standard HTML elements, third-party elements and elements you've defined yourself.
Unfortunately, there are a couple of cases in Polymer 1.x (the
Polymer.dom
API and theset
/notifyPath
API) where implementation details of Polymer-based elements leak out, requiring users of the elements to interact with them in non-standard ways. These "leaks" were by design – compromises we chose to make in the interest of performance – but in hindsight we aren't happy with the tradeoff.In Polymer 2.0 we've found ways to eliminate these leaky abstractions without unduly compromising performance, which means that your Polymer 2.x-based elements will be indistinguishable from "vanilla" elements from a consumer's point of view (unless you leak implementation details of your own).
-
Make targeted improvements to the Polymer data system.
Based on developer feedback and observations of Polymer apps in the wild, we've also made some key improvements to Polymer's data system. These changes are designed to make it easier to reason about and debug the propagation of data through and between elements:
-
Changes are now batched, and the effects of those changes are run in well-defined order.
-
We ensure that multi-property observers run exactly once per turn for any set of changes to dependencies (removing the multi-property undefined rule).
-
To improve compatibility with top-down data-flow approaches (e.g. Flux), we no longer dirty-check properties whose values are objects or arrays.
-
-
Improve factoring of Polymer and the polyfills
We've done some refactoring of Polymer and the webcomponentsjs polyfills to improve efficiency, utility and flexibility:
-
The "Shady DOM" shim that was part of Polymer 1.x has been factored out of Polymer and added to the webcomponentsjs polyfills, along with the related shim for CSS Custom Properties. (As noted above, the Shady DOM shim no longer exposes an alternative API but instead patches the native DOM API.)
-
Polymer itself has been internally factored into several loosely coupled libraries.
-
The new
Polymer.Element
class extends from the nativeHTMLElement
and mixes in functionality from these libraries. -
The idiomatic way of using Polymer 2.0 (assuming you're not using the 1.x compatibility layer) is to define your own custom elements that subclass
Polymer.Element
, using standard ES class definition syntax. -
If you're interested in using pieces of Polymer's functionality in a la carte fashion, you can try defining your own base element class, utilizing a subset of the libraries. For now, this use case should be considered experimental, as the factoring of libraries is subject to change and is not part of the official Polymer 2.0 API.
-
-
You can install Polymer 2.0 using bower:
bower install --save Polymer/polymer#2.0-preview
Polymer 2.0 retains the existing polymer/polymer.html
import that current Polymer 1.0 users can continue to import, which strives to provide a very minimally-breaking change for code written to the Polymer 1.0 API. For the most part, existing users upgrading to Polymer 2.0 will only need to adapt existing code to be compliant with the V1 Shadow DOM API related to content distribution and styling, as well as minor breaking changes introduced due to changes in the V1 Custom Elements spec and data-layer improvements listed below.
With the widespread adoption of ES6 in browsers, as well as the requirement that V1 Custom Elements be defined as ES6 class extensions of HTMLElement
, Polymer 2.0 shifts its primary API for defining new elements to an ES6 class-centric syntax. Using this syntax, users will extend Polymer.Element
(a subclass of HTMLElement
), which provides meta-programming for most of the same features of Polymer 1.0 based on static configuration data supplied on the class definition.
Basic syntax looks like this:
<!-- Load the Polymer.Element base class -->
<link rel="import" href="bower_components/polymer/polymer-element.html">
// Extend Polymer.Element base class
class MyElement extends Polymer.Element {
static get is() { return 'my-element'; }
static get config() {
return { /* properties, observers meta data */ }
}
constructor() {
super();
...
}
connectedCallback() {
super.connectedCallback();
...
}
...
}
// Register custom element definition using standard platform API
customElements.define(MyElement.is, MyElement);
Users can then leverage native subclassing support provided by ES6 to extend and customize existing elements defined using ES6 syntax:
// Subclass existing element
class MyElementSubclass extends MyElement {
static get is() { return 'my-element-subclass'; }
static get config() { ... }
constructor() {
super();
...
}
...
}
// Register custom element definition using standard platform API
customElements.define(MyElementSubclass.is, MyElementSubclass);
Below are the general steps for defining a custom element using this new syntax:
- Extend from
Polymer.Element
. This class provides the minimal surface area to integrate with 2.0's data binding system. It provides only standard custom element lifecycle with the addition ofready
. (You can extend fromPolymer.LegacyElement
to get all of the Polymer 1.0 element api, but since most of this api was rarely used, this should not often be needed.) - Implement "behaviors" as mixins that return class expressions
- Property metadata (
properties
) and multi-property/wildcard observers (observers
) should be put on the class as a static in a property calledconfig
- Element's
is
property should be defined as a static on the class - Implement a static
template
getter to provide a template to stamp inside the element. By default the first<template>
found in a<dom-module>
with anid
matching the element'sis
property is used. listeners
andhostAttributes
have been removed from element metadata; they can be installed how and when needed but for convenienceensureAttribute
is available.
Note that Polymer.Element
provides a cleaner base class void of a lot of sugared utility API that present on elements defined with Polymer()
, such as fire
, transform
, etc. With web platform surface area becoming far more stable across browsers, we intend to hew towards sugaring less and embracing the raw platform API more. So when using Polymer.Element
, instead of using the legacy this.fire('some-event')
API, simply use the equivalent platform API's such as this.dispatchEvent(new CustomEvent('some-event'), {bubbles: true})
. #usetheplatform
See below for a visual guide on migrating Polymer 1.0's declarative syntax to the ES6 class syntax in Polymer 2.0:
![image](img/migration.png)Polymer 2.0 has been developed alongside and tested with a new suite of V1-spec compatible polyfills for Custom Elements and Shadow DOM. Polymer 2.0 can currently be tested by loading the v1
branch of webcomponentsjs/webcomponents-lite.js
, which is included as a bower dependency to Polymer 2.x and loads all necessary polyfills. The polyfills are still under active development and are not fully ready for use in other browsers or for production.
Below is a list of intentional breaking changes made in Polymer 2.0, along with their rationale/justification and migration guidance. If you find changes that broke existing code not documented here, please file an issue and we'll investigate to determine whether they are expected/intentional or not.
On browsers that lack native V1 Shadow DOM support, Polymer 2.0 is designed to be used with the new V1 Shady DOM shim, which patches native DOM API as necessary to be mostly equivalent to native Shadow DOM. This removes the requirement to use the Polymer.dom
API when interacting with the DOM. Polymer.dom
can be eliminated for elements targeting Polymer 2.0, in favor of the native DOM API's.
Note that Polymer.dom
is still provided in the polymer.html
backward-compatibility layer which simply facades the native API, but usage of it in 2.0 can be removed. Note that Polymer.dom
will no longer return Array
s for API's where the platform returns e.g. NodeList
's, so code may need to be updated to avoid direct use of array methods.
Polymer 2.0 elements will stamp their templates into shadow roots created using V1's attachShadow({mode: 'open'})
by default. As such, user code related to scoped styling, distribution, and events must be adapted to native V1 API. For a great writeup on all Shadow DOM V1 spec changes, see this writeup. Required changes for V1 are summarized below:
<content>
insertion points must be changed to<slot>
- Insertion points that selected content via
<content select="...">
must be changed to named slots:<slot name="...">
- Selection of distributed content into named slots must use
slot="..."
rather than tag/class/attributes selected by<content>
- Re-distributing content by placing a
<slot>
into an element that itself has named slots requires placing aname
attribute on the<slot>
to indicate what content it selects from its host children, and placing aslot
attribute to indicate where its selected content should be slotted into its parent - In the V1 "Shady DOM" shim, initial distribution of children into
<slot>
is asynchronous (microtask) to creating theshadowRoot
, meaning distribution occurs after observers/ready
(in Polymer 1.0's shim, initial distribution occurred beforeready
). In order to force distribution synchronously, callShadyDOM.flush()
.
::content
CSS pseudo-selectors must be changed to::slotted
, and may only target immediate children and use no descendant selectors:host-context()
pseudo-selectors have been removed. These were primarily useful for writing bidi rules (e.g.:host-context([dir=rtl])
); these should be replaced with the new:dir(rtl)
selector, which we plan to polyfill in the styling shim soon- The previously deprecated
/deep/
and::shadow
selectors have been completely removed from V1 native support and must not be used (use CSS custom properties to customize styling instead)
- Code using
Polymer.dom(event).localTarget
should change to the V1 standard APIevent.target
- Code using
Polymer.dom(event).path
(aka V0event.path
) should change to the V1 standard APIevent.composedPath()
- Code using
Polymer.dom(event).rootTarget
(aka V0event.path[0]
) should change to the V1 standard APIevent.composedPath()[0]
Polymer 2.0 elements will target the V1 Custom Elements API, which primarily changes the "created" step to actually invoke the class
constructor, imposes new restrictions on what can be done in the constructor
(previously createdCallback
), and introduces different callback names.
-
- When using
Polymer({...})
from the compatibility layer, all callbacks should use legacy Polymer API names (created
,attached
,detached
,attributeChanged
) - When extending from
Polymer.Element
, users should override the V1 standard callback names and callsuper()
:created
changes toconstructor
attached
changes toconnectedCallback
detached
changes todisconnectedCallback
attributeChanged
changes toattributeChangedCallback
- When using
-
The V1 Custom Elements spec forbids reading attributes, children, or parent information from the DOM API in the
constructor
(orcreated
when using the legacy API). Likewise, attributes and children may not be added in theconstructor
. Any such work must be deferred (e.g. untilconnectedCallback
or microtask/setTimeout
/requestAnimationFrame
). -
Polymer will no longer produce type-extension elements (aka
is="..."
). Although they are still included in the V1 Custom Elements spec and scheduled for implementation in Chrome, because Apple has stated it will not implementis
, we will not be encouraging its use to avoid indefinite reliance on the Custom Elements polyfill. Instead, a wrapper custom element can surround a native element, e.g.<a is="my-anchor">...</a>
could become<my-anchor><a>...</a></my-anchor>
. Users will need to change existingis
elements where necessary. -
All template type extensions provided by Polymer have now been changed to standard custom elements that take a
<template>
in their light dom, e.g.<template is="dom-repeat" items="{{items}}">...</template>
should change to
<dom-repeat items="{{items}}"> <template>...</template> </dom-repeat>
For the time being, Polymer (both legacy and class API) will automatically wrap template extensions used in Polymer element templates during template processing for backward-compatibility, although we may decide to remove this auto-wrapping in the future. Templates used in the main document must be manually wrapped.
-
The
custom-style
element has also been changed to a standard custom element that must wrap a style element e.g.<style is="custom-style">...</style>
should change to
<custom-style> <style>...</style> </custom-style>
Polymer 2.0 will continue to use a shim to provide limited CSS Custom Properties support on browsers that do not yet natively support custom properties, to allow an element to expose a custom styling API. The following changes have been made in the shim that Polymer 2.0 will use:
- The shim will now always use native CSS Custom Properties by default on browsers that implement them (this was opt-in in 1.0). The shim will perform a one-time transformation of stylesheets containing CSS Custom Property mixins to leverage individual native CSS properties where possible for better performance. This introduces some limitations to be aware of.
- The following invalid styling syntax was previously accepted by the 1.0 custom property shim. In order to support native CSS Custom Properties, rules should be correct to use only natively valid syntax:
:root {}
- Should be
:host > * {}
(in a shadow root) - Should be
html {}
(in main document) - Thus, cannot share old
:root
styles for use in both main document and shadow root
- Should be
var(--a, --b)
- Should be
var(--a, var(--b))
- Should be
@apply(--foo)
- Should be
@apply --foo;
- Should be
element.customStyle
as an object that can be assigned to has been removed; useelement.updateStyles({...})
instead.<style>
inside of a<dom-module>
, but outside of<template>
is no longer supported- Imperatively created custom-styles (e.g.
document.createElement('style', 'custom-style')
) are no longer supported.
- An element's template is not stamped & data system not initialized (observers, bindings, etc.) until the element has been connected to the main document. This is a direct result of the V1 changes that prevent reading attributes in the constructor.
- Re-setting an object or array no longer dirty checks, meaning you can make deep changes to an object/array and just re-set it, without needing to use
set
/notifyPath
. Although theset
API remains and will often be the more efficient way to make changes, this change removes users of Polymer elements from needing to use this API, making it more compatible with alternate data-binding and state management libraries.- To achieve the same strict equality dirty-checking that 1.x did, simply override the new
_shouldPropertyChange
method as follows:_shouldPropertyChange(property, value, old) { return value !== old; }
- To achieve the same strict equality dirty-checking that 1.x did, simply override the new
- Propagation of data through the binding system is now batched, such that multi-property computing functions and observers run once with a set of coherent changes. Single property accessors still propagate data synchronously, although there is a new
setProperties({...})
API on Polymer elements that can be used to propagate multiple values as a coherent set. - Property change notification event dispatch (
notify: true
) occurs after all other side effects of a property change occurs. In 1.x notification happened after binding side effects, but before observers, which was counter-intuitive. This rationalizes the concept of upward notification to ensure it happens after all local and downward side-effects based on the change occur. Concretely, the order of effect processing in 2.x is as follows:- computed properties (
computed
) - template bindings (both property bindngs
[[...]]
and computed bindings[[compute(...)]]
, including any side-effects on child elements for the bound property/attribute changes) - attribute reflection (
reflectToAttribute: true
) - observers (both single-property
observer
and multi-propertyobservers
) - property-changed event notification (
notify: true
, including any side-effects on host elements for the bound property changes)
- computed properties (
- Multi-property observers, computed methods, and computed bindings are now called once at initialization if any arguments are defined (and will see
undefined
for any undefined arguments). As such, the 1.x rule requiring all properties of a multi-property observer to be defined no longer applies, as this was a major source of confusion and unintended consequences. Subsequently setting multi-property method arguments will cause the method to be called once for each property changed via accessors, or once per batch of changes viasetProperties({...})
. - Declaring a method name used as an observer or computing function in the
properties
block causes the method property itself to become a dependency for any effects it is used in, meaning the effect for that method will run whenever the method is set, similar to 1.x. However, due to removing theundefined
rule noted above, in 2.x if such a method exists when the element is created, it will run with initial values of arguments, even in the case some or all arguments areundefined
. - <a name="breaking-binding-notifications>‘notify’ events are no longer fired when value changes as result of binding from host, as a major performance optimization over 1.x behavior. Use cases such as
<my-el foo="{{bar}}" on-foo-changed="fooChanged">
are no longer supported. In this case you should simply user abar
observer in the host. Use cases such as dynamically adding aproperty-changed
event listener on for properties bound by an element's host by an actor other than the host are no longer supported. - In order for a property to be deserialized from its attribute, it must be declared in the
properties
metadata object - The
Polymer.Collection
and associated key-based path and splice notification for arrays has been eliminated. See explanation here for more details.
Polymer.instanceof
andPolymer.isInstance
: no longer needed, useinstanceof
andinstanceof Polymer.Element
instead.dom-module
: Removed ability to useis
andname
attribute to configure the module name. The only supported declarative way set the module id is to useid
.element.getPropertyInfo
: This api returned unexpected information some of the time and was rarely used.element.getNativePrototype
: Removed because it is no longer needed for internal code and was unused by users.element.beforeRegister
: This was originally added for metadata compatibility with ES6 classes. We now prefer users create ES6 classes by extendingPolymer.Element
, specifying metadata in the staticconfig
property. For legacy use viaPolymer({...})
, dynamic effects may now be added using theregistered
lifecycle method.element.attributeFollows
: Removed due to disuse.element.classFollows
: Removed due to disuse.listeners
: Removed ability to useid.event
to add listeners to elements in local dom. Use declarative template event handlers instead.- Methods starting with
_
are not guaranteed to exist (most have been removed)
- Attached: no longer deferred until first render time. Instead when measurement is needed use... API TBD.
- The legacy
created
callback is no longer called before default values inproperties
have been set. As such, you should not rely on properties set increated
from withinvalue
functions that define property defaults. However, you can now set any property defaults within thecreated
callback (in 1.0 this was forbidden for observed properties) in lieu of using thevalue
function inproperties
. - Binding a default value of
false
via an attribute binding to a boolean property will not override a defaulttrue
property of the target, due to the semantics of boolean attributes. In general, property binding should always be used when possible, and will avoid such situations. lazyRegister
option removed and all meta-programming (parsing template, creating accessors on prototype, etc.) is deferred until the first instance of the element is created- Any attribute values will take priority over property values set prior to upgrade due to V1
attributeChangedCallback
timing semantics. In 1.x properties set prior to upgrade overrode attributes. - Polymer 2.0 uses ES2015 syntax, and can be run without transpilation in current Chrome, Safari 10, Safari Technology Preview, Firefox, and Edge. Transpilation is required to run in IE11 and Safari 9. We will be releasing tooling for development and production time to support this need in the future.
- In Polymer 1.x, the
class
attribute was explicitly blacklisted fromhostAttributes
and never serialized. This is no longer the case using the 2.0 legacy API. Polymer.RenderStatus.afterNextRender(context, callback, args)
is now called only with acallback
argument (Polymer.RenderStatus.afterNextRender(callback)
). The callback should be bound as needed with the correct context and arguments.