Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/fartinmartin/style-parser-plugin-wrap


https://github.com/fartinmartin/style-parser-plugin-wrap

Last synced: 11 days ago
JSON representation

Awesome Lists containing this project

README

        

# style-parser-plugin-wrap

> [!NOTE]
> This plugin is under development!

An example plugin for [`@motiondeveloper/style-parser`](https://github.com/motiondeveloper/style-parser), specifically for https://github.com/motiondeveloper/style-parser/pull/4.

Download [latest release](https://github.com/fartinmartin/style-parser-plugin-wrap/releases), requires `FontMetrics` for each font your text layer supports. Get `FontMetrics` via the [web tool](https://fartinmartin.github.io/style-parser-plugin-wrap/)! Example use:

```js
const { parse, createTextStyle } = footage('style-parser.jsx').sourceData.get();
const { normalizeText, wrap } = footage('style-parser-plugin-wrap.jsx').sourceData;

const { RobotoRegular, RobotoBold, RobotoItalic } = footage('font-metrics.jsx').sourceData;

const inputLayer = thisComp.layer('Input');
const inputText = inputLayer.text.sourceText.value;

const styles = {
font: 'Roboto-Regular',
fontSize: 40,
leading: 48,
};

parse(normalizeText(inputText), {
textStyle: createTextStyle(styles),
fontMap: {
regular: 'Roboto-Regular',
bold: 'Roboto-Bold',
italic: 'Roboto-Italic',
},
plugins: [
wrap({
maxWidth: effect('Max Width')(1).value,
styles,
fontMetrics: {
'Roboto-Regular': RobotoRegular,
'Roboto-Bold': RobotoBold,
'Roboto-Italic': RobotoItalic,
},
}),
],
});
```

---

## What

This plugin wraps the `parsed.text` string to a specified width, inserting line breaks and handling indentation for hanging bullets.

It calculates the width of each character based on the provided `FontMetrics` and the `font` and `fontSize` values at the character's index.

If the plugin adds characters to `parsed.text` (e.g. for hanging bullets) the appropriate `Tansforms` in `parsed.transforms` will be updated to reflect their new index values.

> [!NOTE]
> Currently, "hanging bullets" require your bullet lines to start with the `•` character. Use the `normalizeText` utility on your input text (before passing to `style-parser/parse()`) to convert markdown-like `*` and `-` lines to `•`—see example above for details.

To create `FontMetrics` for your specific fonts, use the [web tool](https://fartinmartin.github.io/style-parser-plugin-wrap/) to generate `jsx` files and use those in expressions as necessary. Each key on `fontMetrics` should be a "PostScript font name" [as recognized by After Effects](https://helpx.adobe.com/after-effects/using/expressions-text-properties.html#PostScriptfontexpressionmenu):

```js
const { RobotoRegular, RobotoBold, RobotoItalic } = footage('font-metrics.jsx').sourceData;

// later...
wrap({
// ...
fontMetrics: {
'Roboto-Regular': RobotoRegular,
'Roboto-Bold': RobotoBold,
'Roboto-Italic': RobotoItalic,
},
});
```

## Why

After Effects has two types of text layers: point text and paragraph text (sometimes called 'box text' or 'area text'). Traditionally, if you want your text to automatically wrap at a given width, you would use paragraph text:

> When you enter _point text_, each line of text is independent—the length of a line increases or decreases as you edit the text, but it doesn’t wrap to the next line.

> When you enter _paragraph text_, the lines of text wrap to fit the dimensions of the bounding box. You can enter multiple paragraphs and apply paragraph formatting.

Creating and editing text layers on Adobe.com

This works great until you need to know when your text is wrapping.

For example, [`@motiondeveloper/aefunctions`](https://github.com/motiondeveloper/aefunctions) provides [`layerRect()`](https://github.com/motiondeveloper/aefunctions/blob/a6a777177fe0e0acb5451a0f0f265fecd41153a1/src/index.ts#L286), a utility to get the height of a text layer using the font's `xHeight` instead of the layer's `sourceRectAtTime()`:

```js
const { fontSize, leading } = thisLayer.text.sourceText.style;
const xHeight = fontSize / 2;
const totalHeight = xHeight + leading * (numLines - 1);
```

Tim Haywood does [an excellent job explaining](https://motiondeveloper.com/blog/dealing-with-descenders) why this is desired.

However! You may have noticed above that in order to calculate the `totalHeight` we need to know the number of lines. `layerRect()` does this by [counting the total number](https://github.com/motiondeveloper/aefunctions/blob/a6a777177fe0e0acb5451a0f0f265fecd41153a1/src/index.ts#L353) of return, new line, and "end of text" characters in our text:

```js
const numLines = (text) => Math.max(text.split(/[^\r\n\3]*/gm).length - 1, 0);
```

If our text layer is a _paragraph text_ layer, After Effects will wrap our text without the use of these characters and we will have no way of knowing how many lines are rendered or where in our text line breaks occur!

Thus, if we want to use `layerRect()`, we should use _point text_. With point text we can manually enter return characters and our calculated height will react accordingly.

However, we're likely using `layerRect()` inside of a template. It would be ideal if users of our template didn't have to be responsible for adding manual line breaks. With `style-parser-plugin-wrap` they won't have to!

Instead of relying on After Effects to automatically wrap our text or on our users to manually add line breaks, `wrap()` will calculate the width of each line based on the character width at each character index. It does this by looking up the `FontMetric` for the character's `font` and scaling it to match the character's `fontSize`. This means we can use _point text_ and automatically add line break characters—the best of both worlds!

This functionality _could_ act on `textLayer.text.sourceText` directly, and therefore NOT be a "plugin" for `style-parser`. However it would need to poll `getStyleAt(index)` for each character, which in early tests seemed very slow! Since `style-parser` tracks styles at index values via the `parsed.transforms` array before they're applied to a `TextStyle` object, we can make these calculations much faster as a `style-parser` plugin.