diff options
Diffstat (limited to 'src/console')
| -rw-r--r-- | src/console/components/AppContext.ts | 13 | ||||
| -rw-r--r-- | src/console/components/Console.tsx | 235 | ||||
| -rw-r--r-- | src/console/index.tsx | 79 | ||||
| -rw-r--r-- | src/console/reducers/index.ts | 2 | 
4 files changed, 174 insertions, 155 deletions
| diff --git a/src/console/components/AppContext.ts b/src/console/components/AppContext.ts new file mode 100644 index 0000000..878d00b --- /dev/null +++ b/src/console/components/AppContext.ts @@ -0,0 +1,13 @@ +import React from "react"; +import { State, defaultState } from "../reducers"; +import { ConsoleAction } from "../actions"; + +const AppContext = React.createContext<{ +  state: State; +  dispatch: React.Dispatch<Promise<ConsoleAction> | ConsoleAction>; +}>({ +  state: defaultState, +  dispatch: () => null, +}); + +export default AppContext; diff --git a/src/console/components/Console.tsx b/src/console/components/Console.tsx index bb9aee7..3cccc43 100644 --- a/src/console/components/Console.tsx +++ b/src/console/components/Console.tsx @@ -1,10 +1,8 @@ -import { connect } from "react-redux";  import React from "react";  import Input from "./console/Input";  import Completion from "./console/Completion";  import Message from "./console/Message";  import * as consoleActions from "../../console/actions/console"; -import { State as AppState } from "../reducers";  import CommandLineParser, {    InputPhase,  } from "../commandline/CommandLineParser"; @@ -14,6 +12,7 @@ import { LightTheme, DarkTheme } from "./Theme";  import styled from "./Theme";  import { ThemeProvider } from "styled-components";  import ConsoleFrameClient from "../clients/ConsoleFrameClient"; +import AppContext from "./AppContext";  const ConsoleWrapper = styled.div`    border-top: 1px solid gray; @@ -21,63 +20,54 @@ const ConsoleWrapper = styled.div`  const COMPLETION_MAX_ITEMS = 33; -type StateProps = ReturnType<typeof mapStateToProps>; -interface DispatchProps { -  dispatch: (action: any) => void; -} -type Props = StateProps & DispatchProps; +const Console: React.FC = () => { +  const { state, dispatch } = React.useContext(AppContext); +  const commandLineParser = new CommandLineParser(); +  const consoleFrameClient = new ConsoleFrameClient(); -class Console extends React.Component<Props> { -  private commandLineParser: CommandLineParser = new CommandLineParser(); -  private consoleFrameClient = new ConsoleFrameClient(); - -  constructor(props: Props) { -    super(props); -  } - -  onBlur() { -    if (this.props.mode === "command" || this.props.mode === "find") { -      return this.props.dispatch(consoleActions.hideCommand()); +  const onBlur = () => { +    if (state.mode === "command" || state.mode === "find") { +      dispatch(consoleActions.hideCommand());      } -  } +  }; -  doEnter(e: React.KeyboardEvent<HTMLInputElement>) { +  const doEnter = (e: React.KeyboardEvent<HTMLInputElement>) => {      e.stopPropagation();      e.preventDefault();      const value = (e.target as HTMLInputElement).value; -    if (this.props.mode === "command") { -      return this.props.dispatch(consoleActions.enterCommand(value)); -    } else if (this.props.mode === "find") { -      return this.props.dispatch( -        consoleActions.enterFind(value === "" ? undefined : value) -      ); +    if (state.mode === "command") { +      dispatch(consoleActions.enterCommand(value)); +    } else if (state.mode === "find") { +      dispatch(consoleActions.enterFind(value === "" ? undefined : value));      } -  } +  }; -  selectNext(e: React.KeyboardEvent<HTMLInputElement>) { -    this.props.dispatch(consoleActions.completionNext()); +  const selectNext = (e: React.KeyboardEvent<HTMLInputElement>) => { +    dispatch(consoleActions.completionNext());      e.stopPropagation();      e.preventDefault(); -  } +  }; -  selectPrev(e: React.KeyboardEvent<HTMLInputElement>) { -    this.props.dispatch(consoleActions.completionPrev()); +  const selectPrev = (e: React.KeyboardEvent<HTMLInputElement>) => { +    dispatch(consoleActions.completionPrev());      e.stopPropagation();      e.preventDefault(); -  } +  }; -  onKeyDown(e: React.KeyboardEvent<HTMLInputElement>) { +  const onKeyDown = (e: React.KeyboardEvent<HTMLInputElement>) => {      switch (e.key) {        case "Escape": -        return this.props.dispatch(consoleActions.hideCommand()); +        dispatch(consoleActions.hideCommand()); +        break;        case "Enter": -        return this.doEnter(e); +        doEnter(e); +        break;        case "Tab":          if (e.shiftKey) { -          this.props.dispatch(consoleActions.completionPrev()); +          dispatch(consoleActions.completionPrev());          } else { -          this.props.dispatch(consoleActions.completionNext()); +          dispatch(consoleActions.completionNext());          }          e.stopPropagation();          e.preventDefault(); @@ -85,126 +75,79 @@ class Console extends React.Component<Props> {        case "[":          if (e.ctrlKey) {            e.preventDefault(); -          return this.props.dispatch(consoleActions.hideCommand()); +          dispatch(consoleActions.hideCommand());          }          break;        case "c":          if (e.ctrlKey) {            e.preventDefault(); -          return this.props.dispatch(consoleActions.hideCommand()); +          dispatch(consoleActions.hideCommand());          }          break;        case "m":          if (e.ctrlKey) { -          return this.doEnter(e); +          doEnter(e);          }          break;        case "n":          if (e.ctrlKey) { -          this.selectNext(e); +          selectNext(e);          }          break;        case "p":          if (e.ctrlKey) { -          this.selectPrev(e); +          selectPrev(e);          }          break;      } -  } +  }; -  onChange(e: React.ChangeEvent<HTMLInputElement>) { +  const onChange = (e: React.ChangeEvent<HTMLInputElement>) => {      const text = e.target.value; -    this.props.dispatch(consoleActions.setConsoleText(text)); -    if (this.props.mode !== "command") { +    dispatch(consoleActions.setConsoleText(text)); +    if (state.mode !== "command") {        return;      } -    this.updateCompletions(text); -  } - -  componentDidUpdate(prevProps: Props) { -    if (prevProps.mode !== "command" && this.props.mode === "command") { -      this.updateCompletions(this.props.consoleText); -      this.focus(); -    } else if (prevProps.mode !== "find" && this.props.mode === "find") { -      this.focus(); +    updateCompletions(text); +  }; + +  const prevState = React.useRef(state); +  React.useEffect(() => { +    if (prevState.current.mode !== "command" && state.mode === "command") { +      updateCompletions(state.consoleText); +      focus(); +    } else if (prevState.current.mode !== "find" && state.mode === "find") { +      focus();      }      const {        scrollWidth: width,        scrollHeight: height,      } = document.getElementById("vimvixen-console")!; -    this.consoleFrameClient.resize(width, height); -  } - -  render() { -    let theme = this.props.colorscheme; -    if (this.props.colorscheme === ColorScheme.System) { -      if ( -        window.matchMedia && -        window.matchMedia("(prefers-color-scheme: dark)").matches -      ) { -        theme = ColorScheme.Dark; -      } else { -        theme = ColorScheme.Light; -      } -    } +    consoleFrameClient.resize(width, height); -    switch (this.props.mode) { -      case "command": -      case "find": -        return ( -          <ThemeProvider -            theme={theme === ColorScheme.Dark ? DarkTheme : LightTheme} -          > -            <ConsoleWrapper> -              <Completion -                size={COMPLETION_MAX_ITEMS} -                completions={this.props.completions} -                select={this.props.select} -              /> -              <Input -                mode={this.props.mode} -                onBlur={this.onBlur.bind(this)} -                onKeyDown={this.onKeyDown.bind(this)} -                onChange={this.onChange.bind(this)} -                value={this.props.consoleText} -              /> -            </ConsoleWrapper> -          </ThemeProvider> -        ); -      case "info": -      case "error": -        return ( -          <ThemeProvider -            theme={theme === ColorScheme.Dark ? DarkTheme : LightTheme} -          > -            <Message mode={this.props.mode}>{this.props.messageText}</Message> -          </ThemeProvider> -        ); -      default: -        return null; -    } -  } +    prevState.current = state; +  }); -  async focus() { -    this.props.dispatch(consoleActions.setColorScheme()); +  const focus = () => { +    dispatch(consoleActions.setColorScheme());      window.focus(); -  } +  }; -  private updateCompletions(text: string) { -    const phase = this.commandLineParser.inputPhase(text); +  const updateCompletions = (text: string) => { +    const phase = commandLineParser.inputPhase(text);      if (phase === InputPhase.OnCommand) { -      return this.props.dispatch(consoleActions.getCommandCompletions(text)); +      dispatch(consoleActions.getCommandCompletions(text));      } else { -      const cmd = this.commandLineParser.parse(text); +      const cmd = commandLineParser.parse(text);        switch (cmd.command) {          case Command.Open:          case Command.TabOpen:          case Command.WindowOpen: -          this.props.dispatch( +          dispatch(              consoleActions.getOpenCompletions( -              this.props.completionTypes, +              state.completionTypes,                text,                cmd.command,                cmd.args @@ -212,32 +155,78 @@ class Console extends React.Component<Props> {            );            break;          case Command.Buffer: -          this.props.dispatch( +          dispatch(              consoleActions.getTabCompletions(text, cmd.command, cmd.args, false)            );            break;          case Command.BufferDelete:          case Command.BuffersDelete: -          this.props.dispatch( +          dispatch(              consoleActions.getTabCompletions(text, cmd.command, cmd.args, true)            );            break;          case Command.BufferDeleteForce:          case Command.BuffersDeleteForce: -          this.props.dispatch( +          dispatch(              consoleActions.getTabCompletions(text, cmd.command, cmd.args, false)            );            break;          case Command.Set: -          this.props.dispatch( +          dispatch(              consoleActions.getPropertyCompletions(text, cmd.command, cmd.args)            );            break;        }      } +  }; + +  let theme = state.colorscheme; +  if (state.colorscheme === ColorScheme.System) { +    if ( +      window.matchMedia && +      window.matchMedia("(prefers-color-scheme: dark)").matches +    ) { +      theme = ColorScheme.Dark; +    } else { +      theme = ColorScheme.Light; +    }    } -} -const mapStateToProps = (state: AppState) => ({ ...state }); +  switch (state.mode) { +    case "command": +    case "find": +      return ( +        <ThemeProvider +          theme={theme === ColorScheme.Dark ? DarkTheme : LightTheme} +        > +          <ConsoleWrapper> +            <Completion +              size={COMPLETION_MAX_ITEMS} +              completions={state.completions} +              select={state.select} +            /> +            <Input +              mode={state.mode} +              onBlur={onBlur} +              onKeyDown={onKeyDown} +              onChange={onChange} +              value={state.consoleText} +            /> +          </ConsoleWrapper> +        </ThemeProvider> +      ); +    case "info": +    case "error": +      return ( +        <ThemeProvider +          theme={theme === ColorScheme.Dark ? DarkTheme : LightTheme} +        > +          <Message mode={state.mode}>{state.messageText}</Message> +        </ThemeProvider> +      ); +    default: +      return null; +  } +}; -export default connect(mapStateToProps)(Console); +export default Console; diff --git a/src/console/index.tsx b/src/console/index.tsx index f9313a0..cf9367b 100644 --- a/src/console/index.tsx +++ b/src/console/index.tsx @@ -1,42 +1,59 @@  import * as messages from "../shared/messages"; -import reducers from "./reducers"; -import { createStore, applyMiddleware } from "redux"; -import promise from "redux-promise"; +import reducers, { defaultState } from "./reducers";  import * as consoleActions from "./actions/console"; -import { Provider } from "react-redux";  import Console from "./components/Console"; +import AppContext from "./components/AppContext";  import "./index.css";  import React from "react";  import ReactDOM from "react-dom"; -const store = createStore(reducers, applyMiddleware(promise)); +const wrapAsync = <T extends unknown>( +  dispatch: React.Dispatch<T> +): React.Dispatch<T | Promise<T>> => { +  return (action: T | Promise<T>) => { +    if (action instanceof Promise) { +      action.then((a) => dispatch(a)).catch(console.error); +    } else { +      dispatch(action); +    } +  }; +}; -window.addEventListener("DOMContentLoaded", () => { -  const wrapper = document.getElementById("vimvixen-console"); -  ReactDOM.render( -    <Provider store={store}> -      <Console></Console> -    </Provider>, -    wrapper -  ); -}); +const RootComponent: React.FC = () => { +  const [state, dispatch] = React.useReducer(reducers, defaultState); + +  React.useEffect(() => { +    const onMessage = async (message: any): Promise<any> => { +      const msg = messages.valueOf(message); +      switch (msg.type) { +        case messages.CONSOLE_SHOW_COMMAND: +          return dispatch(await consoleActions.showCommand(msg.command)); +        case messages.CONSOLE_SHOW_FIND: +          return dispatch(consoleActions.showFind()); +        case messages.CONSOLE_SHOW_ERROR: +          return dispatch(consoleActions.showError(msg.text)); +        case messages.CONSOLE_SHOW_INFO: +          return dispatch(consoleActions.showInfo(msg.text)); +        case messages.CONSOLE_HIDE: +          return dispatch(consoleActions.hide()); +      } +    }; -const onMessage = async (message: any): Promise<any> => { -  const msg = messages.valueOf(message); -  switch (msg.type) { -    case messages.CONSOLE_SHOW_COMMAND: -      return store.dispatch(await consoleActions.showCommand(msg.command)); -    case messages.CONSOLE_SHOW_FIND: -      return store.dispatch(consoleActions.showFind()); -    case messages.CONSOLE_SHOW_ERROR: -      return store.dispatch(consoleActions.showError(msg.text)); -    case messages.CONSOLE_SHOW_INFO: -      return store.dispatch(consoleActions.showInfo(msg.text)); -    case messages.CONSOLE_HIDE: -      return store.dispatch(consoleActions.hide()); -  } +    browser.runtime.onMessage.addListener(onMessage); +    const port = browser.runtime.connect(undefined, { +      name: "vimvixen-console", +    }); +    port.onMessage.addListener(onMessage); +  }, []); + +  return ( +    <AppContext.Provider value={{ state, dispatch: wrapAsync(dispatch) }}> +      <Console /> +    </AppContext.Provider> +  );  }; -browser.runtime.onMessage.addListener(onMessage); -const port = browser.runtime.connect(undefined, { name: "vimvixen-console" }); -port.onMessage.addListener(onMessage); +window.addEventListener("DOMContentLoaded", () => { +  const wrapper = document.getElementById("vimvixen-console"); +  ReactDOM.render(<RootComponent />, wrapper); +}); diff --git a/src/console/reducers/index.ts b/src/console/reducers/index.ts index dbaf97d..49d0de1 100644 --- a/src/console/reducers/index.ts +++ b/src/console/reducers/index.ts @@ -14,7 +14,7 @@ export interface State {    colorscheme: ColorScheme;  } -const defaultState = { +export const defaultState = {    mode: "",    messageText: "",    consoleText: "", | 
