/react-injector

Dependency injection for React

Primary LanguageJavaScript

React Injector

Dependency injection for React

Installation

$ npm install --save react-injector

Getting Started

React Injector provides a root component that can easily be used to set up component injection in React. Begin by importing the ReactInjector component. This should only be done once.

import { ReactInjector } from 'react-injector'

This can then be used to bootstrap your application.

render((
    <ReactInjector>
        <YourApplicationRoot />
    </ReactInjector>
), document.getElementById("mount"))

That's it! Your application is now set up for dependency injection.

Class Dependency Declaration

The easiest way set up dependencies is to leverage the Inject decorator provided by React Injector. Keep in mind that decorators are still experimental. In order to use decorators a library will need to be used for interpretation. Babel, for example, can be used alongside its transform-decorators-legacy plugin

Let's start by declaring a class that we'll then inject as a dependency into another class.

export default class AmazingDependency {
    getAmazingText() {
        return "World!"
    }
}

We'll then create the class that uses AmazingDependency as the dependency we'll inject.

import { Inject } from 'react-injector'
import AmazingDependency from './AmazingDependency'

@Inject([ AmazingDependency ])
export default class Greeter {
    constructor(amazingDependency) {
        this.amazingDependency = amazingDependency
    }
    
    greet() {
        return "Hello, " + this.amazingDependency.getAmazingText()
    }
}

When we call the greet method on the class above it will return Hello, World!. Let's break down the above snippet. The Inject decorator takes an array of dependencies. These dependencies are then injected into the constructor when the class is instantiated.

The dependencies can actually be pretty much anything. Dependencies that are classes and happen to need their own injected dependencies also will be properly injected upon insertion into the constructor. Any other type will be directly translated into the constructor.

Function Dependency Declaration

React Injector allows you to inject functions as dependencies as well. This can be done by specifying that a dependency is a function.

import { Inject } from 'react-injector'

Inject({ type: 'function' })
export default function amazingFunction() {
    return "You just called an amazing function"
}

Functions can also be injected with dependencies.

import { Inject } from 'react-injector'
import AmazingDependency from './AmazingDependency'

Inject({ type: 'function' }, [ AmazingDependency ])
export default function greet(amazingDependency) {
    return () => {
        return "Hello, " + amazingDependency.getAmazingText()
    }
}

Notice that the above greet function is actually a wrapper around another function. The wrapper function is injected with the dependency and invoked by the container. This means that when injected you will actually recieve the inner function having access to the dependencies in scope.

React Component Injection

React components have their dependencies injected in a slightly different way. This is because React itself abstracts the instantiation of components. Due to this restraint dependencies are injected into a component's props.

Using our AmazingDependency from above let's set up a new component for injection.

import { Inject } from 'react-injector'
import AmazingDependency from './AmazingDependency'

@Inject([ AmazingDependency ])
export default class GreeterComponent {
    render() {
        return <div>Hello, { this.props.AmazingDependency.getAmazingText() }</div>
    }
}

Rendering this component will display Hello, World! because the dependency was injected into the props object. Notice that the name on the props object matches the name of the dependency. This is by design. However, if you decide to uglify your code leveraging the mangle option the name of the dependency will change, but props will not be updated. To avoid this either don't use this option or change your code to access dependency props in a different way. Example:

this.props[AmazingDependency.name].getAmazingText()

Decorator Alternative

React Injector provides an alternative to Inject because decorators are still experimental. The InjectDirect function can be used in it's place.

Example:

import { InjectDirect } from 'react-injector'

class DirectGreeter {
    constructor(amazingDependency) {
        this.amazingDependency = amazingDependency
    }
    
    greet() {
        return "Hello, " + this.amazingDependency.getAmazingText()
    }
}
InjectDirect(DirectGreeter, [ AmazingDependency ])

export default DirectGreeter

Container

The internals of React Injector rely on a dependency injection container. This container can be accessed directly.

import { Container } from 'react-injector'

Though the container is generally behind this scenes it can be used, if needed, for direct service location.

Container.get(ThatOneDependency)