2024-03-11 17:02:15 +02:00
|
|
|
import * as React from 'react';
|
2024-05-02 18:05:25 +02:00
|
|
|
import { useCallback, useState } from 'react';
|
2024-03-11 17:02:15 +02:00
|
|
|
import { ConfigScreenStyles } from '../configScreenStyles';
|
|
|
|
import { View } from 'react-native';
|
|
|
|
import { Banner, Button, Text } from 'react-native-paper';
|
|
|
|
import { _ } from '@joplin/lib/locale';
|
2024-04-27 12:45:39 +02:00
|
|
|
import PluginService, { PluginSettings, SerializedPluginSettings } from '@joplin/lib/services/plugins/PluginService';
|
2024-03-11 17:02:15 +02:00
|
|
|
import PluginToggle from './PluginToggle';
|
|
|
|
import SearchPlugins from './SearchPlugins';
|
|
|
|
import { ItemEvent } from '@joplin/lib/components/shared/config/plugins/types';
|
|
|
|
import NavService from '@joplin/lib/services/NavService';
|
|
|
|
import useRepoApi from './utils/useRepoApi';
|
|
|
|
import RepositoryApi from '@joplin/lib/services/plugins/RepositoryApi';
|
2024-05-02 18:05:25 +02:00
|
|
|
import useAsyncEffect from '@joplin/lib/hooks/useAsyncEffect';
|
2024-03-11 17:02:15 +02:00
|
|
|
|
|
|
|
interface Props {
|
|
|
|
themeId: number;
|
|
|
|
styles: ConfigScreenStyles;
|
2024-04-27 12:45:39 +02:00
|
|
|
pluginSettings: SerializedPluginSettings;
|
2024-03-11 17:02:15 +02:00
|
|
|
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;
|
|
|
|
};
|
|
|
|
|
2024-04-27 12:45:39 +02:00
|
|
|
// Loaded plugins: All plugins with available manifests.
|
2024-05-02 18:05:25 +02:00
|
|
|
const useLoadedPluginIds = () => {
|
|
|
|
const getLoadedPlugins = useCallback(() => {
|
|
|
|
return PluginService.instance().pluginIds;
|
|
|
|
}, []);
|
|
|
|
const [loadedPluginIds, setLoadedPluginIds] = useState(getLoadedPlugins);
|
2024-04-27 12:45:39 +02:00
|
|
|
|
2024-05-02 18:05:25 +02:00
|
|
|
useAsyncEffect(async event => {
|
|
|
|
while (!event.cancelled) {
|
|
|
|
await PluginService.instance().waitForLoadedPluginsChange();
|
|
|
|
setLoadedPluginIds(getLoadedPlugins());
|
|
|
|
}
|
|
|
|
}, []);
|
2024-04-27 12:45:39 +02:00
|
|
|
|
|
|
|
return loadedPluginIds;
|
|
|
|
};
|
|
|
|
|
2024-03-11 17:02:15 +02:00
|
|
|
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;
|
|
|
|
});
|
2024-04-03 19:51:09 +02:00
|
|
|
const updatablePluginIds = await repoApi.canBeUpdatedPlugins(manifests);
|
2024-03-11 17:02:15 +02:00
|
|
|
|
|
|
|
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();
|
2024-04-27 12:45:39 +02:00
|
|
|
|
2024-05-02 18:05:25 +02:00
|
|
|
const pluginIds = useLoadedPluginIds();
|
2024-04-27 12:45:39 +02:00
|
|
|
for (const pluginId of pluginIds) {
|
|
|
|
const plugin = pluginService.plugins[pluginId];
|
2024-03-11 17:02:15 +02:00
|
|
|
|
|
|
|
if (!props.shouldShowBasedOnSearchQuery || props.shouldShowBasedOnSearchQuery(plugin.manifest.name)) {
|
|
|
|
installedPluginCards.push(
|
|
|
|
<PluginToggle
|
2024-04-27 12:45:39 +02:00
|
|
|
key={`plugin-${pluginId}`}
|
2024-04-25 15:02:10 +02:00
|
|
|
themeId={props.themeId}
|
2024-04-27 12:45:39 +02:00
|
|
|
pluginId={pluginId}
|
2024-03-11 17:02:15 +02:00
|
|
|
styles={props.styles}
|
|
|
|
pluginSettings={props.pluginSettings}
|
|
|
|
updatablePluginIds={updatablePluginIds}
|
|
|
|
updatePluginStates={props.updatePluginStates}
|
|
|
|
onShowPluginLog={onShowPluginLog}
|
|
|
|
repoApi={repoApi}
|
|
|
|
/>,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const showSearch = (
|
2024-03-29 14:40:54 +02:00
|
|
|
!props.shouldShowBasedOnSearchQuery || props.shouldShowBasedOnSearchQuery(searchInputSearchText())
|
2024-03-11 17:02:15 +02:00
|
|
|
);
|
|
|
|
|
|
|
|
const searchComponent = (
|
|
|
|
<SearchPlugins
|
|
|
|
pluginSettings={props.pluginSettings}
|
|
|
|
themeId={props.themeId}
|
|
|
|
onUpdatePluginStates={props.updatePluginStates}
|
|
|
|
repoApiInitialized={repoApiLoaded}
|
|
|
|
repoApi={repoApi}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<View>
|
|
|
|
{renderRepoApiStatus()}
|
|
|
|
{installedPluginCards}
|
|
|
|
{showSearch ? searchComponent : null}
|
|
|
|
</View>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default PluginStates;
|