Skip to content

Commit

Permalink
feat: slot for markdown links
Browse files Browse the repository at this point in the history
Fix #957
  • Loading branch information
Grafikart committed May 15, 2024
1 parent ea77d27 commit 094894d
Show file tree
Hide file tree
Showing 9 changed files with 82 additions and 138 deletions.
2 changes: 2 additions & 0 deletions src/components/shared/HOC/slottableComponent.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -49,6 +49,7 @@ import type { CustomCheckboxGroup } from '../../CheckboxGroup/CustomCheckboxGrou
import type { RouterLink } from '../MDLabel/RouterLink';
import type { SummaryResponses, SummaryTitle } from '../../Summary/Summary';
import type { LunaticComponentProps } from '../../type';
import type { MarkdownLink } from '../MDLabel/MarkdownLink';

/**
* Contains the type of every customizable component
Expand Down Expand Up @@ -119,6 +120,7 @@ export type LunaticSlotComponents = {
ComponentWrapper: ComponentType<
PropsWithChildren<LunaticComponentProps & { index: number }>
>;
MarkdownLink: typeof MarkdownLink;
};

const empty = {} as Partial<LunaticSlotComponents> | undefined;
Expand Down
54 changes: 28 additions & 26 deletions src/components/shared/MDLabel/MDLabel.tsx
Original file line number Diff line number Diff line change
@@ -1,39 +1,41 @@
import type { ComponentProps } from 'react';
import Markdown from 'react-markdown';
import { MDLabelLink } from './MDLabelLink';
import {
type ComponentProps,
Fragment,
type PropsWithChildren,
type ReactNode,
} from 'react';
import Markdown, { type Components } from 'react-markdown';
import { MarkdownLink } from './MarkdownLink';
import remarkBreaks from 'remark-breaks';
import emoji from 'remark-emoji';

type Props = { expression: string };

export const MDLabel = ({ expression }: Props) => {
export function MDLabel({ expression }: Props) {
const hasParagraphs = /\n\n/.test(expression);
const components = {
p: hasParagraphs ? 'p' : Fragment,
br: 'br',
a: MarkdownA,
} as Partial<Components>;
return (
<Markdown
components={renderComponentsFor(expression)}
components={components}
remarkPlugins={[[emoji, { accessible: true }], remarkBreaks]}
>
{expression}
</Markdown>
);
};

const renderComponentsFor = (
expression: string
): ComponentProps<typeof Markdown>['components'] => {
const components = {
p: (props) => <>{props.children}</>,
br: 'br',
a: (props) => (
<MDLabelLink {...({ ...props } as ComponentProps<typeof MDLabelLink>)} />
),
} satisfies ComponentProps<typeof Markdown>['components'];

if (/\n\n/.test(expression)) {
return {
...components,
p: 'p',
};
}

return components;
}
const MarkdownA = ({
title,
href,
children,
}: PropsWithChildren<{ title?: string; href: string }>) => {
const tooltip = title ? <MDLabel expression={title} /> : null;
return (
<MarkdownLink href={href} tooltip={tooltip}>
{children}
</MarkdownLink>
);
};
59 changes: 0 additions & 59 deletions src/components/shared/MDLabel/MDLabelLink.tsx

This file was deleted.

44 changes: 44 additions & 0 deletions src/components/shared/MDLabel/MarkdownLink.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
import { useState, type PropsWithChildren, type ReactNode } from 'react';
import { Tooltip } from 'react-tooltip';
import { RouterLink } from './RouterLink';
import { slottableComponent } from '../HOC/slottableComponent';

type Props = PropsWithChildren<{ href: string; tooltip?: ReactNode }>;

export const MarkdownLink = slottableComponent<Props>(
'MarkdownLink',
({ href, children, tooltip }) => {
// Must be replace by useId when queen move to React 18.
const [id] = useState(() => generateUniqueId());

const extraProps = tooltip
? { 'data-tooltip-id': `tooltip-${id}`, className: 'link-md' }
: {};
const isAbsoluteLink = href.trim().startsWith('/');

return (
<>
{isAbsoluteLink ? (
<a href={href} target="_blank" rel="noopener noreferrer" id={id}>
{children}
</a>
) : (
<RouterLink to={href} id={id} {...extraProps}>
{children}
</RouterLink>
)}
{tooltip && (
<Tooltip className="tooltip-content" id={`tooltip-${id}`}>
{tooltip}
</Tooltip>
)}
</>
);
}
);

export function generateUniqueId(): string {
const timestamp = Date.now().toString(36);
const randomPart = Math.random().toString(36).substring(2, 8);
return `${timestamp}-${randomPart}`;
}
17 changes: 8 additions & 9 deletions src/stories/utils/orchestrator.jsx
Original file line number Diff line number Diff line change
@@ -1,13 +1,17 @@
import './custom-lunatic.scss';
import './orchestrator.scss';

import { useLunatic, Button, components, ModalControls } from '../..';
import React, { memo, useCallback, useState, useEffect } from 'react';
import {
Button,
components,
LunaticComponents,
ModalControls,
useLunatic,
} from '../..';
import React, { memo, useCallback, useEffect, useState } from 'react';

import { Logger } from '../../utils/logger';
import { Overview } from './overview';
import Waiting from './waiting';
import { LunaticComponents } from '../..';
import { SchemaValidator } from './SchemaValidator.jsx';

const Input = components.Input;
Expand Down Expand Up @@ -247,11 +251,6 @@ function OrchestratorForStories({
isCritical={errorsForModal.isCritical}
/>
)}
<Waiting status={waiting}>
<div className="waiting-orchestrator">
Initialisation des données de suggestion...
</div>
</Waiting>
</aside>
</div>
</Provider>
Expand Down
1 change: 0 additions & 1 deletion src/stories/utils/waiting/index.js

This file was deleted.

1 change: 0 additions & 1 deletion src/stories/utils/waiting/preloader.svg

This file was deleted.

21 changes: 0 additions & 21 deletions src/stories/utils/waiting/waiting.jsx

This file was deleted.

21 changes: 0 additions & 21 deletions src/stories/utils/waiting/waiting.scss

This file was deleted.

0 comments on commit 094894d

Please sign in to comment.