Write PIXI applications using React declarative style đź‘Ś
yarn add @paolotozzo/react-pixi
or
npm install @paolotozzo/react-pixi --save
import { Stage, Sprite } from '@paolotozzo/react-pixi'
const App = () => (
<Stage>
<Sprite image="./bunny.png" x={100} y={100} />
</Stage>
)
This example will render a PIXI.Sprite
object into a
Root Container of a
PIXI.Application
on the page. The Stage
object will
create a valid <canvas />
element to render to.
import { render, Text } from '@paolotozzo/react-pixi'
import { Application } from 'pixi.js'
// Setup PIXI app
const app = new Application({
width: 800,
height: 600,
backgroundColor: 0x10bb99,
view: document.getElementById('container'),
})
// Use the custom renderer to render a valid PIXI object into a PIXI container.
render(<Text text="Hello World" x={200} y={200} />, app.stage)
With the introduction of the className attribute react-pixi supports the use of css files to style pixi components almost in the same way as you would do it for React HTML elements, which means that most of the PIXI properties can be defined in your own class selectors. Names of the class selectors need to be preceded by 'pixi'.
pixi.header {
x: 0;
y: 0;
alpha: 0.8;
}
pixi.title {
fontFamily: Arial;
font-size: 36;
font-style: italic; // both the pixi camel cased and css kebab cased style are supported
fontWeight: bold;
fill: ['#FF0000', '#FFFF00'];
stroke: #FFFFFF;
strokeThickness: 5;
dropShadow: true;
dropShadowColor: #000000;
dropShadowBlur: 4;
dropShadowDistance: 6;
wordWrap: true;
wordWrapWidth: 400;
}
pixi.clickable {
alpha: 0.8;
buttonMode: true;
}
// comma to separate multiple selectors with the same styles.
pixi.hidden, .hide {
alpha: 0;
}
// same declarations get merged, and same rules override each other meaning that the last one wins
pixi.clickable {
alpha: 1;
interactive: true;
}
Then on your React component you could use them as follow:
<Container className='header'>
<Text className='title' text='Welcome' />
</Container>
or with the very convenient classNames library.
const btnClass = classNames({
clickable: clickable,
btn: true
});
<Container className={btnClass}>
Stylesheets can be imported directly in your component as usual (in this case you'll need to install css-loader
and style-loader
if you use a webpack build), as internal by using the <style>
tag in the head, or even as an external stylesheet added as a link to the HTML main document (as long as the .css file is on the same domain of the html that loads it).
There are some properties that are inside sub-objects in PIXI, such as .object.position.x
, object.scale.y
, object.skew.x
, and which are often used. In CSS, in order to be able to parse them you have to use the much easier format: positionX
, positionY
, scaleX
, scaleY
.
Other properties affected are skew
,pivot
,anchor
,tilePosition
and tileScale
.
Note also that rotation
can be written in degress, not in radians.
Same as Dom CSS, inline set props have the priority over the ones defined on the stylesheet, unless they contain the !important
clause, in this case they would override it.
pixi.header {
x: 0;
y: 0;
..
alpha: 0.8 !important;
}
//alpha will be overriden onl
<Container x={10} y={10} alpha={1}>
To use different rules for different devices mediaqueries are also supported
@media (min-width: 1024px) {
pixi.header {
visible: false;
}
}
Very conveniently you can also define animations for your PIXI components with CSS. This feature requires the use of gsap as a project dependency.
@keyframes fromtoptobottom {
from {
y: 0;
}
to {
y: 590;
}
}
@keyframes aroundthescreen {
0% {
x:0;
}
25% {
x: 400;
}
50% {
y: 400;
}
75% {
x: 0;
}
100% {
y: 0;
}
}
pixi.from-top-to-bottom {
animation-name: fromtoptobottom;
animation-duration: 1000;
animation-delay: 1000;
}
pixi.around-the-screen {
animation-name: aroundthescreen;
animation-duration: 2000;
animation-delay: 2000;
}
All components that use a class that is triggering a css animation call an animationEnd
event once the animation has finished. This is true also for the classes that have animation-direction
set as 'reverse', whereas the ones with animation-iteration-count
set as 'infinite' never call animationEnd
<Container className='around-the-screen' animationEnd={this.handleAnimationEnd} />
- The
animation
shorthand property is currently not supported, therefore each animation params need to be declared with its own property. - Almost all animation parameters are supported, with the exception of the
animation-fill-mode
which is currently ignored and behaves as it would do if set asforwards
. - Durations can be currently set only by using milliseconds, therefore css time format (2ms, 1s...) is not currently supported.
- To set the
animation-timing-function
refer to https://greensock.com/ease-visualizer/ as the Dom CSS predefined ones won't work. Example of valid values would beanimation-timing-function: Power1.easeOut
orElastic.easeIn
Together with CSS animations, also CSS transitions can be used.
pixi.movable {
transition-property: x, y;
transition-duration: 1000;
transition-timing-function: Circ.easeIn;
x: 0;
}
pixi.moving {
x: 500;
y: 550;
}
In the same way as Dom CSS, also on react-pixi the element first needs to be set with the class that declares the transition properties, and then any ensuing other class that will be added and that will change the values of those properties will unleash the animations.
Same as Dom CSS animations, also components that are animated with transitions call a transitionEnd
event once the animation has finished.
<Container className='around-the-screen' animationEnd={this.handleAnimationEnd} />
- The
transition
shorthand property is currently not supported, therefore each transition params need to be declared with its own property. - Durations can be currently set only by using milliseconds, therefore css time format (2ms, 1s...) is not currently supported.
- To set the
transition-timing-function
refer to https://greensock.com/ease-visualizer/ as the css predefined ones won't work. Example of valid values would betransition-timing-function: Power1.easeOut
orElastic.easeIn
getCSSBasedProps
is a useful method that returns an object literal with all css properties that the className attribute has set. This allows to use the props in a more imperative way when needed. Examples of that is when using the ref
attribute, or when setting the rules for the drawing API in the Graphics
component such as follows.
pixi.header {
width: 1024;
height: 100;
background-color: 0x555555;
z-index: 2;
x: 0;
y: 0;
}
<Graphics className={'header'} draw={(graphics) => {
const props = graphics.getCSSBasedProps()
const {backgroundColor, x, y, width, height} = props
graphics.beginFill(backgroundColor);
graphics.drawRect(x, y, width, height);
graphics.endFill();
}} />
The supports of the className attributes comes with a huge set of css feature immediately available, which will make the writing your PIXI application with React even more declarative, and many more are on the roadmap. But right now, of course, as the Dom CSS native implementation is huge, and not all of it is useful for the kind of applications that are built with PIXI, many of them are missing. Besides, as react-pixi supports also 'styled-components', it is already easy to write components with a great level of style encapsulation.
Following some Dom CSS features that are currently incomplete, different or totally missing.
Currently the only selector supported, other than the single class one used in all examples above, is the compounding multiple class selector, meaning that you can also define your selector in this way:
pixi.header.desktop {
width: 1024;
height: 100;
background-color: 0x555555;
z-index: 2;
x: 0;
y: 0;
}
pixi.title {
fontFamily: Arial;
font-size: 36;
}
pixi.white {
fill: '#FFFFFF';
}
and then use them like this
<Container className='header desktop'>
<Text className='title white' text='Welcome' />
</Container>
Therefore all the other combining css selectors, such as descendant, next sibling, etc. are not supported. Also selecting by ID, or with pseudo selectors such us :hover
is not yet offered.
CSS specificity between rules is not yet leveraged, therefore all CSS rules are interpreted with the same level of importance. The only levels of specificity which apply are:
- inline attributes prevail over the ones set with css classes
!important
prevail over the inline set attributes- classes triggered by mediaquery matching prevail over the general ones, regardless of the
!important
clause.
Currently the following Components are implemented by default:
- Container
- ParticleContainer
- Sprite
- TilingSprite
- Graphics
- SimpleMesh
- SimpleRope
- Text
- BitmapText
- NineSlicePlane
You can easily add new components to your project:
./components/Rectangle.js
import { Graphics } from 'pixi.js'
import { PixiComponent } from '@paolotozzo/react-pixi'
export default PixiComponent('Rectangle', {
create: props => {
return new Graphics()
},
didMount: (instance, parent) => {
// apply custom logic on mount
},
willUnmount: (instance, parent) => {
// clean up before removal
},
applyProps: (instance, oldProps, newProps) => {
const { fill, x, y, width, height } = newProps
instance.clear()
instance.beginFill(fill)
instance.drawRect(x, y, width, height)
instance.endFill()
},
})
App.js
import { Stage } from '@paolotozzo/react-pixi'
import Rectangle from './components/Rectangle'
export default () => (
<Stage>
<Rectangle x={100} y={100} width={500} height={300} fill={0xff0000} />
</Stage>
)
Props helper
ReactPixi comes with a handy utility method applyDefaultProps
that can help you applying
props directly to a PIXI primitive instance handling events, PIXI props and point-like values.
Here's an example to pass through every other DisplayObject props and handle prop count
separately:
import { Text } from 'pixi.js'
import { Stage, applyDefaultProps, PixiComponent } from '@paolotozzo/react-pixi'
export default PixiComponent('Counter', {
create: ({ count }) => {
return new Text(count.toString())
},
applyProps: (instance, oldProps, newProps) => {
const { count, ...oldP } = oldProps
const { count, ...newP } = newProps
// apply rest props to PIXI.Text
applyDefaultProps(instance, oldP, newP)
// set new count
instance.text = count.toString()
},
})
Consider this rotating bunny example:
./components/RotatingBunny.jsx
import { Sprite } from '@paolotozzo/react-pixi'
class RotatingBunny extends React.Component {
state = { rotation: 0 }
componentDidMount() {
this.props.app.ticker.add(this.tick)
}
componentWillUnmount() {
this.props.app.ticker.remove(this.tick)
}
tick = delta => {
this.setState(({ rotation }) => ({
rotation: rotation + 0.1 * delta,
}))
}
render() {
return <Sprite image="./bunny.png" rotation={this.state.rotation} />
}
}
There are 2 ways of accessing the PIXI.Application
instance.
- Using
AppConsumer
and pass the instance via render props:
App.jsx
import { Stage, Container, AppConsumer } from '@paolotozzo/react-pixi'
import { RotatingBunny } from './components/RotatingBunny'
export default () => (
<Stage>
<Container>
<AppConsumer>{app => <RotatingBunny app={app} />}</AppConsumer>
</Container>
</Stage>
)
- Or use a Higher Order Component:
App.jsx
import { Stage, Container, withPixiApp } from '@paolotozzo/react-pixi'
import { RotatingBunny } from './components/RotatingBunny'
const BunnyWithApp = withPixiApp(RotatingBunny)
export default () => (
<Stage>
<Container>
<BunnyWithApp />
</Container>
</Stage>
)
- Use hooks API in Functional Components
RotatingBunny.jsx
import { useApp } from '@paolotozzo/react-pixi'
function RotatingBunny(props) {
const app = useApp()
// app => PIXI.Application
return (
...
)
}
The component implements the pixi-spine plugin which enables the Spine support, including some nice extra features to make easier to work with dynamic content with Spine.
class Pixie extends React.Component {
constructor(props) {
super(props)
this.state = {
animation: 'running'
}
}
componentDidMount() {
const loader = PixiLoader.shared;
loader
.add('pixie', 'public/spine/pixie.json')
.load((ld, res) => {
this.setState({
resource: res.pixie.spineData
})
});
}
animationEnd = (event, track) => {
console.log(event)
console.log(track)
}
animationStart = (event, track) => {
if(event.animation.name === 'jump') {
this.setState(prevState => {
return {
animation: 'running'
}
})
}
}
onclick = (el) => {
this.setState({
animation: 'jump'
})
}
render() {
const { resource, animation } = this.state
return (
<Stage width={1024} height={1024} resolution = {window.devicePixelRatio || 1} options={{ backgroundColor: 0x012b30}}>
<Container x={520} y={500}>
{resource && (
<SpineContainer
mixes={[{from: 'running', to: 'jump', time: 0.2}, {from: 'jump', to: 'running', time: 0.4}]}
animations={[animation]}
x={0}
y={400}
scale={0.5}
pointerup={this.onclick}
interactive
buttonMode
resource={spineData}
events={
{
complete: this.animationEnd, // internal event
start: this.animationStart // internal event
halfwayJump: this.halfwayJump // custom timeline event
}
}>
<SpineContainer.Animation name={'jump'} play={false} track={0} />
<SpineContainer.Animation name={'running'} queue loop track={0} />
</SpineContainer>
)}
</Container>
</Stage>
)
}
}
The SpineContainer accepts all properties a PIXI DisplayObject
does, and additionally the following ones which are used to actually control the component.
Prop | Default Value | Description |
---|---|---|
spineData object |
The resource spineData loaded with the pixi loader | |
autoPlay boolean |
true | The animation starts as soon as added to the stage |
width number |
Width of the object in pixels | |
height number |
height of the object in pixels | |
events object |
All the animation events to listen to, whether they are the lifecycle ones or the ones emitted by the timelines [see AnimationStateListener] | |
skin text |
The skeleton's skin [see Skin] | |
speed number |
1 | The animation speed [see timeScale] |
mixes array |
Array of objects containing all the crossfades between animations [see mixes] | |
animations array |
Array of the animations on playing |
In order to render and control the animation the following parts of the compound component will be used.
It refers to the actual animations playable in the Spine object. See [Animation].
Prop | Default Value | Description |
---|---|---|
name string |
The name of the animation (mandatory as it indentifies the animation we want to associate with the component) | |
play boolean |
true | If the animation is playing |
trackTime number |
0 | The position in time the animation is. Default is the start time. [see trackTime] |
timeScale number |
1 | The animation speed. Default is 1 [see timeScale] |
animationEnd number |
The ending time of the animation, default is its duration [see animationEnd] | |
events object |
The animation events to listen to, whether they are the lifecycle ones or the ones emitted by the timelines [see AnimationStateListener] | |
track number |
The track index for the animation playing [see tracks] | |
loop boolean |
If the animation repeat after its duration | |
queue boolean |
If we want the animation queued for late or played immediately replacing the current one [see difference between Playback and Queuing] | |
delay number |
0 | Delaying time of the animation playing |
<SpineContainer.Animation
name={'jump'}
play
trackTime={2} // move to the 2th second of animation's duration
timeScale={2} // play it twice as fast
animationEnd={4} // end it at four seconds
track={0} //
events: {
complete: this.animationEnd, // internal event
start: this.animationStart // internal event
halfwayJump: this.halfwayJump // custom timeline event
}
loop
queue
delay={1} // start the animation after one second
/>
It refers to the Slot object. See [Slot].
A Slot component implements also a Content
component, useful to add external elements to the Slot container, therefore making them inherit all the transformations the Slot goes through. In this way you can use Spine to define the animations and then use react-pixi to add content generate dynamically, such as texts or other react-pixi components.
<SpineContainer.Slot name='txt_label_score'>
<SpineContainer.Slot.Content>
<Text text="Score" style={style} />
</SpineContainer.Slot.Content>
</SpineContainer.Slot>
It refers to the Bone object. See [Bone].
As well useful to position external elements using the bone world transform. Its properties can be also changed externally.
<SpineContainer.Bone name={'position_score'} rotation={rotation}>
<Text style={{fill:'#00ffcc'}} text="Score" />
</SpineContainer.Bone>
To access the full spine API the SpineContainer component makes available a spineElement
prop, which, together with a getSpineObject
which returns a reference to the PIXI.spine
component, exposes also some useful methods to control the animations.
import { Loader as PixiLoader } from 'pixi.js'
import spine from 'pixi-spine';
import {
Stage, Container, SpineContainer
} from '@paolotozzo/react-pixi';
import React, { Component } from 'react';
class Pixie extends React.Component {
constructor(props) {
super(props)
this.childRef = React.createRef()
this.state = {
animation: 'running'
}
}
componentDidMount() {
const loader = PixiLoader.shared;
loader
.add('pixie', 'public/spine/pixie.json')
.load((ld, res) => {
this.setState({
spineData: res.pixie.spineData
})
});
}
animationEnd = (event, track) => {
// once the animation end go back to the beginning
const spine = this.childRef.current.spineElement.getSpineObject()
const runningTrack = spine.spineData.findAnimation('running')
runningTrack.trackTime = 0
}
render() {
const { spineData, animation } = this.state
return (
<Stage width={1024} height={1024} resolution = {window.devicePixelRatio || 1} options={{ backgroundColor: 0x012b30}}>
<Container>
{spineData && (
<SpineContainer
ref={this.childRef}
animations={[animation]}
x={300}
y={700}
spineData={spineData}
events={
{
complete: this.animationEnd
}
}>
<SpineContainer.Animation name={'running'} play loop track={0} />
</SpineContainer>
)}
</Container>
</Stage>
)
}
}
Renders Root Container of a PIXI.Application
.
Props:
width
the width of the renderers view, default800
height
the height of the renderers view, default600
onMount
a callback function for the created app instanceonUnMount
a callback function when the Stage gets unmountedraf
use the internal PIXI ticker (requestAnimationFrame) to render the stage, defaulttrue
renderOnComponentChange
render stage when the Stage component updates. This is ignored ifraf
istrue
.options
see PIXI.Application options
The Stage stores the created PIXI.Application
instance to context, which can be accessed using a Provider or a Higher
Order Component.
Pass component options as props, example:
import { Sprite } from '@paolotozzo/react-pixi'
const MyComponent = () => <Sprite image=".image.png" x={100} y={200} />
The image
prop here is a short-hand for PIXI.Sprite.from()
:
import { Sprite } from '@paolotozzo/react-pixi'
const texture = new PIXI.Sprite.fromImage('./image.png')
const MyComponent = () => <Sprite texture={texture} x={100} y={200} />
# compile umd & es builds
yarn build
# compile dev builds
yarn build:dev
# compile production builds
yarn build:prod
# watch development builds
yarn build:watch
# lint code
yarn eslint
# fix linting issues
yarn eslint --fix
# test
yarn test
# watch tests
yarn test:watch