1
0
mirror of https://github.com/laurent22/joplin.git synced 2024-12-18 09:35:20 +02:00
joplin/packages/app-mobile/components/screens/ConfigScreen/plugins/PluginToggle.tsx

107 lines
3.5 KiB
TypeScript

import * as React from 'react';
import { ConfigScreenStyles } from '../configScreenStyles';
import PluginService, { PluginSettings, defaultPluginSetting } from '@joplin/lib/services/plugins/PluginService';
import { useCallback, useMemo, useState } from 'react';
import PluginBox, { UpdateState } from './PluginBox';
import useOnDeleteHandler from '@joplin/lib/components/shared/config/plugins/useOnDeleteHandler';
import { ItemEvent, OnPluginSettingChangeEvent } from '@joplin/lib/components/shared/config/plugins/types';
import useOnInstallHandler from '@joplin/lib/components/shared/config/plugins/useOnInstallHandler';
import RepositoryApi from '@joplin/lib/services/plugins/RepositoryApi';
interface Props {
pluginId: string;
styles: ConfigScreenStyles;
pluginSettings: string;
updatablePluginIds: Record<string, boolean>;
repoApi: RepositoryApi;
onShowPluginLog: (event: ItemEvent)=> void;
updatePluginStates: (settingValue: PluginSettings)=> void;
}
const PluginToggle: React.FC<Props> = props => {
const pluginService = useMemo(() => PluginService.instance(), []);
const plugin = useMemo(() => {
return pluginService.pluginById(props.pluginId);
}, [pluginService, props.pluginId]);
const pluginSettings = useMemo(() => {
const settings = { ...pluginService.unserializePluginSettings(props.pluginSettings) };
if (!settings[props.pluginId]) {
settings[props.pluginId] = defaultPluginSetting();
}
return settings;
}, [props.pluginSettings, pluginService, props.pluginId]);
const onPluginSettingsChange = useCallback((event: OnPluginSettingChangeEvent) => {
props.updatePluginStates(event.value);
}, [props.updatePluginStates]);
const updatePluginEnabled = useCallback((enabled: boolean) => {
const newSettings = { ...pluginSettings };
newSettings[props.pluginId].enabled = enabled;
props.updatePluginStates(newSettings);
}, [pluginSettings, props.pluginId, props.updatePluginStates]);
const pluginId = plugin.manifest.id;
const onToggle = useCallback(() => {
const settings = pluginSettings[pluginId];
updatePluginEnabled(!settings.enabled);
}, [pluginSettings, updatePluginEnabled, pluginId]);
const onDelete = useOnDeleteHandler(pluginSettings, onPluginSettingsChange, true);
const [updatingPluginIds, setUpdatingPluginIds] = useState<Record<string, boolean>>({});
const onUpdate = useOnInstallHandler(setUpdatingPluginIds, pluginSettings, props.repoApi, onPluginSettingsChange, true);
const updateState = useMemo(() => {
const settings = pluginSettings[pluginId];
if (settings.hasBeenUpdated) {
return UpdateState.HasBeenUpdated;
}
if (updatingPluginIds[pluginId]) {
return UpdateState.Updating;
}
if (props.updatablePluginIds[pluginId]) {
return UpdateState.CanUpdate;
}
return UpdateState.Idle;
}, [pluginSettings, updatingPluginIds, pluginId, props.updatablePluginIds]);
const pluginItem = useMemo(() => {
const settings = pluginSettings[pluginId];
return {
manifest: plugin.manifest,
enabled: settings.enabled,
deleted: settings.deleted,
devMode: plugin.devMode,
builtIn: plugin.builtIn,
hasBeenUpdated: settings.hasBeenUpdated,
};
}, [plugin, pluginId, pluginSettings]);
const isCompatible = useMemo(() => {
return PluginService.instance().isCompatible(plugin.manifest);
}, [plugin]);
return (
<PluginBox
item={pluginItem}
isCompatible={isCompatible}
hasErrors={plugin.hasErrors}
onShowPluginLog={props.onShowPluginLog}
onToggle={onToggle}
onDelete={onDelete}
onUpdate={onUpdate}
updateState={updateState}
/>
);
};
export default PluginToggle;