From 60561ec38d4eab0cc39ae9106bd166651e6d2252 Mon Sep 17 00:00:00 2001 From: Mikael Sand Date: Fri, 3 Jan 2020 00:51:31 +0200 Subject: [PATCH] feat(web): Implement support for event, touch & responder handlers --- src/ReactNativeSVG.web.ts | 254 +++++++++++++++++++++++------------ src/elements/Shape.tsx | 15 +-- src/index.js.flow | 7 + src/lib/SvgTouchableMixin.ts | 20 ++- 4 files changed, 197 insertions(+), 99 deletions(-) diff --git a/src/ReactNativeSVG.web.ts b/src/ReactNativeSVG.web.ts index 0885adc14..3a2a7965d 100644 --- a/src/ReactNativeSVG.web.ts +++ b/src/ReactNativeSVG.web.ts @@ -1,18 +1,72 @@ -import { createElement } from 'react-native-web'; +// @ts-ignore +import * as React from 'react'; +import { createElement, GestureResponderEvent } from 'react-native'; +import { NumberProp, TransformProps } from './lib/extract/types'; +import SvgTouchableMixin from './lib/SvgTouchableMixin'; import { resolve } from './lib/resolve'; -import { Component } from 'react'; -import { NumberProp } from './lib/extract/types'; + +type BlurEvent = Object; +type FocusEvent = Object; +type PressEvent = Object; +type LayoutEvent = Object; +type EdgeInsetsProp = Object; + +interface BaseProps { + accessible?: boolean; + accessibilityLabel?: string; + accessibilityHint?: string; + accessibilityIgnoresInvertColors?: boolean; + accessibilityRole?: string; + accessibilityState?: Object; + delayLongPress?: number; + delayPressIn?: number; + delayPressOut?: number; + disabled?: boolean; + hitSlop?: EdgeInsetsProp; + nativeID?: string; + touchSoundDisabled?: boolean; + onBlur?: (e: BlurEvent) => void; + onFocus?: (e: FocusEvent) => void; + onLayout?: (event: LayoutEvent) => object; + onLongPress?: (event: PressEvent) => object; + onClick?: (event: PressEvent) => object; + onPress?: (event: PressEvent) => object; + onPressIn?: (event: PressEvent) => object; + onPressOut?: (event: PressEvent) => object; + pressRetentionOffset?: EdgeInsetsProp; + rejectResponderTermination?: boolean; + + translate: TransformProps; + scale: NumberProp; + rotation: NumberProp; + skewX: NumberProp; + skewY: NumberProp; + originX: NumberProp; + originY: NumberProp; + + fontStyle?: string; + fontWeight?: NumberProp; + fontSize?: NumberProp; + fontFamily?: string; + forwardedRef: {}; + style: {}; +} /** * `react-native-svg` supports additional props that aren't defined in the spec. * This function replaces them in a spec conforming manner. * - * @param {Object} props Properties given to us. - * @returns {Object} Cleaned object. + * @param {WebShape} self Instance given to us. + * @param {Object?} props Optional overridden props given to us. + * @returns {Object} Cleaned props object. * @private */ -function prepare(props) { +const prepare = ( + self: WebShape, + props = self.props, +) => { const { + accessible, translate, scale, rotation, @@ -26,8 +80,31 @@ function prepare(props) { fontStyle, style, forwardedRef, - ...clean + // @ts-ignore + ...rest } = props; + const clean: { + accessible?: boolean; + onStartShouldSetResponder?: (e: GestureResponderEvent) => boolean; + onResponderMove?: (e: GestureResponderEvent) => void; + onResponderGrant?: (e: GestureResponderEvent) => void; + onResponderRelease?: (e: GestureResponderEvent) => void; + onResponderTerminate?: (e: GestureResponderEvent) => void; + onResponderTerminationRequest?: (e: GestureResponderEvent) => boolean; + transform?: string; + style?: object; + ref?: {}; + } = { + ...rest, + accessible: accessible !== false, + onStartShouldSetResponder: self.touchableHandleStartShouldSetResponder, + onResponderTerminationRequest: + self.touchableHandleResponderTerminationRequest, + onResponderGrant: self.touchableHandleResponderGrant, + onResponderMove: self.touchableHandleResponderMove, + onResponderRelease: self.touchableHandleResponderRelease, + onResponderTerminate: self.touchableHandleResponderTerminate, + }; const transform = []; @@ -62,10 +139,6 @@ function prepare(props) { clean.ref = forwardedRef; } - if (props.onPress && !props.onClick) { - clean.onClick = props.onPress; - } - const styles: { fontStyle?: string; fontFamily?: string; @@ -89,153 +162,166 @@ function prepare(props) { clean.style = resolve(style, styles); return clean; +}; + +export class WebShape< + P extends BaseProps = BaseProps, + C = {} +> extends React.Component { + [x: string]: unknown; + constructor(props: P, context: C) { + super(props, context); + SvgTouchableMixin(this); + } } -export class Circle extends Component { - render() { - return createElement('circle', prepare(this.props)); +export class Circle extends WebShape { + render(): JSX.Element { + return createElement('circle', prepare(this)); } } -export class ClipPath extends Component { - render() { - return createElement('clipPath', prepare(this.props)); +export class ClipPath extends WebShape { + render(): JSX.Element { + return createElement('clipPath', prepare(this)); } } -export class Defs extends Component { - render() { - return createElement('defs', prepare(this.props)); +export class Defs extends WebShape { + render(): JSX.Element { + return createElement('defs', prepare(this)); } } -export class Ellipse extends Component { - render() { - return createElement('ellipse', prepare(this.props)); +export class Ellipse extends WebShape { + render(): JSX.Element { + return createElement('ellipse', prepare(this)); } } -export class G extends Component<{ - x?: NumberProp; - y?: NumberProp; - translate?: string; -}> { - render() { +export class G extends WebShape< + BaseProps & { + x?: NumberProp; + y?: NumberProp; + translate?: string; + } +> { + render(): JSX.Element { const { x, y, ...rest } = this.props; if ((x || y) && !rest.translate) { rest.translate = `${x || 0}, ${y || 0}`; } - return createElement('g', prepare(rest)); + return createElement('g', prepare(this, rest)); } } -export class Image extends Component { - render() { - return createElement('image', prepare(this.props)); +export class Image extends WebShape { + render(): JSX.Element { + return createElement('image', prepare(this)); } } -export class Line extends Component { - render() { - return createElement('line', prepare(this.props)); +export class Line extends WebShape { + render(): JSX.Element { + return createElement('line', prepare(this)); } } -export class LinearGradient extends Component { - render() { - return createElement('linearGradient', prepare(this.props)); +export class LinearGradient extends WebShape { + render(): JSX.Element { + return createElement('linearGradient', prepare(this)); } } -export class Path extends Component { - render() { - return createElement('path', prepare(this.props)); +export class Path extends WebShape { + render(): JSX.Element { + return createElement('path', prepare(this)); } } -export class Polygon extends Component { - render() { - return createElement('polygon', prepare(this.props)); +export class Polygon extends WebShape { + render(): JSX.Element { + return createElement('polygon', prepare(this)); } } -export class Polyline extends Component { - render() { - return createElement('polyline', prepare(this.props)); +export class Polyline extends WebShape { + render(): JSX.Element { + return createElement('polyline', prepare(this)); } } -export class RadialGradient extends Component { - render() { - return createElement('radialGradient', prepare(this.props)); +export class RadialGradient extends WebShape { + render(): JSX.Element { + return createElement('radialGradient', prepare(this)); } } -export class Rect extends Component { - render() { - return createElement('rect', prepare(this.props)); +export class Rect extends WebShape { + render(): JSX.Element { + return createElement('rect', prepare(this)); } } -export class Stop extends Component { - render() { - return createElement('stop', prepare(this.props)); +export class Stop extends WebShape { + render(): JSX.Element { + return createElement('stop', prepare(this)); } } -export class Svg extends Component { - render() { - return createElement('svg', prepare(this.props)); +export class Svg extends WebShape { + render(): JSX.Element { + return createElement('svg', prepare(this)); } } -export class Symbol extends Component { - render() { - return createElement('symbol', prepare(this.props)); +export class Symbol extends WebShape { + render(): JSX.Element { + return createElement('symbol', prepare(this)); } } -export class Text extends Component { - render() { - return createElement('text', prepare(this.props)); +export class Text extends WebShape { + render(): JSX.Element { + return createElement('text', prepare(this)); } } -export class TSpan extends Component { - render() { - return createElement('tspan', prepare(this.props)); +export class TSpan extends WebShape { + render(): JSX.Element { + return createElement('tspan', prepare(this)); } } -export class TextPath extends Component { - render() { - return createElement('textPath', prepare(this.props)); +export class TextPath extends WebShape { + render(): JSX.Element { + return createElement('textPath', prepare(this)); } } -export class Use extends Component { - render() { - return createElement('use', prepare(this.props)); +export class Use extends WebShape { + render(): JSX.Element { + return createElement('use', prepare(this)); } } -export class Mask extends Component { - render() { - return createElement('mask', prepare(this.props)); +export class Mask extends WebShape { + render(): JSX.Element { + return createElement('mask', prepare(this)); } } -export class Marker extends Component { - render() { - return createElement('marker', prepare(this.props)); +export class Marker extends WebShape { + render(): JSX.Element { + return createElement('marker', prepare(this)); } } -export class Pattern extends Component { - render() { - return createElement('pattern', prepare(this.props)); +export class Pattern extends WebShape { + render(): JSX.Element { + return createElement('pattern', prepare(this)); } } diff --git a/src/elements/Shape.tsx b/src/elements/Shape.tsx index 50582c088..a73aab412 100644 --- a/src/elements/Shape.tsx +++ b/src/elements/Shape.tsx @@ -8,10 +8,6 @@ import { import { TransformProps } from '../lib/extract/types'; const RNSVGRenderableManager = NativeModules.RNSVGRenderableManager; -const { touchableGetInitialState } = SvgTouchableMixin; -const touchKeys = Object.keys(SvgTouchableMixin); -const touchVals = touchKeys.map(key => SvgTouchableMixin[key]); -const numTouchKeys = touchKeys.length; export interface SVGBoundingBoxOptions { fill?: boolean; @@ -241,16 +237,7 @@ export default class Shape

extends Component

{ root: (Shape

& NativeMethodsMixinStatic) | null = null; constructor(props: P, context: {}) { super(props, context); - for (let i = 0; i < numTouchKeys; i++) { - const key = touchKeys[i]; - const val = touchVals[i]; - if (typeof val === 'function') { - this[key] = val.bind(this); - } else { - this[key] = val; - } - } - this.state = touchableGetInitialState(); + SvgTouchableMixin(this); } refMethod: ( instance: (Shape

& NativeMethodsMixinStatic) | null, diff --git a/src/index.js.flow b/src/index.js.flow index 84d51279e..c682c8a42 100644 --- a/src/index.js.flow +++ b/src/index.js.flow @@ -177,6 +177,12 @@ export type TransformProps = { export interface CommonMaskProps { mask?: string; } +export interface CommonMarkerProps { + marker?: string; + markerStart?: string; + markerMid?: string; + markerEnd?: string; +} export type CommonPathProps = { ... } & FillProps & StrokeProps & ClipProps & @@ -185,6 +191,7 @@ export type CommonPathProps = { ... } & FillProps & ResponderProps & TouchableProps & DefinitionProps & + CommonMarkerProps & CommonMaskProps; export type CircleProps = { cx?: NumberProp, diff --git a/src/lib/SvgTouchableMixin.ts b/src/lib/SvgTouchableMixin.ts index 8170fea01..4285dca59 100644 --- a/src/lib/SvgTouchableMixin.ts +++ b/src/lib/SvgTouchableMixin.ts @@ -10,9 +10,10 @@ const { touchableHandleResponderMove, touchableHandleResponderRelease, touchableHandleResponderTerminate, + touchableGetInitialState, } = Mixin; -export default { +const SvgTouchableMixin = { ...Mixin, touchableHandleStartShouldSetResponder(e: GestureResponderEvent) { @@ -114,3 +115,20 @@ export default { return delayPressOut || 0; }, }; + +const touchKeys = Object.keys(SvgTouchableMixin); +const touchVals = touchKeys.map(key => SvgTouchableMixin[key]); +const numTouchKeys = touchKeys.length; + +export default (target: { [x: string]: unknown; state: unknown }) => { + for (let i = 0; i < numTouchKeys; i++) { + const key = touchKeys[i]; + const val = touchVals[i]; + if (typeof val === 'function') { + target[key] = val.bind(target); + } else { + target[key] = val; + } + } + target.state = touchableGetInitialState(); +};