mirror of
https://github.com/outline/outline.git
synced 2026-06-13 03:14:59 +03:00
0139b91b5d
* chore: Replace lodash with es-toolkit Migrate all direct lodash imports to es-toolkit/compat for a smaller, faster, lodash-compatible utility library. Transitive lodash usage from other packages remains unchanged. * fix: Restore isPlainObject semantics in CanCan policy The lodash migration aliased `isObject` to `lodash/isPlainObject` and the codemod incorrectly mapped the local name to es-toolkit's `isObject`, which also returns true for arrays and functions. This caused condition objects in policy definitions to be skipped, breaking authorization checks across the codebase. * fix: Restore unicode-aware length counting in validators es-toolkit/compat's size() returns string.length, while lodash's _.size() counts unicode code points. Switch to [...value].length to preserve the previous behavior so multi-byte characters like emoji count as one.
58 lines
1.8 KiB
TypeScript
58 lines
1.8 KiB
TypeScript
import { find } from "es-toolkit/compat";
|
|
import { useEffect, useMemo } from "react";
|
|
import embeds from "@shared/editor/embeds";
|
|
import { IntegrationType, TeamPreference } from "@shared/types";
|
|
import type Integration from "~/models/Integration";
|
|
import Logger from "~/utils/Logger";
|
|
import useCurrentTeam from "./useCurrentTeam";
|
|
import useStores from "./useStores";
|
|
|
|
/**
|
|
* Hook to get all embed configuration for the current team
|
|
*
|
|
* @param loadIfMissing Should we load integration settings if they are not locally available
|
|
* @returns A list of embed descriptors
|
|
*/
|
|
export default function useEmbeds(loadIfMissing = false) {
|
|
const { integrations } = useStores();
|
|
const team = useCurrentTeam({ rejectOnEmpty: false });
|
|
|
|
useEffect(() => {
|
|
async function fetchEmbedIntegrations() {
|
|
try {
|
|
await integrations.fetchAll({
|
|
type: IntegrationType.Embed,
|
|
});
|
|
} catch (err) {
|
|
Logger.error("Failed to fetch embed integrations", err);
|
|
}
|
|
}
|
|
|
|
if (!integrations.isLoaded && !integrations.isFetching && loadIfMissing) {
|
|
void fetchEmbedIntegrations();
|
|
}
|
|
}, [integrations, loadIfMissing]);
|
|
|
|
const disabledEmbeds =
|
|
(team?.getPreference(TeamPreference.DisabledEmbeds) as string[]) || [];
|
|
|
|
return useMemo(
|
|
() =>
|
|
embeds.map((e) => {
|
|
// Find any integrations that match this embed and inject the settings
|
|
const integration: Integration<IntegrationType.Embed> | undefined =
|
|
find(integrations.orderedData, (i) => i.service === e.name);
|
|
|
|
if (integration?.settings) {
|
|
e.settings = integration.settings;
|
|
}
|
|
|
|
e.disabled = disabledEmbeds.includes(e.id);
|
|
|
|
return e;
|
|
}),
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
[integrations.orderedData, team?.preferences]
|
|
);
|
|
}
|