diff --git a/packages/app/src/Sidebar.js b/packages/app/src/Sidebar.js
index b6e4d44..00f5961 100644
--- a/packages/app/src/Sidebar.js
+++ b/packages/app/src/Sidebar.js
@@ -4,6 +4,7 @@ import {
View,
Dimensions,
StyleSheet,
+ Button,
registerComponent,
withFocus,
} from '../../react-ape/reactApeEntry';
@@ -58,6 +59,14 @@ class Sidebar extends Component {
return (
{/**/}
+ */}
);
diff --git a/packages/docs/components-button.md b/packages/docs/components-button.md
new file mode 100644
index 0000000..1b40443
--- /dev/null
+++ b/packages/docs/components-button.md
@@ -0,0 +1,29 @@
+---
+id: components-button
+title: Button
+sidebar_label: Button
+---
+
+
+A component for creating a Button.
+
+The following example shows how to render a Button.
+```JS
+import React from "react";
+import { Button, View, render } from "react-ape";
+
+class ImageExample extends React.Component {
+ render() {
+ return (
+
+
+ );
+ }
+}
+```
+
\ No newline at end of file
diff --git a/packages/react-ape/__tests__/render-tests/__snapshots__/render-updates.js.snap b/packages/react-ape/__tests__/render-tests/__snapshots__/render-updates.js.snap
deleted file mode 100644
index c919d7f..0000000
--- a/packages/react-ape/__tests__/render-tests/__snapshots__/render-updates.js.snap
+++ /dev/null
@@ -1,19 +0,0 @@
-// Jest Snapshot v1, https://goo.gl/fbAQLP
-
-exports[`Render Updates Relative View Render relative view with props and children updates 1`] = `""`;
-
-exports[`Render Updates Relative View Render relative view with props and children updates 2`] = `""`;
-
-exports[`Render Updates Text Test "Text" multiples content change 1`] = `""`;
-
-exports[`Render Updates Text Test "Text" multiples content change 2`] = `""`;
-
-exports[`Render Updates Text Test "Text" multiples content change 3`] = `""`;
-
-exports[`Render Updates Text Test "Text" simple text change 1`] = `""`;
-
-exports[`Render Updates Text Test "Text" simple text change 2`] = `""`;
-
-exports[`Render Updates Text Test "Text" style props change 1`] = `""`;
-
-exports[`Render Updates Text Test "Text" style props change 2`] = `""`;
diff --git a/packages/react-ape/reactApeEntry.js b/packages/react-ape/reactApeEntry.js
index a80eb07..0d6b7d8 100644
--- a/packages/react-ape/reactApeEntry.js
+++ b/packages/react-ape/reactApeEntry.js
@@ -25,6 +25,7 @@ export const render = ReactApeRenderer.render;
export const Image = 'Image';
export const View = 'View';
export const Text = 'Text';
+export const Button = 'Button';
export const StyleSheet = StyleSheetModule;
export const Dimensions = DimensionsModule;
diff --git a/packages/react-ape/renderer/constants/index.js b/packages/react-ape/renderer/constants/index.js
index 244fb24..387a7ad 100644
--- a/packages/react-ape/renderer/constants/index.js
+++ b/packages/react-ape/renderer/constants/index.js
@@ -12,6 +12,28 @@ export const ViewDefaults = {
size: 200, // 200x200
lineHeight: 24,
};
+export const ButtonDefaults = {
+ containerStyle: {
+ width: 85,
+ elevation: 4,
+ height: 30,
+ backgroundColor: '#2196F3',
+ borderRadius: 2,
+ },
+ textStyle: {
+ textAlign: 'center',
+ margin: 8,
+ color: 'white',
+ fontSize: 18,
+ },
+ buttonDisabled: {
+ elevation: 0,
+ backgroundColor: '#dfdfdf',
+ },
+ textDisabled: {
+ color: '#cdcdcd',
+ },
+};
// ReactApe Internal Constants
export const _SectionBlockSize: number = 80; // 80x80
diff --git a/packages/react-ape/renderer/elements/Button.js b/packages/react-ape/renderer/elements/Button.js
new file mode 100644
index 0000000..6fb88d0
--- /dev/null
+++ b/packages/react-ape/renderer/elements/Button.js
@@ -0,0 +1,199 @@
+/**
+ * @https://github.com/facebook/react-native/blob/main/Libraries/Components/Button.js
+ *
+ * @flow
+ *
+ */
+
+import {ButtonDefaults} from '../constants';
+import {trackMousePosition, isMouseInside} from '../utils';
+import type {CanvasComponentContext} from '../types';
+
+//TODO adjust Opacity when focus, Blur
+type PressEvent = {||};
+type ButtonProps = {|
+ title: string,
+ onPress: (event?: PressEvent) => mixed,
+ onClick: (event?: SyntheticMouseEvent) => mixed,
+ //handleOnClick:(event?:SyntheticMouseEvent)=>mixed,
+ touchSoundDisabled?: ?boolean,
+ color?: ?string,
+ /**
+ * TV next focus down (see documentation for the View component).
+ *
+ * @platform android
+ */
+ nextFocusDown?: ?number,
+ /**
+ * TV next focus forward (see documentation for the View component).
+ *
+ * @platform android
+ */
+ nextFocusForward?: ?number,
+
+ /**
+ * TV next focus left (see documentation for the View component).
+ *
+ * @platform android
+ */
+ nextFocusLeft?: ?number,
+
+ /**
+ * TV next focus right (see documentation for the View component).
+ *
+ * @platform android
+ */
+ nextFocusRight?: ?number,
+
+ /**
+ * TV next focus up (see documentation for the View component).
+ *
+ * @platform android
+ */
+ nextFocusUp?: ?number,
+
+ /**
+ * Text to display for blindness accessibility features
+ */
+ accessibilityLabel?: ?string,
+
+ /**
+ * If true, disable all interactions for this component.
+ */
+ disabled?: ?boolean,
+ getDimension?: () => mixed,
+ /**
+ * Used to locate this view in end-to-end tests.
+ */
+ testID?: ?string,
+|};
+
+function renderButton(
+ props: ButtonProps,
+ apeContext: CanvasComponentContext,
+ parentLayout
+) {
+ const {ctx} = apeContext;
+
+ // If is relative and x and y haven't be processed, don't render
+ // start drawing the canvas
+ console.log('[PROPS]', props);
+ const {title, color} = props;
+ if (!title) {
+ throw Error('Title required!');
+ }
+ const borderRadius = ButtonDefaults.containerStyle.borderRadius;
+ const backgroundColor = ButtonDefaults.containerStyle.backgroundColor;
+ let x = 40;
+ let y = 300;
+
+ const textWidth = ctx.measureText(title).width;
+ let width = textWidth * 1.5;
+ let height = ButtonDefaults.containerStyle.height;
+ let globalStyle = {
+ width: width,
+ height: height,
+ color: color,
+ borderRadius: borderRadius,
+ backgroundColor: color,
+ lineWidth: 0,
+ borderColor: 'transparent',
+ };
+ const resetStyle = newStyle => {
+ globalStyle = {...globalStyle, newStyle};
+ };
+ // const redrawButton = ctx => {
+ // // TODO reset Style on focus
+ // let newStyle = {
+ // lineWidth: 2,
+ // borderColor: '#ccc',
+ // };
+ // resetStyle(newStyle);
+ // };
+
+ ctx.beginPath();
+ ctx.fillStyle = color || ButtonDefaults.containerStyle.backgroundColor;
+ ctx.moveTo(x, y);
+ /**
+* Top Right Radius
+*/
+ ctx.lineTo(x + width - borderRadius, y);
+ ctx.quadraticCurveTo(x + width, y, x + width, y + borderRadius);
+ /**
+* Bottom right Radius
+*/
+
+ ctx.lineTo(x + width, y + height - borderRadius);
+ ctx.quadraticCurveTo(
+ x + width,
+ y + height,
+ x + width - borderRadius,
+ y + height
+ );
+
+ /**
+* Bottom Left Radius
+*/
+ ctx.lineTo(x + borderRadius, y + height);
+ ctx.quadraticCurveTo(x, y + height, x, y + height - borderRadius);
+ /** Top left Radius */
+ ctx.lineTo(x, y + borderRadius);
+ ctx.quadraticCurveTo(x, y, x + borderRadius, y);
+
+ ctx.fill();
+ ctx.lineWidth = globalStyle.lineWidth;
+ ctx.strokeStyle = globalStyle.borderColor;
+ ctx.stroke();
+ ctx.fillStyle = ButtonDefaults.textStyle.color;
+
+ //set the fontSize
+ const fontArgs = ctx.font.split(' ');
+ const newSize = `${ButtonDefaults.textStyle.fontSize}px`;
+ ctx.font = newSize + ' ' + fontArgs[fontArgs.length - 1];
+
+ // ctx.textAlign = 'center';
+
+ // ctx.fillText(title, 400 / 2, y + height / 2,textWidth);
+ ctx.fillText(title, x + textWidth / 2.5, y + height / 2);
+ ctx.closePath();
+ // if(props.handleOnClick){
+ // onClick()
+ // }
+
+ // const onClick = (event: SyntheticMouseEvent) => {
+ // const rect = {
+ // x,
+ // y,
+ // height,
+ // width,
+ // };
+ // const mousePosition = trackMousePosition(ctx.canvas, event);
+ // if (isMouseInside(mousePosition, rect)) {
+ // redrawButton(ctx);
+ // if (props.onClick && typeof props.onClick === 'function') {
+ // props.onClick(event);
+ // }
+ // }
+ // };
+
+ // TODO:
+ /**
+ * We need to remove addEventListeners from the renderButton
+ * function because this function runs for each state/prop update.
+ * It will keep creating/refreshing listeners for every render.
+
+We can keep this way, if we run this addEventListener
+once by checking if the listener already exist.
+Note onClick will need to share scope with this function to work properly.
+ */
+ // ctx.canvas.addEventListener('click', onClick, false);
+ // ctx.canvas.addEventListener('focus', redrawButton);
+ // ctx.canvas.addEventListener('blur', redrawButton);
+}
+
+export default function createButtonInstance(props: ButtonProps): mixed {
+ return {
+ type: 'Button',
+ render: renderButton.bind(this, props),
+ };
+}
diff --git a/packages/react-ape/renderer/elements/__tests__/Button-test.js b/packages/react-ape/renderer/elements/__tests__/Button-test.js
new file mode 100644
index 0000000..b272171
--- /dev/null
+++ b/packages/react-ape/renderer/elements/__tests__/Button-test.js
@@ -0,0 +1,68 @@
+import React from 'react';
+import renderer from 'react-test-renderer';
+import {ButtonDefaults} from '../../constants';
+import CreateButtonInstance from '../Button';
+
+describe('Button', () => {
+ describe('Call the button with Props', () => {
+ it('Should render properly', () => {
+ const title = 'Press Me';
+ const color = '#f8a978';
+ const x = 40;
+ const y = 300;
+ const width = x + y;
+ const height = ButtonDefaults.containerStyle.height;
+ const props = {title, color};
+ const apeContext = {
+ ctx: {
+ beginPath: jest.fn(),
+ fillStyle: jest.fn(),
+ moveTo: jest.fn(),
+ fillText: jest.fn(),
+ fill: jest.fn(),
+ stroke: jest.fn(),
+ closePath: jest.fn(),
+ lineTo: jest.fn(),
+ quadraticCurveTo: jest.fn(),
+ font: 'Helvetica',
+ measureText: jest.fn(() => {
+ return {
+ width: 100,
+ };
+ }),
+ canvas: {
+ addEventListener: jest.fn(),
+ },
+ },
+ };
+
+ const Button = CreateButtonInstance(props);
+ Button.render(apeContext, {spatialGeometry: {x, y}});
+ const {
+ beginPath,
+ fillStyle,
+ moveTo,
+ fillText,
+ fill,
+ stroke,
+ closePath,
+ lineTo,
+ quadraticCurveTo,
+ font,
+ } = apeContext.ctx;
+ expect(beginPath.mock.calls.length).toBe(1);
+ expect(beginPath).toBeCalledWith();
+ expect(closePath).toBeCalledWith();
+ expect(stroke).toBeCalledWith();
+ expect(moveTo).toBeCalledWith(x, y);
+ expect(lineTo.mock.calls.length).toEqual(4);
+ expect(fill.mock.calls.length).toBe(1);
+ expect(fillText.mock.calls.length).toBe(1);
+ expect(fillText.mock.calls.length).toBe(1);
+ expect(font).toEqual(`${ButtonDefaults.textStyle.fontSize}px Helvetica`);
+ expect(quadraticCurveTo.mock.calls.length).toEqual(4);
+ expect(fillStyle).toEqual('white');
+ expect(Button).toMatchSnapshot();
+ });
+ });
+});
diff --git a/packages/react-ape/renderer/elements/__tests__/__snapshots__/Button-test.js.snap b/packages/react-ape/renderer/elements/__tests__/__snapshots__/Button-test.js.snap
new file mode 100644
index 0000000..7ba1ddc
--- /dev/null
+++ b/packages/react-ape/renderer/elements/__tests__/__snapshots__/Button-test.js.snap
@@ -0,0 +1,8 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`Button Call the button with Props Should render properly 1`] = `
+Object {
+ "render": [Function],
+ "type": "Button",
+}
+`;
diff --git a/packages/react-ape/renderer/reactApeComponent.js b/packages/react-ape/renderer/reactApeComponent.js
index 360a2b5..6bd81df 100644
--- a/packages/react-ape/renderer/reactApeComponent.js
+++ b/packages/react-ape/renderer/reactApeComponent.js
@@ -9,7 +9,9 @@
import Image from './elements/Image';
import Text from './elements/Text';
import View from './elements/View';
+import Button from './elements/Button';
import {CustomComponents} from '../modules/Register';
+import {trackMousePosition, isMouseInside} from './utils';
const CHILDREN = 'children';
const STYLE = 'style';
@@ -31,11 +33,42 @@ const ReactApeComponent = {
);
});
+ const tag = '[CREATE_ELEMENT]';
+ if (type === 'Button') {
+ const {ctx} = apeContextGlobal;
+ ctx.canvas.addEventListener(
+ 'click',
+ event => {
+ props.onClick(event);
+ },
+ false
+ );
+ }
+ /**
+ * Handling click button event
+ * @param {*} event
+ */
+ // const onClick = (event: SyntheticMouseEvent) => {
+ // const rect = {
+ // x,
+ // y,
+ // height,
+ // width,
+ // };
+ // const mousePosition = trackMousePosition(ctx.canvas, event);
+ // if (isMouseInside(mousePosition, rect)) {
+ // //redrawButton(ctx);
+ // if (props.onClick && typeof props.onClick === 'function') {
+ // props.onClick(event);
+ // }
+ // }
+ // };
const COMPONENTS = {
...customDict,
Image: Image(props),
Text: Text(props),
View: new View(props),
+ Button: Button(props),
};
if (!COMPONENTS[type]) {
diff --git a/packages/react-ape/renderer/utils.js b/packages/react-ape/renderer/utils.js
index a1d2618..231ea3c 100644
--- a/packages/react-ape/renderer/utils.js
+++ b/packages/react-ape/renderer/utils.js
@@ -6,3 +6,21 @@
export function unsafeCreateUniqueId(): string {
return (Math.random() * 10e18 + Date.now()).toString(36);
}
+type MouseEventType = {|
+ x: number,
+ y: number,
+|};
+export function trackMousePosition(canvas, event): MouseEventType {
+ return {
+ x: event.clientX - canvas.offsetLeft,
+ y: event.clientY - canvas.offsetTop,
+ };
+}
+export const isMouseInside = (pos, rect): boolean => {
+ return (
+ pos.x > rect.x &&
+ pos.x < rect.x + rect.width &&
+ pos.y < rect.y + rect.height &&
+ pos.y > rect.y
+ );
+};
diff --git a/packages/website/sidebars.json b/packages/website/sidebars.json
index 0d7234b..f29b8de 100644
--- a/packages/website/sidebars.json
+++ b/packages/website/sidebars.json
@@ -16,8 +16,10 @@
"Components": [
"components-image",
"components-text",
- "components-view"
+ "components-view",
+ "components-button"
],
+
"APIs": [
"apis-dimensions",
"apis-platform",