Skip to content

Commit

Permalink
fix: improve heading-analyzer reliability
Browse files Browse the repository at this point in the history
Use new status value from usePuck with a MutationObserver.
  • Loading branch information
chrisvxd committed Dec 27, 2024
1 parent 2048dd3 commit 39a6e17
Show file tree
Hide file tree
Showing 3 changed files with 32 additions and 33 deletions.
2 changes: 1 addition & 1 deletion packages/core/components/AutoFrame/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -347,7 +347,7 @@ function AutoFrame({
{...props}
className={className}
id={id}
srcDoc='<!DOCTYPE html><html><head></head><body><div id="frame-root"></div></body></html>'
srcDoc='<!DOCTYPE html><html><head></head><body><div id="frame-root" data-puck-entry></div></body></html>'
ref={frameRef}
onLoad={() => {
setLoaded(true);
Expand Down
7 changes: 6 additions & 1 deletion packages/core/components/Puck/components/Preview/index.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -125,7 +125,12 @@ export const Preview = ({ id = "puck-preview" }: { id?: string }) => {
</autoFrameContext.Consumer>
</AutoFrame>
) : (
<div id="preview-frame" className={getClassName("frame")} ref={ref}>
<div
id="preview-frame"
className={getClassName("frame")}
ref={ref}
data-puck-entry
>
<Page
{...rootProps}
puck={{ renderDropZone: DropZone, isEditing: true, dragRef: null }}
Expand Down
56 changes: 25 additions & 31 deletions packages/plugin-heading-analyzer/src/HeadingAnalyzer.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -22,25 +22,20 @@ const ReactFromJSON =
(ReactFromJSONModule as unknown as { default: typeof ReactFromJSONModule })
.default || ReactFromJSONModule;

const dataAttr = "data-puck-heading-analyzer-id";

const getOutline = ({
addDataAttr = false,
frame,
}: { addDataAttr?: boolean; frame?: Element | Document } = {}) => {
const getOutline = ({ frame }: { frame?: Element | Document } = {}) => {
const headings = frame?.querySelectorAll("h1,h2,h3,h4,h5,h6") || [];

const _outline: { rank: number; text: string; analyzeId: string }[] = [];
const _outline: {
rank: number;
text: string;
element: HTMLElement;
}[] = [];

headings.forEach((item, i) => {
if (addDataAttr) {
item.setAttribute(dataAttr, i.toString());
}

_outline.push({
rank: parseInt(item.tagName.split("H")[1]),
text: item.textContent!,
analyzeId: i.toString(),
element: item as HTMLElement,
});
});

Expand All @@ -53,10 +48,11 @@ type Block = {
children?: Block[];
missing?: boolean;
analyzeId?: string;
element?: HTMLElement;
};

function buildHierarchy(frame: Element | Document): Block[] {
const headings = getOutline({ addDataAttr: true, frame });
const headings = getOutline({ frame });

const root = { rank: 0, children: [], text: "" }; // Placeholder root node
let path: Block[] = [root];
Expand All @@ -65,8 +61,8 @@ function buildHierarchy(frame: Element | Document): Block[] {
const node: Block = {
rank: heading.rank,
text: heading.text,
analyzeId: heading.analyzeId,
children: [],
element: heading.element,
};

// When encountering an h1, reset the path to only the root
Expand Down Expand Up @@ -102,23 +98,25 @@ function buildHierarchy(frame: Element | Document): Block[] {
export const HeadingAnalyzer = () => {
const { appState } = usePuck();
const [hierarchy, setHierarchy] = useState<Block[]>([]);
const [firstRender, setFirstRender] = useState(true);

// Re-render when content changes
useEffect(() => {
const frame = getFrame();
const entry = frame?.querySelector(`[data-puck-entry]`);

if (!frame) return;
if (!entry) return;

// We need to delay to allow remainder of page to render first
if (firstRender) {
setTimeout(() => {
setHierarchy(buildHierarchy(frame));
setFirstRender(false);
}, 100);
} else {
setHierarchy(buildHierarchy(frame));
}
setHierarchy(buildHierarchy(entry));

const observer = new MutationObserver(() => {
setHierarchy(buildHierarchy(entry));
});

observer.observe(entry, { subtree: true, childList: true });

return () => {
observer.disconnect();
};
}, [appState.data]);

return (
Expand Down Expand Up @@ -153,16 +151,12 @@ export const HeadingAnalyzer = () => {
<small
className={getClassNameItem({ missing: props.missing })}
onClick={
typeof props.analyzeId == "undefined"
typeof props.element == "undefined"
? undefined
: (e) => {
e.stopPropagation();

const frame = getFrame();

const el = frame?.querySelector(
`[${dataAttr}="${props.analyzeId}"]`
) as HTMLElement;
const el = props.element;

const oldStyle = { ...el.style };

Expand Down

0 comments on commit 39a6e17

Please sign in to comment.