When working on larger applications, we observed substantial benefits from creating a separation between "passive view" React components, and those that interacted directly with Microcosm (and other non-presentational entities).
The Presenter add-on makes it easier to keep application logic high
within a component tree. It is designed specifically to extract and
compute properties coming from a Microcosm instance and efficiently
send them down as props
to child "passive view" React components.
Additionally, Presenters expose an "intent" API
(see withIntent
and <Form />
). Intents
allow components deep within a component tree the ability to
broadcast messages through Presenters, eventually passing straight
into a Microcosm repo if no Presenter's register
method intercepts it.
We'll cover both of these features within this document
Presenter extends from React.Component
, and can be used just like a
React component:
import React from 'react'
import DOM from 'react-dom'
import Presenter from 'microcosm/addons/presenter'
import Microcosm from 'microcosm'
const repo = new Microcosm()
repo.patch({ planets: [ 'Mercury', 'Venus', 'Earth' ]})
class PlanetsPresenter extends Presenter {
model (props, state) {
return {
planets: data => data.planets
}
}
view ({ planets }) {
return <p>{ planets.join(', ') }</p>
}
}
DOM.render(<PlanetsPresenter repo={ repo } />)
In the example above, the PlanetsPresenter
will extract a list of
planets from the Microcosm instance provided to it via the repo
prop. This is available as state, which the Presenter can send into a
child component.
While we do not recommend it for large repos, some times it's simply easier
to subscribe to all repo changes. model
can also return a function, which
will be called with state:
class PlanetsPresenter extends Presenter {
model () {
return data => data
}
view ({ planets }) {
return <p>{ planets.join(', ') }</p>
}
}
Though explicit, passing event callbacks down through a deep component
hierarchy can be cumbersome and brittle. Presenters expose a method on
context
that enable child components to declare intents
receivable
by Presenters.
The Form add-on can be used to broadcast intents to Presenters:
import React from 'react'
import DOM from 'react-dom'
import Presenter from 'microcosm/addons/presenter'
import Form from 'microcosm/addons/form'
import Microcosm from 'microcosm'
const repo = new Microcosm()
const increaseCount = n => n
repo.addDomain('count', {
getInitialState() {
return 0
},
increase(count, amount) {
return count + amount
},
register() {
return {
[increaseCount] : this.increase
}
}
})
function StepperForm ({ count }) {
return (
<Form intent="increaseCount">
<input type="hidden" name="amount" value="1" />
<p>The current count is { count }</p>
<button>+ 1</button>
</Form>
)
}
class CountPresenter extends Presenter {
model () {
return {
count: data => data.count
}
}
register () {
return {
increaseCount: this.increaseCount
}
}
increaseCount (repo, { amount }) {
return repo.push(increment, amount)
}
view ({ count }) {
return <StepperForm count={ count } />
}
}
DOM.render(<CountPresenter repo={ repo } />, document.getElementById('container'))
Whenever the form is submitted, an increaseCount
intent will bubble
up to the associated Presenter including the serialized parameters of
the form. Since this Presenter's register method includes increaseCount
, it will
invoke the method with the associated parameters.
If a Presenter does not implement an intent, it will bubble up to any parent Presenters. If no Presenter implements the intent, an exception will raise. This is useful for broadcasting action intents. For example, we could replace the prior Form example with:
function StepperForm ({ count }) {
return (
<Form intent={ increaseCount }>
<input type="hidden" name="amount" value="1" />
<p>The current count is { count }</p>
<button>+ 1</button>
</Form>
)
}
Called when a presenter is created, useful for initial data fetching and other prep work.
Called when a presenter gets new props. This is useful for secondary data fetching and other work that must happen when a Presenter receives new information.
If pure (by default) this will only get called if properties are shallowly different.
Runs when the presenter unmounts. Useful for tearing down subscriptions and other setup behavior.
Builds a view model for the current props and state. This must return an object of key/value pairs. If the value is a function, it will be calculated by passing in the repo's current state:
class PlanetPresenter extends Presenter {
model (props, state) {
return {
planet : data => data.planets.find(p => p.id === props.planetId)
}
}
// ...
}
If the Presenter is pure (passed in as either a prop or as an option to the associated repo), this will only update state if shallowly equal.
When not implemented, model returns all repo state.
A special render method that is given the current result of model
.
class Greeter extends Presenter {
model ({ greet })
return {
message: "Hello, " + greet
}
}
view ({ message }) {
return <p>{message}</p>
}
}
Views may also be assigned as getters, or as properties:
class Example extends Presenter {
get view {
return MyView
}
}
// Or use newer class features:
class Example extends Presenter {
view = MyView
}
If a view is a React component, it will invoke it with the Presenter's model as props (including children).
Expose "intent" subscriptions to child components. This is used with the
add-on to improve the ergonomics of presenter/view communication (though this only occurs from the view to the presenter).import Form from 'microcosm/addons/form'
class HelloWorldPresenter extends Presenter {
register() {
return {
'greet': this.greet
}
}
greet() {
alert("hello world!")
}
view () {
return (
<Form intent="greet">
<button>Greet</button>
</Form>
)
}
}