forked from Github/frigate
* use react-use-websockets * check ready state * match context shape * jsonify dispatch * remove unnecessary ready check * bring back h * non-working tests * skip failing tests * upgrade some dependencies --------- Co-authored-by: Josh Hawkins <32435876+hawkeye217@users.noreply.github.com>
135 lines
3.7 KiB
JavaScript
135 lines
3.7 KiB
JavaScript
import { h, createContext } from 'preact';
|
|
import { baseUrl } from './baseUrl';
|
|
import { produce } from 'immer';
|
|
import { useCallback, useContext, useEffect, useReducer } from 'preact/hooks';
|
|
import useWebSocket, { ReadyState } from 'react-use-websocket';
|
|
|
|
const initialState = Object.freeze({ __connected: false });
|
|
export const WS = createContext({ state: initialState, readyState: null, sendJsonMessage: () => {} });
|
|
|
|
function reducer(state, { topic, payload, retain }) {
|
|
switch (topic) {
|
|
case '__CLIENT_CONNECTED':
|
|
return produce(state, (draftState) => {
|
|
draftState.__connected = true;
|
|
});
|
|
|
|
default:
|
|
return produce(state, (draftState) => {
|
|
let parsedPayload = payload;
|
|
try {
|
|
parsedPayload = payload && JSON.parse(payload);
|
|
} catch (e) {}
|
|
draftState[topic] = {
|
|
lastUpdate: Date.now(),
|
|
payload: parsedPayload,
|
|
retain,
|
|
};
|
|
});
|
|
}
|
|
}
|
|
|
|
export function WsProvider({
|
|
config,
|
|
children,
|
|
wsUrl = `${baseUrl.replace(/^http/, 'ws')}ws`,
|
|
}) {
|
|
const [state, dispatch] = useReducer(reducer, initialState);
|
|
|
|
const { sendJsonMessage, readyState } = useWebSocket(wsUrl, {
|
|
|
|
onMessage: (event) => {
|
|
dispatch(JSON.parse(event.data));
|
|
},
|
|
onOpen: () => dispatch({ topic: '__CLIENT_CONNECTED' }),
|
|
shouldReconnect: () => true,
|
|
});
|
|
|
|
useEffect(() => {
|
|
Object.keys(config.cameras).forEach((camera) => {
|
|
const { name, record, detect, snapshots, audio } = config.cameras[camera];
|
|
dispatch({ topic: `${name}/recordings/state`, payload: record.enabled ? 'ON' : 'OFF', retain: false });
|
|
dispatch({ topic: `${name}/detect/state`, payload: detect.enabled ? 'ON' : 'OFF', retain: false });
|
|
dispatch({ topic: `${name}/snapshots/state`, payload: snapshots.enabled ? 'ON' : 'OFF', retain: false });
|
|
dispatch({ topic: `${name}/audio/state`, payload: audio.enabled ? 'ON' : 'OFF', retain: false });
|
|
});
|
|
}, [config]);
|
|
|
|
return <WS.Provider value={{ state, readyState, sendJsonMessage }}>{children}</WS.Provider>;
|
|
}
|
|
|
|
export function useWs(watchTopic, publishTopic) {
|
|
const { state, readyState, sendJsonMessage } = useContext(WS);
|
|
|
|
const value = state[watchTopic] || { payload: null };
|
|
|
|
const send = useCallback(
|
|
(payload, retain = false) => {
|
|
if (readyState === ReadyState.OPEN) {
|
|
sendJsonMessage({
|
|
topic: publishTopic || watchTopic,
|
|
payload,
|
|
retain,
|
|
});
|
|
}
|
|
},
|
|
[sendJsonMessage, readyState, watchTopic, publishTopic]
|
|
);
|
|
|
|
return { value, send, connected: state.__connected };
|
|
}
|
|
|
|
export function useDetectState(camera) {
|
|
const {
|
|
value: { payload },
|
|
send,
|
|
connected,
|
|
} = useWs(`${camera}/detect/state`, `${camera}/detect/set`);
|
|
return { payload, send, connected };
|
|
}
|
|
|
|
export function useRecordingsState(camera) {
|
|
const {
|
|
value: { payload },
|
|
send,
|
|
connected,
|
|
} = useWs(`${camera}/recordings/state`, `${camera}/recordings/set`);
|
|
return { payload, send, connected };
|
|
}
|
|
|
|
export function useSnapshotsState(camera) {
|
|
const {
|
|
value: { payload },
|
|
send,
|
|
connected,
|
|
} = useWs(`${camera}/snapshots/state`, `${camera}/snapshots/set`);
|
|
return { payload, send, connected };
|
|
}
|
|
|
|
export function useAudioState(camera) {
|
|
const {
|
|
value: { payload },
|
|
send,
|
|
connected,
|
|
} = useWs(`${camera}/audio/state`, `${camera}/audio/set`);
|
|
return { payload, send, connected };
|
|
}
|
|
|
|
export function usePtzCommand(camera) {
|
|
const {
|
|
value: { payload },
|
|
send,
|
|
connected,
|
|
} = useWs(`${camera}/ptz`, `${camera}/ptz`);
|
|
return { payload, send, connected };
|
|
}
|
|
|
|
export function useRestart() {
|
|
const {
|
|
value: { payload },
|
|
send,
|
|
connected,
|
|
} = useWs('restart', 'restart');
|
|
return { payload, send, connected };
|
|
}
|