import { Platform } from '@unimodules/core'; import * as React from 'react'; import { findDOMNode } from 'react-dom'; import { LayoutChangeEvent, PixelRatio, StyleSheet, View, ViewProps } from 'react-native'; import createElement from 'react-native-web/dist/exports/createElement'; function getElement(component) { try { return findDOMNode(component); } catch (e) { return component; } } function setRef(refProp: React.Ref, ref: T | null) { if (!refProp) return; if (typeof refProp === 'function') { refProp(ref); } else if ('current' in refProp) { // @ts-ignore refProp.current = ref; } } const Canvas = React.forwardRef( (props: React.ComponentProps, ref: React.Ref) => createElement('canvas', { ...props, ref }) ); const CanvasWrapper: React.FunctionComponent; }> = ({ pointerEvents, children, ...props }) => { const [size, setSize] = React.useState<{ width: number; height: number } | null>(null); const ref = React.useRef(null); const _canvasRef = React.useRef(null); function updateCanvasSize(): void { const canvas = _canvasRef.current; // eslint-disable-next-line no-undef if (typeof HTMLCanvasElement !== 'undefined' && canvas instanceof HTMLCanvasElement) { const size = getSize(); const scale = PixelRatio.get(); canvas.style.width = `${size.width}px`; canvas.style.height = `${size.height}px`; canvas.width = size.width * scale; canvas.height = size.height * scale; } } function getSize(): { width: number; height: number } { if (size) { return size; } else if (!ref.current || !Platform.isDOMAvailable) { return { width: 0, height: 0 }; } const element = getElement(ref.current); const { offsetWidth: width = 0, offsetHeight: height = 0 } = element; return { width, height }; } function onLayout(event: LayoutChangeEvent): void { const { nativeEvent: { layout: { width, height }, }, } = event; if (width !== size?.width || height !== size.height) { setSize({ width, height }); if (props.onLayout) { props.onLayout(event); } } } React.useEffect(() => { if (ref.current != null) { setSize(getSize()); } }, [ref]); React.useEffect(() => { updateCanvasSize(); }, [size]); React.useEffect(() => { const canvas = _canvasRef.current; if (canvas) { updateCanvasSize(); } setRef(props.canvasRef, canvas); }, [_canvasRef]); return ( {children} ); }; export default CanvasWrapper;