Skip to content
This repository has been archived by the owner on Dec 6, 2023. It is now read-only.

Commit

Permalink
feat: Adding functionality to support standalone reactotron
Browse files Browse the repository at this point in the history
BREAKING CHANGE: The ReactotronProvider has been renamed to ReactotronAppProvider and a new ReactotronProvider has been created serving a different function
  • Loading branch information
rmevans9 authored Jan 26, 2020
1 parent 2dc0d17 commit 22c404f
Show file tree
Hide file tree
Showing 26 changed files with 992 additions and 36 deletions.
6 changes: 2 additions & 4 deletions .storybook/config.js
Original file line number Diff line number Diff line change
Expand Up @@ -2,11 +2,9 @@ import React from "react"
import { configure, addDecorator } from "@storybook/react"
import { withKnobs } from "@storybook/addon-knobs"

import ReactotronProvider from "../src/components/ReactotronProvider"
import ReactotronAppProvider from "../src/components/ReactotronAppProvider"

import theme from "../src/theme"

const StyledDecorator = storyFn => <ReactotronProvider>{storyFn()}</ReactotronProvider>
const StyledDecorator = storyFn => <ReactotronAppProvider>{storyFn()}</ReactotronAppProvider>
addDecorator(StyledDecorator)

addDecorator(withKnobs)
Expand Down
14 changes: 11 additions & 3 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -55,6 +55,7 @@
"@storybook/addon-links": "5.3.9",
"@storybook/addons": "5.3.9",
"@storybook/react": "5.3.9",
"@testing-library/react-hooks": "^3.2.1",
"@types/jest": "24.9.1",
"@types/react-modal": "3.10.2",
"@types/react-motion": "0.0.29",
Expand All @@ -70,6 +71,8 @@
"eslint-plugin-import": "2.20.0",
"eslint-plugin-node": "11.0.0",
"eslint-plugin-promise": "4.2.1",
"eslint-plugin-react": "^7.18.0",
"eslint-plugin-react-hooks": "^2.3.0",
"eslint-plugin-standard": "4.0.1",
"jest": "25.1.0",
"npm-run-all": "4.1.5",
Expand All @@ -78,6 +81,7 @@
"react-icons": "3.8.0",
"react-modal": "3.11.1",
"react-motion": "0.5.2",
"react-test-renderer": "^16.12.0",
"react-tooltip": "3.11.2",
"rollup": "1.29.1",
"rollup-plugin-babel": "4.3.3",
Expand All @@ -96,6 +100,7 @@
"parser": "@typescript-eslint/parser",
"extends": [
"plugin:@typescript-eslint/recommended",
"plugin:react/recommended",
"standard",
"prettier"
],
Expand All @@ -106,9 +111,13 @@
"project": "./tsconfig.json"
},
"plugins": [
"@typescript-eslint"
"@typescript-eslint",
"react-hooks"
],
"rules": {
"react-hooks/rules-of-hooks": "error",
"react-hooks/exhaustive-deps": "warn",
"react/prop-types": 0,
"no-unused-vars": 0,
"no-undef": 0,
"space-before-function-paren": 0,
Expand All @@ -119,8 +128,7 @@
"@typescript-eslint/no-object-literal-type-assertion": 0,
"@typescript-eslint/no-empty-interface": 0,
"@typescript-eslint/no-var-requires": 0,
"@typescript-eslint/member-delimiter-style": 0,
"@typescript-eslint/no-unused-vars": 0
"@typescript-eslint/member-delimiter-style": 0
}
},
"jest": {
Expand Down
15 changes: 15 additions & 0 deletions src/components/EmptyState/EmptyState.story.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,15 @@
/* eslint-disable @typescript-eslint/no-empty-function */
import React from "react"
import { MdReorder } from "react-icons/md"

import EmptyState from "./index"

export default {
title: "Empty State",
}

export const Default = () => (
<EmptyState icon={MdReorder} title="Empty!">
Some more information
</EmptyState>
)
42 changes: 42 additions & 0 deletions src/components/EmptyState/index.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,42 @@
import React, { FunctionComponent } from "react"
import styled from "styled-components"

const Container = styled.div`
height: 100%;
flex: 1;
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
color: ${props => props.theme.foregroundLight};
`

const Title = styled.div`
font-size: 2rem;
padding-bottom: 50px;
padding-top: 10px;
`

const Message = styled.div`
color: ${props => props.theme.foreground};
max-width: 400px;
line-height: 1.4;
text-align: center;
`

interface Props {
icon?: any // TODO: Type Better?
title: string
}

const EmptyState: FunctionComponent<Props> = ({ title, icon: Icon, children }) => {
return (
<Container>
{Icon && <Icon size={100} />}
<Title>{title}</Title>
<Message>{children}</Message>
</Container>
)
}

export default EmptyState
Original file line number Diff line number Diff line change
Expand Up @@ -7,14 +7,15 @@ const ReactotronContainer = styled.div`
font-family: ${props => props.theme.fontFamily};
width: 100%;
height: 100%;
user-select: none;
`

const ReactotronProvider: FunctionComponent = ({ children }) => {
const ReactotronAppProvider: FunctionComponent = ({ children }) => {
return (
<ThemeProvider theme={theme}>
<ReactotronContainer>{children}</ReactotronContainer>
</ThemeProvider>
)
}

export default ReactotronProvider;
export default ReactotronAppProvider;
41 changes: 41 additions & 0 deletions src/contexts/CustomCommands/index.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,41 @@
import React, { FunctionComponent, useContext, useCallback } from "react"

import ReactotronContext from "../Reactotron"

import useCustomCommands, { CustomCommand } from "./useCustomCommands"

interface Context {
customCommands: CustomCommand[]
sendCustomCommand: (command: any, args: any) => void
}

const CustomCommandsContext = React.createContext<Context>({
customCommands: [],
sendCustomCommand: null,
})

const Provider: FunctionComponent<any> = ({ children }) => {
const { sendCommand } = useContext(ReactotronContext)
const { customCommands } = useCustomCommands()

const sendCustomCommand = useCallback(
(command, args) => {
sendCommand("custom", { command, args })
},
[sendCommand]
)

return (
<CustomCommandsContext.Provider
value={{
customCommands,
sendCustomCommand,
}}
>
{children}
</CustomCommandsContext.Provider>
)
}

export default CustomCommandsContext
export const CustomCommandsProvider = Provider
109 changes: 109 additions & 0 deletions src/contexts/CustomCommands/useCustomCommands.test.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,109 @@
/* eslint-disable react/display-name */
import React from "react"
import { renderHook } from "@testing-library/react-hooks"

import ReactotronContext from "../Reactotron"
import { CommandType } from "../../types"

import useCustomCommands from "./useCustomCommands"

function buildContextValues({ addCommandListener = null } = {}) {
return {
commands: [],
sendCommand: jest.fn(),
clearCommands: jest.fn(),
addCommandListener: addCommandListener || jest.fn(),
isDispatchModalOpen: false,
dispatchModalInitialAction: "",
openDispatchModal: jest.fn(),
closeDispatchModal: jest.fn(),
isSubscriptionModalOpen: false,
openSubscriptionModal: jest.fn(),
closeSubscriptionModal: jest.fn(),
}
}

describe("contexts/CustomCommands/useCustomCommands", () => {
it("should add and remove custom commands", () => {
let addCallback = null

const contextValues = buildContextValues({
addCommandListener: callback => {
addCallback = callback
},
})

const { result } = renderHook(() => useCustomCommands(), {
wrapper: ({ children }) => (
<ReactotronContext.Provider value={contextValues}>{children}</ReactotronContext.Provider>
),
})

expect(result.current.customCommands).toEqual([])

addCallback({
type: CommandType.CustomCommandRegister,
clientId: "1234",
payload: {
id: 0,
},
})

expect(result.current.customCommands).toEqual([
{
clientId: "1234",
id: 0,
},
])

addCallback({
type: CommandType.CustomCommandUnregister,
clientId: "1234",
payload: {
id: 0,
},
})

expect(result.current.customCommands).toEqual([])
})

it("should clear all commands after a reconnect", () => {
let addCallback = null

const contextValues = buildContextValues({
addCommandListener: callback => {
addCallback = callback
},
})

const { result } = renderHook(() => useCustomCommands(), {
wrapper: ({ children }) => (
<ReactotronContext.Provider value={contextValues}>{children}</ReactotronContext.Provider>
),
})

expect(result.current.customCommands).toEqual([])

addCallback({
type: CommandType.CustomCommandRegister,
clientId: "1234",
payload: {
id: 0,
},
})

expect(result.current.customCommands).toEqual([
{
clientId: "1234",
id: 0,
},
])

addCallback({
type: CommandType.ClientIntro,
clientId: "1234",
})

expect(result.current.customCommands).toEqual([])
})
})
Loading

0 comments on commit 22c404f

Please sign in to comment.