Security News
Fluent Assertions Faces Backlash After Abandoning Open Source Licensing
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
react-habitat
Advanced tools
A React DOM Bootstrapper designed to harmonise a hybrid application
React Habitat is designed for integrating React with your CMS using the DOM as the interface. It's based of some basic container programming principles and brings peace and order to multi page apps.
This framework exists so you can get on with the fun stuff!
You should use React Habitat any time there is a framework or CMS rendering your HTML and you want one or multiple React components on the page(s). For example sometimes there are only sections of your page that you want to be a React Component, then this framework is perfect for that.
The idea behind this is that, rather than trying to initiate one or many React components; by either hard coding or using a Router. You switch it around so components "new up" themselves when required.
React Habitat works great with:
Typically if you're using a router... then this framework isn't really going to bring much benefit to you. However you are definitely invited to use it if you want to.
Polyfills
We highly recommend you use something like WebPack or Browserify when using this framework.
Install with NPM
npm install --save-dev react-habitat
This assumes that you’re using a package manager with a module bundler like Webpack or Browserify.
If you don’t use a module bundler, and would prefer a single-file UMD build that makes ReactHabitat
available as a global object, you can grab a pre-built version from the dist folder.
Using ES5? Read the ES5 version here.
The basic pattern for integrating React Habitat into your application is:
This getting started guide walks you through these steps for a simple React application. This document assumes you already know:
The class must extend ReactHabitat.Bootstrapper
and is intended to be an entry point of your bundled app. So if you're using something like webpack or browserify then this is file to point it too.
In the constructor() of the class you need to register your React components with it and then set the container. The container is later bound to the DOM automatically so your React components self-initiate.
In React Habitat, you'd register a component for a key something like this
container.register('SomeReactComponent', SomeReactComponent);
So for our sample application we need to register all of our components (classes) to be exposed to the DOM so things get wired up nicely. Note in this example you can also define split points using React Habitat dynamic imports.
import ReactHabitat from 'react-habitat';
import SomeReactComponent from './SomeReactComponent';
import AnotherReactComponent from './AnotherReactComponent';
class MyApp extends ReactHabitat.Bootstrapper {
constructor(){
super();
// Create a new container builder
var container = new ReactHabitat.Container();
// Register top level component(s) (ie mini/child apps)
container.register('SomeReactComponent', SomeReactComponent);
container.register('AnotherReactComponent', AnotherReactComponent);
// Register a dynamic import
container.register('AsyncReactComponent', import('./AsyncReactComponent'));
// Finally, set the container
this.setContainer(container);
}
}
// Always export a 'new' instance so it immediately evokes
export default new MyApp();
You can also register multiple component's all at once with registerAll
like this
container.registerAll({
'SomeReactComponent': SomeReactComponent,
'AnotherReactComponent': AnotherReactComponent
});
If you are using Redux
You will need to use a different container. Please install & configure the react-habitat-redux library. Then continue with step 2 below.
During the web application execution you will want to make use of the components you registered. You do this by resolving them in the DOM from a scope.
When you resolve a component, a new instance of the object gets created (Resolving a component is roughly equivalent to calling 'new').
To resolve new instances of your components you need to attach a data-component
attribute to a div
or a span
element in the HTML.
Any child components should be nested inside the React components themselves.
Set the data-component
value to equal a component name you have registered in the container.
For instance:
<div data-component="SomeReactComponent"></div>
Will be resolved by the following registration.
container.register('SomeReactComponent', SomeReactComponent);
So, for our sample app we would do something like this
<html>
<body>
<div data-component="SomeReactComponent"></div>
<script src="myBundle.js" />
</body>
</html>
When you view this page you will see a instance of SomeReactComponent
automatically rendered in the div's
place. In fact, you can add as many as you like and it will render multiple instances.
For example. This is perfectly valid.
<html>
<body>
<div data-component="SomeReactComponent"></div>
<div data-component="SomeReactComponent"></div>
<div data-component="SomeReactComponent"></div>
<script src="myBundle.js" />
</body>
</html>
Will render 3 instances of your component.
Note It's important that the output built javascript file is included at the end of the DOM just before the closing body tag.
Resolving and registering components alone is not all that special, but passing data to it via html attributes is pretty useful. This allows the backend to easily pass data to your components in a modular fashion.
To set props you have a few choices. You can use all of these or only some (they merge) so just use what's suits you best for setting properties.
Attribute | Description |
---|---|
data-props | Maps encoded JSON to props. |
data-prop-* | This prefix maps in strings, booleans, null, array or encoded JSON to a prop. |
data-n-prop-* | This prefix maps in numbers and floats to a prop. |
data-r-prop-* | This prefix in a reference to an object that exists on the global scope (window) to a prop. |
With an attribute prefix the * may be replaced by any name. This allow's you to define the property name. Property names must be all lower case and hyphens will be automatically converted to camel case.
For example
data-prop-title
would expose title
on the props object inside the component.
data-prop-my-title
would expose myTitle
on the props object inside the component.
Set component props via an encoded JSON string on the data-props
attribute.
For example
<div data-component="SomeReactComponent" data-props='{"title": "A nice title"}'></div>
Set an component prop via prefixing attributes with data-prop-
.
For example
data-prop-title
would expose title
as a property inside the component.
Please note: JSON, booleans & null are automatically parsed. Eg data-prop-my-bool="true"
would expose the value of true
, NOT the string representation "true"
.
Passing in an array of objects will require you to use html encoded characters for quotes etc i.e "foo" will replace "foo"
Simple Example
<div data-component="SomeReactComponent"
data-prop-title="A nice title"
data-prop-show-title="true">
</div>
Would expose props as
class SomeReactComponent extends React.Component {
constructor(props) {
super(props);
props.title === "A nice title"; //> true
props.showTitle === true; //> true
}
render() {
return <div>{ this.props.showTitle ? this.props.title : null }</div>;
}
}
JSON Example
<div data-component="SomeReactComponent"
data-prop-person='{"name": "john", "age": 22}'>
</div>
Would expose as
class MyReactComponent extends React.Component {
constructor(props) {
super(props);
return (
<div>
Name: {this.props.person.name}
Age: {this.props.person.age}
</div>
);
}
}
Set an component prop with type [number] via prefixing attributes with data-n-prop-
.
For example data-n-prop-temperature="33.3"
would expose the float value of 33.3 and not the string representation '33.3'.
This is handy if you know that a property is always going to be a number or float.
Referenced a global variable in your component prop via prefixing attributes with data-r-prop-
.
For Example
<script>
var foo = window.foo = 'bar';
</script>
<div data-component="SomeReactComponent" data-r-prop-foo="foo"></div>
This is handy if you need to share properties between habitats or you need to set JSON onto the page.
It can be handy to pass values back again, particularly for inputs so the backend frameworks can see any changes or read data.
Every React Habitat instance is passed in a prop named proxy
, this is a reference the original dom element.
Please note only <inputs />
are left in the DOM by default. To keep a generic element in the DOM, set the data-habitat-no-replace="true"
attribute.
So for example, we could use proxy
to update the value of an input like so
<input id="personId" type="hidden" data-component="personLookup" />
Somewhere inside the component
this.props.proxy.value = '1234'
Sometimes you may additionally need to call this.props.proxy.onchange()
if you have other scripts listening for this event.
Using ES5? Read the ES5 version here.
You can set a custom css class on the habitat element by setting the data-habitat-class
attribute on the target element.
Example
<div data-component="MyComponent" data-habitat-class="my-css-class"></div>
Will result in the following being rendered
<div data-habitat="C1" class="my-css-class">...</div>
By default only <inputs />
are left in the DOM when a React Habitat is created.
To keep a generic element in the DOM, set the data-habitat-no-replace="true"
attribute.
Default: 'data-component'
By default React Habitat will resolve components via the data-component
attribute. You can configure this by assigning
the componentSelector
property in your constructor.
It will accept any string containing any valid attribute name.
Example
class MyApp extends ReactHabitat.Bootstrapper {
constructor(){
super();
this.componentSelector = 'data-myComponents';
}
}
update()
The update method will scan the DOM and for any components that require wiring up (i.e after ajaxing in some HTML). This can be evoked automatically by using a watcher.
Example
class MyApp extends ReactHabitat.Bootstrapper {
someMethod() {
// This will scan the entire document body
this.update();
}
}
By default update() will scan the entire body, however a parent node can optionally be passed in for better performance if you know where the update has occurred.
Example
class MyApp extends ReactHabitat.Bootstrapper {
someMethod() {
// Will scan just the children of the element with id 'content'
this.update(window.document.getElementById('content'))
}
}
You can call this method from somewhere else in your app by importing it
import MyApp from './MyApp';
// ...
MyApp.update();
React Habitat applications have update "lifecycle methods" that you can override to run code at particular times in the process.
shouldUpdate(node)
Called when an update has been requested. Return false to cancel the update.
willUpdate(node)
Called when an update is about to take place.
didUpdate(node)
Called after an update has taken place.
Example
class MyApp extends ReactHabitat.Bootstrapper {
shouldUpdate(node) {
// Dont allow updates on div's
if (node.tagName === 'div') {
return false;
}
}
willUpdate(node) {
console.log('Im about to update', node);
}
didUpdate(node) {
console.log('I just updated', node);
}
}
Please Note IE 9 & 10 will require a MutationObserver polyfill to use this feature. An alternative is to call update manually.
Start watching the DOM for any changes and wire up future components automatically (eg ajaxed HTML).
Example
class MyApp extends ReactHabitat.Bootstrapper {
constructor(){
this.setContainer(myContainer);
// Wire up any future habitat elements automatically
this.startWatcher();
}
}
Will stop watching the DOM for any changes.
Example
class MyApp extends ReactHabitat.Bootstrapper {
someMethod(){
this.stopWatcher();
}
}
To unload the container and remove all React Habitat instances. Call the dispose()
method.
Example
class MyApp extends ReactHabitat.Bootstrapper {
constructor(){
super();
//...
this.dispose();
}
}
React Habitat supports resolving components asynchronously by using Promises. To define async registrations, register a Promise (that resolves to a component) instead of a component.
for example
container.register('AsynReactComponent', new Promise((resolve, reject) => {
// .. do async work to get 'component', then
resolve(component);
}));
or with registerAll
container.registerAll({
'SomeReactComponent': new Promise((resolve, reject) => {
// .. do async work to get 'component', then
resolve(component);
})
});
React Habitat has no restrictions on how you want to resolve your components however this does enable you to define code split points.
Code splitting is one great feature that means our visitors dont need to download the entire app before they can use it. Think of code splitting as incrementally download your application only as its needed.
While there are other methods for code splitting we will use Webpack for these examples.
Webpack 2 treats import()
as a split-point and puts the requested module into a separate chunk.
So for example, we could create a split point using import()
like this:
container.register('AsynReactComponent', new Promise((resolve, reject) => {
import('./components/MyComponent').then((MyComponent) => {
resolve(MyComponent);
}).catch((err) => {
reject(err);
})
}));
However, since import()
returns a Promise, we can actually simplify the above to:
container.register('AsynReactComponent', import('./components/MyComponent'));
Here is an example using require.ensure()
to define a split-point in webpack 1
container.register('AsynReactComponent', new Promise((resolve) => {
require.ensure(['./components/MyComponent'], () => {
resolve(require('./components/MyComponent'));
});
}));
When passing JSON to an attribute, you will need to encode the value so that content can be preserved and properly rendered.
As a general rule, escape the following characters with HTML entity encoding:
&
--> &
<
--> <
>
--> >
"
--> "
'
--> '
/
--> /
Example:
<div data-props="{"foo": "bar"}"></div>
Additionally, an encoder may replace extended ASCII characters with the equivalent HTML entity encoding.
Most backend systems are capable of doing this automatically. An alternative is to use the data-r-prop-* option.
Single of Double Quotes?
Double quotes around attributes values are the most common and our recommendation for setting properties with React Habitat.
However, there is a known hack of wrapping JSON attributes with single quotes and escaping nested single quotes.
example
<div data-props='{"restaurant": "Bob\'s bar and grill"}'></div>
We will use this method in the docs to maintain readability. However, we strongly recommend you encode in production code.
Please don't hesitate to raise an issue through GitHub or open a pull request to show off your fancy pants coding skills - we'll really appreciate it!
Part Business. Part Creative. Part Technology. One hundred per cent digital.
Pioneered in Australia, Deloitte Digital is committed to helping clients unlock the business value of emerging technologies. We provide clients with a full suite of digital services, covering digital strategy, user experience, content, creative, engineering and implementation across mobile, web and social media channels.
http://www.deloittedigital.com/au
Copyright (C) 2015, Deloitte Digital. All rights reserved.
React Habitat can be downloaded from: https://github.com/DeloitteDigitalAPAC/react-habitat
Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
Neither the name of the copyright holder nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
FAQs
A React DOM Bootstrapper designed to harmonise a hybrid application
The npm package react-habitat receives a total of 1,670 weekly downloads. As such, react-habitat popularity was classified as popular.
We found that react-habitat demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 4 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
Fluent Assertions is facing backlash after dropping the Apache license for a commercial model, leaving users blindsided and questioning contributor rights.
Research
Security News
Socket researchers uncover the risks of a malicious Python package targeting Discord developers.
Security News
The UK is proposing a bold ban on ransomware payments by public entities to disrupt cybercrime, protect critical services, and lead global cybersecurity efforts.