Skip to content

Commit

Permalink
chore: PEM-6153 fixed the fragment url to follow the actual content (#…
Browse files Browse the repository at this point in the history
…4697) (#4862)

* PEM-6153-fixed the fragment url to follow the actual content

* ci: auto-formatting prettier issues

* changed the click as scrollIntoView

* reverting back to click

* fixed the fragment url for main Readme as well

* ci: auto-formatting prettier issues

* cleaned up

* ci: auto-formatting prettier issues

* addressed the review comments

* added fragment link support for h1 as well

---------

Co-authored-by: nage1234 <[email protected]>
Co-authored-by: Karl Cardenas <[email protected]>
(cherry picked from commit db85b2d)

Co-authored-by: Nageswari Raghunathan <[email protected]>
  • Loading branch information
1 parent 5d7ede7 commit 9fd0c1e
Show file tree
Hide file tree
Showing 5 changed files with 156 additions and 66 deletions.
1 change: 0 additions & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -56,7 +56,6 @@
"docusaurus-plugin-sass": "^0.2.5",
"docusaurus-theme-openapi-docs": "0.0.0-949",
"fuse.js": "^6.6.2",
"markdown-to-jsx": "^7.0.0",
"node-fetch": "^3.1.0",
"p-ratelimit": "^1.0.1",
"prism-react-renderer": "^2.1.0",
Expand Down
2 changes: 1 addition & 1 deletion plugins/packs-integrations.js
Original file line number Diff line number Diff line change
Expand Up @@ -285,7 +285,7 @@ function generateRoutes(packsAllData) {
metadata: {
sourceFilePath: "../docs/docs-content/integrations/packs.mdx",
},
data: { name: pack.name, version: pack.latestVersion, parent: parentVersion?.title },
data: { name: pack.name, version: pack.latestVersion, parent: parentVersion?.title, tab: "main" },
};
});
}
Expand Down
5 changes: 4 additions & 1 deletion src/components/PacksInformation/PacksInformation.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ interface PacksData {
name: string;
version: string;
parent: string;
tab: string;
};
}

Expand All @@ -30,7 +31,9 @@ export default function Packs(props: Packs) {
const data = props?.route?.data;
return data ? (
<Switch>
<Redirect to={`/integrations/packs?pack=${data.name}&version=${data.version}&parent=${data.parent}`} />
<Redirect
to={`/integrations/packs?pack=${data.name}&version=${data.version}&parent=${data.parent}&tab=${data.tab}`}
/>
</Switch>
) : (
<PacksReadme />
Expand Down
212 changes: 150 additions & 62 deletions src/components/PacksReadme/PacksReadme.tsx
Original file line number Diff line number Diff line change
@@ -1,9 +1,8 @@
import React, { useEffect, useState, useMemo, ReactElement } from "react";
import styles from "./PacksReadme.module.scss";
import { Tabs, ConfigProvider, theme, TreeSelect } from "antd";
import { ConfigProvider, theme, TreeSelect, Tabs } from "antd";
import CustomLabel from "../Technologies/CategorySelector/CustomLabel";
import PackCardIcon from "../Technologies/PackCardIcon";
import Markdown from "markdown-to-jsx";
import { useHistory } from "react-router-dom";
import "./PacksReadme.antd.css";
import { usePluginData } from "@docusaurus/useGlobalData";
Expand All @@ -14,10 +13,12 @@ import { useColorMode } from "@docusaurus/theme-common";
import { packTypeNames, cloudDisplayNames } from "../../constants/packs";
import Admonition from "@theme/Admonition";
import { Redirect } from "react-router-dom";

import useIsBrowser from "@docusaurus/useIsBrowser";
import ReactMarkDown from "react-markdown";
import remarkGfm from "remark-gfm";
interface PackReadmeProps {
customDescription: string;
packUidMap: Record<string, { deprecated?: boolean; readme?: string; registryUid?: string }>;
packUidMap: Record<string, { deprecated?: boolean; readme?: ReactElement; registryUid?: string }>;
versions: Version[];
title: string;
logoUrl: string;
Expand Down Expand Up @@ -62,11 +63,12 @@ function versionChange(
setSelectedPackUid: React.Dispatch<React.SetStateAction<string>>,
getParentVersion: (version: string) => Version | undefined,
history: ReturnType<typeof useHistory>,
packName: string
packName: string,
selectedTabPane: string = "main"
) {
const [version, packUid] = item.split("===");
const parentVersion = getParentVersion(version)?.title || "";
history.replace({ search: `?pack=${packName}&version=${version}&parent=${parentVersion}` });
history.replace({ search: `?pack=${packName}&version=${version}&parent=${parentVersion}&tab=${selectedTabPane}` });
setSelectedVersion(version);
setSelectedPackUid(packUid);
}
Expand Down Expand Up @@ -110,63 +112,70 @@ function renderVersionOptions(packData: PackData) {
}));
}

function renderTabs(selectedPackUid: string, packData: PackData, customReadme: ReactElement<any, any> | null) {
const empty_icon_light = useBaseUrl("/img/empty_icon_table_light.svg");
const empty_icon_dark = useBaseUrl("/img/empty_icon_table_dark.svg");
const readme = selectedPackUid ? packData.packUidMap[selectedPackUid]?.readme : "";
const tabs = [
readme && {
label: `README`,
key: "1",
children: <Markdown>{readme}</Markdown>,
},
customReadme && {
label: `Additional Details`,
key: "2",
children: customReadme,
},
].filter(Boolean) as { label: string; key: string; children: JSX.Element }[];

if (tabs.length > 1) {
return (
<Tabs defaultActiveKey="1">
{tabs.map((item) => (
<Tabs.TabPane tab={item.label} key={item.key}>
{item.children}
</Tabs.TabPane>
))}
</Tabs>
);
}
if (tabs.length === 1) {
return tabs[0].children;
}
return (
<div className={styles.emptyContent}>
<ThemedImage
alt="Docusaurus themed image"
sources={{
light: empty_icon_light,
dark: empty_icon_dark,
}}
width={120}
height={120}
/>
<div className={styles.emptyContentTitle}>No README found</div>
</div>
);
}

function getProviders(packData: PackData) {
if (packData.provider.includes("all")) {
return "All";
}

return packData.provider
.map((provider) => cloudDisplayNames[provider as keyof typeof cloudDisplayNames] || provider)
.join(", ");
}

function processPackUiMap(
packUidMap: Record<string, { readme?: string }>
): Record<string, { deprecated?: boolean; readme?: ReactElement; registryUid?: string }> {
const newPackUidMap: Record<string, { deprecated?: boolean; readme?: ReactElement; registryUid?: string }> = {};
Object.entries(packUidMap).map(([key, value]) => {
if (key) {
const readmeFileName = packUidMap[key].readme;
let module;
if (readmeFileName) {
module = (
<ReactMarkDown
remarkPlugins={[remarkGfm]}
components={{
h1: (props) => {
const headingId = props.children?.toString().replace(/\s+/g, "-").toLowerCase();
return (
<h1 id={headingId}>
{props.children}
<a href={`#${headingId}`} className="hash-link" />
</h1>
);
},
h2: (props) => {
const headingId = props.children?.toString().replace(/\s+/g, "-").toLowerCase();
return (
<h2 id={headingId}>
{props.children}
<a href={`#${headingId}`} className="hash-link" />
</h2>
);
},
h3: (props) => {
const headingId = props.children?.toString().replace(/\s+/g, "-").toLowerCase();
return (
<h3 id={headingId}>
{props.children}
<a href={`#${headingId}`} className="hash-link" />
</h3>
);
},
}}
>
{readmeFileName}
</ReactMarkDown>
);
}
newPackUidMap[key] = {
...value,
readme: module ? (module as ReactElement) : undefined,
};
}
});
return newPackUidMap;
}

function getRegistries(packData: PackData, selectedVersion: string, selectedPackUid: string) {
if (selectedVersion && !selectedVersion.endsWith(".x")) {
const registryUid = packData.packUidMap[selectedPackUid]?.registryUid || "";
Expand All @@ -186,19 +195,28 @@ function getRegistries(packData: PackData, selectedVersion: string, selectedPack
export default function PacksReadme() {
try {
const { packs, repositories } = usePluginData("plugin-packs-integrations") as PacksIntegrationsPluginData;

const [fragmentIdentifier, setFragmentIdentifier] = useState<string>("");
const [customReadme, setCustomReadme] = useState<ReactElement<any, any> | null>(null);
const [packName, setPackName] = useState<string>("");
const [selectedPackUid, setSelectedPackUid] = useState<string>("");
const { colorMode } = useColorMode();
const { defaultAlgorithm, darkAlgorithm } = theme;
const [selectedTabPane, setSelectedTabPane] = useState<string>("");
const [selectedVersion, setSelectedVersion] = useState<string>("");
const history = useHistory();
const isBrowser = useIsBrowser();

useEffect(() => {
const searchParams = window ? new URLSearchParams(window.location.search) : null;
const pckName = searchParams?.get("pack") || "";
setPackName(pckName);
let pckName = "";
if (isBrowser) {
const searchParams = window ? new URLSearchParams(window.location.search) : null;
const hashLocationValue = window ? window.location.hash : "";
pckName = searchParams?.get("pack") || "";
setPackName(pckName);
if (hashLocationValue) {
setFragmentIdentifier(hashLocationValue);
}
}
const importComponent = async () => {
try {
const module: MarkdownFile = await import(`../../../docs/docs-content/integrations/${pckName}.md`);
Expand All @@ -223,7 +241,7 @@ export default function PacksReadme() {
if (pack) {
const packDataInfo: PackReadmeProps = {
customDescription: pack.description,
packUidMap: pack.packUidMap,
packUidMap: processPackUiMap(pack.packUidMap),
versions: pack.versions,
title: pack.title,
logoUrl: pack.logoUrl,
Expand Down Expand Up @@ -252,8 +270,24 @@ export default function PacksReadme() {
}, [packName]);

useEffect(() => {
const searchParams = window ? new URLSearchParams(window.location.search) : null;
if (isBrowser && fragmentIdentifier) {
const timeoutId = setTimeout(() => {
const elementId = fragmentIdentifier.replace("#", "");
const parent = document.getElementById?.(elementId);
parent?.querySelector?.("a")?.click();
}, 1000);
return () => clearTimeout(timeoutId);
}
}, [fragmentIdentifier]);

useEffect(() => {
let searchParams;
if (isBrowser) {
searchParams = new URLSearchParams(window.location.search);
}
const urlParamVersion = searchParams?.get("version");
const urlParamTag = searchParams?.get("tab");
setSelectedTabPane(urlParamTag || "main");
const version = urlParamVersion || packData?.latestVersion || packData?.versions[0]?.title || "";
if (version && !version.endsWith(".x")) {
const parentVersionObj = getParentVersion(version, packData);
Expand All @@ -265,6 +299,59 @@ export default function PacksReadme() {
}
}, [packData]);

const renderTabs = () => {
const empty_icon_light = useBaseUrl("/img/empty_icon_table_light.svg");
const empty_icon_dark = useBaseUrl("/img/empty_icon_table_dark.svg");
const ReadMe = selectedPackUid ? packData.packUidMap[selectedPackUid]?.readme : null;

const tabs = [
ReadMe && {
label: `README`,
key: "main",
children: ReadMe,
},
customReadme && {
label: `Additional Details`,
key: "custom",
children: customReadme,
},
].filter(Boolean) as { label: string; key: string; children: JSX.Element }[];
if (tabs.length > 1) {
return (
<div>
<Tabs
defaultActiveKey={selectedTabPane || "main"}
items={tabs}
onChange={(key) => {
setSelectedTabPane(key);
const parentVersion = getParentVersion(selectedVersion, packData)?.title || "";
history.replace({
search: `?pack=${packName}&version=${selectedVersion}&parent=${parentVersion}&tab=${key}`,
});
}}
/>
</div>
);
}
if (tabs.length === 1) {
return tabs[0].children;
}
return (
<div className={styles.emptyContent}>
<ThemedImage
alt="Docusaurus themed image"
sources={{
light: empty_icon_light,
dark: empty_icon_dark,
}}
width={120}
height={120}
/>
<div className={styles.emptyContentTitle}>No README found</div>
</div>
);
};

return (
<div className={styles.wrapper}>
<div className={styles.description}>
Expand Down Expand Up @@ -292,7 +379,8 @@ export default function PacksReadme() {
setSelectedPackUid,
(version) => getParentVersion(version, packData),
history,
packName
packName,
selectedTabPane
)
}
treeData={renderVersionOptions(packData)}
Expand Down Expand Up @@ -322,7 +410,7 @@ export default function PacksReadme() {
</div>
<div className={styles.tabPane}>
<ConfigProvider theme={{ algorithm: colorMode === "dark" ? darkAlgorithm : defaultAlgorithm }}>
{renderTabs(selectedPackUid, packData, customReadme)}
{renderTabs()}
</ConfigProvider>
</div>
</div>
Expand Down
2 changes: 1 addition & 1 deletion src/components/Technologies/TechnologyCard.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ export default function TechnologyCard({ name, title, logoUrl, type, slug, versi
versions?.find((tagVersion) => tagVersion.children.find((child) => child.title === version))?.title || "";
return (
<div className={styles.card}>
<Link to={slug || `/integrations/packs?pack=${name}&version=${version}&parent=${parentVersion}`}>
<Link to={slug || `/integrations/packs?pack=${name}&version=${version}&parent=${parentVersion}&tab=main`}>
<div className={styles.cardContent}>
<PackCardIcon appType={slug ? "app" : "integration"} logoUrl={logoUrl} type={type} />
<div className={styles.title}>{title}</div>
Expand Down

0 comments on commit 9fd0c1e

Please sign in to comment.