mirror of
https://github.com/blakeblackshear/frigate.git
synced 2024-11-26 19:06:11 +01:00
af3f6dadcb
* Ensure viewport is always full screen * Protect against hour with no cards and ensure data is consistent * Reduce grouped up image refreshes * Include current hour and fix scrubbing bugginess * Scroll initially selected timeline in to view * Expand timelne class type * Use poster image for preview on video player instead of using separate image view * Fix available streaming modes * Incrase timing for grouping timline items * Fix audio activity listener * Fix player not switching views correctly * Use player time to convert to timeline time * Update sub labels for previous timeline items * Show mini timeline bar for non selected items * Rewrite desktop timeline to use separate dynamic video player component * Extend improvements to mobile as well * Improve time formatting * Fix scroll * Fix no preview case * Mobile fixes * Audio toggle fixes * More fixes for mobile * Improve scaling of graph motion activity * Add keyboard shortcut hook and support shortcuts for playback page * Fix sizing of dialog * Improve height scaling of dialog * simplify and fix layout system for timeline * Fix timeilne items not working * Implement basic Frigate+ submitting from timeline
166 lines
4.2 KiB
TypeScript
166 lines
4.2 KiB
TypeScript
import useSWR from "swr";
|
|
import * as monaco from "monaco-editor";
|
|
import { configureMonacoYaml } from "monaco-yaml";
|
|
import { useCallback, useEffect, useRef, useState } from "react";
|
|
import { useApiHost } from "@/api";
|
|
import Heading from "@/components/ui/heading";
|
|
import ActivityIndicator from "@/components/ui/activity-indicator";
|
|
import { Button } from "@/components/ui/button";
|
|
import axios from "axios";
|
|
import copy from "copy-to-clipboard";
|
|
import { useTheme } from "@/context/theme-provider";
|
|
|
|
type SaveOptions = "saveonly" | "restart";
|
|
|
|
function ConfigEditor() {
|
|
const apiHost = useApiHost();
|
|
|
|
const { data: config } = useSWR<string>("config/raw");
|
|
|
|
const { theme } = useTheme();
|
|
const [success, setSuccess] = useState<string | undefined>();
|
|
const [error, setError] = useState<string | undefined>();
|
|
|
|
const editorRef = useRef<monaco.editor.IStandaloneCodeEditor | null>(null);
|
|
const modelRef = useRef<monaco.editor.IEditorModel | null>(null);
|
|
const configRef = useRef<HTMLDivElement | null>(null);
|
|
|
|
const onHandleSaveConfig = useCallback(
|
|
async (save_option: SaveOptions) => {
|
|
if (!editorRef.current) {
|
|
return;
|
|
}
|
|
|
|
axios
|
|
.post(
|
|
`config/save?save_option=${save_option}`,
|
|
editorRef.current.getValue(),
|
|
{
|
|
headers: { "Content-Type": "text/plain" },
|
|
}
|
|
)
|
|
.then((response) => {
|
|
if (response.status === 200) {
|
|
setError("");
|
|
setSuccess(response.data.message);
|
|
}
|
|
})
|
|
.catch((error) => {
|
|
setSuccess("");
|
|
|
|
if (error.response) {
|
|
setError(error.response.data.message);
|
|
} else {
|
|
setError(error.message);
|
|
}
|
|
});
|
|
},
|
|
[editorRef]
|
|
);
|
|
|
|
const handleCopyConfig = useCallback(async () => {
|
|
if (!editorRef.current) {
|
|
return;
|
|
}
|
|
|
|
copy(editorRef.current.getValue());
|
|
}, [editorRef]);
|
|
|
|
useEffect(() => {
|
|
if (!config) {
|
|
return;
|
|
}
|
|
|
|
if (modelRef.current != null) {
|
|
// we don't need to recreate the editor if it already exists
|
|
editorRef.current?.layout();
|
|
return;
|
|
}
|
|
|
|
const modelUri = monaco.Uri.parse("a://b/api/config/schema.json");
|
|
|
|
if (monaco.editor.getModels().length > 0) {
|
|
modelRef.current = monaco.editor.getModel(modelUri);
|
|
} else {
|
|
modelRef.current = monaco.editor.createModel(config, "yaml", modelUri);
|
|
}
|
|
|
|
configureMonacoYaml(monaco, {
|
|
enableSchemaRequest: true,
|
|
hover: true,
|
|
completion: true,
|
|
validate: true,
|
|
format: true,
|
|
schemas: [
|
|
{
|
|
uri: `${apiHost}api/config/schema.json`,
|
|
fileMatch: [String(modelUri)],
|
|
},
|
|
],
|
|
});
|
|
|
|
const container = configRef.current;
|
|
|
|
if (container != null) {
|
|
editorRef.current = monaco.editor.create(container, {
|
|
language: "yaml",
|
|
model: modelRef.current,
|
|
scrollBeyondLastLine: false,
|
|
theme: theme == "dark" ? "vs-dark" : "vs-light",
|
|
});
|
|
}
|
|
|
|
return () => {
|
|
configRef.current = null;
|
|
editorRef.current = null;
|
|
modelRef.current = null;
|
|
};
|
|
});
|
|
|
|
if (!config) {
|
|
return <ActivityIndicator />;
|
|
}
|
|
|
|
return (
|
|
<div className="absolute top-28 bottom-16 right-0 left-0 md:left-24 lg:left-60">
|
|
<div className="lg:flex justify-between mr-1">
|
|
<Heading as="h2">Config</Heading>
|
|
<div>
|
|
<Button
|
|
size="sm"
|
|
className="mx-1"
|
|
onClick={(_) => handleCopyConfig()}
|
|
>
|
|
Copy Config
|
|
</Button>
|
|
<Button
|
|
size="sm"
|
|
className="mx-1"
|
|
onClick={(_) => onHandleSaveConfig("restart")}
|
|
>
|
|
Save & Restart
|
|
</Button>
|
|
<Button
|
|
size="sm"
|
|
className="mx-1"
|
|
onClick={(_) => onHandleSaveConfig("saveonly")}
|
|
>
|
|
Save Only
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
|
|
{success && <div className="max-h-20 text-success">{success}</div>}
|
|
{error && (
|
|
<div className="p-4 overflow-scroll text-danger whitespace-pre-wrap">
|
|
{error}
|
|
</div>
|
|
)}
|
|
|
|
<div ref={configRef} className="h-full mt-2" />
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default ConfigEditor;
|