/** @jsx jsx */ import { jsx, css } from '@emotion/react' import React, { useState, useEffect, useReducer, useContext, useRef } from "react"; import { ButtonsSetting } from "../components/buttons_setting"; import { Button, buttons } from "../types/button"; import { LayerKey, layerKeys } from "../types/layer_key"; import { ButtonInLayer, ButtonsInLayer, ButtonsSettingType, Layers, Flip, Macro, StructMacro, ModeTable, StructMode } from "../types/buttons_setting_type"; import { HttpClient, SettingApiResponse } from "../lib/http_client"; import { ButtonState } from "./../lib/button_state"; import { ButtonStateDiff } from "./../lib/button_state_diff"; import { ButtonsSettingContext, } from "./../contexts/buttons_setting"; import { ButtonsSettingConverter } from "./../lib/buttons_setting_converter"; import { disableFlipType, alwaysFlipType, flipIfPressedSelfType, flipIfPressedSomeButtonsType, ignoreButtonsInFlipingType, remapType, closeMenuType, applyMacroType, installMacroType, installModeType, applyModeType } from "../reducers/layer_reducer"; import { ButtonsModal } from "../components/buttons_modal"; import { InstallableMacros } from "../components/installable_macros"; import { InstallableModes } from "../components/installable_modes"; import _ from 'lodash'; import md5 from 'md5'; const httpClient = new HttpClient(); interface LayerRef { setVisibility(status: string): string; }; export const ButtonsSettingPage = () => { const [loaded, DidLoad] = useReducer(() => { return true; }, false); const { layers, layersDispatch, prefixKeys, setPrefixKeys } = useContext(ButtonsSettingContext); const [selectedLayer, setSelectedLayer] = useState("up"); const layerRefs = layerKeys.map((l) => ({} as LayerRef)); const [initializedSetting, setInitializedSetting] = useState({} as ButtonsSettingType) const [infoMessage, setInfoMessage] = useState(undefined as undefined | string) const switchLayer = (event: React.MouseEvent) => { if (event.target instanceof HTMLElement) { setSelectedLayer(event.target.dataset.layerKey as LayerKey); layerRefs.forEach(r => r.setVisibility("hidden")); layerRefs[ Number(event.target.dataset.layerKeyIndex) ].setVisibility("shown"); } } const exportSetting = () => { const body = ButtonsSettingConverter({ prefixKeys: prefixKeys, layers: layers }) var data = new Blob([body], { type: 'text/yaml' }) var csvURL = window.URL.createObjectURL(data); const tempLink = document.createElement('a'); tempLink.href = csvURL; tempLink.setAttribute('download', 'setting.yml'); tempLink.click(); } const applySetting = (e: React.MouseEvent) => { e.preventDefault(); httpClient.postSetting( ButtonsSettingConverter({ prefixKeys: prefixKeys, layers: layers }) ).then( function (response) { alert("設定ファイルのパスへ書き込みが完了しました"); reset(); }); } const changes = (): Array => { return ButtonStateDiff({ before: initializedSetting, after: { prefix_keys_for_changing_layer: prefixKeys, layers: layers }, }) } const reset = () => { setInitializedSetting({ prefix_keys_for_changing_layer: prefixKeys, layers: _.cloneDeep(layers), }); isUptodate() } const isUptodate = () => { httpClient.getSettingDigest().then(function (response) { const digest = md5( ButtonsSettingConverter({ prefixKeys: prefixKeys, layers: layers }) ) console.log("digest:", digest); if(digest === response.data.digest) { console.log("最新です") setInfoMessage("設定が反映済みです。") } else { setInfoMessage("設定が未反映です. PBMの再起動が必要です。") } }).catch(function (error) { if (error.response.status === 404) { setInfoMessage("PBMを起動していません。") } else { console.log("想定外のエラーです"); } }) } useEffect(() => { if (loaded) { reset(); layerRefs[0].setVisibility("shown"); return; } httpClient.getSetting() .then(function (response) { const body = response.data.setting_group_by_button as SettingApiResponse; if(body.installed_macros) { body.installed_macros.forEach((installed_macro: string) => { layersDispatch({ type: installMacroType, payload: { installed_macro: installed_macro }}); }) } if(body.installed_modes) { body.installed_modes.forEach((modeName: string) => { layersDispatch({ type: installModeType, payload: { installed_mode: modeName } }); }) } setPrefixKeys(response.data.setting.prefix_keys_for_changing_layer.sort()); const layers = layerKeys.reduce((a, key) => { a[key] = response.data.setting_group_by_button.layers[key]; return a; }, {} as Layers) layerKeys.forEach((layerKey) => { const macroTable = layers[layerKey].macro as Macro if(macroTable) { Object.entries(macroTable).forEach((val, i) => { const name = val[0]; const ifPressed = val[1]; const macro = { name: name, if_pressed: ifPressed } as StructMacro layersDispatch({ type: applyMacroType, payload: { layerKey: layerKey, macro: macro }}); }) } const modeTable = layers[layerKey].mode as ModeTable; if(modeTable) { Object.entries(modeTable).forEach((val, i) => { const name = val[0]; const mode = { name: name } as StructMode; layersDispatch({ type: applyModeType, payload: { layerKey: layerKey, mode: mode }}); }) } else { layersDispatch({ type: applyModeType, payload: { layerKey: layerKey, mode: { name: "disable" } }}); } buttons.forEach((button) => { const buttonState = new ButtonState( button, layers[layerKey][button]?.flip, layers[layerKey][button]?.remap, ); const flip = layers[layerKey][button]?.flip || {} as Flip if(layers[layerKey][button] === undefined) { layersDispatch({ type: closeMenuType, payload: { layerKey: layerKey, button: button }}); } else if (layers[layerKey][button].remap?.to) { layersDispatch({ type: remapType, payload: { layerKey: layerKey, button: button, targetButtons: layers[layerKey][button].remap?.to }}); } else if (buttonState.hasFlipSetting()) { layersDispatch({ type: ignoreButtonsInFlipingType, payload: { layerKey: layerKey, button: button, targetButtons: flip.force_neutral }}); if(buttonState.isDisabledFlip()) { layersDispatch({ type: disableFlipType, payload: { layerKey: layerKey, button: button }}); } else if (buttonState.isAlwaysFlip()) { layersDispatch({ type: alwaysFlipType, payload: { layerKey: layerKey, button: button }}); } else if (buttonState.isFlipIfPressedSelf()) { layersDispatch({ type: flipIfPressedSelfType, payload: { layerKey: layerKey, button: button }}); } else if (buttonState.isFlipIfPressedSomeButtons()) { layersDispatch({ type: flipIfPressedSomeButtonsType, payload: { layerKey: layerKey, button: button, targetButtons: layers[layerKey][button].flip?.if_pressed }}); } } else { console.log("unexpectですです!!!!!!!!!!!!!"); } }); }); DidLoad(); }).catch(function (error) { if (error.response.status === 404) { setInfoMessage("設定ファイルのパスが未設定です") } else { console.log("想定外のエラーです"); } }) }, [loaded]); const layersTabStyle = () => { return css` list-style: none; display:flex; margin: 0; padding: 0; border-left: 1px solid #aaa; margin-bottom: 30px; li { border-top: 1px solid #aaa; border-right: 1px solid #aaa; &.active { border-bottom: 1px solid #white; } &.inactive { border-bottom: 1px solid #aaa; } a { padding: 20px; display: block; &:hover { cursor:pointer; } } } `; } const liClassName = (layer: LayerKey) => { if(layer === selectedLayer) { return "active" } else { return "inactive" }; } const handlePrefixKeysField = () => { toggleModal(); setModalTitle("キープレフィックスの変更") setModalPrefillButtons(prefixKeys); setModalCallbackOnSubmit(() => setPrefixKeys); setModalCloseCallback(() => toggleModal); } // for modal const [isOpenModal, toggleModal] = useReducer(((m) => { return !m; }), false); const [modalCallbackOnSubmit, setModalCallbackOnSubmit] = useState(undefined as any) const [modalCloseCallback, setModalCloseCallback] = useState(undefined as any) const [modalTitle, setModalTitle] = useState("") const [modalPrefillButtons, setModalPrefillButtons] = useState>([]) if(!loaded) { return null; }; return( <>

設定ファイルの変更

インストール可能なマクロ

{}

インストール可能なモード

{}

インストール可能なマクロ

{loaded && }

設定中のプレフィックスキー

{isOpenModal && }

変更した設定でsetting.ymlへ上書きする
{infoMessage}
    {changes().map((c, i) =>
  • {c}
  • )}
    {layerKeys.map((l, index) => (
  • {l}
  • ))}
{layerKeys.map((l, index) => ())} ) }