2.1.4 • Published 5 months ago

@sourcetoad/react-native-sketch-canvas v2.1.4

Weekly downloads
-
License
MIT
Repository
github
Last release
5 months ago

react-native-sketch-canvas

Forked from terrylinla/react-native-sketch-canvas as package abandoned in 2018.


A React Native component for drawing by touching on both iOS and Android, with TypeScript support.

    

Features

  • Supports New Architecture
  • Support iOS and Android
  • Full TypeScript support
  • Stroke thickness and color are changeable while drawing
  • Can undo strokes one by one
  • Can serialize path data to JSON for syncing between devices
  • Save drawing to a non-transparent image (png or jpg) or a transparent image (png only)
  • Use vector concept - sketches won't be cropped in different sizes of canvas
  • Support translucent colors and eraser
  • Support drawing on an image
  • High performance
  • Can draw multiple canvases in the same screen
  • Can draw multiple multiline text on canvas
  • Support for custom UI components
  • Permission handling for Android image saving

Installation


Install from yarn (only support RN >= 0.40)

yarn install @sourcetoad/react-native-sketch-canvas

Usage


● Using without UI component (for customizing UI)

import React from 'react';
import {
  AppRegistry,
  StyleSheet,
  View,
} from 'react-native';

import { SketchCanvas } from '@sourcetoad/react-native-sketch-canvas';

export default function Example() {
  return (
    <View style={styles.container}>
      <View style={{ flex: 1, flexDirection: 'row' }}>
        <SketchCanvas
          style={{ flex: 1 }}
          strokeColor={'red'}
          strokeWidth={7}
        />
      </View>
    </View>
  );
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    justifyContent: 'center',
    alignItems: 'center',
    backgroundColor: '#F5FCFF',
  },
});

AppRegistry.registerComponent('example', () => Example);

Properties


PropTypeDescription
styleobjectStyles to be applied on canvas component
strokeColorstringSet the color of stroke, which can be #RRGGBB or #RRGGBBAA. If strokeColor is set to #00000000, it will automatically become an eraser. NOTE: Once an eraser path is sent to Android, Android View will disable hardware acceleration automatically. It might reduce the canvas performance afterward.
strokeWidthnumberThe thickness of stroke
onStrokeStartfunctionAn optional function which accepts 2 arguments x and y. Called when user's finger touches the canvas (starts to draw)
onStrokeChangedfunctionAn optional function which accepts 2 arguments x and y. Called when user's finger moves
onStrokeEndfunctionAn optional function called when user's finger leaves the canvas (end drawing)
onSketchSavedfunctionAn optional function which accepts 2 arguments success and path. If success is true, image is saved successfully and the saved image path might be in second argument. In Android, image path will always be returned. In iOS, image is saved to camera roll or file system, path will be set to null or image location respectively.
onPathsChangefunctionAn optional function which accepts 1 argument pathsCount, which indicates the number of paths. Useful for UI controls.
userstringAn identifier to identify who draws the path. Useful when undo between two users
touchEnabledboolIf false, disable touching. Default is true.
localSourceImageobjectRequire an object (see below) which consists of filename, directory(optional) and mode(optional). If set, the image will be loaded and display as a background in canvas.
permissionDialogTitlestringAndroid Only: Provide a Dialog Title for the Image Saving PermissionDialog. Defaults to empty string if not set
permissionDialogMessagestringAndroid Only: Provide a Dialog Message for the Image Saving PermissionDialog. Defaults to empty string if not set
onGenerateBase64functionAn optional function which accepts 1 argument result containing the base64 string of the canvas. Called when getBase64() is invoked.
onCanvasReadyfunctionAn optional function called when the canvas is ready for interaction.

Methods


MethodDescription
clear()Clear all the paths
undo()Delete the latest path. Can undo multiple times.
addPath(path)Add a path (see below) to canvas.
deletePath(id)Delete a path with its id
save(imageType, transparent, folder, filename, includeImage, cropToImageSize)Save image to camera roll or filesystem. If localSourceImage is set and a background image is loaded successfully, set includeImage to true to include background image and set cropToImageSize to true to crop output image to background image.Android: Save image in imageType format with transparent background (if transparent sets to True) to /sdcard/Pictures/folder/filename (which is Environment.DIRECTORY_PICTURES).iOS: Save image in imageType format with transparent background (if transparent sets to True) to camera roll or file system. If folder and filename are set, image will save to temporary directory/folder/filename (which is NSTemporaryDirectory())
getPaths()Get the paths that drawn on the canvas
getBase64(imageType, transparent, includeImage, includeText, cropToImageSize)Get the base64 string of the canvas. The result will be sent through the onGenerateBase64 event handler. Parameters:- imageType: "png" or "jpg"- transparent: whether to include transparency- includeImage: whether to include background image- includeText: whether to include text- cropToImageSize: whether to crop to background image size

Constants


ConstantDescription
MAIN_BUNDLEAndroid: empty string, '' iOS: equivalent to [NSBundle mainBundle bundlePath]
DOCUMENTAndroid: empty string, '' iOS: equivalent to NSDocumentDirectory
LIBRARYAndroid: empty string, '' iOS: equivalent to NSLibraryDirectory
CACHESAndroid: empty string, '' iOS: equivalent to NSCachesDirectory

● Using with build-in UI components

import React from 'react';
import {
  AppRegistry,
  StyleSheet,
  Text,
  View,
} from 'react-native';

import RNSketchCanvas from '@sourcetoad/react-native-sketch-canvas';

export default function Example() {
  return (
    <View style={styles.container}>
      <View style={{ flex: 1, flexDirection: 'row' }}>
        <RNSketchCanvas
          containerStyle={{ backgroundColor: 'transparent', flex: 1 }}
          canvasStyle={{ backgroundColor: 'transparent', flex: 1 }}
          defaultStrokeIndex={0}
          defaultStrokeWidth={5}
          closeComponent={<View style={styles.functionButton}><Text style={{color: 'white'}}>Close</Text></View>}
          undoComponent={<View style={styles.functionButton}><Text style={{color: 'white'}}>Undo</Text></View>}
          clearComponent={<View style={styles.functionButton}><Text style={{color: 'white'}}>Clear</Text></View>}
          eraseComponent={<View style={styles.functionButton}><Text style={{color: 'white'}}>Eraser</Text></View>}
          strokeComponent={color => (
            <View style={[{ backgroundColor: color }, styles.strokeColorButton]} />
          )}
          strokeSelectedComponent={(color, index, changed) => {
            return (
              <View style={[{ backgroundColor: color, borderWidth: 2 }, styles.strokeColorButton]} />
            )
          }}
          strokeWidthComponent={(w) => {
            return (<View style={styles.strokeWidthButton}>
              <View  style={{
                backgroundColor: 'white', marginHorizontal: 2.5,
                width: Math.sqrt(w / 3) * 10, height: Math.sqrt(w / 3) * 10, borderRadius: Math.sqrt(w / 3) * 10 / 2
              }} />
            </View>
          )}}
          saveComponent={<View style={styles.functionButton}><Text style={{color: 'white'}}>Save</Text></View>}
          savePreference={() => {
            return {
              folder: 'RNSketchCanvas',
              filename: String(Math.ceil(Math.random() * 100000000)),
              transparent: false,
              imageType: 'png'
            }
          }}
        />
      </View>
    </View>
  );
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    justifyContent: 'center',
    alignItems: 'center',
    backgroundColor: '#F5FCFF',
  },
  strokeColorButton: {
    marginHorizontal: 2.5,
    marginVertical: 8,
    width: 30,
    height: 30,
    borderRadius: 15,
  },
  strokeWidthButton: {
    marginHorizontal: 2.5,
    marginVertical: 8,
    width: 30,
    height: 30,
    borderRadius: 15,
    justifyContent: 'center',
    alignItems: 'center',
    backgroundColor: '#39579A'
  },
  functionButton: {
    marginHorizontal: 2.5,
    marginVertical: 8,
    height: 30,
    width: 60,
    backgroundColor: '#39579A',
    justifyContent: 'center',
    alignItems: 'center',
    borderRadius: 5,
  }
});

AppRegistry.registerComponent('example', () => Example);

Properties


PropTypeDescription
containerStyleobjectStyles to be applied on container
canvasStyleobjectStyles to be applied on canvas component
onStrokeStartfunctionSee above
onStrokeChangedfunctionSee above
onStrokeEndfunctionSee above
onPathsChangefunctionSee above
onClosePressedfunctionAn optional function called when user taps closeComponent
onUndoPressedfunctionAn optional function that accepts a argument id (the deleted id of path) and is called when user taps "undo"
onClearPressedfunctionAn optional function called when user taps clearComponent
userstringSee above
closeComponentcomponentAn optional component for closing
eraseComponentcomponentAn optional component for eraser
undoComponentcomponentAn optional component for undoing
clearComponentcomponentAn optional component for clearing
saveComponentcomponentAn optional component for saving
strokeComponentfunctionAn optional function which accepts 1 argument color and should return a component.
strokeSelectedComponentfunctionAn optional function which accepts 3 arguments color, selectedIndex, isColorChanged and should return a component. isColorChanged is useful for animating when changing color. Because rerendering also calls this function, we need isColorChanged to determine whether the component is rerendering or the selected color is changed.
strokeWidthComponentfunctionAn optional function which accepts 1 argument width and should return a component.
strokeColorsarrayAn array of colors. Example: [{ color: '#000000' }, {color: '#FF0000'}]
defaultStrokeIndexnumbberThe default index of selected stroke color
defaultStrokeWidthnumberThe default thickness of stroke
minStrokeWidthnumberThe minimum value of thickness
maxStrokeWidthnumberThe maximum value of thickness
strokeWidthStepnumberThe step value of thickness when tapping strokeWidthComponent.
savePreferencefunctionA function which is called when saving image and should return an object (see below).
onSketchSavedfunctionSee above
onCanvasReadyfunctionAn optional function called when the canvas is ready for interaction.

Methods


MethodDescription
clear()See above
undo()See above
addPath(path)See above
deletePath(id)See above
save()

Constants


ConstantDescription
MAIN_BUNDLESee above
DOCUMENTSee above
LIBRARYSee above
CACHESSee above

Background Image


To use an image as background, localSourceImage(see below) reqires an object, which consists of filename, directory(optional) and mode(optional). Note: Because native module cannot read the file in JS bundle, file path cannot be relative to JS side. For example, '../assets/image/image.png' will fail to load image.

Typical Usage

  • Load image from app native bundle
    • Android:
      1. Put your images into android/app/src/main/res/drawable.
      2. Set filename to the name of image files with or without file extension.
      3. Set directory to ''
    • iOS:
      1. Open Xcode and add images to project by right-clicking Add Files to [YOUR PROJECT NAME].
      2. Set filename to the name of image files with file extension.
      3. Set directory to MAIN_BUNDLE (e.g. RNSketchCanvas.MAIN_BUNDLE or SketchCanvas.MAIN_BUNDLE)
  • Load image from camera
    1. Retrieve photo complete path (including file extension) after snapping.
    2. Set filename to that path.
    3. Set directory to ''

Content Mode

  • AspectFill
  • AspectFit (default)
  • ScaleToFill

Objects


SavePreference object

{
  folder: 'RNSketchCanvas',
  filename: 'image',
  transparent: true,
  imageType: 'jpg',
  includeImage: true,
  includeText: false,
  cropToImageSize: true
}
PropertyTypeDescription
folder?stringAndroid: the folder name in Pictures directoryiOS: if filename is not null, image will save to temporary directory with folder and filename, otherwise, it will save to camera roll
filename?stringthe file name of imageiOS: Set to null to save image to camera roll.
transparentbooleansave canvas with transparent background, ignored if imageType is jpg
imageTypestringimage file formatOptions: png, jpg
includeImage?booleanSet to true to include the image loaded from LocalSourceImage. (Default is true)
includeText?booleanSet to true to include the text drawn from Text. (Default is true)
cropToImageSize?booleanSet to true to crop output image to the image loaded from LocalSourceImage. (Default is false)

Path object

{
  drawer: 'user1',
  size: { // the size of drawer's canvas
    width: 480,
    height: 640
  },
  path: {
    id: 8979841, // path id
    color: '#FF000000', // ARGB or RGB
    width: 5,
    data: [
      "296.11,281.34",  // x,y
      "293.52,284.64",
      "290.75,289.73"
    ]
  }
}

LocalSourceImage object

{
  filename: 'image.png',  // e.g. 'image.png' or '/storage/sdcard0/Pictures/image.png'
  directory: '', // e.g. SketchCanvas.MAIN_BUNDLE or '/storage/sdcard0/Pictures/'
  mode: 'AspectFill'
}
PropertyTypeDescriptionDefault
filenamestringthe fold name of the background image file (can be a full path)
directory?stringthe directory of the background image file (usually used with constants)''
mode?booleanSpecify how the background image resizes itself to fit or fill the canvas.Options: AspectFill, AspectFit, ScaleToFillAspectFit

CanvasText object

{
  text: 'TEXT',
  font: '',
  fontSize: 20,
  fontColor: 'red',
  overlay: 'TextOnSketch',
  anchor: { x: 0, y: 1 },
  position: { x: 100, y: 200 },
  coordinate: 'Absolute',
  alignment: 'Center',
  lineHeightMultiple: 1.2
}
PropertyTypeDescriptionDefault
textstringthe text to display (can be multiline by \n)
font?stringAndroid: You can set font to fonts/[filename].ttf to load font in android/app/src/main/assets/fonts/ in your Android projectiOS: Set font that included with iOS
fontSize?numberfont size12
fontColor?stringtext colorblack
overlay?stringSet to TextOnSketch to overlay drawing with text, otherwise the text will be overlaid with drawing.Options: TextOnSketch, SketchOnTextSketchOnText
anchor?objectSet the origin point of the image. (0, 0) to (1, 1). (0, 0) and (1, 1) indicate the top-left and bottom-right point of the image respectively.{ x: 0, y: 0 }
positionobjectSet the position of the image on canvas. If coordinate is Ratio, (0, 0) and (1, 1) indicate the top-left and bottom-right point of the canvas respectively.{ x: 0, y: 0 }
coordinate?stringSet to Absolute and Ratio to treat position as absolute position (in point) and proportion respectively.Options: Absolute, RatioAbsolute
alignment?stringSpecify how the text aligns inside container. Only work when text is multiline text.Left
lineHeightMultiple?numberMultiply line height by this factor. Only work when text is multiline text.1.0

Example


The source code includes 7 examples, using build-in UI components, using with only canvas, and sync between two canvases.

Check full example app in the example folder

Jest Setup

If you're using Jest in your project, you'll need to mock the TurboModule registry. Add the following to your Jest setup file:

jest.mock('react-native/Libraries/TurboModule/TurboModuleRegistry', () => {
  const turboModuleRegistry = jest.requireActual(
    'react-native/Libraries/TurboModule/TurboModuleRegistry'
  );
  return {
    ...turboModuleRegistry,
    getEnforcing: (name) => {
      // List of TurboModules libraries to mock
      const modulesToMock = ['SketchCanvasModule'];
      if (modulesToMock.includes(name)) {
        return {
          getConstants: jest.fn(() => ({
            MainBundlePath: 'test',
            NSDocumentDirectory: 'test',
            NSLibraryDirectory: 'test',
            NSCachesDirectory: 'test',
          })),
        };
      }
      return turboModuleRegistry.getEnforcing(name);
    },
  };
});

This mock ensures that the native module constants are available during testing.

1.3.2

10 months ago

1.3.1

11 months ago

2.1.2

6 months ago

2.1.1

6 months ago

2.1.4

5 months ago

2.1.3

5 months ago

2.1.0

7 months ago

2.0.0

7 months ago

1.3.0

1 year ago

1.2.0

1 year ago

1.2.1

1 year ago

1.1.2

2 years ago

1.1.1

2 years ago

1.0.4

2 years ago

1.0.3

3 years ago

1.0.2

3 years ago