react-three-texture
TypeScript icon, indicating that this package has built-in type declarations

1.1.18 • Public • Published

Texture Tinker Tool

Table of Contents

Introduction

The 3D drawing tool ThreeJS, its React rendering solution React Three Fiber, and on its turn the R3F abstraction library Drei can be useful in making quick and beautiful 3D creations. And there are many other useful ThreeJS tools, helpers, libraries and documentations available around the web. However, on the level of drawing and handling images and textures there is not much available, except for professional designers and experts in handling expensive photo drawing and editing programs.

So I decided to create a tool where you can draw, edit, optimize and even animate your textures in an easy-to-use React environment. In a similar way popular drawing programs can handle photos and images, you can stack layers and blend them together, transform, filter and retouch them, add effects to them and draw shapes. To increase the performance, once an image, layer or texture is loaded or rendered, it is cached and reused for when the exact same input data is called a second time.

The Texture Tinker Tool is a fast, powerful, elegant and dynamic texture editor, to use together with ThreeJS and React Three Fiber. It helps you to dynamically optimize your textures thanks to the many features available in the Canvas Rendering Context interface, like filters, composite operations (blending), gradient and additional effects like bloom, alpha channel mapping, and outline effects.

Demo

Texture Tinker Tool Demo

Installation

npm install react three @react-three/fiber react-three-texture

...or...

yarn add react three @react-three/fiber react-three-texture

Prerequisites

Before you start you need to know a bit about React, ThreeJS and React Three Fiber. If you need any additional help on these topics, please refer to the following links:

Quickstart Example

Quick Example

import { TextureSet, Layer } from "react-three-texture";

const ExampleBox = () => (
  <mesh castShadow>
    <boxBufferGeometry attach="geometry" />
    <meshStandardMaterial attach="material">
    <TextureSet>
      <Layer fill="blue" />
      <Layer gradient blend="lighter" />
      <Layer src="burger.svg" scale={[0.5, 0.5]} image="center middle" blend="hard-light" />
    </TextureSet>
  </mesh>
)

Documentation

Default Texture Settings

To apply texture settings globally and for all textures defined in the texture sets, call the texture default function at the top level of your application. All the available layer settings can be defined here.

textureGlobals({
  dimensions: 512,
  filter: "contrast(85%) brightness(110%)",
});

Texture Set

A texture containing a set of layers. A map propertly can be added to determine the type of map. Also the dimensions can be defined for this texture. Additionally, all features like offset, repeat and rotation that are normally available for the basic Texture object, can also be used here. Its children can only be layers. Read more about ThreeJS Textures.

<TextureSet offset={[0.5, 0.5]}>...</TextureSet>

<TextureSet map="bump">...</TextureSet>

<TextureSet map="env" dimensions={1024}>...</TextureSet>

Map types

The following map types are available. When no map property is defined, by default the color map is used. Technically, all Material properties ending with Map can be used as a TextureSet property.

  • env
  • specular
  • displacement
  • normal
  • bump
  • roughness
  • metalness
  • alpha
  • light
  • emissive
  • clearcoat
  • clearcoatNormal
  • clearcoatRoughness
  • sheenRoughness
  • sheenColor
  • specularIntensity
  • specularColor
  • thickness
  • transmission
  • ao

The useTextureSet hook

Alternatively, the texture can be built by using the useTextureSet hook. The hook returns a CanvasTexture object, which can directly be used as a texture map. When applying the texture to a material, be sure to update the material, using material.needsUpdate = true.

const texture = useTextureSet(
  <>
    <Layer fill="darkBlue" />
    <Layer circle={[0.5, 0.5, 0.4]} color="lightBlue" />
  </>,
  1024
);

<meshStandardMaterial attach="material" map={texture} bumpMap={texture} />;
arguments example type description
layer <Layer fill="red" /> JSX A React Node object containing all the layers
dimensions? 512 number The width and height of the texture, in pixels
isEnvMap? true boolean When the type is an environment map, set this to true to be sure the PMREM is correctly generated.

Layer

Layers are like a stack of transparent film paper, used in old animated movies. The bottom layer will be shown at the back, then the second layer, etcetera. All layers together will form one single texture.

<Layer {...props1} /> // Will be drawn first and is displayed in the back
<Layer {...props2} />
<Layer {...props3} />
<Layer {...props4} /> // Will be drawn last and is displayed in the front

Src

Src Example

Will directly load an image source onto the texture. All loaded images will be cached, to increase performance. I.e. when the same image is used somewhere else there will be no reloading. It is also possible to load SVG files, and even external sources can be used. The root folder for the source files is /src/assets.

<Layer src="image.png" />

<Layer src="image.svg" />

<Layer src="https://images.unsplash.com/photo-1581373449483-37449f962b6c" />

Dimensions

Dimension Example

You can set the width and the height (in pixels) for the layer. By default the texture dimensions are 512x512 pixels. Note, that if you wish to set the dimensions higher than the Default Texture Settings, you also need to increase it there.

<Layer src="image.png" dimensions={64} />

<Layer src="image.png" dimensions={256} />

Basic transformations

Transformation Example

All layers can separately be moved, scaled and rotated, relative to the texture. It is also possible to flip (mirror) the layer either horizontally or vertically

<Layer src="image.png" position={[0.5, 0.5]} />

<Layer src="image.png" position={[0.2, 0.2]} scale={[0.8, 0.8]} />

<Layer src="image.png" rotation={Math.PI * 0.5} />
property default value arguments description
position [0, 0] x, y The position of the layer.
scale [1, 1] w, h The scale of the layer.
rotation 0 rad The rotation of the layer in radians. The rotation will be done relative to the center of the texture.
flipX false boolean Flip the layer horizontally.
flipY false boolean Flip the layer vertically.

Fit transformations

Image Example

Layers and images that are smaller or larger than the layer can be fit into the layer as we like, without getting deformed. We can choose to fit a rectangular sized image on a square sized texture, while keeping the aspect ratio of the image. Also, when the image is too big to fit on the texture we can choose to position the image to a side.

<Layer src="image.png" image /> // "size-max center middle"

<Layer src="image.png" image="size-max" />

<Layer src="image.png" image="size-x center middle" />
string description
size-fill ** Image will be resized to match the given dimension, and may stretch it to fit.
size-max * Will fit the image to whichever side (width or height) is smaller, and keeping the image's aspect ratio. This may clip the image.
size-min Will fit the image to whichever side (width or height) is larger, and keeping the image's aspect ratio.
size-x Will fit the image to its width, while keeping the image's aspect ratio.
size-y Will fit the image to its height, while keeping the image's aspect ratio.
size-none Will not fit the image and will keep its original size.
top ** Will align the image to the top.
right Will align the image to the right.
bottom ** Will align the image to the bottom.
left Will align the image to the left.
center * Will horizontally align the image to the center.
middle * Will vertically align the image to the middle.

* default, when defining the property "image" without a value

** default, when not defining the property "image"

Repeat

Repeat Example

Instead of a single render of a layer, it is also possible to repeat it as a pattern. When scaling the layer down, the pattern will become visible.

<Layer src="image.png" scale={[0.2, 0.2]} repeat />

Color

Outline Example

Overwrite the colors in your layers with a single color. Useful to colorize your monochrome SVG images. When setting the alpha channel lower than 100% it will blend the original colors with the overwritten color.

<Layer src="image.png" color /> // "white"

<Layer src="image.png" color="red" />

<Layer src="image.png" color="#ff000080" />

<Layer src="image.png" color="#rgba(255, 255, 0, 0.7)" />

Fill

Fill Example

Will fill the layer with a single color. The alpha channel can also be used to make the fill color (semi-)transparent.

<Layer fill /> // "black"

<Layer fill="darkBlue" />

<Layer fill="#00ff00" />

<Layer fill="rgb(128, 0, 128)" />

Gradient

Gradient Example

Gradients is an operation to smoothly blend two or more colors into each other. See the linear gradient page and the radial gradient page of the MDN web docs for more information.

<Layer gradient /> // { type: "linear", from: [0, 0], to: [0, 1], stops: [[0, "white"], [1, "black"]] }

<Layer gradient={{ stops: [[0, "#008800"], [1, "#88ff88"]] }} />

<Layer
  gradient={{
    type: "radial",
    from: [0.5, 0.5, 0],
    to: [0.5, 0.5, 0.75],
    stops: [
      [0, "yellow"],
      [0.4, "red"],
      [1, "blue"],
    ],
  }}
/>
key default value arguments description
type "linear" linear or radial the type of gradient. Can be linear or radial.
from [0, 0] or [0.5, 0.5, 0] x, y, r? The starting point of the gradient. The radius r only applies for radial gradients.
to [0, 1] or [0.5, 0.5, 1] x, y, r? The end point of the gradient. The radius r only applies for radial gradients.
stops [[0, "white"], [1, "black"]] index, color The stop index and color between the start and the end point. Multiply stops and indexes can be used.

Image smoothing**

Image smoothing Example

Turn off image smoothing for the layer (the "nearest neighbour" algorithm), when scaling/fitting the image. See the image smoothing page of the MDN web docs for more information.

<Layer img="image.png" nearest /> // true

** Unfortunately, currently image smoothing is not working in most of the browsers

Shadow and glow

Shadow and glow Example

Apply a shadow (or glow) effect to the layer. See the shadow page of the MDN web docs for more information.

<Layer src="image.svg" shadow /> // { color: "black", blur: 20, offset: [0, 0] }

<Layer src="image.svg" shadow={{ color: "white", blur: 30, offset: [-5, 5] }} />
key default value arguments description
color "black" The color of the effect
blur 20 0 to x The blur radius of the effect
offset [0, 0] x, y The position offset from the center of the effect.

Outline

Outline Example

You can give your image an outline effect. Other than shadow or glow, outline will not be smooth and does not have an offset.

<Layer src="image.svg" outline /> // { color: "black", size: 1, detail: 8 }

<Layer src="image.svg" outline={{ color: "#ff00ff", size: 10, detail: 10 }} />
key default value arguments description
color "black" The color of the outline.
size 1 1 to x The thickness of the outline.
quality 8 1 to x The detail of the outline effect. The higher the more instances of the effect will be drawn.

Bloom

Outline Example

Bloom is an effect, well known to photographers and videogame developers, to intensify the light parts of an image with a bleeding effect of brightness.

<Layer src="image.png" bloom /> // { size: 30, strength: 0.4, softness: 0.7, detail: 10, darken: false }

<Layer src="image.png" bloom={{ size: 10, strength: 0.5, softness: 0.3 }} />

<Layer src="image.png" bloom={{ size: 10, strength: 0.5, detail: 4, darken: true }} />
key default value arguments description
size 30 0 to x Indicates the radius of the bloom effect. The higher the larger the scale of the effect.
strength 0.4 0.0 to 1.0 The power of the effect. The higher the value the brighter the effect will remain, moving further away from the center.
softness 0.7 0.0 to 1.0 The fallout of the effect. The higher the value the faster the effect will fade, moving further away from the center.
detail 10 1 to x The detail of the bloom effect. The higher the more instances of the effect will be drawn.
darken false boolean Optionally darken the effect.

Filters

Filters Example

You can change brightness, contrast, saturation, blur, and many other filters. See the filter page of the MDN web docs for more information.

<Layer src="image.png" filter="brightness(250%)" />

<Layer src="image.png" filter="blur(5px)" />

<Layer src="image.png" filter="saturate(0)" />

<Layer src="image.png" filter="hue-rotate(180deg) sepia(0.5)" />

Blending

Blending Example

Blending is a method to mix the colors of two or more layers together by a mathematic operation. Also it is possible to change the transparency by color. See the composition operation page of the MDN web docs for more information. Blending only has effect with two or more layers.

<Layer color="red">
<Layer src="image.svg" color="blue" blend="lighter" />

<Layer gradient />
<Layer src="image.png" blend="overlay" />

<Layer src="image.png" />
<Layer src="image.svg" blend="xor" />

Alpha

Alpha Example

The alpha (transparency) channel can be altered in many ways. It is possible to alter the alpha channel in such a way that darker colors become more transparent. Or vice versa.

<Layer src="image.png" alpha={0.5} /> // level: 0.5

<Layer src="image.png" alpha /> // { level: 1, power: 1, offset: 0, reverse: false }

<Layer src="image.png" alpha={{ level: 0.5}} />

<Layer src="image.png" alpha={{ power: 10, offset: 0.5, reverse: true }} />
key default value arguments description
level 1 0 to 1 The opacity of the layer. 0 means transparent, 1 means opaque.
power 1 0 to x The power of the alpha map levels. Higher power means more difference between the alpha levels.
offset 0 0 to 1 Shift the alpha levels down or up, making respectively less or more colors transparent.
reverse false boolean Reverse the alpha channels to make lighter colors more transparent.

Shapes

Shapes Example

A variety of basic shapes can be drawn on a layer. Think of lines, curved lines, circles or rectanges.

<Layer line={[0.1, 0.1, 0.9, 0.9, 0.9, 0.1]} shapeThickness={32} shapeRounded />

<Layer curved={[0.1, 0.1, 0.9, 0.9, 0.9, 0.1]} shapeThickness={16} />

<Layer circle={[0.5, 0.5, 0.7, 0.4]} />

<Layer rect={[0.2, 0.3, 0.6, 0.4, 0.1]} />
property default value arguments description
shapeThickness 0 0 to x The thickness of the shape. Will work with all shapes. A shape thickness of 0 will fill the shape.
shapeRounded false boolean Will make lines rounded in the corners, and lines and curved lines rounded on the start and end points.
line x1, y1, x2, y2, ... Draw a line from one coordinate to the next. Ending with the same point as the starting point will close the path.
curved x1, y1, x2, y2, x3, y3, x4?, y4? Draw a curved line from one coordinate to the next. Can be a bezier curve (3 points), or a quadratic curve (4 points).
circle x, y, rx, ry?, rotation?, start?, end?, cc? Will draw a circle at a certain point x, y and a radius of rx. When a second radius ry is defined the circle will become an ellipse, and with the start and end point defined the circle is only partly drawn. Optionally the circle can be drawn counter-clockwise with cc.
rect x, y, w, h?, rounded? Will draw a rectangle at a certain point x, y and a fixed size of w and and optional height of h. It is also possible to give the rectangle rounded corners.

Text

Texts can be created in all sorts of types, styles, colors and sizes. Font faces can also be retreived from files or a url. The property shapeThickness can also be used here, to create outlined text.

<Layer text={{ value: "Lorem ipsum" }}  />

<Layer text={{ value: "This is a nice text-ure", font: "Arial", weight: "bold" }} color="red" position={[0.5, 0.5]} scale={[0.1, 0.1]} />

<Layer text={{
    value: "Texts can be created in all sorts of types, styles, colors and sizes.",
    font: "fonts/myLovelyFont.otf",
    width: {0.8}
    height: {1.0}
    align: "center",
    base: "middle"
  }}
  position={[0.5, 0.5]}
  scale={[0.1, 0.1]}
/>
key default value arguments description
value string The text value.
font serif string The font family of the text. System fonts can be used, but also a path to a custom font file, or even a url can be used to retreive a font face. The root folder for the source files is /src/assets.
style string The font style. For example italic or oblique.
weight string The font weight. Examples are bold or lighter, but you can use number values to define the strength of the weight.
width 0 number The maximum width of the text. When the text exceeds this width, the words will wrap down to the next line. The value is a factor of the full width of the layer. A value of 0 will not wrap the text.
height 1.3 number The line height of the text. The number is a multiplier for the height of the font.
align center left, center of right The horizontal alignment of the text. The text will orientate around the position of the layer.
base middle top, middle or bottom The vertical alignment of the text. The text will orientate around the position of the layer.

Seamless texture

Seamless Example

Seamless textures are textures that can be repeated without leaving an ugly "seam" between the copies of the image. Finding a well created seamless image on the internet can be either expensive or difficult. And manually adjusting an image in a photo editing tool can sometimes be a time-consuming and difficult job. Here, without much effort it is possible to make a texture seamless.

<Layer seamless /> //
key default value arguments description
offset [0.3, 0.3] 0 to 1 The offset of the seamless edges.
size [0.2, 0.2] 0 to 0.5 The size of the seamless edges.
both false boolean Both edges (x and y) are affected by the offset.
flipX false boolean Flip the seamless edge horizontally.
flipY false boolean Flip the seamless edge vertically.
alphaOffset 0 0 to 1 The edges are affected by an alpha channel offset. Is disabled when the value is 0. See also alpha.
alphaReverse false boolean When alphaOffset is larger than 0, set this to reverse the alpha channel. See also alpha.

Q and A

Disabling the smooth effect does not always work.

When using the nearest property, the texture's smooth effect is not always disabled. Then try to use the ThreeJS texture.magFilter = THREE.NearestFilter value instead.

Known issues and Roadmap

New layer effects

  • [ ] Flip layer (x, y, both)
  • [x] Repeat layer and pattern drawing
  • [x] Drawing shapes and text
  • [ ] Inner shadow/glow
  • [ ] Bevel and emboss effect
  • [ ] Image scaling while maintaining aspect ratio
  • [ ] Seamless rendering
  • [ ] Procedural rendering
    • [ ] clouds
    • [ ] perlin noise
    • [ ] distortion
    • [ ] walls
  • [ ] Sharpen effect
  • [x] Initial setup to add more sizes than 512x512

Technical upgrades

  • [x] Migration to Typescript
  • [ ] Adding JSDoc
  • [ ] Use "offscreenCanvas" for better performance
  • [ ] Layer groups

Known bugs

  • [x] Cached large images should keep their original size, instead of scaling them down to 512x512.

  • [ ] to do
  • [x] completed

Package Sidebar

Install

npm i react-three-texture

Weekly Downloads

9

Version

1.1.18

License

MIT

Unpacked Size

303 kB

Total Files

107

Last publish

Collaborators

  • jnieberg