DEV Community

Ryan John
Ryan John

Posted on • Originally published at vureact.top

How does VuReact compile Vue 3's defineExpose() to React?

VuReact is a compiler toolchain for migrating from Vue to React — and for writing React with Vue 3 syntax.

In this article, we will look at how Vue 3's defineExpose() macro is mapped into React.

Before We Start

To keep the examples easy to read, this article follows two simple conventions:

  1. All Vue and React snippets focus on core logic only, with full component wrappers and unrelated configuration omitted.
  2. The discussion assumes you are already familiar with the API shape and core behavior of Vue 3 defineExpose().

Compilation Mapping

Vue defineExpose() -> React forwardRef() + useImperativeHandle()

defineExpose() is the macro used inside Vue 3 <script setup> to expose internal state or methods from a child component to its parent.

VuReact compiles that pattern into React's forwardRef() plus useImperativeHandle(), allowing the parent component to access the exposed object through a ref.

  • Vue
<script setup lang="ts">
import { ref, defineExpose } from 'vue';

defineProps<{ title: string }>();

const count = ref(0);
const increment = () => count.value++;

defineExpose({
  count,
  increment,
});
</script>
Enter fullscreen mode Exit fullscreen mode
  • Compiled React
import { forwardRef, useCallback, useImperativeHandle, memo } from 'react';
import { useVRef } from '@vureact/runtime-core';

type IComponentProps = { title: string };

const Component = memo(
  forwardRef<any, IComponentProps>((props, expose) => {
    const count = useVRef(0);

    const increment = useCallback(() => {
      count.value++;
    }, [count.value]);

    useImperativeHandle(expose, () => ({
      count,
      increment,
    }));

    return <div>{count.value}</div>;
  }),
);

export default Component;
Enter fullscreen mode Exit fullscreen mode

As the example shows, Vue defineExpose() is compiled into React's forwardRef() and useImperativeHandle() combination.

VuReact preserves the structure of the exposed object, and exposed refs still use .value, which keeps the interaction model close to Vue.

Parent access: Vue ref + expose -> React ref.current

In Vue, parent components access exposed child values through ref and expose. In React, VuReact maps that pattern to useRef() plus ref.current.

  • Vue parent
<template>
  <Component ref="childRef" />
</template>

<script setup lang="ts">
import { onMounted, ref } from 'vue';

const childRef = ref();

onMounted(() => {
  childRef.value?.count.value; // 0
  childRef.value?.increment();
  childRef.value?.count.value; // 1
});
</script>
Enter fullscreen mode Exit fullscreen mode
  • Compiled React parent
const Parent = () => {
  const childRef = useRef();

  useMounted(() => {
    childRef.current?.count.value; // 0
    childRef.current?.increment();
    childRef.current?.count.value; // 1
  });

  return <Component ref={childRef} />;
};
Enter fullscreen mode Exit fullscreen mode

VuReact keeps the parent access path aligned with the original Vue intent, so exposed child refs and methods remain straightforward to use.

Related Links

Top comments (0)