Skip to content

Latest commit

 

History

History
579 lines (436 loc) · 16.1 KB

README.md

File metadata and controls

579 lines (436 loc) · 16.1 KB

XComponent

This is a micro framework that brings together MobX and React in order to solve performance, boilerplate and lifecycle issues with React.

1. Why

React's ecosystem is a great place to be, so we do not want to leave it just because of how annoying hooks, state management and render performance tweaking is.

So we use an signals/observables solution, like mobx.

It solves the performance problems, but it also adds boilerplate and isn't cleanly interoperable with React's lifecycle. There is also no "correct" convention, so your team is left to "figure it out" on each new project, which further deteriorates consistency.

This is a micro framework which exists to define a convention while also being simple enough to review or copy paste parts of it into your own project as desired.

2. How does it compare?

2.1. Drop in replacement for observer

import { observer } from 'mobx-react-lite'
const MyComponent = observer((props: { someProp: number }) => <>{props.someProp}</>)

import { X } from '@n4s/xcomponent'
const MyComponent = X((props: { someProp: number }) => <>{props.someProp}</>)

2.2. Full example comparison

//
// BEFORE:
//

import { observer } from 'mobx-react-lite'
import React from 'react'

// Standard mobx component with some inline class state
export const MyComponent = observer<{ someProp: number }>((props) => {
  const [state] = React.useState(() => new (class {
    constructor() {
      makeAutoObservable(this)
    }

    count = 0
    someProp = props.someProp

    get combinedNumber() {
      return this.count + this.someProp
    }

    setCount = (value: number) => this.count = value
    setSomeProp = (value: number) => this.someProp = value
    increment = () => this.setCount(this.count + 1)
    
  })())

  useEffect(() => {
    console.log('mounted')
    return () => {
      console.log('unmounted')
    }
  }, [])

  useEffect(() => {
    if (state.someProp !== props.someProp)
      state.setSomeProp(props.someProp)
  }, [state.someProp])
  

  return <>
    <div>{props.someProp}</div>
    <div>{state.count}</div>
    <div>{state.combinedNumber}</div>
    <button onClick={state.increment}>Incr</button>
  </>
})

//
// AFTER:
//

import { X } from '@n4s/xcomponent'

export const MyComponent = X<{ someProp: number }>((props) => {
  const state = X.useState(props, (props) => class {
    count = new X.Value(0)

    get combinedNumber() {
      return this.count.value + props.someProp // must access props through `props.` for reactivity
    }

    increment = () => this.count.set(this.count.value + 1)
  })

  X.useOnMounted(() => { console.log('mounted, do some setup') })
  X.useOnUnmounted(() => { console.log('unmounted, do some cleanup') })

  return <>
    <div>{props.someProp}</div>
    <div>{state.count.value}</div>
    <div>{state.combinedNumber}</div>
    <button onClick={state.increment}>Incr</button>
  </>
})

You'll notice first off a few things:

  • Far less boilerplate
  • Specific hooks for specific lifecycles
  • Props are made and kept observable
    • Instead of using complicated useEffect to sync props with state, props are stored in a store and updated on each render based on a diff algo
    • Then you can use X.useReaction or X.useAutorun to react to specific prop changes, just like any other observable, anywhere you want to, not just within a react hook.

2.3. Extended example pattern

Want to emulate Vue's setup() pattern?

import { X } from '@n4s/xcomponent'

type Props = { someProp: number }

// By making this another function we have nicely seperated concerns
// And we can make additional hooks calls/lifecycle management which is all state-related
// So that we do not pollute the view component with state management
const useMyComponentState = (props: Props) => {
  const state = X.useState(props, (props) => class {
    props = props // So that we can react to prop changes in our reactions below
    count = new X.Value(0)

    get combinedNumber() {
      return this.count.value + props.value.someProp
    }

    increment = () => this.count.set(this.count.value + 1)
  })

  X.useOnMounted(() => { console.log('mounted, do some setup') })
  X.useOnUnmounted(() => { console.log('unmounted, do some cleanup') })

  // This is a custom hook that will run whenever `state.count` changes
  X.useReaction(
    () => state.count,
    () => console.log('count changed to', state.count.value)
  )

  // This is a custom hook that will run whenever `props.someProp` changes
  X.useReaction(
    () => state.props.value.someProp,
    () => console.log('someProp changed to', state.props.value.someProp)
  )

  return state
}

export const MyComponent = X<Props>((props) => {
  const state = useMyComponentState(props)
  
  return <>
    <div>{props.someProp}</div>
    <div>{state.count.value}</div>
    <div>{state.combinedNumber}</div>
    <button onClick={state.increment}>Incr</button>
  </>
})

2.4. Inline class based state with computed JSX

This is a little more on the crazy side, use this only as... Inspiration for now. Now lets create computed properties which return JSX. Note that in this pattern, this ui state class is actively coupled to the view - we aren't trying to decouple it, but rather to avoid useCallback/useMemo related hackery and to also avoid needing to create a new component for each computed property.

It's mostly a convenience pattern, but it can be useful to organize your view logic in a performant way.

I'm not totally sure if MobX does anything funny with JSX return values, or if memory usage becomes an issue here, so I'll have to do some benchmarks.

  • mobxjs/mobx#485
    • This thread posits that there is a performance improvement
    • The performance isn't huge because basically react is slow anyway
    • Likely not harmful
import { X } from '@n4s/xcomponent'

type Props = { someProp: number }

export const MyComponent = X<Props>((props) => {
  const state = X.useState(props, (props) => class {
    props = props // Store it so we can use it in other reactive contexts
    count = new X.Value(0)

    get combinedNumber() {
      return this.count.value + this.props.value.someProp
    }

    increment = () => this.count.set(this.count.value + 1)
  })

  // This ui state accesses props via `state.props`, so doesn't need to be passed props
  const ui = X.useState(() => class {
    // This will recompute whenever the below state observables change
    get infoList() {
      return <ul>
        <li>Count: {state.count.value}</li>
        <li>Some prop: {state.props.someProp}</li>
        <li>Combined number: {state.combinedNumber}</li>
      </ul>
    }

    // In theory this should never recompute
    get incrButton() {
      return <button onClick={state.increment}>Incr</button>
    }

    // Should recompute if state.combinedNumber changes
    get bigNumberMessage() {
      return state.combinedNumber > 10 && <div>Wow, that's a big number!</div>
    }
  })


  return <>
    {ui.infoList}
    {ui.incrButton}
    {ui.bigNumberMessage}
  </>
})

3. Details of the API

You can use a class based styled (my preference currently!), or a functional style.

// Functional style
export const MyStateFn = () => {
  const count = new X.Value(0)
  const increment = () => count.set(count.value + 1)

  return { count, increment }
}

// Class style
export class MyStateClass {
  count = new X.Value(0)
  increment = () => this.count.set(this.count.value + 1)
}

// and both inside a component, inlined looks like this:
export const MyComponent = X(() => {
  // Functional
  const state = X.useState(() => {
    const count = new X.Value(0)
    const increment = () => count.set(count.value + 1)

    return {
      count,
      get someComputed() { return count.value + 999 }
      increment,
    }
  })

  // Class
  const state2 = X.useState(() => class {
    count = new X.Value(0)
    get someComputed() { return this.count.value + 999 }
    increment = () => this.count.set(this.count.value + 1)
  })

  return <></>
})

Note: When defining GLOBAL state functions or classes you will still need to add in makeAutoObservable or makeObservable as needed - it's just standard mobx at this point, no react compatability needed - these mobx utilities could be added to X, too, for convenience.

import { makeAutoObservable } from 'mobx'

// standard mobx class state
export class MyGlobalState {
  constructor() {
    makeAutoObservable(this)
  }

  count = new X.Value(0)
  get someComputed() { return this.count.value + 999 }
  increment = () => this.count.set(this.count.value + 1)
}

export function MyGlobalStateFn() {
  const count = new X.Value(0)
  const increment = () => count.set(count.value + 1)

  return makeAutoObservable({
    count,
    increment, 
    get someComputed() { return count.value + 999 }})
   })
}

4. State can be decoupled from the component

State can be decoupled completely from the component so that it may be reasoned with effectively. This keeps things sane as a project grows.

In this example the state lives in another file, and the component is just a view into that state.

  • ./src/MyComponentState.ts
import { X } from '@n4s/xcomponent'

export class MyComponentState {
  constructor(public props: { someProp: number }) {
    makeAutoObservable(this)
  }

  count = new X.Value(0)

  get combinedNumber() {
    return this.count.value + this.props.value.someProp
  }

  increment = () => this.count.set(this.count.value + 1)
}

export const MyComponentStateAsFn = (props: { someProp: number }) => {
  // ^ props is an observable store already.
  const count = new X.Value(0)

  return {
    count,
    get combinedNumber() { return count.value + props.someProp },
    increment: () => count.set(count.value + 1)
  }
}
  • ./src/MyComponent.tsx
import { X } from '@n4s/xcomponent'
import { MyComponentState, MyComponentStateAsFn } from './MyComponentState'

// Can re-use the type defined by the state
type MyComponentProps = typeof MyComponentState['props']['value']

export const MyComponent = X((props: MyComponentProps) => {
  // Classes
  const state = X.useState(props, (props) => new MyComponentState(props))
  // Fn if you prefer
  const stateFromFn = X.useState(props, MyComponentStateAsFn)

  return <>
    <div>{props.someProp}</div>
    <div>{state.count.value}</div>
    <div>{state.combinedNumber}</div>
    <button onClick={state.increment}>Incr</button>
  </>
})

4.1. Component composition extender

//
// BEFORE:
//

import { observer } from 'mobx-react-lite'

const Dialog = observer<{ children: ReactNode }>((props) => <> {props.children} </> )
const DialogHead = observer<{ children: ReactNode }>((props) => <h2>{props.children}</h2> )
const Example = () => <Dialog><DialogHead>Title</DialogHead>Content</Dialog>

//
// AFTER:
//

import { X } from '@n4s/xcomponent'

const DialogBody = X((props) => <h2 className={Dialog.classes.body}>{props.children}</h2> )

const Dialog = X((props) => <>{props.children}</> )
  .with({
    // Inline component
    Head: X((props) => <h2 className={Dialog.classes.head}>{props.children}</h2> ),
    // Referenced
    Body: DialogBody,

    // Lets add some class names to make it easier to style externally
    classes: {
      head: 'my-fancy-dialog-head',
      body: 'my-fancy-dialog-body',
    }
  })

const Example = () =>
  <Dialog>
    <Dialog.Head>Title</Dialog.Head>
    <Dialog.Body>Content</Dialog.Body>  
  </Dialog>

5. Dismiss unecessary React hooks

Moving logic to mobx allows for the majority of React state-related hooks to be dismissed. The nature of observables means that many of React's lifecycle hooks corrupt the state lifecycle, and should be avoided. We want to let mobx handle it.

  • Good:
    • useEffect
      • X Alternatives:
        • useOnMounted
        • useOnUnmounted
        • useReaction
        • useAutorun
    • useLayoutEffect
    • useRef
  • Unecessary:
    • useCallback
    • useState
    • useReducer
    • useMemo
    • etc.

6. Helper models

6.1. Value

The Value class is effectively observable.box of interface { value: T, set: (value: T) => void }.

Features:

  • Type inferrence
  • Async mobx actions (no need to wrap in runInAction or use flow generators)
  • Terseness
  • Avoids reading value until necessary during prop-passing
  • Supports two way binding patterns
const selectedFruit = new Value<'banana'|'apple'|undefined>(undefined)
selectedFruit.set('test') // TS error
selectedFruit.set('banana') // Valid
selectedFruit.value // 'banana'

6.2. AsyncValue

Think of react-query for this one. It is a Value that can be in a loading state, and can be awaited.

Features:

  • Ergonomic types
  • Async mobx actions
  • Queuing
  • Promise cancellation
  • Pending state
  • Error state
  • Success state
  • Progress state (eg. for uploads)
async function fetchFiles(c: { userId: string; foo: number }): Promise<{ name: string }[]> {
  return []
}

class ExampleModel {
  constructor() { makeAutoObservable(this) }
  activeUserId = '22'
  files = new AsyncValue(async ({ foo }: { foo: number }) =>
    fetchFiles({ userId: this.activeUserId, foo })
  )
}

const example = new ExampleModel()
example.files.value?.[0]?.name // undefined - missing data
await example.files.query({ foo: 22 }) // foo is strongly typed, inferred!
example.files.value?.[0]?.name // 'myFile.txt' - has data!
example.files.error // undefined - no error
example.files.isPending // false - we already awaited it

const v = new AsyncValue(() => fetchUsersList())
v.value // undefined
const promise = v.query() // Don't need to provide params as none are defined
v.isPending // true
await promise
v.isPending // false
v.value // [{ id: 1, name: 'John' }, { id: 2, name: 'Jane' }]

6.3. BoxedValue

Very similar to Value, however, allows for the getter and setter to be defined seperately, and additionally encapsulates the observable value inside the closure.

const blah = { something: 'banana' }

const somethingFromUri = new BoxedValue(
  // getter
  () => uriRoutes.someRoute.search.something,
  // setter
  (newValue) => uriRoutes.someRoute.push((uri) => ({ search: { something: newValue } })),
)

somethingFromUri.value // 'foo'
somethingFromUri.set('bar')
somethingFromUri.value // 'bar'

// Here we omit the setter, so the value is read-only
// This is effectively just a container encapsulating the value
const somethingWrappedToOptimizeObservability = new BoxedValue(
  () => blah.something,
)

somethingWrappedToOptimizeObservability.value // 'banana'
somethingWrappedToOptimizeObservability.set('banana') // does nothing, because no setter
  

6.4. BoolValue

A Value that is specifically for boolean values. It has a few additional methods to make working with booleans easier.

const isOpen = new BoolValue(true)

isOpen.toggle() // false
isOpen.toggle() // true
isOpen.setFalse()
isOpen.value // false
isOpen.isTrue // false
isOpen.setTrue()
isOpen.isTrue // true
isOpen.value // false

const Example = X(() => 
  <>
    <button onClick={isOpen.toggle}>Open</button>
    <Dialog onClose={isOpen.setFalse}>...</Dialog>
  </>
)

Guides

Working with class-based state

Scaffolding a new project

Please check out this folder for a (currently WIP) full project structure example: