Security News
pnpm 10.0.0 Blocks Lifecycle Scripts by Default
pnpm 10 blocks lifecycle scripts by default to improve security, addressing supply chain attack risks but sparking debate over compatibility and workflow changes.
A library for wiring up Dojo 2 applications.
WARNING This is alpha software. It is not yet production ready, so you should use at your own risk.
Dojo 2 applications consist of widgets, stores and actions. Stores provide data to widgets, widgets call actions, and actions mutate data in stores.
This library provides an application factory which lets you define actions, stores and widgets. It takes care of lazily loading them, wiring actions to events emitted by widgets, and making widgets and actions observe stores that hold their state.
It provides an implementation of Custom Elements, allowing you to attach previously registered widgets or indeed register widget factories for custom elements.
The examples below are provided in TypeScript syntax. The package does work under JavaScript, but for clarity, the examples will only include one syntax.
import createApp from 'dojo-app/createApp';
const app = createApp();
You can also define a default action and widget stores at creation time:
import createMemoryStore from 'dojo-widgets/util/createMemoryStore';
const defaultActionStore = createMemoryStore();
const defaultWidgetStore = createMemoryStore();
const app = createApp({ defaultActionStore, defaultWidgetStore });
Or you can, just once, assign default stores:
import createMemoryStore from 'dojo-widgets/util/createMemoryStore';
const app = createApp();
app.defaultActionStore = createMemoryStore();
app.defaultWidgetStore = createMemoryStore();
This store will be used as the stateFrom
option to widget and custom element factories, unless another store is specified.
If you already have instantiated an action:
import createAction from 'dojo-actions/createAction';
const action = createAction({
do() {
// something
}
});
app.registerAction('my-action', action);
You can also register a factory method which creates the action only when needed:
app.registerActionFactory('my-lazy-action', () => {
return createAction({
do() {
// something else
}
});
});
Note that an action instance may only be registered once. A factory is not allowed to return a previously registered instance.
The factory is called with a registry object and potentially a default action store. The registry object contains the same has*()
and get*()
methods that are available on the application factory itself.
import createWidget from 'dojo-widgets/createWidget';
app.registerCustomElementFactory('tag-name', createWidget);
A factory for a custom element should return a unique widget instance. It receives an options
object with an optional id
property and other options from the custom element.
Tag names must be valid according to the Custom Elements spec. Additionally app-projector
and app-widget
names are reserved. Names are automatically lowercased before the factory is registered.
If you already have instantiated a store:
import createMemoryStore from 'dojo-widgets/util/createMemoryStore';
const store = createMemoryStore();
app.registerStore('my-store', store);
You can also register a factory method which creates the store only when needed:
app.registerStoreFactory('my-lazy-store', () => {
return createMemoryStore();
});
Note that a store instance may only be registered once. A factory is not allowed to return a previously registered instance.
If you already have instantiated a widget:
import createWidget from 'dojo-widgets/createWidget';
const widget = createWidget();
app.registerWidget('my-widget', widget);
You can also register a factory method which creates the widget only when needed:
app.registerWidgetFactory('my-lazy-widget', (options) => {
return createWidget(options);
});
The options
object will have an id
property set to my-lazy-widget
. A registry provider is available under the registryProvider
property.
Note that a widget instance may only be registered once. A factory is not allowed to return a previously registered instance.
To see whether a particular action, store or widget is registered, use the has*()
methods:
app.hasAction('my-action');
app.hasStore('my-store');
app.hasWidget('my-widget');
Each method returns true
if the respective item was registered, and false
if not.
Besides checking app.defaultActionStore
or app.defaultWidgetStore
you can use the DEFAULT_ACTION_STORE
and DEFAULT_WIDGET_STORE
symbols to see if the respective default store was provided:
import { DEFAULT_ACTION_STORE, DEFAULT_WIDGET_STORE } from 'dojo-app/createApp';
app.hasStore(DEFAULT_ACTION_STORE);
app.hasStore(DEFAULT_WIDGET_STORE);
To find the ID under which a particular action, store or widget instance was registered, use the identify*()
methods:
app.identifyAction(action);
app.identifyStore(store);
app.identifyWidget(widget);
Each method returns the ID string if the respective instance was registered, or throws an error if not.
Note that the default action store, if any, is registered under the DEFAULT_ACTION_STORE
symbol, not an ID string. The same goes for the default widget store, which is registered under the DEFAULT_WIDGET_STORE
symbol.
You can load previously registered actions, stores and widgets using the get*()
methods:
app.getAction('my-action');
app.getStore('my-store');
app.getWidget('my-widget');
Each method returns a promise for the respective item. If the item was not registered or could not be loaded, the promise is rejected.
Besides accessing app.defaultActionStore
or app.defaultWidgetStore
you can use the DEFAULT_ACTION_STORE
and DEFAULT_WIDGET_STORE
symbols to get the respective default store:
import { DEFAULT_ACTION_STORE, DEFAULT_WIDGET_STORE } from 'dojo-app/createApp';
app.getStore(DEFAULT_ACTION_STORE);
app.getStore(DEFAULT_WIDGET_STORE);
You might want to export an action from a module, and then register this module with the app factory:
// my-action.ts
import createAction from 'dojo-actions/createAction';
export default createAction({
do() {
// something
}
});
// my-app.ts
import createApp from 'dojo-app/createApp';
import myAction from './my-action';
const app = createApp();
app.registerAction('my-action', myAction);
However if the action needs to access a store that is lazily loaded you'd then need a reference to the application factory in order to access the store.
To make this easier the application factory calls configure()
on actions after they've been loaded. The configuration object contains the same has*()
and get*()
methods that are available on the application factory itself.
For example, to store a reference to a particular store, you could do:
// my-action.ts
import createAction from 'dojo-actions/createAction';
import { CombinedRegistry } from 'dojo-app/createApp';
import { MemoryStore } from 'dojo-widgets/util/createMemoryStore';
interface MyAction {
store: MemoryStore<Object>;
}
export default createAction.extend<MyAction>({})({
configure(registry: CombinedRegistry) {
return registry.getStore('my-store').then((store) => {
(<MyAction> this).store = store;
});
},
do() {
const { store } = <MyAction> this;
return store.patch({ id: 'some-object', value: 'some-value' });
}
});
If you registered an action factory you can do something similar, without having to implement the configure()
method:
// my-action-factory.ts
import createAction from 'dojo-actions/createAction';
import { CombinedRegistry } from 'dojo-app/createApp';
import { MemoryStore } from 'dojo-widgets/util/createMemoryStore';
export default function(registry: CombinedRegistry) {
return registry.getStore('my-store').then((store) => {
return createAction({
do() {
return (<MemoryStore<Object>> store).patch({ id: 'some-object', value: 'some-value' });
}
});
});
}
The above examples show how to register individual objects and factories. However you can also describe and load entire applications with a single method call:
import createApp from 'dojo-app/createApp';
const app = createApp();
app.loadDefinition({
actions: [
// describe actions here
],
customElements: [
// describe custom elements here
],
stores: [
// describe stores here
],
widgets: [
// describe widgets here
]
});
Each action, store and widget definition is an object. It must have an id
property which uniquely identifies that particular item. (You can have an action and a store with the same ID however, or a store and a widget.)
Further, definitions must either have a factory
or an instance
property. With factory
you must specify either a module identifier string or a factory function that can create the appropriate action, store or widget. With instance
you can also specify a module identifier string, or alternatively an action, store or widget instance as appropriate.
Note that if an action, store or widget instance is provided to the instance
option, the identify*()
methods will not return their associated ID until the respective get*()
methods have been called.
The application factory must be loaded with dojo-loader
in order to resolve module identifiers. Both ES and UMD modules are supported. The default export is used as the factory
or instance
value.
Custom element definitions must have a name
property, which must be a valid custom element name (and not app-projector
or app-widget
). They must also have the factory
property, but not the id
and instance
properties
You might be tempted to specify dojo-actions/createAction
as the factory
in action definitions. However actions must be created with a do()
implementation and this implementation cannot be specified in the definition object. You'll have to use your own factory method, like in the registerActionFactory()
example above, or point directly at an action instance.
If you use your own factory method you can use the stateFrom
option in the action definition. This can be an actual store or a string identifier for a store that is registered with the application factory. The store will be lazily loaded when the action is needed, and is then passed to the factory method as its second argument. You can now set up your action so it observes the store for its state.
If the stateFrom
option is not used, but a default action store is provided, that default action store will be passed to the factory.
If you use factory
in your store definition you can use the options
property to specify an object that is passed when the factory is called.
Like stores, widget factories typically take an options argument. Widget definitions too support the options
property, letting you specify the object that is passed when the factory is called.
The options
object must not contain id
, listeners
, state
and stateFrom
properties. These need to be specified in the widget definition itself.
Use the listeners
object to automatically wire events emitted by the widget. Keys are event types. Values are event listeners, actions, string identifiers for actions that are registered with the application factory, or an array containing such values.
Use the state
property to define an initial state that is added to the widget's store before the widget is created, if any. This will be done lazily once the widget is needed. The store is assumed to reject the initial state if it already contains state for the widget. This error will be ignored and the widget will be created with whatever state was already in the store.
Use the stateFrom
property to specify the store that the widget should observe for its state. It can be an actual store or a string identifier for a store that is registered with the application factory.
These actions and stores will be lazily loaded when the widget is needed.
Module identifiers are resolved relative to the dojo-app/createApp
module. You can provide a toAbsMid()
function when creating the application factory to implement your own module resolution logic.
For instance if you bootstrap your application in my-app/main
, and you want to use module identifiers relative to that module, you could do:
import createApp from 'dojo-app/createApp';
const app = createApp({ toAbsMid: require.toAbsMid });
app.loadDefinition({
actions: [
{
id: 'my-action',
factory: './actions/mine'
}
]
});
This uses the require()
function available to my-app/main
, which will resolve module identifiers relative to itself.
The various register*()
and register*Factory()
methods return a handle. Call destroy()
on this handle to deregister the action, custom element, store or widget from the application factory.
loadDefinition()
also returns a handle. Destroying it will deregister all actions, stores and widgets that were registered as part of the definition.
Note that destroying handles will not destroy any action, store or widget instances.
The registry provider can provide read-only registries for actions, stores and widgets. It's available under app.registryProvider
and passed to widget factories as the registryProvider
option.
Use registryProvider.get('actions')
to get an action registry. registryProvider.get('stores')
gives you a store registry, and registryProvider.get('widgets')
a widget registry.
Each registry has get()
and identify()
methods. These behave the same as the get*()
and identify*()
methods of the application.
Use the App#realize(root: Element)
method to realize custom elements inside a DOM element.
Widgets are rendered inside a projector. You can declare (multiple) projector slots in your DOM tree using the app-projector
custom element. These projectors must not be nested. Other custom elements can only occur within a app-projector
. You can pass a single app-projector
element as the root
argument to App#realize()
.
All descending custom elements are replaced by rendered widgets. Widgets for nested elements are appended to their parent widget. Regular (non-custom) DOM nodes inside custom elements are not preserved, however regular DOM nodes within a app-projector
element are.
Custom elements are matched (case-insensitively) to registered factories. First the tag name is matched. If no factory is found, and the element has an is
attribute, that value is used to find a factory. Unrecognized elements are left in the DOM where possible.
A factory options object can be provided in the DOM by setting the data-options
attribute to a JSON string. The options object may have a stateFrom
property containing a store identifier. It may also have a listeners
property containing a widget listener map. Values for each event type can be action identifiers or arrays thereof. These properties are resolved to the actual store and action instances before the factory is called. Additional properties are passed to the factory as-is.
Widgets can be identified through the id
property on the options object, a data-uid
attribute, or an id
attribute. The options object takes precedence over the data-uid
attribute, which takes precedence over the id
attribute. It's valid to use the different attributes, but only the most specific ID will be passed to the factory (in the options
object).
The data-state-from
attribute may be used on custom elements to specify a store identifier. This will only take effect if a widget ID is also specified. The stateFrom
property on the options object that is passed to the factory will be set to the referenced store. Any stateFrom
property in the data-options
object still takes precedence.
A default widget store may be configured by setting the data-state-from
attribute on the app-projector
custom element. It applies to all descendant elements that have IDs, though they can override it by setting their own data-state-from
attribute or configuring stateFrom
in their data-options
.
Custom elements that have widget IDs and a stateFrom
store may set their data-state
attribute to an initial state object, encoded as a JSON string. This initial state will be added to the store before the widget is created. The store is assumed to reject the initial state if it already contains state for the widget. This error will be ignored and the widget will be created with whatever state was already in the store.
The special app-widget
custom element can be used to render a previously registered widget. The data-uid
or id
attribute is used to retrieve the widget. The data-state
, data-state-from
and data-options
attributes are ignored. No default widget store is applied.
A widget ID can only be used once within an application. Similarly a widget instance can only be rendered once. The getWidget()
, hasWidget()
and identifyWidget()
methods will work with widgets created by custom element factories.
App#realize()
returns a promise. It is rejected when errors occur (e.g. bad data-options
values, or a factory throws an error). Otherwise it is fulfilled with a Handle
object. Use the destroy()
method to destroy the created projectors and widgets. Widgets rendered through app-widget
are left as-is.
Given this application definition:
import createContainer from 'dojo-widgets/createContainer';
import createWidget from 'dojo-widgets/createWidget';
import createMemoryStore from 'dojo-widgets/util/createMemoryStore';
app.loadDefinition({
actions: [
{
id: 'an-action',
instance: './my-action'
}
],
customElements: [
{
name: 'dojo-container',
factory: createContainer
},
{
name: 'a-widget',
factory: createWidget
}
],
stores: [
{
id: 'widget-state',
instance: createMemoryStore({
data: [
{ id: 'widget-1', classes: [ 'awesome' ] }
]
})
}
],
widgets: [
{
id: 'widget-2',
instance: createWidget({ tagName: 'strong' })
}
]
});
app.realize(document.body);
And this <body>
:
<body>
<app-projector>
<div>
<dojo-container>
<a-widget data-options='{"id":"widget-1","tagName":"mark","stateFrom":"widget-state","listeners":{"click":"an-action"}}'></a-widget>
<div>
<div is="app-widget" id="widget-2"></div>
</div>
</dojo-container>
</div>
</app-projector>
</body>
The realized DOM will be:
<body>
<app-projector>
<div>
<dojo-container>
<mark class="awesome"></mark>
<strong></strong>
</dojo-container>
</div>
</app-projector>
</body>
TODO: Add appropriate usage and instruction guidelines
We appreciate your interest! Please see the Dojo 2 Meta Repository for the Contributing Guidelines and Style Guide.
Test cases MUST be written using Intern using the Object test interface and Assert assertion interface.
90% branch coverage MUST be provided for all code submitted to this repository, as reported by istanbul’s combined coverage results for all supported platforms.
To test locally in node run:
grunt test
To test against browsers with a local selenium server run:
grunt test:local
To test against BrowserStack or Sauce Labs run:
grunt test:browserstack
or
grunt test:saucelabs
© 2004–2016 Dojo Foundation & contributors. New BSD license.
FAQs
An application framework for Dojo 2
The npm package dojo-app receives a total of 10 weekly downloads. As such, dojo-app popularity was classified as not popular.
We found that dojo-app demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 2 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Security News
pnpm 10 blocks lifecycle scripts by default to improve security, addressing supply chain attack risks but sparking debate over compatibility and workflow changes.
Product
Socket now supports uv.lock files to ensure consistent, secure dependency resolution for Python projects and enhance supply chain security.
Research
Security News
Socket researchers have discovered multiple malicious npm packages targeting Solana private keys, abusing Gmail to exfiltrate the data and drain Solana wallets.