Skip to content

I want to develop using a component-based approach.

Thinking based on organizing existing implementations

So far, we have implemented createApp API, Reactivity System, and Virtual DOM system in a small scale.
With the current implementation, we can dynamically change the UI using the Reactivity System and perform efficient rendering using the Virtual DOM system.
However, as a developer interface, everything is written in createAppAPI.
In reality, I want to divide the files more and implement generic components for reusability.
First, let's review the parts that are currently messy in the existing implementation. Please take a look at the render function in renderer.ts.

ts
const render: RootRenderFunction = (rootComponent, container) => {
  const componentRender = rootComponent.setup!()

  let n1: VNode | null = null
  let n2: VNode = null!

  const updateComponent = () => {
    const n2 = componentRender()
    patch(n1, n2, container)
    n1 = n2
  }

  const effect = new ReactiveEffect(updateComponent)
  effect.run()
}

In the render function, information about the root component is directly defined.
In reality, n1, n2, updateComponent, and effect exist for each component.
In fact, from now on, I want to define the component (in a sense, the constructor) on the user side and instantiate it.
And I want the instance to have properties such as n1, n2, and updateComponent.
So, let's think about encapsulating these as a component instance.

Let's define something called ComponentInternalInstance in ~/packages/runtime-core/component.ts.
This will be the type of the instance.

ts
export interface ComponentInternalInstance {
  type: Component // The original user-defined component (old rootComponent (actually not just the root component))
  vnode: VNode // To be explained later
  subTree: VNode // Old n1
  next: VNode | null // Old n2
  effect: ReactiveEffect // Old effect
  render: InternalRenderFunction // Old componentRender
  update: () => void // Old updateComponent
  isMounted: boolean
}

export type InternalRenderFunction = {
  (): VNodeChild
}

The vnode, subTree, and next properties that this instance has are a bit complicated, but from now on, we will implement it so that ConcreteComponent can be specified as the type of VNode.
In instance.vnode, we will keep the VNode itself.
And subTree and next will hold the rendering result VNode of that component. (This is the same as before with n1 and n2)

In terms of image,

ts
const MyComponent = {
  setup() {
    return h('p', {}, ['hello'])
  },
}

const App = {
  setup() {
    return h(MyComponent, {}, [])
  },
}

You can use it like this, and if you let the instance be instance of MyComponent, instance.vnode will hold the result of h(MyComponent, {}, []), and instance.subTree will hold the result of h("p", {}, ["hello"]).

For now, let's implement it so that you can specify a component as the first argument of the h function.
However, it's just a matter of receiving an object that defines the component as the type.
In ~/packages/runtime-core/vnode.ts

ts
export type VNodeTypes = string | typeof Text | object // Add object;

In ~/packages/runtime-core/h.ts

ts
export function h(
  type: string | object, // Add object
  props: VNodeProps
) {..}

Let's also make sure that VNode has a component instance.

ts
export interface VNode<HostNode = any> {
  // .
  // .
  // .
  component: ComponentInternalInstance | null // Add
}

As a result, the renderer also needs to handle components.
Implement processComponent similar to processElement and processText for handling components, and also implement mountComponent and patchComponent (or updateComponent).

First, let's start with an overview and detailed explanation.

ts
const patch = (n1: VNode | null, n2: VNode, container: RendererElement) => {
  const { type } = n2
  if (type === Text) {
    processText(n1, n2, container)
  } else if (typeof type === 'string') {
    processElement(n1, n2, container)
  } else if (typeof type === 'object') {
    // Add branching
    processComponent(n1, n2, container)
  } else {
    // do nothing
  }
}

const processComponent = (
  n1: VNode | null,
  n2: VNode,
  container: RendererElement,
) => {
  if (n1 == null) {
    mountComponent(n2, container)
  } else {
    updateComponent(n1, n2)
  }
}

const mountComponent = (initialVNode: VNode, container: RendererElement) => {
  // TODO:
}

const updateComponent = (n1: VNode, n2: VNode) => {
  // TODO:
}

Now, let's take a look at mountComponent. There are three things to do.

  1. Create an instance of the component.
  2. Execute the setup function and store the result in the instance.
  3. Create a ReactiveEffect and store it in the instance.

First, let's implement a function in component.ts to create an instance of the component (similar to a constructor).

ts
export function createComponentInstance(
  vnode: VNode,
): ComponentInternalInstance {
  const type = vnode.type as Component

  const instance: ComponentInternalInstance = {
    type,
    vnode,
    next: null,
    effect: null!,
    subTree: null!,
    update: null!,
    render: null!,
    isMounted: false,
  }

  return instance
}

Although the type of each property is non-null, we initialize them with null when creating the instance (following the design of the original Vue.js).

ts
const mountComponent = (initialVNode: VNode, container: RendererElement) => {
  const instance: ComponentInternalInstance = (initialVNode.component =
    createComponentInstance(initialVNode))
  // TODO: setup component
  // TODO: setup effect
}

Next is the setup function.
We need to move the code that was previously written directly in the render function to here and store the result in the instance instead of using variables.

ts
const mountComponent = (initialVNode: VNode, container: RendererElement) => {
  const instance: ComponentInternalInstance = (initialVNode.component =
    createComponentInstance(initialVNode))

  const component = initialVNode.type as Component
  if (component.setup) {
    instance.render = component.setup() as InternalRenderFunction
  }

  // TODO: setup effect
}

Finally, let's combine the code for creating the effect into a function called setupRenderEffect.
Again, the main task is to move the code that was previously implemented directly in the render function to here, while utilizing the state of the instance.

ts
const mountComponent = (initialVNode: VNode, container: RendererElement) => {
  const instance: ComponentInternalInstance = (initialVNode.component =
    createComponentInstance(initialVNode))

  const component = initialVNode.type as Component
  if (component.setup) {
    instance.render = component.setup() as InternalRenderFunction
  }

  setupRenderEffect(instance, initialVNode, container)
}

const setupRenderEffect = (
  instance: ComponentInternalInstance,
  initialVNode: VNode,
  container: RendererElement,
) => {
  const componentUpdateFn = () => {
    const { render } = instance

    if (!instance.isMounted) {
      // mount process
      const subTree = (instance.subTree = normalizeVNode(render()))
      patch(null, subTree, container)
      initialVNode.el = subTree.el
      instance.isMounted = true
    } else {
      // patch process
      let { next, vnode } = instance

      if (next) {
        next.el = vnode.el
        next.component = instance
        instance.vnode = next
        instance.next = null
      } else {
        next = vnode
      }

      const prevTree = instance.subTree
      const nextTree = normalizeVNode(render())
      instance.subTree = nextTree

      patch(prevTree, nextTree, hostParentNode(prevTree.el!)!) // ※ 1
      next.el = nextTree.el
    }
  }

  const effect = (instance.effect = new ReactiveEffect(componentUpdateFn))
  const update = (instance.update = () => effect.run()) // Register to instance.update
  update()
}

※ 1: Please implement a function called parentNode in nodeOps that retrieves the parent Node.

ts
parentNode: (node) => {
    return node.parentNode;
},

I think it's not particularly difficult, although it's a bit long.
In the setupRenderEffect function, a function for updating is registered as the update method of the instance, so in updateComponent, we just need to call that function.

ts
const updateComponent = (n1: VNode, n2: VNode) => {
  const instance = (n2.component = n1.component)!
  instance.next = n2
  instance.update()
}

Finally, since the implementation that was defined in the render function so far is no longer needed, we will remove it.

ts
const render: RootRenderFunction = (rootComponent, container) => {
  const vnode = createVNode(rootComponent, {}, [])
  patch(null, vnode, container)
}

Now we can render components. Let's try creating a playground component as an example.
In this way, we can divide the rendering into components.

ts
import { createApp, h, reactive } from 'chibivue'

const CounterComponent = {
  setup() {
    const state = reactive({ count: 0 })
    const increment = () => state.count++

    return () =>
      h('div', {}, [
        h('p', {}, [`count: ${state.count}`]),
        h('button', { onClick: increment }, ['increment']),
      ])
  },
}

const app = createApp({
  setup() {
    return () =>
      h('div', { id: 'my-app' }, [
        h(CounterComponent, {}, []),
        h(CounterComponent, {}, []),
        h(CounterComponent, {}, []),
      ])
  },
})

app.mount('#app')

Source code up to this point: chibivue (GitHub)

Released under the MIT License.