UNPKG

popmotion-pose

Version:

A declarative animation library for HTML and SVG

150 lines (100 loc) 4.58 kB
--- title: Passive values description: Learn how to create passive values that only change when others do category: vue next: vue-flip --- # Passive values Sometimes we don't want to explicitly define a state for a value, we might just want it to change whenever another value does. For instance, we might want an element to disappear as it moves beyond certain boundaries: <CodeSandbox vue id="848v06y8yj" /> For this, we can use **passive values**. In this tutorial we'll see how to define them, and how to make them respond to changes in parent values too. <TOC /> ## Defining a passive value Open the [this draggable example](https://codesandbox.io/s/848v06y8yj?module=%2Fsrc%2FApp.vue) and replace the posed component config with this: ```javascript Box: posed.div({ draggable: 'x' }); ``` The dragging motion of the element is locked to the `x` axis. We can actually lock movement to the diagonal by defining `y` as a `passive` value. The `passive` syntax will become clearer in the future but for now they're defined as tuples, like this: ```javascript const Box = posed.div({ draggable: 'x', passive: { y: ['x', v => v] } }); ``` The first item in the tuple is the name of the value we want to link to. In this case, that's `'x'`. The second item is a mapping function. This takes the output of the linked value and returns our passive value. In this example, we're simply returning `x`, and creating this motion: <CodeSandbox vue id="qzz3l8j32w" /> By using this mapping function we can start creating new effects. For instance, returning the negative of `x` creates diagonal movement in the opposite direction: ```javascript y: ['x', v => -v] ``` Or by using `Math.sin` we can make wavey behaviour: ```javascript y: ['x', v => v * Math.sin(v * 0.01)] ``` ## Changing non-numerical values So far, we've mapped two pixel values. But we can set any kind of value with any other. Long-time users of [Popmotion Pure](/pure) will recognise the signature of the mapping function. It accepts one value, and returns another. Which means we can compose this function using [Popmotion's transformers](/api/transformers). For instance, instead of `y` let's create a function that will map `x` to `backgroundColor`. For this we'll need to import four functions from `popmotion.transform`: ```javascript import transform from 'popmotion'; const { pipe, clamp, interpolate, blendColor } = transform; ``` Our steps will be: 1) Convert the output of `x` from pixels to a `0` to `1` range 2) Clamp that output to within `0` and `1` 3) Use that progress number to blend between two colors Which means our function will look like this: ```javascript backgroundColor: ['x', pipe( interpolate([-200, 200], [0, 1]), clamp(0, 1), blendColor('#FF1C68', '#198FE3') )] ``` <CodeSandbox id="ovj5wvoq2z" vue /> ## Linking to ancestors We can also link a passive value to a value in one of the poser's ancestors. Let's revist our [sidebar example](https://codesandbox.io/s/qq667ljpz4?module=%2Fsrc%2FApp.vue) from earlier. Currently, we're actively animating the children by setting poses on both the parent and the children. But, it's possible to change the opacity of the items as the `x` of their sidebar parent changes. To do this, we pass `true` as the third and final argument of the tuple. `true` says "link to my parent". Add a slower `transition` to `Sidebar`'s `'visible'` pose to help us see this in effect. ```javascript transition: { duration: 1000 } ``` Now, import `pipe` and `interpolate` from `'popmotion'`: ```javascript import { transform } from 'popmotion'; const { pipe, interpolate } = transform; ``` And replace `Item`'s config with this: ```javascript Item: posed.li({ passive: { opacity: ['x', pipe( parseFloat, interpolate([-100, 0], [0, 1]) ), true] } }); ``` As you can see, we're passing in a third parameter to the passive tuple, `true`. This says "listen to the `x` value, but do so on my immediate parent". ### Distant ancestors Using `true` is fine if we want to look just one part up the ancestor chain. But it's also possible to go much further up using `label`. By explicitly naming our posers with a `label`, we can refer to any poser in the ancestor chain. Add the label `'sidebar'` to our `Sidebar` config: ```javascript Sidebar: posed.ul({ label: 'sidebar', /* other props */ }); ``` Now replace `true` in the `Item` config with `'sidebar'`. It still works, and it will still work if you decide to put a different posed component between `Sidebar` and `Item`.