erudit 3.0.0-dev.4 → 3.0.0-dev.5
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/app/app.vue +195 -172
- package/app/components/Loading.vue +23 -23
- package/app/components/SiteAside.vue +393 -382
- package/app/components/SiteMain.vue +35 -35
- package/app/components/ads/BannerTemplate.vue +51 -51
- package/app/components/ads/BottomBanner.vue +45 -45
- package/app/components/ads/LeftBanner.vue +50 -50
- package/app/components/aside/AsideListItem.vue +74 -74
- package/app/components/aside/AsideMajor.vue +56 -56
- package/app/components/aside/AsideMinor.vue +71 -71
- package/app/components/aside/major/PaneContentScroll.vue +23 -23
- package/app/components/aside/major/PaneSwitch.vue +54 -54
- package/app/components/aside/major/PaneSwitchButton.vue +63 -63
- package/app/components/aside/major/SiteInfo.vue +85 -85
- package/app/components/aside/major/panes/Language.vue +79 -79
- package/app/components/aside/major/panes/Pages.vue +34 -34
- package/app/components/aside/major/panes/Search.vue +11 -11
- package/app/components/aside/major/panes/nav/Nav.vue +87 -91
- package/app/components/aside/major/panes/nav/NavBook.vue +87 -87
- package/app/components/aside/major/panes/nav/NavBookLoading.vue +24 -24
- package/app/components/aside/major/panes/nav/NavGlobal.vue +16 -16
- package/app/components/aside/major/panes/nav/fnav/FNav.vue +105 -105
- package/app/components/aside/major/panes/nav/fnav/FNavBook.vue +32 -32
- package/app/components/aside/major/panes/nav/fnav/FNavFlags.vue +40 -40
- package/app/components/aside/major/panes/nav/fnav/FNavFolder.vue +60 -60
- package/app/components/aside/major/panes/nav/fnav/FNavItem.vue +34 -34
- package/app/components/aside/major/panes/nav/fnav/FNavSeparator.vue +80 -80
- package/app/components/aside/major/panes/nav/fnav/FNavTopic.vue +24 -24
- package/app/components/aside/major/panes/other/ItemContent.vue +29 -29
- package/app/components/aside/major/panes/other/ItemGenerator.vue +15 -15
- package/app/components/aside/major/panes/other/ItemTheme.vue +54 -54
- package/app/components/aside/major/panes/other/Other.vue +16 -16
- package/app/components/aside/minor/AsideMinorContributor.vue +5 -5
- package/app/components/aside/minor/AsideMinorNews.vue +11 -11
- package/app/components/aside/minor/AsideMinorPane.vue +15 -15
- package/app/components/aside/minor/AsideMinorTopLink.vue +67 -67
- package/app/components/aside/minor/Contribute.vue +145 -145
- package/app/components/aside/minor/content/AsideMinorContent.vue +92 -92
- package/app/components/aside/minor/topic/AsideMinorTopic.vue +32 -32
- package/app/components/aside/minor/topic/TopicContributors.vue +177 -177
- package/app/components/aside/minor/topic/TopicNav.vue +49 -49
- package/app/components/aside/minor/topic/TopicToc.vue +219 -203
- package/app/components/aside/minor/topic/TopicTocItem.vue +30 -31
- package/app/components/aside/utils/AsideOverlayPane.vue +40 -40
- package/app/components/bitran/BitranContent.vue +70 -63
- package/app/components/bitran/RenderWrapper.vue +10 -10
- package/app/components/contributor/ContributorAvatar.vue +43 -43
- package/app/components/contributor/ContributorListItem.vue +35 -35
- package/app/components/main/topic/MainTopic.vue +79 -79
- package/app/components/main/topic/TopicPartSwitch.vue +118 -118
- package/app/components/main/utils/Breadcrumb.vue +75 -75
- package/app/components/main/utils/ContentDecoration.vue +29 -29
- package/app/components/main/utils/ContentDescription.vue +19 -19
- package/app/components/main/utils/ContentPopover.vue +188 -176
- package/app/components/main/utils/ContentPopovers.vue +105 -105
- package/app/components/main/utils/ContentReferences.vue +70 -70
- package/app/components/main/utils/ContentSection.vue +45 -45
- package/app/components/main/utils/ContentTitle.vue +63 -39
- package/app/components/main/utils/reference/ReferenceGroup.vue +38 -38
- package/app/components/main/utils/reference/ReferenceItem.vue +68 -68
- package/app/components/main/utils/reference/ReferenceSource.vue +116 -116
- package/app/components/preview/Preview.vue +186 -177
- package/app/components/preview/PreviewDisplay.vue +139 -139
- package/app/components/preview/PreviewFooterAction.vue +73 -73
- package/app/components/preview/PreviewLoading.vue +14 -14
- package/app/components/preview/PreviewScreen.vue +141 -99
- package/app/components/preview/display/Alert.vue +50 -50
- package/app/components/preview/display/Custom.vue +18 -18
- package/app/components/preview/display/GenericLink.vue +48 -48
- package/app/components/preview/display/PageLink.vue +20 -20
- package/app/components/preview/display/Unique.vue +49 -49
- package/app/components/transition/Fade.vue +19 -19
- package/app/components/tree/TreeContainer.vue +11 -11
- package/app/components/tree/TreeItem.vue +89 -89
- package/app/composables/bitran.ts +129 -127
- package/app/composables/bitranContent.ts +39 -39
- package/app/composables/bitranLocation.ts +7 -7
- package/app/composables/contentData.ts +36 -36
- package/app/composables/contentPage.ts +157 -156
- package/app/composables/contentRoute.ts +45 -45
- package/app/composables/darkMagic.ts +24 -24
- package/app/composables/externalApi.ts +63 -63
- package/app/composables/favicon.ts +8 -8
- package/app/composables/formatText.ts +86 -86
- package/app/composables/majorPane.ts +60 -60
- package/app/composables/phrases.ts +81 -81
- package/app/composables/theme.ts +29 -29
- package/app/composables/url.ts +33 -33
- package/app/pages/_test/preview.vue +110 -110
- package/app/pages/article/[...articleId].vue +3 -3
- package/app/pages/book/[...bookId].vue +47 -47
- package/app/pages/group/[...groupId].vue +65 -65
- package/app/pages/index.vue +32 -32
- package/app/pages/members.vue +6 -6
- package/app/pages/practice/[...practice].vue +3 -3
- package/app/pages/summary/[...summaryId].vue +3 -3
- package/app/public/favicon/article.svg +9 -9
- package/app/public/favicon/default.svg +3 -3
- package/app/public/favicon/practice.svg +3 -3
- package/app/public/favicon/summary.svg +4 -4
- package/app/public/logotype.svg +2 -2
- package/app/scripts/_immediate.js +9 -9
- package/app/scripts/aside/index.ts +59 -59
- package/app/scripts/aside/major/nav.ts +26 -26
- package/app/scripts/aside/minor/state.ts +37 -37
- package/app/scripts/aside/minor/topic.ts +3 -3
- package/app/scripts/flag.ts +28 -28
- package/app/scripts/og.ts +27 -27
- package/app/scripts/preview/build.ts +73 -73
- package/app/scripts/preview/data/alert.ts +19 -19
- package/app/scripts/preview/data/custom.ts +8 -8
- package/app/scripts/preview/data/genericLink.ts +24 -24
- package/app/scripts/preview/data/pageLink.ts +22 -22
- package/app/scripts/preview/data/unique.ts +71 -71
- package/app/scripts/preview/data.ts +24 -24
- package/app/scripts/preview/display.ts +37 -37
- package/app/scripts/preview/footer.ts +9 -9
- package/app/scripts/preview/request.ts +51 -51
- package/app/scripts/preview/state.ts +63 -63
- package/app/styles/_immediate.css +7 -7
- package/app/styles/_util.scss +43 -43
- package/app/styles/_utils.scss +44 -44
- package/app/styles/app.scss +91 -91
- package/app/styles/def/_bp.scss +27 -27
- package/app/styles/def/_size.scss +7 -7
- package/app/styles/def/_z.scss +5 -5
- package/app/styles/normalize.scss +63 -63
- package/app/styles/partials/_darkMagic.scss +5 -5
- package/app/styles/partials/_fnav.scss +15 -15
- package/app/styles/partials/_preview.scss +5 -5
- package/globalPath.ts +21 -21
- package/globals/bitran.ts +2 -47
- package/globals/content.ts +22 -22
- package/globals/contributor.ts +5 -5
- package/globals/erudit.ts +5 -5
- package/globals/register.ts +18 -18
- package/languages/en.ts +95 -95
- package/languages/ru.ts +99 -99
- package/module/bitran.ts +35 -34
- package/module/config.ts +34 -34
- package/module/imports.ts +50 -46
- package/module/index.ts +47 -47
- package/module/logger.ts +10 -10
- package/module/paths.ts +22 -22
- package/module/restart.ts +61 -61
- package/nuxt.config.ts +138 -112
- package/package.json +7 -8
- package/server/api/aside/major/nav/bookIds.ts +5 -5
- package/server/api/aside/major/nav/bookNav/[...bookId].ts +20 -20
- package/server/api/aside/major/nav/global.ts +7 -7
- package/server/api/aside/minor/news.ts +7 -7
- package/server/api/aside/minor/path.ts +78 -78
- package/server/api/bitran/content/[location].ts +8 -8
- package/server/api/bitran/toc/[location].ts +7 -7
- package/server/api/content/data.ts +72 -72
- package/server/api/contributor/count.ts +6 -6
- package/server/api/fake/content.ts +11 -11
- package/server/api/fake/shared/languages.ts +12 -12
- package/server/api/language/functions.ts +12 -12
- package/server/api/language/phrase/[phraseId].ts +19 -19
- package/server/api/language/phraseIds.ts +8 -8
- package/server/api/preview/page/[...parts].ts +51 -51
- package/server/api/preview/unique/[location].ts +57 -57
- package/server/plugin/bitran/content.ts +187 -187
- package/server/plugin/bitran/location.ts +25 -25
- package/server/plugin/bitran/products/include.ts +230 -230
- package/server/plugin/bitran/products/link.ts +116 -116
- package/server/plugin/bitran/setup.ts +11 -9
- package/server/plugin/bitran/toc.ts +83 -83
- package/server/plugin/bitran/transpiler.ts +48 -46
- package/server/plugin/build/close.ts +10 -10
- package/server/plugin/build/jobs/content/builderArgs.ts +8 -8
- package/server/plugin/build/jobs/content/generic.ts +176 -176
- package/server/plugin/build/jobs/content/parse.ts +100 -100
- package/server/plugin/build/jobs/content/path.ts +6 -6
- package/server/plugin/build/jobs/content/type/book.ts +9 -9
- package/server/plugin/build/jobs/content/type/group.ts +37 -37
- package/server/plugin/build/jobs/content/type/topic.ts +36 -36
- package/server/plugin/build/jobs/contributors.ts +66 -66
- package/server/plugin/build/jobs/language.ts +36 -36
- package/server/plugin/build/jobs/nav.ts +210 -210
- package/server/plugin/build/process.ts +25 -25
- package/server/plugin/build/rebuild.ts +55 -55
- package/server/plugin/build/setup.ts +21 -21
- package/server/plugin/content/absoluteId.ts +94 -94
- package/server/plugin/content/context.ts +112 -112
- package/server/plugin/db/entities/Book.ts +7 -7
- package/server/plugin/db/entities/Content.ts +49 -49
- package/server/plugin/db/entities/Contribution.ts +10 -10
- package/server/plugin/db/entities/Contributor.ts +16 -16
- package/server/plugin/db/entities/Group.ts +14 -14
- package/server/plugin/db/entities/Hash.ts +15 -15
- package/server/plugin/db/entities/Topic.ts +20 -20
- package/server/plugin/db/entities/Unique.ts +21 -21
- package/server/plugin/db/setup.ts +34 -34
- package/server/plugin/global.ts +17 -18
- package/server/plugin/importer.ts +12 -12
- package/server/plugin/index.ts +9 -9
- package/server/plugin/logger.ts +23 -23
- package/server/plugin/nav/node.ts +26 -26
- package/server/plugin/nav/utils.ts +129 -129
- package/server/plugin/repository/book.ts +21 -21
- package/server/plugin/repository/content.ts +238 -238
- package/server/plugin/repository/contributor.ts +8 -8
- package/server/plugin/repository/frontNav.ts +148 -148
- package/server/plugin/repository/topic.ts +32 -32
- package/server/tsconfig.json +9 -9
- package/shared/aside/minor.ts +50 -50
- package/shared/asset.ts +15 -15
- package/shared/bitran/context.ts +8 -8
- package/shared/bitran/default.ts +46 -46
- package/shared/bitran/link/Link.vue +166 -167
- package/shared/bitran/link/factory.ts +24 -24
- package/shared/bitran/link/languages/en.ts +7 -7
- package/shared/bitran/link/languages/ru.ts +7 -7
- package/shared/bitran/link/renderer.ts +21 -21
- package/shared/bitran/link/shared.ts +17 -17
- package/shared/bitran/link/target.ts +134 -134
- package/shared/bitran/link/transpiler.ts +10 -10
- package/shared/bitran/location.ts +166 -166
- package/shared/bitran/toc.ts +8 -8
- package/shared/content/context.ts +9 -9
- package/shared/content/data/base.ts +32 -32
- package/shared/content/data/index.ts +5 -5
- package/shared/content/data/type/book.ts +5 -5
- package/shared/content/data/type/group.ts +6 -6
- package/shared/content/data/type/topic.ts +11 -11
- package/shared/content/previousNext.ts +9 -9
- package/shared/contributor.ts +5 -5
- package/shared/frontNav.ts +41 -41
- package/shared/icons.ts +38 -38
- package/shared/image.ts +5 -5
- package/shared/link.ts +25 -25
- package/shared/popover.ts +8 -0
- package/shared/types/language.ts +75 -75
- package/shared/utils/objectsEqual.ts +4 -4
- package/shared/utils/stringColor.ts +9 -9
- package/test/bitran/link/target.test.ts +141 -141
- package/test/bitran/location.test.ts +143 -143
- package/tsconfig.json +8 -8
- package/utils/stress.ts +9 -9
- package/app/components/main/utils/ContentFlag.vue +0 -15
- package/app/styles/default.scss +0 -83
|
@@ -1,238 +1,238 @@
|
|
|
1
|
-
import type { ContentType } from 'erudit-cog/schema';
|
|
2
|
-
import { Like } from 'typeorm';
|
|
3
|
-
|
|
4
|
-
import { ERUDIT_SERVER } from '@server/global';
|
|
5
|
-
import { DbContent } from '@server/db/entities/Content';
|
|
6
|
-
import { DbContribution } from '@server/db/entities/Contribution';
|
|
7
|
-
|
|
8
|
-
import { getContentContext } from '@server/content/context';
|
|
9
|
-
import type { ContentGenericData } from '@shared/content/data/base';
|
|
10
|
-
import { getIdsUp, getPreviousNextNav } from '../nav/utils';
|
|
11
|
-
import { createContentLink, createTopicPartLink } from '@erudit/shared/link';
|
|
12
|
-
import { getTopicPart } from './topic';
|
|
13
|
-
import type { PreviousNextItem } from '@erudit/shared/content/previousNext';
|
|
14
|
-
import type { NavNode } from '../nav/node';
|
|
15
|
-
import { toAbsoluteContentId } from '../content/absoluteId';
|
|
16
|
-
|
|
17
|
-
import type { ContentContributor } from '@erudit/shared/contributor';
|
|
18
|
-
import { DbContributor } from '../db/entities/Contributor';
|
|
19
|
-
|
|
20
|
-
export async function getContentGenericData(
|
|
21
|
-
contentId: string,
|
|
22
|
-
): Promise<ContentGenericData> {
|
|
23
|
-
const dbContent = await ERUDIT_SERVER.DB.manager.findOne(DbContent, {
|
|
24
|
-
where: { contentId },
|
|
25
|
-
});
|
|
26
|
-
|
|
27
|
-
if (!dbContent)
|
|
28
|
-
throw createError({
|
|
29
|
-
statusCode: 404,
|
|
30
|
-
message: `Content item "${contentId}" not found!`,
|
|
31
|
-
});
|
|
32
|
-
|
|
33
|
-
const previousNext = await getPreviousNext(contentId);
|
|
34
|
-
const decoration = await getContentDecoration(contentId);
|
|
35
|
-
const flags = await getContentFlags(contentId);
|
|
36
|
-
|
|
37
|
-
const contentPage: ContentGenericData = {
|
|
38
|
-
contentId,
|
|
39
|
-
type: dbContent.type,
|
|
40
|
-
title: dbContent.title || undefined,
|
|
41
|
-
description: dbContent.description || undefined,
|
|
42
|
-
flags,
|
|
43
|
-
previousNext,
|
|
44
|
-
decoration,
|
|
45
|
-
context: await getContentContext(contentId),
|
|
46
|
-
seo: dbContent.seo,
|
|
47
|
-
ogImage: dbContent.ogImage,
|
|
48
|
-
dependencies: dbContent.dependencies
|
|
49
|
-
? await getContentDependencies(contentId, dbContent.dependencies)
|
|
50
|
-
: undefined,
|
|
51
|
-
references: dbContent.references,
|
|
52
|
-
};
|
|
53
|
-
|
|
54
|
-
return contentPage;
|
|
55
|
-
}
|
|
56
|
-
|
|
57
|
-
export async function getPreviousNext(contentId: string) {
|
|
58
|
-
const { previousNav, nextNav } = await getPreviousNextNav(contentId);
|
|
59
|
-
|
|
60
|
-
async function getItemData(navNode: NavNode): Promise<PreviousNextItem> {
|
|
61
|
-
const title = await getContentTitle(navNode.id);
|
|
62
|
-
|
|
63
|
-
const link = await (async () => {
|
|
64
|
-
if (navNode.type === 'topic')
|
|
65
|
-
return createTopicPartLink(
|
|
66
|
-
await getTopicPart(navNode.id),
|
|
67
|
-
navNode.id,
|
|
68
|
-
);
|
|
69
|
-
|
|
70
|
-
return createContentLink(navNode.type, navNode.id);
|
|
71
|
-
})();
|
|
72
|
-
|
|
73
|
-
return {
|
|
74
|
-
title,
|
|
75
|
-
link,
|
|
76
|
-
};
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
return {
|
|
80
|
-
previous: previousNav ? await getItemData(previousNav) : undefined,
|
|
81
|
-
next: nextNav ? await getItemData(nextNav) : undefined,
|
|
82
|
-
};
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
export async function getContentDecoration(contentId: string) {
|
|
86
|
-
const idsUp = await getIdsUp(contentId);
|
|
87
|
-
|
|
88
|
-
for (const id of idsUp) {
|
|
89
|
-
const decoration = (
|
|
90
|
-
await ERUDIT_SERVER.DB.manager.findOne(DbContent, {
|
|
91
|
-
select: ['decoration'],
|
|
92
|
-
where: { contentId: id },
|
|
93
|
-
})
|
|
94
|
-
)?.decoration;
|
|
95
|
-
|
|
96
|
-
if (decoration) return decoration;
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
return undefined;
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
export async function getContentFlags(contentId: string) {
|
|
103
|
-
const idsUp = await getIdsUp(contentId);
|
|
104
|
-
const flags: any = {};
|
|
105
|
-
|
|
106
|
-
for (const id of idsUp.reverse()) {
|
|
107
|
-
const dbFlags = (
|
|
108
|
-
await ERUDIT_SERVER.DB.manager.findOne(DbContent, {
|
|
109
|
-
select: ['flags'],
|
|
110
|
-
where: { contentId: id },
|
|
111
|
-
})
|
|
112
|
-
)?.flags;
|
|
113
|
-
|
|
114
|
-
if (dbFlags)
|
|
115
|
-
for (const [flagName, flagValue] of Object.entries(dbFlags))
|
|
116
|
-
flags[flagName] = flagValue;
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
for (const [flagName, flagValue] of Object.entries(flags))
|
|
120
|
-
if (flagValue === false) delete flags[flagName];
|
|
121
|
-
|
|
122
|
-
return flags;
|
|
123
|
-
}
|
|
124
|
-
|
|
125
|
-
export async function getContentDependencies(
|
|
126
|
-
contentId: string,
|
|
127
|
-
strDependencies: string[],
|
|
128
|
-
) {
|
|
129
|
-
const dependencyIds = strDependencies.map((rawDependency) =>
|
|
130
|
-
toAbsoluteContentId(rawDependency, contentId),
|
|
131
|
-
);
|
|
132
|
-
const dependencies: Record<string, string> = {};
|
|
133
|
-
|
|
134
|
-
for (const dependencyId of dependencyIds)
|
|
135
|
-
dependencies[await getContentLink(dependencyId)] =
|
|
136
|
-
await getContentTitle(dependencyId);
|
|
137
|
-
|
|
138
|
-
return dependencies;
|
|
139
|
-
}
|
|
140
|
-
|
|
141
|
-
export async function getContentTitle(contentId: string) {
|
|
142
|
-
const dbContent = await ERUDIT_SERVER.DB.manager.findOne(DbContent, {
|
|
143
|
-
select: ['title'],
|
|
144
|
-
where: { contentId },
|
|
145
|
-
});
|
|
146
|
-
|
|
147
|
-
return dbContent?.title || contentId.split('/').pop()!;
|
|
148
|
-
}
|
|
149
|
-
|
|
150
|
-
export async function getContentLink(contentId: string) {
|
|
151
|
-
const dbContent = await ERUDIT_SERVER.DB.manager.findOne(DbContent, {
|
|
152
|
-
select: ['type'],
|
|
153
|
-
where: { contentId },
|
|
154
|
-
});
|
|
155
|
-
|
|
156
|
-
if (!dbContent)
|
|
157
|
-
throw createError({
|
|
158
|
-
statusCode: 404,
|
|
159
|
-
statusText: `Missing "${contentId}" content item!`,
|
|
160
|
-
});
|
|
161
|
-
|
|
162
|
-
if (dbContent.type !== 'topic')
|
|
163
|
-
return createContentLink(dbContent.type, contentId);
|
|
164
|
-
|
|
165
|
-
const topicPart = await getTopicPart(contentId);
|
|
166
|
-
|
|
167
|
-
return createTopicPartLink(topicPart, contentId);
|
|
168
|
-
}
|
|
169
|
-
|
|
170
|
-
export async function getFullContentId(contentId: string) {
|
|
171
|
-
const dbContent = await ERUDIT_SERVER.DB.manager.findOne(DbContent, {
|
|
172
|
-
select: ['fullId'],
|
|
173
|
-
where: { contentId },
|
|
174
|
-
});
|
|
175
|
-
|
|
176
|
-
return dbContent?.fullId;
|
|
177
|
-
}
|
|
178
|
-
|
|
179
|
-
export async function getContentContributors(contentId: string) {
|
|
180
|
-
const contributorIds = await (async () => {
|
|
181
|
-
const idHash: Record<string, true> = {};
|
|
182
|
-
|
|
183
|
-
(
|
|
184
|
-
await ERUDIT_SERVER.DB.manager.find(DbContribution, {
|
|
185
|
-
select: ['contributorId'],
|
|
186
|
-
where: { contentId },
|
|
187
|
-
})
|
|
188
|
-
).forEach(
|
|
189
|
-
(contribution) => (idHash[contribution.contributorId] = true),
|
|
190
|
-
);
|
|
191
|
-
|
|
192
|
-
const type: ContentType = (await ERUDIT_SERVER.DB.manager.findOne(
|
|
193
|
-
DbContent,
|
|
194
|
-
{
|
|
195
|
-
select: ['type'],
|
|
196
|
-
where: { contentId },
|
|
197
|
-
},
|
|
198
|
-
))!.type;
|
|
199
|
-
|
|
200
|
-
if (type === 'book' || type === 'group') {
|
|
201
|
-
// Collecting contributors for all sub-content items
|
|
202
|
-
(
|
|
203
|
-
await ERUDIT_SERVER.DB.manager.find(DbContribution, {
|
|
204
|
-
select: ['contributorId'],
|
|
205
|
-
where: { contentId: Like(`${contentId}/%`) },
|
|
206
|
-
})
|
|
207
|
-
).forEach(
|
|
208
|
-
(contribution) => (idHash[contribution.contributorId] = true),
|
|
209
|
-
);
|
|
210
|
-
}
|
|
211
|
-
|
|
212
|
-
return Object.keys(idHash);
|
|
213
|
-
})();
|
|
214
|
-
|
|
215
|
-
const contentContributors: ContentContributor[] = [];
|
|
216
|
-
|
|
217
|
-
for (const contributorId of contributorIds) {
|
|
218
|
-
const dbContributor = (await ERUDIT_SERVER.DB.manager.findOne(
|
|
219
|
-
DbContributor,
|
|
220
|
-
{
|
|
221
|
-
select: ['contributorId', 'displayName', 'avatar'],
|
|
222
|
-
where: { contributorId },
|
|
223
|
-
},
|
|
224
|
-
))!;
|
|
225
|
-
|
|
226
|
-
const contentContributor: ContentContributor = { contributorId };
|
|
227
|
-
|
|
228
|
-
if (dbContributor.displayName)
|
|
229
|
-
contentContributor.displayName = dbContributor.displayName;
|
|
230
|
-
|
|
231
|
-
if (dbContributor.avatar)
|
|
232
|
-
contentContributor.avatar = dbContributor.avatar;
|
|
233
|
-
|
|
234
|
-
contentContributors.push(contentContributor);
|
|
235
|
-
}
|
|
236
|
-
|
|
237
|
-
return contentContributors.length ? contentContributors : undefined;
|
|
238
|
-
}
|
|
1
|
+
import type { ContentType } from 'erudit-cog/schema';
|
|
2
|
+
import { Like } from 'typeorm';
|
|
3
|
+
|
|
4
|
+
import { ERUDIT_SERVER } from '@server/global';
|
|
5
|
+
import { DbContent } from '@server/db/entities/Content';
|
|
6
|
+
import { DbContribution } from '@server/db/entities/Contribution';
|
|
7
|
+
|
|
8
|
+
import { getContentContext } from '@server/content/context';
|
|
9
|
+
import type { ContentGenericData } from '@shared/content/data/base';
|
|
10
|
+
import { getIdsUp, getPreviousNextNav } from '../nav/utils';
|
|
11
|
+
import { createContentLink, createTopicPartLink } from '@erudit/shared/link';
|
|
12
|
+
import { getTopicPart } from './topic';
|
|
13
|
+
import type { PreviousNextItem } from '@erudit/shared/content/previousNext';
|
|
14
|
+
import type { NavNode } from '../nav/node';
|
|
15
|
+
import { toAbsoluteContentId } from '../content/absoluteId';
|
|
16
|
+
|
|
17
|
+
import type { ContentContributor } from '@erudit/shared/contributor';
|
|
18
|
+
import { DbContributor } from '../db/entities/Contributor';
|
|
19
|
+
|
|
20
|
+
export async function getContentGenericData(
|
|
21
|
+
contentId: string,
|
|
22
|
+
): Promise<ContentGenericData> {
|
|
23
|
+
const dbContent = await ERUDIT_SERVER.DB.manager.findOne(DbContent, {
|
|
24
|
+
where: { contentId },
|
|
25
|
+
});
|
|
26
|
+
|
|
27
|
+
if (!dbContent)
|
|
28
|
+
throw createError({
|
|
29
|
+
statusCode: 404,
|
|
30
|
+
message: `Content item "${contentId}" not found!`,
|
|
31
|
+
});
|
|
32
|
+
|
|
33
|
+
const previousNext = await getPreviousNext(contentId);
|
|
34
|
+
const decoration = await getContentDecoration(contentId);
|
|
35
|
+
const flags = await getContentFlags(contentId);
|
|
36
|
+
|
|
37
|
+
const contentPage: ContentGenericData = {
|
|
38
|
+
contentId,
|
|
39
|
+
type: dbContent.type,
|
|
40
|
+
title: dbContent.title || undefined,
|
|
41
|
+
description: dbContent.description || undefined,
|
|
42
|
+
flags,
|
|
43
|
+
previousNext,
|
|
44
|
+
decoration,
|
|
45
|
+
context: await getContentContext(contentId),
|
|
46
|
+
seo: dbContent.seo,
|
|
47
|
+
ogImage: dbContent.ogImage,
|
|
48
|
+
dependencies: dbContent.dependencies
|
|
49
|
+
? await getContentDependencies(contentId, dbContent.dependencies)
|
|
50
|
+
: undefined,
|
|
51
|
+
references: dbContent.references,
|
|
52
|
+
};
|
|
53
|
+
|
|
54
|
+
return contentPage;
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
export async function getPreviousNext(contentId: string) {
|
|
58
|
+
const { previousNav, nextNav } = await getPreviousNextNav(contentId);
|
|
59
|
+
|
|
60
|
+
async function getItemData(navNode: NavNode): Promise<PreviousNextItem> {
|
|
61
|
+
const title = await getContentTitle(navNode.id);
|
|
62
|
+
|
|
63
|
+
const link = await (async () => {
|
|
64
|
+
if (navNode.type === 'topic')
|
|
65
|
+
return createTopicPartLink(
|
|
66
|
+
await getTopicPart(navNode.id),
|
|
67
|
+
navNode.id,
|
|
68
|
+
);
|
|
69
|
+
|
|
70
|
+
return createContentLink(navNode.type, navNode.id);
|
|
71
|
+
})();
|
|
72
|
+
|
|
73
|
+
return {
|
|
74
|
+
title,
|
|
75
|
+
link,
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
|
|
79
|
+
return {
|
|
80
|
+
previous: previousNav ? await getItemData(previousNav) : undefined,
|
|
81
|
+
next: nextNav ? await getItemData(nextNav) : undefined,
|
|
82
|
+
};
|
|
83
|
+
}
|
|
84
|
+
|
|
85
|
+
export async function getContentDecoration(contentId: string) {
|
|
86
|
+
const idsUp = await getIdsUp(contentId);
|
|
87
|
+
|
|
88
|
+
for (const id of idsUp) {
|
|
89
|
+
const decoration = (
|
|
90
|
+
await ERUDIT_SERVER.DB.manager.findOne(DbContent, {
|
|
91
|
+
select: ['decoration'],
|
|
92
|
+
where: { contentId: id },
|
|
93
|
+
})
|
|
94
|
+
)?.decoration;
|
|
95
|
+
|
|
96
|
+
if (decoration) return decoration;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
return undefined;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
export async function getContentFlags(contentId: string) {
|
|
103
|
+
const idsUp = await getIdsUp(contentId);
|
|
104
|
+
const flags: any = {};
|
|
105
|
+
|
|
106
|
+
for (const id of idsUp.reverse()) {
|
|
107
|
+
const dbFlags = (
|
|
108
|
+
await ERUDIT_SERVER.DB.manager.findOne(DbContent, {
|
|
109
|
+
select: ['flags'],
|
|
110
|
+
where: { contentId: id },
|
|
111
|
+
})
|
|
112
|
+
)?.flags;
|
|
113
|
+
|
|
114
|
+
if (dbFlags)
|
|
115
|
+
for (const [flagName, flagValue] of Object.entries(dbFlags))
|
|
116
|
+
flags[flagName] = flagValue;
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
for (const [flagName, flagValue] of Object.entries(flags))
|
|
120
|
+
if (flagValue === false) delete flags[flagName];
|
|
121
|
+
|
|
122
|
+
return flags;
|
|
123
|
+
}
|
|
124
|
+
|
|
125
|
+
export async function getContentDependencies(
|
|
126
|
+
contentId: string,
|
|
127
|
+
strDependencies: string[],
|
|
128
|
+
) {
|
|
129
|
+
const dependencyIds = strDependencies.map((rawDependency) =>
|
|
130
|
+
toAbsoluteContentId(rawDependency, contentId),
|
|
131
|
+
);
|
|
132
|
+
const dependencies: Record<string, string> = {};
|
|
133
|
+
|
|
134
|
+
for (const dependencyId of dependencyIds)
|
|
135
|
+
dependencies[await getContentLink(dependencyId)] =
|
|
136
|
+
await getContentTitle(dependencyId);
|
|
137
|
+
|
|
138
|
+
return dependencies;
|
|
139
|
+
}
|
|
140
|
+
|
|
141
|
+
export async function getContentTitle(contentId: string) {
|
|
142
|
+
const dbContent = await ERUDIT_SERVER.DB.manager.findOne(DbContent, {
|
|
143
|
+
select: ['title'],
|
|
144
|
+
where: { contentId },
|
|
145
|
+
});
|
|
146
|
+
|
|
147
|
+
return dbContent?.title || contentId.split('/').pop()!;
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
export async function getContentLink(contentId: string) {
|
|
151
|
+
const dbContent = await ERUDIT_SERVER.DB.manager.findOne(DbContent, {
|
|
152
|
+
select: ['type'],
|
|
153
|
+
where: { contentId },
|
|
154
|
+
});
|
|
155
|
+
|
|
156
|
+
if (!dbContent)
|
|
157
|
+
throw createError({
|
|
158
|
+
statusCode: 404,
|
|
159
|
+
statusText: `Missing "${contentId}" content item!`,
|
|
160
|
+
});
|
|
161
|
+
|
|
162
|
+
if (dbContent.type !== 'topic')
|
|
163
|
+
return createContentLink(dbContent.type, contentId);
|
|
164
|
+
|
|
165
|
+
const topicPart = await getTopicPart(contentId);
|
|
166
|
+
|
|
167
|
+
return createTopicPartLink(topicPart, contentId);
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
export async function getFullContentId(contentId: string) {
|
|
171
|
+
const dbContent = await ERUDIT_SERVER.DB.manager.findOne(DbContent, {
|
|
172
|
+
select: ['fullId'],
|
|
173
|
+
where: { contentId },
|
|
174
|
+
});
|
|
175
|
+
|
|
176
|
+
return dbContent?.fullId;
|
|
177
|
+
}
|
|
178
|
+
|
|
179
|
+
export async function getContentContributors(contentId: string) {
|
|
180
|
+
const contributorIds = await (async () => {
|
|
181
|
+
const idHash: Record<string, true> = {};
|
|
182
|
+
|
|
183
|
+
(
|
|
184
|
+
await ERUDIT_SERVER.DB.manager.find(DbContribution, {
|
|
185
|
+
select: ['contributorId'],
|
|
186
|
+
where: { contentId },
|
|
187
|
+
})
|
|
188
|
+
).forEach(
|
|
189
|
+
(contribution) => (idHash[contribution.contributorId] = true),
|
|
190
|
+
);
|
|
191
|
+
|
|
192
|
+
const type: ContentType = (await ERUDIT_SERVER.DB.manager.findOne(
|
|
193
|
+
DbContent,
|
|
194
|
+
{
|
|
195
|
+
select: ['type'],
|
|
196
|
+
where: { contentId },
|
|
197
|
+
},
|
|
198
|
+
))!.type;
|
|
199
|
+
|
|
200
|
+
if (type === 'book' || type === 'group') {
|
|
201
|
+
// Collecting contributors for all sub-content items
|
|
202
|
+
(
|
|
203
|
+
await ERUDIT_SERVER.DB.manager.find(DbContribution, {
|
|
204
|
+
select: ['contributorId'],
|
|
205
|
+
where: { contentId: Like(`${contentId}/%`) },
|
|
206
|
+
})
|
|
207
|
+
).forEach(
|
|
208
|
+
(contribution) => (idHash[contribution.contributorId] = true),
|
|
209
|
+
);
|
|
210
|
+
}
|
|
211
|
+
|
|
212
|
+
return Object.keys(idHash);
|
|
213
|
+
})();
|
|
214
|
+
|
|
215
|
+
const contentContributors: ContentContributor[] = [];
|
|
216
|
+
|
|
217
|
+
for (const contributorId of contributorIds) {
|
|
218
|
+
const dbContributor = (await ERUDIT_SERVER.DB.manager.findOne(
|
|
219
|
+
DbContributor,
|
|
220
|
+
{
|
|
221
|
+
select: ['contributorId', 'displayName', 'avatar'],
|
|
222
|
+
where: { contributorId },
|
|
223
|
+
},
|
|
224
|
+
))!;
|
|
225
|
+
|
|
226
|
+
const contentContributor: ContentContributor = { contributorId };
|
|
227
|
+
|
|
228
|
+
if (dbContributor.displayName)
|
|
229
|
+
contentContributor.displayName = dbContributor.displayName;
|
|
230
|
+
|
|
231
|
+
if (dbContributor.avatar)
|
|
232
|
+
contentContributor.avatar = dbContributor.avatar;
|
|
233
|
+
|
|
234
|
+
contentContributors.push(contentContributor);
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
return contentContributors.length ? contentContributors : undefined;
|
|
238
|
+
}
|
|
@@ -1,8 +1,8 @@
|
|
|
1
|
-
import { DbContributor } from '@server/db/entities/Contributor';
|
|
2
|
-
import { ERUDIT_SERVER } from '@server/global';
|
|
3
|
-
|
|
4
|
-
export async function contributorExists(contributorId: string) {
|
|
5
|
-
return await ERUDIT_SERVER.DB.manager.exists(DbContributor, {
|
|
6
|
-
where: { contributorId },
|
|
7
|
-
});
|
|
8
|
-
}
|
|
1
|
+
import { DbContributor } from '@server/db/entities/Contributor';
|
|
2
|
+
import { ERUDIT_SERVER } from '@server/global';
|
|
3
|
+
|
|
4
|
+
export async function contributorExists(contributorId: string) {
|
|
5
|
+
return await ERUDIT_SERVER.DB.manager.exists(DbContributor, {
|
|
6
|
+
where: { contributorId },
|
|
7
|
+
});
|
|
8
|
+
}
|