Improve editor + added imports

This commit is contained in:
DrMint 2023-10-23 22:39:17 +02:00
parent 509cbdba9b
commit 4f807c410b
31 changed files with 5043 additions and 2971 deletions

View File

@ -4,6 +4,7 @@ MONGODB_PORT=27017
PAYLOAD_URI=https://payload.domain.com
PAYLOAD_SECRET=payloadsecreta5e6ea45ef4e66eaa151612bdcb599df
PAYLOAD_PORT=3000
RECORDER_DEFAULT_PASSWORD=somepassword
STRAPI_URI=https://strapi.domain.com
STRAPI_TOKEN=strapisecreta5e6ea45ef4e66eaa151612bdcb599df

BIN
bun.lockb

Binary file not shown.

7126
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@ -22,27 +22,28 @@
"start": "sudo docker compose up"
},
"dependencies": {
"@fontsource/vollkorn": "^5.0.14",
"@payloadcms/bundler-webpack": "^1.0.3",
"@payloadcms/db-mongodb": "^1.0.3",
"@payloadcms/richtext-lexical": "^0.1.8",
"@fontsource/vollkorn": "^5.0.17",
"@payloadcms/bundler-webpack": "^1.0.4",
"@payloadcms/db-mongodb": "^1.0.4",
"@payloadcms/richtext-lexical": "^0.1.15",
"clean-deep": "^3.4.0",
"cross-env": "^7.0.3",
"language-tags": "^1.0.9",
"luxon": "^3.4.3",
"payload": "^2.0.5",
"styled-components": "^6.0.9"
"payload": "^2.0.12",
"styled-components": "^6.1.0"
},
"devDependencies": {
"@types/dotenv": "^8.2.0",
"@types/express": "^4.17.18",
"@types/language-tags": "^1.0.2",
"@types/luxon": "^3.3.2",
"@types/qs": "^6.9.8",
"@types/express": "^4.17.20",
"@types/language-tags": "^1.0.3",
"@types/luxon": "^3.3.3",
"@types/qs": "^6.9.9",
"@types/react-router-dom": "^5.3.3",
"@types/styled-components": "^5.1.28",
"@types/styled-components": "^5.1.29",
"copyfiles": "^2.4.1",
"nodemon": "^3.0.1",
"npm-check-updates": "^16.14.6",
"prettier": "^3.0.3",
"ts-node": "^10.9.1",
"ts-unused-exports": "^10.0.1",

View File

@ -1,4 +1,5 @@
import { Block } from "payload/types";
import { createEditor } from "../utils/editor";
export const cueBlock: Block = {
slug: "cueBlock",
@ -8,12 +9,12 @@ export const cueBlock: Block = {
{
name: "content",
label: false,
type: "textarea",
type: "richText",
required: true,
admin: {
description:
"Parenthesis will automatically be added around cues. You don't have to include them here.",
className: "reduced-margins",
},
editor: createEditor({ inlines: true, lists: true, links: true }),
},
],
};

View File

@ -1,4 +1,5 @@
import { Block } from "payload/types";
import { createEditor } from "../utils/editor";
export const lineBlock: Block = {
slug: "lineBlock",
@ -10,6 +11,10 @@ export const lineBlock: Block = {
label: false,
type: "richText",
required: true,
admin: {
className: "reduced-margins",
},
editor: createEditor({ inlines: true, lists: true, links: true }),
},
],
};

View File

@ -0,0 +1,31 @@
import { Block } from "payload/types";
import { createEditor } from "../utils/editor";
import { transcriptBlock } from "./transcriptBlock";
const generateRecursiveSectionBlock = (depth = 1, maxDepth = 5): Block => ({
slug: "sectionBlock",
interfaceName: "SectionBlock",
labels: { singular: "Section", plural: "Sections" },
fields: [
{
name: "lines",
type: "richText",
label: false,
required: true,
editor: createEditor({
images: true,
inlines: true,
lists: true,
links: true,
relations: true,
alignment: true,
blocks: [
transcriptBlock,
...(depth < maxDepth ? [generateRecursiveSectionBlock(depth + 1, maxDepth)] : []),
],
}),
},
],
});
export const sectionBlock: Block = generateRecursiveSectionBlock();

View File

@ -1,6 +1,6 @@
import { Block } from "payload/types";
import { lineBlock } from "./lineBlock";
import { cueBlock } from "./cueBlock";
import { lineBlock } from "./lineBlock";
export const transcriptBlock: Block = {
slug: "transcriptBlock",
@ -12,7 +12,7 @@ export const transcriptBlock: Block = {
type: "blocks",
required: true,
minRows: 1,
admin: { initCollapsed: true },
admin: { initCollapsed: true, className: "no-label" },
blocks: [lineBlock, cueBlock],
},
],

View File

@ -5,6 +5,7 @@ import { backPropagationField } from "../../fields/backPropagationField/backProp
import { slugField } from "../../fields/slugField/slugField";
import { translatedFields } from "../../fields/translatedFields/translatedFields";
import { buildCollectionConfig } from "../../utils/collectionConfig";
import { createEditor } from "../../utils/editor";
import { getAllEndpoint } from "./endpoints/getAllEndpoint";
import { importFromStrapi } from "./endpoints/importFromStrapi";
import { beforeValidateEndingGreaterThanStarting } from "./hooks/beforeValidateEndingGreaterThanStarting";
@ -68,7 +69,8 @@ export const ChronologyEras: CollectionConfig = buildCollectionConfig({
{ name: fields.translationsTitle, type: "text", required: true },
{
name: fields.translationsDescription,
type: "textarea",
type: "richText",
editor: createEditor({ inlines: true, lists: true, links: true }),
},
],
}),

View File

@ -6,6 +6,7 @@ import {
} from "../../components/QuickFilters";
import { CollectionGroups, Collections } from "../../constants";
import { translatedFields } from "../../fields/translatedFields/translatedFields";
import { createEditor } from "../../utils/editor";
import { buildVersionedCollectionConfig } from "../../utils/versionedCollectionConfig";
import { importFromStrapi } from "./endpoints/importFromStrapi";
import { beforeValidatePopulateNameField } from "./hooks/beforeValidatePopulateNameField";
@ -127,9 +128,14 @@ export const ChronologyItems: CollectionConfig = buildVersionedCollectionConfig(
{
name: fields.eventsTranslationsDescription,
validate: validateEventsTranslationsDescription,
type: "textarea",
type: "richText",
editor: createEditor({ inlines: true, lists: true, links: true }),
},
{
name: fields.eventsTranslationsNotes,
type: "richText",
editor: createEditor({ inlines: true, lists: true, links: true }),
},
{ name: fields.eventsTranslationsNotes, type: "textarea" },
],
}),
],

View File

@ -1,3 +1,5 @@
import { sectionBlock } from "../../blocks/sectionBlock";
import { transcriptBlock } from "../../blocks/transcriptBlock";
import { CollectionGroups, Collections, FileTypes, KeysTypes } from "../../constants";
import { fileField } from "../../fields/fileField/fileField";
import { imageField } from "../../fields/imageField/imageField";
@ -8,7 +10,9 @@ import { beforeDuplicateAddCopyTo } from "../../hooks/beforeDuplicateAddCopyTo";
import { beforeDuplicatePiping } from "../../hooks/beforeDuplicatePiping";
import { beforeDuplicateUnpublish } from "../../hooks/beforeDuplicateUnpublish";
import { isDefined } from "../../utils/asserts";
import { createEditor } from "../../utils/editor";
import { buildVersionedCollectionConfig } from "../../utils/versionedCollectionConfig";
import { importFromStrapi } from "./endpoints/importFromStrapi";
const fields = {
slug: "slug",
@ -61,6 +65,7 @@ export const Contents = buildVersionedCollectionConfig({
},
preview: (doc) => `https://accords-library.com/contents/${doc.slug}`,
},
endpoints: [importFromStrapi],
fields: [
{
type: "row",
@ -103,7 +108,11 @@ export const Contents = buildVersionedCollectionConfig({
{ name: fields.subtitle, type: "text" },
],
},
{ name: fields.summary, type: "textarea" },
{
name: fields.summary,
type: "richText",
editor: createEditor({ inlines: true, lists: true, links: true }),
},
{
type: "tabs",
admin: {
@ -114,7 +123,20 @@ export const Contents = buildVersionedCollectionConfig({
{
label: "Text",
fields: [
{ name: fields.textContent, type: "richText" },
{
name: fields.textContent,
type: "richText",
label: false,
editor: createEditor({
blocks: [sectionBlock, transcriptBlock],
images: true,
inlines: true,
lists: true,
links: true,
relations: true,
alignment: true,
}),
},
{
type: "row",
fields: [
@ -155,7 +177,8 @@ export const Contents = buildVersionedCollectionConfig({
{
name: fields.textNotes,
label: "Notes",
type: "textarea",
type: "richText",
editor: createEditor({ inlines: true, lists: true, links: true }),
},
],
},
@ -173,7 +196,8 @@ export const Contents = buildVersionedCollectionConfig({
{
name: fields.videoNotes,
label: "Notes",
type: "textarea",
type: "richText",
editor: createEditor({ inlines: true, lists: true, links: true }),
admin: { width: "0%" },
},
],
@ -194,7 +218,8 @@ export const Contents = buildVersionedCollectionConfig({
{
name: fields.audioNotes,
label: "Notes",
type: "textarea",
type: "richText",
editor: createEditor({ inlines: true, lists: true, links: true }),
admin: { width: "0%" },
},
],

View File

@ -0,0 +1,121 @@
import type { MarkOptional } from "ts-essentials";
import { Collections } from "../../../constants";
import { createStrapiImportEndpoint } from "../../../endpoints/createStrapiImportEndpoint";
import { Content } from "../../../types/collections";
import { StrapiImage, StrapiLanguage, StrapiRecorders } from "../../../types/strapi";
import { isNotEmpty, isUndefined } from "../../../utils/asserts";
import {
findCategory,
findContentType,
findRecorder,
uploadStrapiImage,
} from "../../../utils/localApi";
import { plainTextToLexical } from "../../../utils/string";
type StrapiContent = {
slug: string;
categories: { data?: { attributes: { slug: string } }[] };
type: { data?: { attributes: { slug: string } } };
thumbnail: StrapiImage;
translations: {
title: string;
subtitle?: string;
pre_title?: string;
description?: string;
language: StrapiLanguage;
text_set?: {
text: string;
notes?: string;
source_language: StrapiLanguage;
transcribers: StrapiRecorders;
translators: StrapiRecorders;
proofreaders: StrapiRecorders;
};
}[];
};
export const importFromStrapi = createStrapiImportEndpoint<StrapiContent>({
strapi: {
collection: "contents",
params: {
populate: [
"type",
"categories",
"thumbnail",
"translations",
"translations.language",
"translations.text_set",
"translations.text_set.source_language",
"translations.text_set.transcribers",
"translations.text_set.translators",
"translations.text_set.proofreaders",
],
},
},
payload: {
collection: Collections.Contents,
convert: async ({ slug, categories, type, thumbnail, translations }) => {
const thumbnailId = await uploadStrapiImage({
collection: Collections.ContentsThumbnails,
image: thumbnail,
});
const data: MarkOptional<Content, "createdAt" | "id" | "updatedAt" | "updatedBy"> = {
slug,
categories:
categories.data &&
(await Promise.all(
categories.data.map(async (category) => await findCategory(category.attributes.slug))
)),
type: type.data && (await findContentType(type.data?.attributes.slug)),
thumbnail: thumbnailId,
translations: await Promise.all(
translations.map(
async ({ language, title, description, pre_title, subtitle, text_set }) => {
if (isUndefined(language.data))
throw new Error("A language is required for a content translation");
if (isUndefined(text_set))
throw new Error("Only content with text_set are supported");
if (isUndefined(text_set.source_language.data))
throw new Error("A language is required for a content translation text_set");
return {
language: language.data.attributes.code,
sourceLanguage: text_set.source_language.data.attributes.code,
title,
pretitle: pre_title,
subtitle,
summary: isNotEmpty(description) ? plainTextToLexical(description) : undefined,
textContent: plainTextToLexical(text_set.text),
textNotes: isNotEmpty(text_set.notes)
? plainTextToLexical(text_set.notes)
: undefined,
textTranscribers:
text_set.transcribers.data &&
(await Promise.all(
text_set.transcribers.data?.map(async (recorder) =>
findRecorder(recorder.attributes.username)
)
)),
textTranslators:
text_set.translators.data &&
(await Promise.all(
text_set.translators.data?.map(async (recorder) =>
findRecorder(recorder.attributes.username)
)
)),
textProofreaders:
text_set.proofreaders.data &&
(await Promise.all(
text_set.proofreaders.data?.map(async (recorder) =>
findRecorder(recorder.attributes.username)
)
)),
};
}
)
),
};
return data;
},
},
});

View File

@ -2,6 +2,7 @@ import { CollectionGroups, Collections } from "../../constants";
import { slugField } from "../../fields/slugField/slugField";
import { translatedFields } from "../../fields/translatedFields/translatedFields";
import { buildCollectionConfig } from "../../utils/collectionConfig";
import { importFromStrapi } from "./endpoints/importFromStrapi";
const fields = {
slug: "slug",
@ -24,13 +25,13 @@ export const ContentsFolders = buildCollectionConfig({
disableDuplicate: true,
group: CollectionGroups.Collections,
},
endpoints: [importFromStrapi],
timestamps: false,
versions: false,
fields: [
slugField({ name: fields.slug }),
translatedFields({
name: fields.translations,
interfaceName: "ContentFoldersTranslation",
admin: {
useAsTitle: fields.name,
},

View File

@ -0,0 +1,107 @@
import payload from "payload";
import QueryString from "qs";
import { Collections } from "../../../constants";
import { CollectionEndpoint } from "../../../types/payload";
import { StrapiLanguage } from "../../../types/strapi";
import { isUndefined } from "../../../utils/asserts";
type StrapiContentsFolder = {
id: string;
attributes: {
slug: string;
titles?: { title: string; language: StrapiLanguage }[];
subfolders: { data: StrapiContentsFolder[] };
contents: { data: { id: number }[] };
};
};
const getStrapiContentFolder = async (id: number): Promise<StrapiContentsFolder> => {
const paramsWithPagination = QueryString.stringify({
populate: [
"subfolders",
"subfolders.contents",
"subfolders.titles",
"subfolders.titles.language",
"subfolders.subfolders",
"subfolders.subfolders.contents",
"subfolders.subfolders.titles",
"subfolders.subfolders.titles.language",
"subfolders.subfolders.subfolders",
"subfolders.subfolders.subfolders.contents",
"subfolders.subfolders.subfolders.titles",
"subfolders.subfolders.subfolders.titles.language",
"subfolders.subfolders.subfolders.subfolders",
"subfolders.subfolders.subfolders.subfolders.contents",
"subfolders.subfolders.subfolders.subfolders.titles",
"subfolders.subfolders.subfolders.subfolders.titles.language",
"subfolders.subfolders.subfolders.subfolders.subfolders",
"subfolders.subfolders.subfolders.subfolders.subfolders.contents",
"subfolders.subfolders.subfolders.subfolders.subfolders.titles",
"subfolders.subfolders.subfolders.subfolders.subfolders.titles.language",
"subfolders.subfolders.subfolders.subfolders.subfolders.subfolders",
"subfolders.subfolders.subfolders.subfolders.subfolders.subfolders.contents",
"subfolders.subfolders.subfolders.subfolders.subfolders.subfolders.titles",
"subfolders.subfolders.subfolders.subfolders.subfolders.subfolders.titles.language",
"subfolders.subfolders.subfolders.subfolders.subfolders.subfolders.subfolders",
"subfolders.subfolders.subfolders.subfolders.subfolders.subfolders.subfolders.contents",
"subfolders.subfolders.subfolders.subfolders.subfolders.subfolders.subfolders.titles",
"subfolders.subfolders.subfolders.subfolders.subfolders.subfolders.subfolders.titles.language",
],
});
const uri = `${process.env.STRAPI_URI}/api/contents-folders/${id}?${paramsWithPagination}`;
const fetchResult = await fetch(uri, {
method: "get",
headers: { authorization: `Bearer ${process.env.STRAPI_TOKEN}` },
});
const { data } = await fetchResult.json();
return data;
};
export const importFromStrapi: CollectionEndpoint = {
method: "post",
path: "/strapi",
handler: async (req, res) => {
if (!req.user) {
return res.status(403).send({
errors: [
{
message: "You are not allowed to perform this action.",
},
],
});
}
let foldersCreated = 0;
const createContentFolder = async (data: StrapiContentsFolder): Promise<string> => {
const subfolders = await Promise.all(
data.attributes.subfolders.data.map(createContentFolder)
);
const { slug, titles } = data.attributes;
const result = await payload.create({
collection: Collections.ContentsFolders,
data: {
slug,
subfolders,
translations: titles?.map(({ title, language }) => {
if (isUndefined(language.data))
throw new Error("A language is required for a content folder translation");
return { language: language.data.attributes.code, name: title };
}),
},
user: req.user,
});
foldersCreated++;
return result.id;
};
const rootFolder = await getStrapiContentFolder(72);
try {
await createContentFolder(rootFolder);
} catch (e) {
res.status(500).json({ message: "Something went wrong", error: e });
}
res.status(200).json({ message: `${foldersCreated} entries have been added successfully.` });
},
};

View File

@ -0,0 +1,67 @@
import { CollectionGroups, Collections } from "../../constants";
import { slugField } from "../../fields/slugField/slugField";
import { translatedFields } from "../../fields/translatedFields/translatedFields";
import { buildCollectionConfig } from "../../utils/collectionConfig";
import { createEditor } from "../../utils/editor";
const fields = {
slug: "slug",
translations: "translations",
name: "name",
description: "description",
subfolders: "subfolders",
items: "items",
} as const satisfies Record<string, string>;
export const LibraryFolders = buildCollectionConfig({
slug: Collections.LibraryFolders,
labels: {
singular: "Library Folder",
plural: "Library Folders",
},
defaultSort: fields.slug,
admin: {
useAsTitle: fields.slug,
defaultColumns: [fields.slug, fields.translations],
disableDuplicate: true,
group: CollectionGroups.Collections,
},
timestamps: false,
versions: false,
fields: [
slugField({ name: fields.slug }),
translatedFields({
name: fields.translations,
admin: {
useAsTitle: fields.name,
},
fields: [
{ name: fields.name, type: "text", required: true },
{
name: fields.description,
type: "richText",
editor: createEditor({ inlines: true, lists: true, links: true }),
},
],
}),
{
type: "row",
fields: [
{
type: "relationship",
name: fields.subfolders,
relationTo: Collections.LibraryFolders,
hasMany: true,
admin: { width: "0%" },
},
{
type: "relationship",
name: fields.items,
relationTo: Collections.LibraryItems,
hasMany: true,
admin: { width: "0%" },
},
],
},
],
});

View File

@ -19,6 +19,7 @@ import { beforeDuplicatePiping } from "../../hooks/beforeDuplicatePiping";
import { beforeDuplicateUnpublish } from "../../hooks/beforeDuplicateUnpublish";
import { LibraryItem } from "../../types/collections";
import { isDefined } from "../../utils/asserts";
import { createEditor } from "../../utils/editor";
import { buildVersionedCollectionConfig } from "../../utils/versionedCollectionConfig";
import { RowLabel } from "./components/RowLabel";
@ -146,15 +147,32 @@ export const LibraryItems = buildVersionedCollectionConfig({
},
fields: [
{
name: fields.itemType,
type: "radio",
options: Object.entries(LibraryItemsTypes).map(([value, label]) => ({
label,
value,
})),
admin: {
layout: "horizontal",
},
type: "row",
fields: [
{
name: fields.itemType,
type: "radio",
options: Object.entries(LibraryItemsTypes).map(([value, label]) => ({
label,
value,
})),
admin: {
layout: "horizontal",
width: "0%",
},
},
{
name: fields.digital,
type: "checkbox",
required: true,
defaultValue: false,
admin: {
description:
"The item is the digital version of another item, or the item is sold only digitally.",
width: "0%",
},
},
],
},
{
type: "tabs",
@ -209,27 +227,6 @@ export const LibraryItems = buildVersionedCollectionConfig({
width: "0%",
},
},
{
name: fields.digital,
type: "checkbox",
required: true,
defaultValue: false,
admin: {
description:
"The item is the digital version of another item, or the item is sold only digitally.",
width: "0%",
},
},
{
name: fields.downloadable,
type: "checkbox",
required: true,
defaultValue: false,
admin: {
description: "Are the scans available for download?",
width: "0%",
},
},
],
},
],
@ -505,6 +502,16 @@ export const LibraryItems = buildVersionedCollectionConfig({
},
],
},
{
name: fields.downloadable,
type: "checkbox",
required: true,
defaultValue: false,
admin: {
description: "Are the scans available for download?",
width: "0%",
},
},
],
}),
],
@ -559,6 +566,7 @@ export const LibraryItems = buildVersionedCollectionConfig({
})
),
admin: {
condition: (data: Partial<LibraryItem>) => !data.digital,
layout: "horizontal",
width: "0%",
},
@ -654,11 +662,18 @@ export const LibraryItems = buildVersionedCollectionConfig({
name: fields.translations,
label: "Descriptions",
admin: { initCollapsed: true, useAsTitle: fields.translationsDescription },
fields: [{ name: fields.translationsDescription, type: "textarea", required: true }],
fields: [
{
name: fields.translationsDescription,
required: true,
type: "richText",
editor: createEditor({ inlines: true, lists: true, links: true }),
},
],
}),
optionalGroupField({
name: fields.size,
admin: { condition: (data) => !data.digital },
admin: { condition: (data: Partial<LibraryItem>) => !data.digital },
fields: [
{
type: "row",
@ -763,7 +778,8 @@ export const LibraryItems = buildVersionedCollectionConfig({
},
{
name: fields.contentsNote,
type: "textarea",
type: "richText",
editor: createEditor({ inlines: true, lists: true, links: true }),
admin: {
condition: ({ itemType }) =>
itemType === LibraryItemsTypes.Game || itemType === LibraryItemsTypes.Other,

View File

@ -0,0 +1,20 @@
import { CollectionConfig } from "payload/types";
import { Collections } from "../../constants";
import { buildCollectionConfig } from "../../utils/collectionConfig";
import { createEditor } from "../../utils/editor";
export const Notes: CollectionConfig = buildCollectionConfig({
slug: Collections.Notes,
labels: { singular: "Note", plural: "Notes" },
admin: {
// TODO: Reenable when we can use rich text as titles useAsTitle: fields.biography,
},
fields: [
{
name: "note",
type: "richText",
required: true,
editor: createEditor({ inlines: true, lists: true, links: true }),
},
],
});

View File

@ -7,6 +7,7 @@ import { beforeDuplicateAddCopyTo } from "../../hooks/beforeDuplicateAddCopyTo";
import { beforeDuplicatePiping } from "../../hooks/beforeDuplicatePiping";
import { beforeDuplicateUnpublish } from "../../hooks/beforeDuplicateUnpublish";
import { isDefined, isUndefined } from "../../utils/asserts";
import { createEditor } from "../../utils/editor";
import { buildVersionedCollectionConfig } from "../../utils/versionedCollectionConfig";
const fields = {
@ -97,7 +98,11 @@ export const Posts = buildVersionedCollectionConfig({
minRows: 1,
fields: [
{ name: fields.title, type: "text", required: true },
{ name: fields.summary, type: "textarea" },
{
name: fields.summary,
type: "richText",
editor: createEditor({ inlines: true, lists: true, links: true }),
},
{
type: "row",
fields: [

View File

@ -6,6 +6,7 @@ import { CollectionGroups, Collections, RecordersRoles } from "../../constants";
import { imageField } from "../../fields/imageField/imageField";
import { translatedFields } from "../../fields/translatedFields/translatedFields";
import { buildCollectionConfig } from "../../utils/collectionConfig";
import { createEditor } from "../../utils/editor";
import { importFromStrapi } from "./endpoints/importFromStrapi";
import { beforeLoginMustHaveAtLeastOneRole } from "./hooks/beforeLoginMustHaveAtLeastOneRole";
@ -107,11 +108,18 @@ export const Recorders = buildCollectionConfig({
name: fields.biographies,
interfaceName: "RecorderBiographies",
admin: {
useAsTitle: fields.biography,
// TODO: Reenable when we can use rich text as titles useAsTitle: fields.biography,
description:
"A short personal description about you or your involvement with this project or the franchise",
},
fields: [{ name: fields.biography, required: true, type: "textarea" }],
fields: [
{
name: fields.biography,
required: true,
type: "richText",
editor: createEditor({ inlines: true, lists: true, links: true }),
},
],
}),
{
name: fields.role,

View File

@ -5,6 +5,7 @@ import { Recorder } from "../../../types/collections";
import { StrapiImage, StrapiLanguage } from "../../../types/strapi";
import { isDefined, isUndefined } from "../../../utils/asserts";
import { uploadStrapiImage } from "../../../utils/localApi";
import { plainTextToLexical } from "../../../utils/string";
type StrapiRecorder = {
username: string;
@ -19,7 +20,7 @@ export const importFromStrapi = createStrapiImportEndpoint<StrapiRecorder>({
strapi: {
collection: "recorders",
params: {
populate: "bio.language,languages,avatar",
populate: ["bio.language", "languages", "avatar"],
},
},
payload: {
@ -50,9 +51,9 @@ export const importFromStrapi = createStrapiImportEndpoint<StrapiRecorder>({
if (isUndefined(language.data))
throw new Error("A language is required for a Recorder biography");
if (isUndefined(bio)) throw new Error("A bio is required for a Recorder biography");
return {
return {
language: language.data.attributes.code,
biography: bio,
biography: plainTextToLexical(bio),
};
}),
},
@ -72,10 +73,9 @@ export const importFromStrapi = createStrapiImportEndpoint<StrapiRecorder>({
if (isUndefined(bio)) throw new Error("A bio is required for a Recorder biography");
return {
language: language.data.attributes.code,
biography: bio,
biography: plainTextToLexical(bio),
};
}),
email: `${anonymous_code}@accords-library.com`,
password: process.env.RECORDER_DEFAULT_PASSWORD,
},

View File

@ -4,6 +4,7 @@ import { imageField } from "../../fields/imageField/imageField";
import { keysField } from "../../fields/keysField/keysField";
import { slugField } from "../../fields/slugField/slugField";
import { translatedFields } from "../../fields/translatedFields/translatedFields";
import { createEditor } from "../../utils/editor";
import { buildVersionedCollectionConfig } from "../../utils/versionedCollectionConfig";
import { AppearanceRowLabel } from "./components/AppearanceRowLabel";
import { getBySlugEndpoint } from "./endpoints/getBySlugEndpoint";
@ -113,7 +114,8 @@ export const Weapons = buildVersionedCollectionConfig({
},
{
name: fields.appearancesTranslationsDescription,
type: "textarea",
type: "richText",
editor: createEditor({ inlines: true }),
admin: { width: "0%" },
},
],
@ -124,13 +126,15 @@ export const Weapons = buildVersionedCollectionConfig({
{
name: fields.appearancesTranslationsLevel1,
label: "Level 1",
type: "textarea",
type: "richText",
editor: createEditor({ inlines: true }),
admin: { width: "0%" },
},
{
name: fields.appearancesTranslationsLevel2,
label: "Level 2",
type: "textarea",
type: "richText",
editor: createEditor({ inlines: true }),
admin: { width: "0%" },
},
],
@ -141,13 +145,15 @@ export const Weapons = buildVersionedCollectionConfig({
{
name: fields.appearancesTranslationsLevel3,
label: "Level 3",
type: "textarea",
type: "richText",
editor: createEditor({ inlines: true }),
admin: { width: "0%" },
},
{
name: fields.appearancesTranslationsLevel4,
label: "Level 4",
type: "textarea",
type: "richText",
editor: createEditor({ inlines: true }),
admin: { width: "0%" },
},
],

View File

@ -8,10 +8,12 @@ export enum Collections {
Files = "files",
Keys = "keys",
Languages = "languages",
LibraryFolders = "library-folders",
LibraryItems = "library-items",
LibraryItemsThumbnails = "library-items-thumbnails",
LibraryItemsScans = "library-items-scans",
LibraryItemsGallery = "library-items-gallery",
Notes = "Notes",
Posts = "posts",
PostsThumbnails = "posts-thumbnails",
Recorders = "recorders",

View File

@ -43,7 +43,7 @@ type Params<S> = {
convert?: (
strapiObject: S,
user: any
) => Parameters<BasePayload<GeneratedTypes>["create"]>[0]["data"];
) => Promise<Parameters<BasePayload<GeneratedTypes>["create"]>[0]["data"]>;
};
};
@ -64,7 +64,7 @@ export const importStrapiEntries = async <S>({
} else if (isDefined(payloadParams.convert)) {
await payload.create({
collection: payloadParams.collection,
data: payloadParams.convert(attributes, user),
data: await payloadParams.convert(attributes, user),
user,
});
} else {

View File

@ -28,7 +28,7 @@ const languageField: Field = {
type: "relationship",
relationTo: Collections.Languages,
required: true,
admin: { allowCreate: false, width: "0%" },
admin: { allowCreate: false },
};
const sourceLanguageField: Field = {
@ -36,7 +36,7 @@ const sourceLanguageField: Field = {
type: "relationship",
relationTo: Collections.Languages,
required: true,
admin: { allowCreate: false, width: "0%" },
admin: { allowCreate: false },
};
const creditFields: Field = {

View File

@ -1,11 +1,9 @@
import { webpackBundler } from "@payloadcms/bundler-webpack";
import { mongooseAdapter } from "@payloadcms/db-mongodb";
import { BlocksFeature, lexicalEditor } from "@payloadcms/richtext-lexical";
import path from "path";
import { buildConfig } from "payload/config";
import { ChronologyEras } from "./collections/ChronologyEras/ChronologyEras";
import { ChronologyItems } from "./collections/ChronologyItems/ChronologyItems";
import { transcriptBlock } from "./collections/Contents/Blocks/transcriptBlock";
import { Contents } from "./collections/Contents/Contents";
import { ContentsFolders } from "./collections/ContentsFolders/ContentsFolders";
import { ContentsThumbnails } from "./collections/ContentsThumbnails/ContentsThumbnails";
@ -13,10 +11,12 @@ import { Currencies } from "./collections/Currencies/Currencies";
import { Files } from "./collections/Files/Files";
import { Keys } from "./collections/Keys/Keys";
import { Languages } from "./collections/Languages/Languages";
import { LibraryFolders } from "./collections/LibraryFolders/LibraryFolders";
import { LibraryItems } from "./collections/LibraryItems/LibraryItems";
import { LibraryItemsGallery } from "./collections/LibraryItemsGallery/LibraryItemsGallery";
import { LibraryItemsScans } from "./collections/LibraryItemsScans/LibraryItemsScans";
import { LibraryItemsThumbnails } from "./collections/LibraryItemsThumbnails/LibraryItemsThumbnails";
import { Notes } from "./collections/Notes/Notes";
import { Posts } from "./collections/Posts/Posts";
import { PostsThumbnails } from "./collections/PostsThumbnails/PostsThumbnails";
import { Recorders } from "./collections/Recorders/Recorders";
@ -29,6 +29,7 @@ import { WeaponsThumbnails } from "./collections/WeaponsThumbnails/WeaponsThumbn
import { Icon } from "./components/Icon";
import { Logo } from "./components/Logo";
import { Collections } from "./constants";
import { createEditor } from "./utils/editor";
export default buildConfig({
serverURL: process.env.PAYLOAD_URI,
@ -43,13 +44,9 @@ export default buildConfig({
css: path.resolve(__dirname, "styles.scss"),
bundler: webpackBundler(),
},
editor: lexicalEditor({
features: ({ defaultFeatures }) => [
...defaultFeatures,
BlocksFeature({ blocks: [transcriptBlock] }),
],
}),
editor: createEditor({}),
collections: [
LibraryFolders,
LibraryItems,
Contents,
ContentsFolders,
@ -66,6 +63,7 @@ export default buildConfig({
RecordersThumbnails,
PostsThumbnails,
Files,
Notes,
Videos,
VideosChannels,
Languages,

View File

@ -56,6 +56,23 @@ html[data-theme="light"] {
}
}
.field-type.no-label > header {
display: none;
}
.lexical-block__block-pill-transcriptBlock + .section-title {
display: none;
}
.blocks-field__block-pill-cueBlock + .section-title {
display: none;
}
.rich-text-lexical.field-type.reduced-margins {
margin-top: -0.75em;
margin-bottom: -2rem;
}
.field-type.array-field.group-array {
> .array-field__header {
.array-field__header-actions {

View File

@ -14,17 +14,15 @@ export type CategoryTranslations = {
}[];
export type RecorderBiographies = {
language: string | Language;
biography: string;
id?: string;
}[];
export type ContentFoldersTranslation = {
language: string | Language;
name: string;
biography: {
[k: string]: unknown;
}[];
id?: string;
}[];
export interface Config {
collections: {
'library-folders': LibraryFolder;
'library-items': LibraryItem;
contents: Content;
'contents-folders': ContentsFolder;
@ -41,6 +39,7 @@ export interface Config {
'recorders-thumbnails': RecordersThumbnail;
'posts-thumbnails': PostThumbnail;
files: File;
Notes: Note;
videos: Video;
'videos-channels': VideosChannel;
languages: Language;
@ -52,9 +51,28 @@ export interface Config {
};
globals: {};
}
export interface LibraryFolder {
id: string;
slug: string;
translations?: {
language: string | Language;
name: string;
description?: {
[k: string]: unknown;
}[];
id?: string;
}[];
subfolders?: string[] | LibraryFolder[];
items?: string[] | LibraryItem[];
}
export interface Language {
id: string;
name: string;
}
export interface LibraryItem {
id: string;
itemType?: 'Textual' | 'Audio' | 'Video' | 'Game' | 'Other';
digital: boolean;
slug: string;
thumbnail?: string | LibraryItemThumbnail;
pretitle?: string;
@ -62,8 +80,6 @@ export interface LibraryItem {
subtitle?: string;
rootItem: boolean;
primary: boolean;
digital: boolean;
downloadable: boolean;
gallery?: {
image?: string | LibraryItemGallery;
id?: string;
@ -111,6 +127,7 @@ export interface LibraryItem {
image: string | LibraryItemScans;
id?: string;
}[];
downloadable: boolean;
id?: string;
}[];
textual?: {
@ -132,7 +149,9 @@ export interface LibraryItem {
categories?: string[] | Key[];
translations?: {
language: string | Language;
description: string;
description: {
[k: string]: unknown;
}[];
id?: string;
}[];
size?: {
@ -156,7 +175,9 @@ export interface LibraryItem {
pageEnd?: number;
timeStart?: number;
timeEnd?: number;
note?: string;
note?: {
[k: string]: unknown;
}[];
id?: string;
}[];
updatedBy: string | Recorder;
@ -292,10 +313,6 @@ export interface Key {
| 'Wordings';
translations?: CategoryTranslations;
}
export interface Language {
id: string;
name: string;
}
export interface File {
id: string;
filename: string;
@ -318,16 +335,22 @@ export interface Content {
pretitle?: string;
title: string;
subtitle?: string;
summary?: string;
summary?: {
[k: string]: unknown;
}[];
textContent?: {
[k: string]: unknown;
}[];
textTranscribers?: string[] | Recorder[];
textTranslators?: string[] | Recorder[];
textProofreaders?: string[] | Recorder[];
textNotes?: string;
textNotes?: {
[k: string]: unknown;
}[];
video?: string | File;
videoNotes?: string;
videoNotes?: {
[k: string]: unknown;
}[];
audio?: string | File;
id?: string;
}[];
@ -424,7 +447,11 @@ export interface RecordersThumbnail {
export interface ContentsFolder {
id: string;
slug: string;
translations?: ContentFoldersTranslation;
translations?: {
language: string | Language;
name: string;
id?: string;
}[];
subfolders?: string[] | ContentsFolder[];
contents?: string[] | Content[];
}
@ -454,7 +481,9 @@ export interface Post {
language: string | Language;
sourceLanguage: string | Language;
title: string;
summary?: string;
summary?: {
[k: string]: unknown;
}[];
translators?: string[] | Recorder[];
proofreaders?: string[] | Recorder[];
content?: {
@ -529,8 +558,12 @@ export interface ChronologyItem {
language: string | Language;
sourceLanguage: string | Language;
title?: string;
description?: string;
notes?: string;
description?: {
[k: string]: unknown;
}[];
notes?: {
[k: string]: unknown;
}[];
transcribers?: string[] | Recorder[];
translators?: string[] | Recorder[];
proofreaders?: string[] | Recorder[];
@ -551,7 +584,9 @@ export interface ChronologyEra {
translations?: {
language: string | Language;
title: string;
description?: string;
description?: {
[k: string]: unknown;
}[];
id?: string;
}[];
events?: string[] | ChronologyItem[];
@ -570,11 +605,21 @@ export interface Weapon {
language: string | Language;
sourceLanguage: string | Language;
name: string;
description?: string;
level1?: string;
level2?: string;
level3?: string;
level4?: string;
description?: {
[k: string]: unknown;
}[];
level1?: {
[k: string]: unknown;
}[];
level2?: {
[k: string]: unknown;
}[];
level3?: {
[k: string]: unknown;
}[];
level4?: {
[k: string]: unknown;
}[];
transcribers?: string[] | Recorder[];
translators?: string[] | Recorder[];
proofreaders?: string[] | Recorder[];
@ -643,6 +688,14 @@ export interface WeaponsGroup {
}[];
weapons?: string[] | Weapon[];
}
export interface Note {
id: string;
note: {
[k: string]: unknown;
}[];
updatedAt: string;
createdAt: string;
}
export interface Video {
id: string;
uid: string;
@ -690,33 +743,5 @@ export interface PayloadMigration {
declare module 'payload' {
export interface GeneratedTypes {
collections: {
'library-items': LibraryItem
'contents': Content
'contents-folders': ContentsFolder
'posts': Post
'chronology-items': ChronologyItem
'chronology-eras': ChronologyEra
'weapons': Weapon
'weapons-groups': WeaponsGroup
'weapons-thumbnails': WeaponsThumbnail
'contents-thumbnails': ContentsThumbnail
'library-items-thumbnails': LibraryItemThumbnail
'library-items-scans': LibraryItemScans
'library-items-gallery': LibraryItemGallery
'recorders-thumbnails': RecordersThumbnail
'posts-thumbnails': PostThumbnail
'files': File
'videos': Video
'videos-channels': VideosChannel
'languages': Language
'currencies': Currency
'recorders': Recorder
'keys': Key
'payload-preferences': PayloadPreference
'payload-migrations': PayloadMigration
}
}
export interface GeneratedTypes extends Config {}
}

View File

@ -2,6 +2,10 @@ export type StrapiLanguage = {
data?: { attributes: { code: string } };
};
export type StrapiRecorders = {
data?: { attributes: { username: string } }[];
};
export type StrapiImage = {
data?: {
attributes: {
@ -9,6 +13,8 @@ export type StrapiImage = {
mime: string;
name: string;
size: number;
hash: string;
ext: string;
};
};
};

75
src/utils/editor.ts Normal file
View File

@ -0,0 +1,75 @@
import {
AdapterProps,
AlignFeature,
BlocksFeature,
BoldTextFeature,
CheckListFeature,
FeatureProvider,
// IndentFeature,
HeadingFeature,
InlineCodeTextFeature,
ItalicTextFeature,
LinkFeature,
// BlockQuoteFeature,
OrderedListFeature,
ParagraphFeature,
RelationshipFeature,
StrikethroughTextFeature,
SubscriptTextFeature,
SuperscriptTextFeature,
TreeviewFeature,
UnderlineTextFeature,
UnoderedListFeature,
UploadFeature,
lexicalEditor,
} from "@payloadcms/richtext-lexical";
import { Block, RichTextAdapter } from "payload/types";
interface EditorOptions {
debugs: boolean;
blocks: Block[];
headings: boolean;
lists: boolean;
inlines: boolean;
images: boolean;
relations: boolean;
links: boolean;
alignment: boolean;
}
export const createEditor = ({
debugs = false,
blocks = [],
headings = false,
images = false,
inlines = false,
lists = false,
links = false,
relations = false,
alignment = false,
}: Partial<EditorOptions>): RichTextAdapter<any, AdapterProps> => {
const enabledFeatures: FeatureProvider[] = [];
if (lists) enabledFeatures.push(OrderedListFeature(), UnoderedListFeature(), CheckListFeature());
if (blocks.length > 0) enabledFeatures.push(BlocksFeature({ blocks }));
if (headings) enabledFeatures.push(ParagraphFeature(), HeadingFeature({}));
if (debugs) enabledFeatures.push(TreeviewFeature());
if (images) enabledFeatures.push(UploadFeature({ collections: [] }));
if (links) enabledFeatures.push(LinkFeature({}));
if (relations) enabledFeatures.push(RelationshipFeature());
if (alignment) enabledFeatures.push(AlignFeature());
if (inlines)
enabledFeatures.push(
BoldTextFeature(),
ItalicTextFeature(),
UnderlineTextFeature(),
StrikethroughTextFeature(),
SubscriptTextFeature(),
SuperscriptTextFeature(),
InlineCodeTextFeature()
);
return lexicalEditor({
features: enabledFeatures,
});
};

View File

@ -21,6 +21,24 @@ export const findCategory = async (name: string): Promise<string> => {
return key.docs[0]?.id;
};
export const findRecorder = async (name: string): Promise<string> => {
const recorder = await payload.find({
collection: Collections.Recorders,
where: { username: { equals: name } },
});
if (!recorder.docs[0]) throw new Error(`Recorder ${name} wasn't found`);
return recorder.docs[0]?.id;
};
export const findContentType = async (name: string): Promise<string> => {
const key = await payload.find({
collection: Collections.Keys,
where: { name: { equals: name }, type: { equals: KeysTypes.Contents } },
});
if (!key.docs[0]) throw new Error(`Content type ${name} wasn't found`);
return key.docs[0]?.id;
};
type UploadStrapiImage = {
image: StrapiImage;
collection: Collections;
@ -31,6 +49,16 @@ export const uploadStrapiImage = async ({
image,
}: UploadStrapiImage): Promise<string | undefined> => {
if (isDefined(image.data)) {
const filename = image.data.attributes.hash + image.data.attributes.ext;
const existingImage = await payload.find({
collection,
where: { filename: { equals: filename } },
});
if (existingImage.docs[0]) {
return existingImage.docs[0].id;
}
const url = `${process.env.STRAPI_URI}${image.data.attributes.url}`;
const blob = await (await fetch(url)).blob();
@ -41,7 +69,7 @@ export const uploadStrapiImage = async ({
file: {
data: buffer,
mimetype: image.data.attributes.mime,
name: image.data.attributes.name,
name: filename,
size: image.data.attributes.size,
},
data: {},

View File

@ -21,3 +21,37 @@ export const formatToCamelCase = (name: string): string =>
.join("");
export const formatToPascalCase = (name: string): string => capitalize(formatToCamelCase(name));
export const plainTextToLexical = (
text: string
): {
[k: string]: unknown;
}[] => ({
root: {
type: "root",
format: "",
indent: 0,
version: 1,
children: [
{
children: [
{
detail: 0,
format: 0,
mode: "normal",
style: "",
text,
type: "text",
version: 1,
},
],
direction: "ltr",
format: "",
indent: 0,
type: "paragraph",
version: 1,
},
],
direction: "ltr",
},
});