Reanimated Babel Plugin
What is Reanimated Babel Plugin?
The Reanimated Babel Plugin transforms your code so that it can run on the UI thread. It looks for functions marked with a 'worklet';
directive and converts them into serializable objects. We call this process workletization.
In short, a worklet is:
- A function that contains a
'worklet'
directive at its very top, i.e.:
function foo() {
'worklet';
console.log('Hello from worklet');
}
- A function that is autoworkletizable, i.e.:
useAnimatedStyle(() => {
// This function will be ran on the UI thread,
// hence it's in a workletizable context and will be
// autoworkletized. You don't need to add the 'worklet' directive here.
return {
width: 100,
};
});
What can be a worklet?
JavaScript terms
Reanimated Babel Plugin supports the following terms as worklets:
Function Declarations
function foo() {
'worklet';
console.log('Hello from FunctionDeclaration');
}
Function Expressions
const foo = function () {
'worklet';
console.log('Hello from FunctionExpression');
};
Arrow Function Expressions
const foo = () => {
'worklet';
console.log('Hello from ArrowFunctionExpression');
};
Object Methods
const obj = {
foo() {
'worklet';
console.log('Hello from ObjectMethod');
},
};
Reanimated terms
[Experimental] Worklet Context Objects
Object methods called on UI thread lose their this
binding.
const obj = {
foo: 1,
bar() {
'worklet';
console.log(this.foo); // undefined - the binding was lost.
},
};
Worklet Context Objects are special terms that preserve that binding. Don't mistake them for objects created with useSharedValue
. All changes to Worklet Context Objects on the UI thread are visible only on the UI thread. The same applies to the JS thread.
const obj = {
__workletContextObject: true,
foo: 1,
bar() {
console.log(this.foo);
},
};
obj.foo = 2;
obj.bar(); // Logs 2
runOnUI(() => obj.bar())(); // Logs 1
runOnUI(() => (obj.foo = 3))();
obj.bar(); // Logs 2
runOnUI(() => obj.bar())(); // Logs 3
__workletContextObject
is a special property that marks an object as a Worklet Context Object. It's value doesn't matter, but it's a good practice to use true
as a value. 'worklet'
directive in methods will be ignored if the object has this property.
const workletContextObject = {
__workletContextObject: true,
message: 'Hello from WorkletContextObject',
foo() {
console.log(this.message);
},
};
[Experimental] Worklet Classes
Hermes, the JavaScript engine used by React Native, doesn't support classes. Class syntax requires polyfilling before it can be used, which is problematic for the UI thread. To work around this, we coined the term of Worklet Classes. Worklet classes can be instantiated on the UI thread.
__workletClass
is a special property that marks a class as a Worklet Class. It's value doesn't matter, but it's a good practice to use true
as a value. 'worklet'
directive in methods will be ignored if the class has this property.
class Clazz {
__workletClass = true;
message = 'Hello from WorkletClass';
foo() {
console.log(this.message);
}
}
runOnUI(() => new Clazz().foo())(); // Logs 'Hello from WorkletClass'
Pitfalls:
- Worklet Classes don't support inheritance.
- Worklet Classes don't support static methods and properties.
- Class instances cannot be shared between JS and UI threads.
Autoworkletization
To reduce boilerplate code and provide a safer API, Reanimated Babel Plugin detects automatically whether a function should be workletized. Thanks to that, you don't need to add the 'worklet'
directive to your callbacks:
const style = useAnimatedStyle(() => {
// You don't need to add the 'worklet' directive here,
// since plugin detects this callback as autoworkletizable.
return {
width: 100,
};
});
This isn't limited to useAnimatedStyle
hook - Reanimated Babel Plugin autoworkletizes all callbacks for the API of Reanimated. The whole list can be found in the plugin source code.
Keep in mind that in more advanced use cases, you might still need to manually mark a function as a worklet.
Referencing worklets
You can define worklets before they are used and the plugin will autoworkletize them too:
function foo() {
// You don't need to add
// the 'worklet' directive here.
return { width: 100 };
}
// You don't need to define an inline function here,
// a reference is enough.
const style = useAnimatedStyle(foo);
Objects aggregating worklets
In some APIs, like useAnimatedScrollHandler
you can pass an object that contains worklets instead of a function:
const handlerObject = {
// You don't need to mark these methods as worklets.
onBeginDrag() {
console.log('Dragging...');
},
onScroll() {
console.log('Scrolling...');
},
};
const handler = useAnimatedScrollHandler(handlerObject);
[Experimental] Workletizing whole files
You can mark a file as a workletizable file by adding the 'worklet'
directive to the top of the file.
This will workletize all top-level JavaScript terms and Reanimated terms automatically. It can come in handy for files that contain multiple worklets.
// file.ts
'worklet';
function foo() {
// Function 'foo' will be autoworkletized.
return { width: 100 };
}
function bar() {
// Function 'bar' will be autoworkletized.
function foobar() {
// Function 'foobar' won't since it's not defined in top-level scope.
console.log("I'm not a worklet");
}
return { width: 100 };
}
Limits of autoworkletization
The plugin cannot infer whether a function is autoworkletizable or not in some contexts.
Imports
When importing a function from another file or a module and using it as a worklet, you must manually add the 'worklet'
directive to the function:
// foo.ts
import { bar } from './bar';
// ...
const style = useAnimatedStyle(bar);
// bar.ts
export function bar() {
'worklet'; // Won't work without it.
return {
width: 100,
};
}
Custom hooks
Currently Reanimated hasn't exposed APIs that would allow you to register your custom hooks for callback workletization. This however, might change in the future.
Expressions
A function won't get automatically workletized when it's a result of an expression. You have to add the 'worklet';
directive to make it work:
const foo = someCondition
? () => {
'worklet'; // Won't work without it.
return { width: 100 };
}
: () => {
'worklet'; // Won't work without it.
return { width: 200 };
};
const style = useAnimatedStyle(foo);
In such cases we recommend either handling the conditional logic in the worklet itself or refactoring your code to eliminate the need for conditional worklets.
Pitfalls
There are some patterns that won't work with the plugin.
Hoisting worklets
Worklets aren't hoisted. This means that you can't use worklets before they are defined:
// The following line crashes,
// even though 'foo' is marked as a worklet.
const style = useAnimatedStyle(foo);
function foo() {
'worklet';
return { width: 100 };
}
Notes
Babel is a powerful tool that can be explored to implement numerous useful features. If you feel like Reanimated Babel plugin could make use of some new functionality or that its pitfalls are too severe, feel free to let us know on GitHub, via an issue or a discussion thread - and as always, PRs are welcome!