1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-01-11 18:24:43 +02:00
joplin/packages/app-mobile/components/screens/ConfigScreen/plugins/PluginStates.tsx
Henry Heino 55cafb8891
Android: Add support for Markdown editor plugins (#10086)
Co-authored-by: Laurent Cozic <laurent22@users.noreply.github.com>
2024-03-11 15:02:15 +00:00

148 lines
4.7 KiB
TypeScript

import * as React from 'react';
import { useCallback, useState } from 'react';
import { ConfigScreenStyles } from '../configScreenStyles';
import { View } from 'react-native';
import { Banner, Button, Text } from 'react-native-paper';
import { _ } from '@joplin/lib/locale';
import PluginService, { PluginSettings } from '@joplin/lib/services/plugins/PluginService';
import PluginToggle from './PluginToggle';
import SearchPlugins from './SearchPlugins';
import shim from '@joplin/lib/shim';
import { ItemEvent } from '@joplin/lib/components/shared/config/plugins/types';
import NavService from '@joplin/lib/services/NavService';
import isInstallingPluginsAllowed from './utils/isPluginInstallingAllowed';
import useRepoApi from './utils/useRepoApi';
import RepositoryApi from '@joplin/lib/services/plugins/RepositoryApi';
interface Props {
themeId: number;
styles: ConfigScreenStyles;
pluginSettings: string;
settingsSearchQuery?: string;
updatePluginStates: (settingValue: PluginSettings)=> void;
shouldShowBasedOnSearchQuery: ((relatedText: string|string[])=> boolean)|null;
}
// Text used for determining whether to display the setting or not.
const searchInputSearchText = () => [_('Search'), _('Plugin search')];
export const getSearchText = () => {
const plugins = PluginService.instance().plugins;
const searchText = [];
for (const key in plugins) {
const plugin = plugins[key];
searchText.push(plugin.manifest.name);
}
searchText.push(...searchInputSearchText());
return searchText;
};
const PluginStates: React.FC<Props> = props => {
const [repoApiError, setRepoApiError] = useState(null);
const [repoApiLoaded, setRepoApiLoaded] = useState(false);
const [reloadRepoCounter, setRepoReloadCounter] = useState(0);
const [updatablePluginIds, setUpdatablePluginIds] = useState<Record<string, boolean>>({});
const onRepoApiLoaded = useCallback(async (repoApi: RepositoryApi) => {
const manifests = Object.values(PluginService.instance().plugins)
.filter(plugin => !plugin.builtIn && !plugin.devMode)
.map(plugin => {
return plugin.manifest;
});
const updatablePluginIds = await repoApi.canBeUpdatedPlugins(manifests, PluginService.instance().appVersion);
const conv: Record<string, boolean> = {};
for (const id of updatablePluginIds) {
conv[id] = true;
}
setRepoApiLoaded(true);
setUpdatablePluginIds(conv);
}, []);
const repoApi = useRepoApi({ setRepoApiError, onRepoApiLoaded, reloadRepoCounter });
const reloadPluginRepo = useCallback(() => {
setRepoReloadCounter(reloadRepoCounter + 1);
}, [reloadRepoCounter, setRepoReloadCounter]);
const renderRepoApiStatus = () => {
if (repoApiLoaded) {
if (!repoApi.isUsingDefaultContentUrl) {
const url = new URL(repoApi.contentBaseUrl);
return (
<Banner visible={true} icon='alert'>{_('Content provided by: %s', url.hostname)}</Banner>
);
}
return null;
}
if (repoApiError) {
return <View style={{ flexDirection: 'row' }}>
<Text>{_('Plugin repository failed to load')}</Text>
<Button onPress={reloadPluginRepo}>{_('Retry')}</Button>
</View>;
} else {
return <Text>{_('Loading plugin repository...')}</Text>;
}
};
const onShowPluginLog = useCallback((event: ItemEvent) => {
const pluginId = event.item.manifest.id;
void NavService.go('Log', { defaultFilter: pluginId });
}, []);
const installedPluginCards = [];
const pluginService = PluginService.instance();
for (const key in pluginService.plugins) {
const plugin = pluginService.plugins[key];
if (!props.shouldShowBasedOnSearchQuery || props.shouldShowBasedOnSearchQuery(plugin.manifest.name)) {
installedPluginCards.push(
<PluginToggle
key={`plugin-${key}`}
pluginId={plugin.id}
styles={props.styles}
pluginSettings={props.pluginSettings}
updatablePluginIds={updatablePluginIds}
updatePluginStates={props.updatePluginStates}
onShowPluginLog={onShowPluginLog}
repoApi={repoApi}
/>,
);
}
}
const showSearch = (
isInstallingPluginsAllowed() && (
!props.shouldShowBasedOnSearchQuery || props.shouldShowBasedOnSearchQuery(searchInputSearchText())
)
);
const renderIosSearchWarning = () => {
if (shim.mobilePlatform() !== 'ios' || !showSearch) return null;
return <Banner visible={true} icon='information'>{'Note: Plugin search is usually disabled on iOS (and only enabled in dev mode).'}</Banner>;
};
const searchComponent = (
<SearchPlugins
pluginSettings={props.pluginSettings}
themeId={props.themeId}
onUpdatePluginStates={props.updatePluginStates}
repoApiInitialized={repoApiLoaded}
repoApi={repoApi}
/>
);
return (
<View>
{renderRepoApiStatus()}
{renderIosSearchWarning()}
{installedPluginCards}
{showSearch ? searchComponent : null}
</View>
);
};
export default PluginStates;