mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-17 18:44:45 +02:00
79 lines
2.3 KiB
TypeScript
79 lines
2.3 KiB
TypeScript
import { useState } from 'react';
|
|
import { ListRenderer } from '@joplin/lib/services/plugins/api/noteListType';
|
|
import Note from '@joplin/lib/models/Note';
|
|
import { NoteEntity, TagEntity } from '@joplin/lib/services/database/types';
|
|
import useAsyncEffect from '@joplin/lib/hooks/useAsyncEffect';
|
|
import { createHash } from 'crypto';
|
|
import getNoteTitleHtml from './getNoteTitleHtml';
|
|
import prepareViewProps from './prepareViewProps';
|
|
import * as Mustache from 'mustache';
|
|
import Tag from '@joplin/lib/models/Tag';
|
|
|
|
interface RenderedNote {
|
|
id: string;
|
|
hash: string;
|
|
html: string;
|
|
}
|
|
|
|
const hashContent = (content: any) => {
|
|
return createHash('sha1').update(JSON.stringify(content)).digest('hex');
|
|
};
|
|
|
|
export default (note: NoteEntity, isSelected: boolean, isWatched: boolean, listRenderer: ListRenderer, highlightedWords: string[]) => {
|
|
const [renderedNote, setRenderedNote] = useState<RenderedNote>(null);
|
|
|
|
useAsyncEffect(async (event) => {
|
|
const renderNote = async (): Promise<void> => {
|
|
let noteTags: TagEntity[] = [];
|
|
|
|
if (listRenderer.dependencies.includes('note.tags')) {
|
|
noteTags = await Tag.tagsByNoteId(note.id, { fields: ['id', 'title'] });
|
|
}
|
|
|
|
// Note: with this hash we're assuming that the list renderer
|
|
// properties never changes. It means that later if we support
|
|
// dynamic list renderers, we should include these into the hash.
|
|
const viewHash = hashContent([
|
|
listRenderer.id,
|
|
note.updated_time,
|
|
isSelected,
|
|
isWatched,
|
|
highlightedWords,
|
|
note.encryption_applied,
|
|
noteTags.map(t => t.title).sort().join(','),
|
|
]);
|
|
|
|
if (renderedNote && renderedNote.hash === viewHash) return null;
|
|
|
|
// console.info('RENDER', note.id, renderedNote ? renderedNote.hash : 'NULL', viewHash);
|
|
|
|
const titleHtml = getNoteTitleHtml(highlightedWords, Note.displayTitle(note));
|
|
const viewProps = await prepareViewProps(
|
|
listRenderer.dependencies,
|
|
note,
|
|
listRenderer.itemSize,
|
|
isSelected,
|
|
titleHtml,
|
|
isWatched,
|
|
noteTags,
|
|
);
|
|
|
|
if (event.cancelled) return null;
|
|
|
|
const view = await listRenderer.onRenderNote(viewProps);
|
|
|
|
if (event.cancelled) return null;
|
|
|
|
setRenderedNote({
|
|
id: note.id,
|
|
hash: viewHash,
|
|
html: Mustache.render(listRenderer.itemTemplate, view),
|
|
});
|
|
};
|
|
|
|
void renderNote();
|
|
}, [note, isSelected, isWatched, listRenderer, renderedNote]);
|
|
|
|
return renderedNote;
|
|
};
|