mirror of https://github.com/usememos/memos
feat: implement embedded memo renderer
parent
67f5ac3657
commit
8a34013558
@ -0,0 +1,51 @@
|
|||||||
|
package parser
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
|
||||||
|
"github.com/usememos/memos/plugin/gomark/ast"
|
||||||
|
"github.com/usememos/memos/plugin/gomark/parser/tokenizer"
|
||||||
|
)
|
||||||
|
|
||||||
|
type EmbeddedContentParser struct{}
|
||||||
|
|
||||||
|
func NewEmbeddedContentParser() *EmbeddedContentParser {
|
||||||
|
return &EmbeddedContentParser{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (*EmbeddedContentParser) Match(tokens []*tokenizer.Token) (int, bool) {
|
||||||
|
lines := tokenizer.Split(tokens, tokenizer.Newline)
|
||||||
|
if len(lines) < 1 {
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
firstLine := lines[0]
|
||||||
|
if len(firstLine) < 5 {
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
if firstLine[0].Type != tokenizer.ExclamationMark || firstLine[1].Type != tokenizer.LeftSquareBracket || firstLine[2].Type != tokenizer.LeftSquareBracket {
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
matched := false
|
||||||
|
for index, token := range firstLine[:len(firstLine)-1] {
|
||||||
|
if token.Type == tokenizer.RightSquareBracket && firstLine[index+1].Type == tokenizer.RightSquareBracket && index+1 == len(firstLine)-1 {
|
||||||
|
matched = true
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !matched {
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
|
||||||
|
return len(firstLine), true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *EmbeddedContentParser) Parse(tokens []*tokenizer.Token) (ast.Node, error) {
|
||||||
|
size, ok := p.Match(tokens)
|
||||||
|
if size == 0 || !ok {
|
||||||
|
return nil, errors.New("not matched")
|
||||||
|
}
|
||||||
|
|
||||||
|
return &ast.EmbeddedContent{
|
||||||
|
ResourceName: tokenizer.Stringify(tokens[3 : size-2]),
|
||||||
|
}, nil
|
||||||
|
}
|
@ -0,0 +1,51 @@
|
|||||||
|
package parser
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
|
||||||
|
"github.com/usememos/memos/plugin/gomark/ast"
|
||||||
|
"github.com/usememos/memos/plugin/gomark/parser/tokenizer"
|
||||||
|
"github.com/usememos/memos/plugin/gomark/restore"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestEmbeddedContentParser(t *testing.T) {
|
||||||
|
tests := []struct {
|
||||||
|
text string
|
||||||
|
embeddedContent ast.Node
|
||||||
|
}{
|
||||||
|
{
|
||||||
|
text: "![[Hello world]",
|
||||||
|
embeddedContent: nil,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: "![[Hello world]]",
|
||||||
|
embeddedContent: &ast.EmbeddedContent{
|
||||||
|
ResourceName: "Hello world",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: "![[memos/1]]",
|
||||||
|
embeddedContent: &ast.EmbeddedContent{
|
||||||
|
ResourceName: "memos/1",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: "![[resources/101]] \n123",
|
||||||
|
embeddedContent: nil,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: "![[resources/101]]\n123",
|
||||||
|
embeddedContent: &ast.EmbeddedContent{
|
||||||
|
ResourceName: "resources/101",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, test := range tests {
|
||||||
|
tokens := tokenizer.Tokenize(test.text)
|
||||||
|
node, _ := NewEmbeddedContentParser().Parse(tokens)
|
||||||
|
require.Equal(t, restore.Restore([]ast.Node{test.embeddedContent}), restore.Restore([]ast.Node{node}))
|
||||||
|
}
|
||||||
|
}
|
File diff suppressed because it is too large
Load Diff
@ -0,0 +1,32 @@
|
|||||||
|
import { useContext, useEffect } from "react";
|
||||||
|
import { useMemoStore } from "@/store/v1";
|
||||||
|
import MemoContent from "..";
|
||||||
|
import { RendererContext } from "../types";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
memoId: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
const EmbeddedMemo = ({ memoId }: Props) => {
|
||||||
|
const context = useContext(RendererContext);
|
||||||
|
const memoStore = useMemoStore();
|
||||||
|
const memo = memoStore.getMemoById(memoId);
|
||||||
|
const resourceName = `memos/${memoId}`;
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
memoStore.getOrFetchMemoById(memoId);
|
||||||
|
}, [memoId]);
|
||||||
|
|
||||||
|
if (memoId === context.memoId || context.embeddedMemos.has(resourceName)) {
|
||||||
|
return <p>Nested Rendering Error: {`![[${resourceName}]]`}</p>;
|
||||||
|
}
|
||||||
|
context.embeddedMemos.add(resourceName);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="embedded-memo">
|
||||||
|
<MemoContent nodes={memo.nodes} memoId={memoId} embeddedMemos={context.embeddedMemos} />
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default EmbeddedMemo;
|
@ -0,0 +1,20 @@
|
|||||||
|
import EmbeddedMemo from "./EmbeddedMemo";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
resourceName: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const extractResourceTypeAndId = (resourceName: string) => {
|
||||||
|
const [resourceType, resourceId] = resourceName.split("/");
|
||||||
|
return { resourceType, resourceId };
|
||||||
|
};
|
||||||
|
|
||||||
|
const EmbeddedContent = ({ resourceName }: Props) => {
|
||||||
|
const { resourceType, resourceId } = extractResourceTypeAndId(resourceName);
|
||||||
|
if (resourceType === "memos") {
|
||||||
|
return <EmbeddedMemo memoId={Number(resourceId)} />;
|
||||||
|
}
|
||||||
|
return <p>Unknown resource: {resourceName}</p>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export default EmbeddedContent;
|
Loading…
Reference in New Issue