mirror of
https://github.com/lordmathis/lemma.git
synced 2025-11-06 16:04:23 +00:00
Compare commits
4 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| f9ce8b9e9f | |||
| a6d2663a7d | |||
| 071e99f4da | |||
| b13ee987c7 |
@@ -108,29 +108,3 @@ $navbar-height: 64px;
|
|||||||
.tree {
|
.tree {
|
||||||
padding-top: $padding;
|
padding-top: $padding;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Syntax highlighting themes
|
|
||||||
@import 'highlight.js/styles/github.css' layer(light-theme);
|
|
||||||
@import 'highlight.js/styles/github-dark.css' layer(dark-theme);
|
|
||||||
|
|
||||||
// Show light theme by default
|
|
||||||
@layer light-theme {
|
|
||||||
[data-mantine-color-scheme='light'] .markdown-preview {
|
|
||||||
pre code.hljs {
|
|
||||||
display: block;
|
|
||||||
overflow-x: auto;
|
|
||||||
padding: 1em;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Show dark theme in dark mode
|
|
||||||
@layer dark-theme {
|
|
||||||
[data-mantine-color-scheme='dark'] .markdown-preview {
|
|
||||||
pre code.hljs {
|
|
||||||
display: block;
|
|
||||||
overflow-x: auto;
|
|
||||||
padding: 1em;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -120,6 +120,34 @@ describe('MarkdownPreview', () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('renders code blocks with correct structure for theme switching', async () => {
|
||||||
|
const content = '```javascript\nconst hello = "world";\n```';
|
||||||
|
|
||||||
|
render(
|
||||||
|
<MarkdownPreview
|
||||||
|
content={content}
|
||||||
|
handleFileSelect={mockHandleFileSelect}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
await waitFor(() => {
|
||||||
|
// Check that rehype-highlight generates the correct structure
|
||||||
|
const preElement = screen
|
||||||
|
.getByRole('code', { hidden: true })
|
||||||
|
.closest('pre');
|
||||||
|
const codeElement = preElement?.querySelector('code');
|
||||||
|
|
||||||
|
expect(preElement).toBeInTheDocument();
|
||||||
|
expect(codeElement).toBeInTheDocument();
|
||||||
|
|
||||||
|
// The code element should have hljs class for theme switching to work
|
||||||
|
expect(codeElement).toHaveClass('hljs');
|
||||||
|
|
||||||
|
// Should also have language class
|
||||||
|
expect(codeElement).toHaveClass('language-javascript');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
it('handles image loading errors gracefully', async () => {
|
it('handles image loading errors gracefully', async () => {
|
||||||
const content = '';
|
const content = '';
|
||||||
|
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ import * as prod from 'react/jsx-runtime';
|
|||||||
import { notifications } from '@mantine/notifications';
|
import { notifications } from '@mantine/notifications';
|
||||||
import { remarkWikiLinks } from '../../utils/remarkWikiLinks';
|
import { remarkWikiLinks } from '../../utils/remarkWikiLinks';
|
||||||
import { useWorkspace } from '../../hooks/useWorkspace';
|
import { useWorkspace } from '../../hooks/useWorkspace';
|
||||||
|
import { useHighlightTheme } from '../../hooks/useHighlightTheme';
|
||||||
|
|
||||||
interface MarkdownPreviewProps {
|
interface MarkdownPreviewProps {
|
||||||
content: string;
|
content: string;
|
||||||
@@ -28,12 +29,6 @@ interface MarkdownLinkProps {
|
|||||||
[key: string]: unknown;
|
[key: string]: unknown;
|
||||||
}
|
}
|
||||||
|
|
||||||
interface MarkdownCodeProps {
|
|
||||||
children: ReactNode;
|
|
||||||
className?: string;
|
|
||||||
[key: string]: unknown;
|
|
||||||
}
|
|
||||||
|
|
||||||
const MarkdownPreview: React.FC<MarkdownPreviewProps> = ({
|
const MarkdownPreview: React.FC<MarkdownPreviewProps> = ({
|
||||||
content,
|
content,
|
||||||
handleFileSelect,
|
handleFileSelect,
|
||||||
@@ -42,7 +37,10 @@ const MarkdownPreview: React.FC<MarkdownPreviewProps> = ({
|
|||||||
null
|
null
|
||||||
);
|
);
|
||||||
const baseUrl = window.API_BASE_URL;
|
const baseUrl = window.API_BASE_URL;
|
||||||
const { currentWorkspace } = useWorkspace();
|
const { currentWorkspace, colorScheme } = useWorkspace();
|
||||||
|
|
||||||
|
// Use the highlight theme hook
|
||||||
|
useHighlightTheme(colorScheme === 'auto' ? 'light' : colorScheme);
|
||||||
|
|
||||||
const processor = useMemo(() => {
|
const processor = useMemo(() => {
|
||||||
const handleLinkClick = (
|
const handleLinkClick = (
|
||||||
@@ -107,13 +105,6 @@ const MarkdownPreview: React.FC<MarkdownPreviewProps> = ({
|
|||||||
{children}
|
{children}
|
||||||
</a>
|
</a>
|
||||||
),
|
),
|
||||||
code: ({ children, className, ...props }: MarkdownCodeProps) => {
|
|
||||||
return (
|
|
||||||
<pre className={className}>
|
|
||||||
<code {...props}>{children}</code>
|
|
||||||
</pre>
|
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
} as Options);
|
} as Options);
|
||||||
}, [currentWorkspace?.name, baseUrl, handleFileSelect]);
|
}, [currentWorkspace?.name, baseUrl, handleFileSelect]);
|
||||||
|
|||||||
36
app/src/hooks/useHighlightTheme.ts
Normal file
36
app/src/hooks/useHighlightTheme.ts
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
import { useEffect } from 'react';
|
||||||
|
// Import theme CSS as text that will be bundled
|
||||||
|
import atomOneLightTheme from 'highlight.js/styles/atom-one-light.css?inline';
|
||||||
|
import atomOneDarkTheme from 'highlight.js/styles/atom-one-dark.css?inline';
|
||||||
|
|
||||||
|
export const useHighlightTheme = (colorScheme: 'light' | 'dark') => {
|
||||||
|
useEffect(() => {
|
||||||
|
// Remove existing highlight theme
|
||||||
|
const existingStylesheet = document.querySelector(
|
||||||
|
'style[data-highlight-theme]'
|
||||||
|
);
|
||||||
|
if (existingStylesheet) {
|
||||||
|
existingStylesheet.remove();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add new theme stylesheet using bundled CSS
|
||||||
|
const style = document.createElement('style');
|
||||||
|
style.setAttribute('data-highlight-theme', 'true');
|
||||||
|
|
||||||
|
if (colorScheme === 'dark') {
|
||||||
|
style.textContent = atomOneDarkTheme as string;
|
||||||
|
} else {
|
||||||
|
style.textContent = atomOneLightTheme as string;
|
||||||
|
}
|
||||||
|
|
||||||
|
document.head.appendChild(style);
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
// Cleanup on unmount
|
||||||
|
const stylesheet = document.querySelector('style[data-highlight-theme]');
|
||||||
|
if (stylesheet) {
|
||||||
|
stylesheet.remove();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}, [colorScheme]);
|
||||||
|
};
|
||||||
5
app/src/types/css-inline.d.ts
vendored
Normal file
5
app/src/types/css-inline.d.ts
vendored
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
// Type declarations for CSS imports with ?inline modifier
|
||||||
|
declare module '*.css?inline' {
|
||||||
|
const content: string;
|
||||||
|
export default content;
|
||||||
|
}
|
||||||
File diff suppressed because it is too large
Load Diff
Reference in New Issue
Block a user