@nuxtjs/prismic 1.4.2 → 3.0.0-alpha.10

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/src/module.ts ADDED
@@ -0,0 +1,157 @@
1
+ import { join } from 'path'
2
+
3
+ import {
4
+ defineNuxtModule,
5
+ createResolver,
6
+ addTemplate,
7
+ addPlugin,
8
+ addAutoImport,
9
+ addComponent,
10
+ extendPages
11
+ } from '@nuxt/kit'
12
+ import { cleanDoubleSlashes } from 'ufo'
13
+
14
+ import { isRepositoryEndpoint, getRepositoryName } from '@prismicio/client'
15
+ import * as prismicVue from '@prismicio/vue'
16
+
17
+ import { name as pkgName, version as pkgVersion } from '../package.json'
18
+ import type { PrismicModuleOptions } from './types'
19
+ import { fileExists } from './utils'
20
+ import { logger } from './logger'
21
+
22
+ // Options export
23
+ export type { PrismicModuleOptions } from './types'
24
+ export type { PrismicModuleOptions as ModuleOptions } from './types'
25
+
26
+ // Module export
27
+ export default defineNuxtModule<PrismicModuleOptions>({
28
+ meta: {
29
+ name: pkgName,
30
+ version: pkgVersion,
31
+ configKey: 'prismic',
32
+ compatibility: { nuxt: '^3.0.0' }
33
+ },
34
+ defaults: nuxt => ({
35
+ endpoint: '',
36
+ clientConfig: {},
37
+ client: cleanDoubleSlashes(`~/${nuxt.options.dir.app}/prismic/client`),
38
+ linkResolver: cleanDoubleSlashes(`~/${nuxt.options.dir.app}/prismic/linkResolver`),
39
+ htmlSerializer: cleanDoubleSlashes(`~/${nuxt.options.dir.app}/prismic/htmlSerializer`),
40
+ injectComponents: true,
41
+ components: {},
42
+ preview: '/preview'
43
+ }),
44
+ hooks: {},
45
+ setup (mergedOptions, nuxt) {
46
+ if (!mergedOptions.endpoint) {
47
+ logger.warn('Options `endpoint` is required, disabling module...')
48
+ return
49
+ }
50
+
51
+ // Runtime dir boilerplate
52
+ const resolver = createResolver(import.meta.url)
53
+ nuxt.options.build.transpile.push(resolver.resolve('runtime'), '@nuxtjs/prismic', '@prismicio/vue')
54
+
55
+ // Add runtime user code
56
+ const proxyUserFileWithUndefinedFallback = (filename: string, path?: string, extensions = ['js', 'mjs', 'ts']) => {
57
+ const resolvedFilename = `prismic/proxy/${filename}.ts`
58
+ const resolvedPath = path
59
+ ? path.replace(/^(~~|@@)/, nuxt.options.rootDir).replace(/^(~|@)/, nuxt.options.srcDir)
60
+ : undefined
61
+ const maybeUserFile = fileExists(resolvedPath, extensions)
62
+
63
+ if (maybeUserFile) {
64
+ // If user file exists, proxy it with vfs
65
+ logger.info(`Using user-defined \`${filename}\` at \`${maybeUserFile.replace(nuxt.options.srcDir, '~').replace(nuxt.options.rootDir, '~~').replace(/\\/g, '/')}\``)
66
+
67
+ addTemplate({
68
+ filename: resolvedFilename,
69
+ getContents: () => `export { default } from '${path}'`
70
+ })
71
+ } else {
72
+ // Else provide `undefined` fallback
73
+ addTemplate({
74
+ filename: resolvedFilename,
75
+ getContents: () => 'export default undefined'
76
+ })
77
+ }
78
+ }
79
+ proxyUserFileWithUndefinedFallback('client', mergedOptions.client)
80
+ proxyUserFileWithUndefinedFallback('linkResolver', mergedOptions.linkResolver)
81
+ proxyUserFileWithUndefinedFallback('htmlSerializer', mergedOptions.htmlSerializer)
82
+
83
+ // Expose options through public runtime config
84
+ nuxt.options.runtimeConfig.public ||= {} as typeof nuxt.options.runtimeConfig.public
85
+ nuxt.options.runtimeConfig.public[pkgName] = mergedOptions
86
+
87
+ // Add plugin
88
+ addPlugin(resolver.resolve('runtime/plugin'))
89
+ addPlugin(resolver.resolve('runtime/plugin.client'))
90
+
91
+ // Add components auto import
92
+ if (mergedOptions.injectComponents) {
93
+ [
94
+ 'PrismicEmbed',
95
+ 'PrismicImage',
96
+ 'PrismicLink',
97
+ 'PrismicText',
98
+ 'PrismicRichText',
99
+ 'SliceZone'
100
+ ].forEach((component) => {
101
+ addComponent({
102
+ name: component,
103
+ export: component,
104
+ filePath: '@prismicio/vue'
105
+ })
106
+ })
107
+ }
108
+
109
+ // Add composable auto import
110
+ const composableAutoImports = Object
111
+ .keys(prismicVue)
112
+ .filter(key => key.startsWith('use'))
113
+ .map((key) => {
114
+ return {
115
+ name: key,
116
+ as: key,
117
+ from: '@prismicio/vue'
118
+ }
119
+ })
120
+ addAutoImport(composableAutoImports)
121
+ addAutoImport({
122
+ name: 'usePrismicPreview',
123
+ as: 'usePrismicPreview',
124
+ from: resolver.resolve('runtime/usePrismicPreview')
125
+ })
126
+
127
+ // Add preview route
128
+ if (mergedOptions.preview) {
129
+ const maybeUserPreviewPage = fileExists(join(nuxt.options.srcDir, nuxt.options.dir.pages, mergedOptions.preview), ['js', 'ts', 'vue'])
130
+
131
+ if (maybeUserPreviewPage) {
132
+ logger.info(`Using user-defined preview page at \`${maybeUserPreviewPage.replace(join(nuxt.options.srcDir), '~').replace(nuxt.options.rootDir, '~~').replace(/\\/g, '/')
133
+ }\`, available at \`${mergedOptions.preview}\``)
134
+ } else {
135
+ logger.info(`Using default preview page, available at \`${mergedOptions.preview}\``)
136
+
137
+ extendPages((pages) => {
138
+ pages.unshift({
139
+ name: 'prismic-preview',
140
+ path: mergedOptions.preview as string, // Checked before
141
+ file: resolver.resolve('runtime/preview.vue')
142
+ })
143
+ })
144
+ }
145
+
146
+ // Add toolbar
147
+ const repositoryName = isRepositoryEndpoint(mergedOptions.endpoint)
148
+ ? getRepositoryName(mergedOptions.endpoint)
149
+ : mergedOptions.endpoint
150
+ nuxt.options.app.head ||= {}
151
+ nuxt.options.app.head.script ||= []
152
+ nuxt.options.app.head.script.push({
153
+ src: `https://static.cdn.prismic.io/prismic.min.js?repo=${repositoryName}&new=true`
154
+ })
155
+ }
156
+ }
157
+ })
@@ -0,0 +1,22 @@
1
+ import { defineNuxtPlugin, refreshNuxtData } from '#app'
2
+
3
+ // import { name as pkgName } from '../../package.json'
4
+ import { PrismicModuleOptions } from '../types'
5
+
6
+ // TODO: Revert when fixed
7
+ const pkgName = '@nuxtjs/prismic'
8
+
9
+ export default defineNuxtPlugin((nuxtApp) => {
10
+ const mergedOptions: PrismicModuleOptions =
11
+ nuxtApp.payload.config[pkgName] ??
12
+ nuxtApp.payload.config.public[pkgName] ??
13
+ {}
14
+
15
+ // Hot reload preview updates
16
+ if (mergedOptions.preview) {
17
+ window.addEventListener('prismicPreviewUpdate', (event) => {
18
+ event.preventDefault()
19
+ refreshNuxtData()
20
+ })
21
+ }
22
+ })
@@ -0,0 +1,66 @@
1
+ import { defineNuxtPlugin, useCookie, useRequestEvent, refreshNuxtData } from '#app'
2
+ import NuxtLink from '#app/components/nuxt-link'
3
+
4
+ import { createPrismic } from '@prismicio/vue'
5
+
6
+ // import { name as pkgName } from '../../package.json'
7
+ import { PrismicModuleOptions } from '../types'
8
+
9
+ // @ts-expect-error vfs cannot be resolved here
10
+ import client from '#build/prismic/proxy/client'
11
+ // @ts-expect-error vfs cannot be resolved here
12
+ import linkResolver from '#build/prismic/proxy/linkResolver'
13
+ // @ts-expect-error vfs cannot be resolved here
14
+ import htmlSerializer from '#build/prismic/proxy/htmlSerializer'
15
+
16
+ // TODO: Revert when fixed
17
+ const pkgName = '@nuxtjs/prismic'
18
+
19
+ export default defineNuxtPlugin((nuxtApp) => {
20
+ const mergedOptions: PrismicModuleOptions =
21
+ nuxtApp.payload.config[pkgName] ??
22
+ nuxtApp.payload.config.public[pkgName] ??
23
+ {}
24
+
25
+ const prismicPlugin = createPrismic({
26
+ ...mergedOptions,
27
+ client,
28
+ linkResolver,
29
+ htmlSerializer,
30
+ injectComponents: false, // Handled at module level
31
+ components: {
32
+ linkInternalComponent: NuxtLink,
33
+ linkExternalComponent: NuxtLink,
34
+ ...mergedOptions.components
35
+ }
36
+ })
37
+
38
+ nuxtApp.vueApp.use(prismicPlugin)
39
+
40
+ if (mergedOptions.preview) {
41
+ const previewCookie = useCookie('io.prismic.preview').value
42
+
43
+ // Update client with req when running server side
44
+ if (process.server) {
45
+ prismicPlugin.client.enableAutoPreviewsFromReq(useRequestEvent()?.req)
46
+ }
47
+
48
+ if (previewCookie) {
49
+ try {
50
+ const session = typeof previewCookie === 'string' ? JSON.parse(decodeURIComponent(previewCookie)) : previewCookie
51
+
52
+ if (Object.keys(session).some(key =>
53
+ key in session &&
54
+ typeof session[key] === 'object' &&
55
+ session[key] !== null &&
56
+ 'preview' in session[key] &&
57
+ session[key].preview)
58
+ ) {
59
+ refreshNuxtData()
60
+ }
61
+ } catch (error) {
62
+ // noop
63
+ }
64
+ }
65
+ }
66
+ })
@@ -0,0 +1,45 @@
1
+ <template>
2
+ <section class="prismic-preview">
3
+ <figure>
4
+ <img
5
+ src=""
6
+ alt="Prismic">
7
+ <figcaption>
8
+ Loading preview...
9
+ </figcaption>
10
+ </figure>
11
+ </section>
12
+ </template>
13
+
14
+ <script setup>
15
+ import { useHead, usePrismicPreview } from '#imports'
16
+
17
+ useHead({
18
+ title: 'Prismic Preview - Loading'
19
+ })
20
+
21
+ usePrismicPreview()
22
+ </script>
23
+
24
+ <style scoped>
25
+ .prismic-preview {
26
+ width: 100%;
27
+ height: 100vh;
28
+ position: fixed;
29
+ top: 0;
30
+ left: 0;
31
+ display: flex;
32
+ justify-content: center;
33
+ align-items: center;
34
+ background: #ffffff;
35
+ color: #666666;
36
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Oxygen,
37
+ Ubuntu, Cantarell, "Open Sans", "Helvetica Neue", sans-serif;
38
+ font-size: 1rem;
39
+ text-align: center;
40
+ }
41
+
42
+ img {
43
+ margin-bottom: 1.25rem;
44
+ }
45
+ </style>
@@ -0,0 +1,21 @@
1
+ import { onMounted } from 'vue'
2
+
3
+ import { useRouter, usePrismic } from '#imports'
4
+
5
+ /**
6
+ * Resolves Prismic previews on the preview entry page (`/preview`)
7
+ *
8
+ * @param defaultURL - The default URL to redirect to if the previewed document doesn't map to one.
9
+ */
10
+ export const usePrismicPreview = (defaultURL = '/'): void => {
11
+ const { client, options: { linkResolver } } = usePrismic()
12
+ const { push } = useRouter()
13
+ onMounted(async () => {
14
+ const redirectURL = await client.resolvePreviewURL({
15
+ linkResolver,
16
+ defaultURL
17
+ })
18
+
19
+ push(redirectURL)
20
+ })
21
+ }
package/src/types.ts ADDED
@@ -0,0 +1,9 @@
1
+ import { PrismicPluginOptions } from '@prismicio/vue'
2
+
3
+ export type PrismicModuleOptions = Omit<PrismicPluginOptions, 'endpoint' | 'client' | 'linkResolver' | 'htmlSerializer'> & {
4
+ endpoint: string;
5
+ client?: string;
6
+ linkResolver?: string;
7
+ htmlSerializer?: string;
8
+ preview?: string | false;
9
+ };
package/src/utils.ts ADDED
@@ -0,0 +1,14 @@
1
+ import { existsSync } from 'fs'
2
+
3
+ export const fileExists = (path?: string, extensions = ['js', 'ts']): string | null => {
4
+ if (!path) {
5
+ return null
6
+ } else if (existsSync(path)) {
7
+ // If path already contains/forces the extension
8
+ return path
9
+ }
10
+
11
+ const extension = extensions.find(extension => existsSync(`${path}.${extension}`))
12
+
13
+ return extension ? `${path}.${extension}` : null
14
+ }
package/src/generator.js DELETED
@@ -1,37 +0,0 @@
1
- const Prismic = require('@prismicio/client')
2
- const logger = require('./logger')
3
-
4
- function generate (options) {
5
- this.nuxt.hook('generate:before', () => {
6
- const client = Prismic.client(options.endpoint, options.apiOptions)
7
- const maybeF = this.options.generate.routes || []
8
-
9
- const fetchRoutes = async (page = 1, routes = []) => {
10
- const response = await client.query('', {
11
- pageSize: 100,
12
- lang: '*',
13
- page
14
- })
15
- const allRoutes = routes.concat(response.results.map(e => e.url))
16
- /* istanbul ignore next */
17
- if (response.results_size + routes.length < response.total_results_size) {
18
- return fetchRoutes(page + 1, allRoutes)
19
- }
20
- return [...new Set(allRoutes)]
21
- }
22
-
23
- this.options.generate.routes = async () => {
24
- try {
25
- const prismicRoutes = await fetchRoutes()
26
- const userRoutes =
27
- typeof maybeF === 'function' ? await maybeF(options) : maybeF
28
- return [...new Set(prismicRoutes.concat(userRoutes))].filter(e => e)
29
- } catch (e) /* istanbul ignore next */ {
30
- logger.error(e)
31
- return []
32
- }
33
- }
34
- })
35
- }
36
-
37
- module.exports = generate
package/src/logger.js DELETED
@@ -1 +0,0 @@
1
- module.exports = require('consola').withScope('@nuxtjs/prismic')
package/src/module.js DELETED
@@ -1,98 +0,0 @@
1
- const path = require('path')
2
- const fs = require('fs')
3
- const logger = require('./logger')
4
- const generate = require('./generator')
5
-
6
- function install (moduleOptions) {
7
- const options = {
8
- preview: true,
9
- previewReloadType: 'hot',
10
- components: true,
11
- modern: false,
12
- ...moduleOptions,
13
- ...(this.options.prismic || {})
14
- }
15
- if (options.preview === true) {
16
- options.preview = '/preview'
17
- }
18
- if (options.modern === true && typeof options.disableGenerator === 'undefined') {
19
- options.disableGenerator = true
20
- }
21
- if (!options.endpoint) {
22
- logger.warn('Options `endpoint` is required, disabling module...')
23
- return
24
- }
25
- const repo = options.endpoint.replace(/^https?:\/\//, '').replace(/(\.cdn)?\.prismic.+/, '')
26
- const app = this.options.dir.app || 'app'
27
-
28
- // Add in Prismic libraries to enable preview
29
- if (options.preview) {
30
- // Add /preview
31
- const userPreviewPage = path.join(this.options.srcDir, app, 'prismic', 'pages', 'preview.vue')
32
- const userPreviewPageExists = fs.existsSync(userPreviewPage)
33
-
34
- this.addTemplate({
35
- fileName: 'prismic/pages/preview.vue',
36
- src: userPreviewPageExists ? userPreviewPage : path.join(__dirname, '../templates/pages/preview.vue')
37
- })
38
- this.extendRoutes((routes, resolve) => {
39
- routes.unshift({
40
- name: 'prismic-preview',
41
- path: options.preview,
42
- component: resolve(this.options.buildDir, 'prismic/pages/preview.vue')
43
- })
44
- })
45
- }
46
-
47
- // Add components
48
- if (options.components) {
49
- this.addPlugin({
50
- fileName: 'prismic/plugins/prismic-components.js',
51
- src: path.resolve(__dirname, '../templates/plugins/prismic-components.js')
52
- })
53
- }
54
-
55
- // Add templates & prismic plugin
56
- const userLinkResolver = path.join(this.options.srcDir, app, 'prismic', 'link-resolver.js')
57
- const userLinkResolverExists = fs.existsSync(userLinkResolver)
58
- const userHtmlSerializer = path.join(this.options.srcDir, app, 'prismic', 'html-serializer.js')
59
-
60
- const apiOptions = options.apiOptions || {}
61
- if (!userLinkResolverExists && !options.linkResolver && !apiOptions.routes) {
62
- logger.warn('Please create ~/app/prismic/link-resolver.js')
63
- }
64
- this.addTemplate({
65
- fileName: 'prismic/link-resolver.js',
66
- src: userLinkResolverExists ? userLinkResolver : path.join(__dirname, '../templates/link-resolver.js'),
67
- options
68
- })
69
- this.addTemplate({
70
- fileName: 'prismic/html-serializer.js',
71
- src: fs.existsSync(userHtmlSerializer) ? userHtmlSerializer : path.join(__dirname, '../templates/html-serializer.js'),
72
- options
73
- })
74
- this.addPlugin({
75
- fileName: 'prismic/plugins/prismic.js',
76
- src: path.resolve(__dirname, '../templates/plugins/prismic.js'),
77
- options: {
78
- preview: options.preview,
79
- previewReloadType: options.previewReloadType,
80
- endpoint: options.endpoint,
81
- modern: options.modern,
82
- apiOptions,
83
- repo,
84
- script: `//static.cdn.prismic.io/prismic.min.js?repo=${repo}&new=true`
85
- }
86
- })
87
-
88
- if (
89
- options.apiOptions &&
90
- options.apiOptions.routes &&
91
- !options.disableGenerator
92
- ) {
93
- generate.call(this, options)
94
- }
95
- }
96
-
97
- module.exports = install
98
- module.exports.meta = require('../package.json')
@@ -1,2 +0,0 @@
1
- <% if (typeof options.htmlSerializer === 'string') { %>export { default } from '<%= options.htmlSerializer %>'<% }
2
- else {%>export default <%= options.htmlSerializer ? serialize(options.htmlSerializer) : 'null' %><% } %>
@@ -1,2 +0,0 @@
1
- <% if (typeof options.linkResolver === 'string') { %>export { default } from '<%= options.linkResolver %>'<% }
2
- else { %>export default <%= options.linkResolver ? serialize(options.linkResolver) : "function (doc) { return null }" %><% } %>
@@ -1,43 +0,0 @@
1
- <template>
2
- <div class="preview-overlay">
3
- <div class="box">
4
- <img src="" alt="Prismic">
5
- <h1>Loading preview...</h1>
6
- </div>
7
- </div>
8
- </template>
9
-
10
- <script>
11
- export default {
12
- nuxtI18n: false,
13
- head: {
14
- title: 'Prismic Preview'
15
- },
16
- mounted () {
17
- this.$prismic.preview()
18
- }
19
- }
20
- </script>
21
-
22
- <style scoped>
23
- .preview-overlay {
24
- width: 100%;
25
- height: 100vh;
26
- position: fixed;
27
- top: 0;
28
- left: 0;
29
- background: white;
30
- font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, Oxygen, Ubuntu, Cantarell, 'Open Sans', 'Helvetica Neue', sans-serif;
31
- color: #666;
32
- font-size: 20px;
33
- text-align: center;
34
- display: table;
35
- }
36
- .box {
37
- display: table-cell;
38
- vertical-align: middle;
39
- }
40
- img {
41
- margin-bottom: 20px;
42
- }
43
- </style>
@@ -1,6 +0,0 @@
1
- import Vue from 'vue'
2
- import { common, nuxt } from '@prismicio/vue/components/common'
3
-
4
- Object.entries({ ...common, ...nuxt }).forEach(([_, c]) => {
5
- Vue.component(c.name, c)
6
- })