Skip to content

Commit b2ed420

Browse files
authoredJan 8, 2025··
feat: vanilla function resolves (#443)
* feat: vanilla function resolves Fixes #435 * chore: broken test
1 parent 13db811 commit b2ed420

File tree

8 files changed

+59
-21
lines changed

8 files changed

+59
-21
lines changed
 

‎packages/schema/src/schema.ts

+11-11
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import type { Base as _Base, HtmlAttributes as _HtmlAttributes, Meta as _Meta, Noscript as _Noscript, Style as _Style, BaseBodyAttributes, BodyEvents, DataKeys, DefinedValueOrEmptyObject, HttpEventAttributes, LinkBase, Merge, MergeHead, MetaFlatInput, ScriptBase, Stringable } from 'zhead'
22
import type { InnerContent, ResolvesDuplicates, TagPosition, TagPriority, TagUserProperties, TemplateParams } from './tags'
3-
import type { FalsyEntries, Never } from './util'
3+
import type { Falsey, MaybeFunction, Never, ResolvableValues } from './util'
44

55
export type UserTagConfigWithoutInnerContent = TagPriority & TagPosition & ResolvesDuplicates & Never<InnerContent> & { processTemplateParams?: false } // only allow opt-out
66
export type UserAttributesConfig = ResolvesDuplicates & TagPriority & Never<InnerContent & TagPosition>
@@ -54,22 +54,22 @@ export interface BaseMeta extends Omit<_Meta, 'content'> {
5454

5555
export type EntryAugmentation = undefined | Record<string, any>
5656

57-
export type MaybeFunctionEntries<T> = {
57+
export type MaybeEventFnHandlers<T> = {
5858
[key in keyof T]?: T[key] | ((e: Event) => void)
5959
}
6060

6161
type TitleTemplateResolver = string | ((title?: string) => string | null)
6262

63-
export type Title = string | FalsyEntries<({ textContent: string } & SchemaAugmentations['title']) | null>
63+
export type Title = MaybeFunction<number | string | Falsey> | ResolvableValues<({ textContent: string } & SchemaAugmentations['title'])>
6464
export type TitleTemplate = TitleTemplateResolver | null | ({ textContent: TitleTemplateResolver } & SchemaAugmentations['titleTemplate'])
65-
export type Base<E extends EntryAugmentation = Record<string, any>> = Partial<Merge<SchemaAugmentations['base'], FalsyEntries<_Base>>> & DefinedValueOrEmptyObject<E>
66-
export type Link<E extends EntryAugmentation = Record<string, any>> = FalsyEntries<LinkBase> & MaybeFunctionEntries<HttpEventAttributes> & DataKeys & SchemaAugmentations['link'] & DefinedValueOrEmptyObject<E>
67-
export type Meta<E extends EntryAugmentation = Record<string, any>> = FalsyEntries<BaseMeta> & DataKeys & SchemaAugmentations['meta'] & DefinedValueOrEmptyObject<E>
68-
export type Style<E extends EntryAugmentation = Record<string, any>> = FalsyEntries<_Style> & DataKeys & SchemaAugmentations['style'] & DefinedValueOrEmptyObject<E>
69-
export type Script<E extends EntryAugmentation = Record<string, any>> = FalsyEntries<ScriptBase> & MaybeFunctionEntries<HttpEventAttributes> & DataKeys & SchemaAugmentations['script'] & DefinedValueOrEmptyObject<E>
70-
export type Noscript<E extends EntryAugmentation = Record<string, any>> = FalsyEntries<_Noscript> & DataKeys & SchemaAugmentations['noscript'] & DefinedValueOrEmptyObject<E>
71-
export type HtmlAttributes<E extends EntryAugmentation = Record<string, any>> = FalsyEntries<HtmlAttr> & DataKeys & SchemaAugmentations['htmlAttrs'] & DefinedValueOrEmptyObject<E>
72-
export type BodyAttributes<E extends EntryAugmentation = Record<string, any>> = FalsyEntries<BodyAttr> & MaybeFunctionEntries<BodyEvents> & DataKeys & SchemaAugmentations['bodyAttrs'] & DefinedValueOrEmptyObject<E>
65+
export type Base<E extends EntryAugmentation = Record<string, any>> = Partial<Merge<SchemaAugmentations['base'], ResolvableValues<_Base>>> & DefinedValueOrEmptyObject<E>
66+
export type Link<E extends EntryAugmentation = Record<string, any>> = ResolvableValues<LinkBase> & MaybeEventFnHandlers<HttpEventAttributes> & DataKeys & SchemaAugmentations['link'] & DefinedValueOrEmptyObject<E>
67+
export type Meta<E extends EntryAugmentation = Record<string, any>> = ResolvableValues<BaseMeta> & DataKeys & SchemaAugmentations['meta'] & DefinedValueOrEmptyObject<E>
68+
export type Style<E extends EntryAugmentation = Record<string, any>> = ResolvableValues<_Style> & DataKeys & SchemaAugmentations['style'] & DefinedValueOrEmptyObject<E>
69+
export type Script<E extends EntryAugmentation = Record<string, any>> = ResolvableValues<ScriptBase> & MaybeEventFnHandlers<HttpEventAttributes> & DataKeys & SchemaAugmentations['script'] & DefinedValueOrEmptyObject<E>
70+
export type Noscript<E extends EntryAugmentation = Record<string, any>> = ResolvableValues<_Noscript> & DataKeys & SchemaAugmentations['noscript'] & DefinedValueOrEmptyObject<E>
71+
export type HtmlAttributes<E extends EntryAugmentation = Record<string, any>> = ResolvableValues<HtmlAttr> & DataKeys & SchemaAugmentations['htmlAttrs'] & DefinedValueOrEmptyObject<E>
72+
export type BodyAttributes<E extends EntryAugmentation = Record<string, any>> = ResolvableValues<BodyAttr> & MaybeEventFnHandlers<BodyEvents> & DataKeys & SchemaAugmentations['bodyAttrs'] & DefinedValueOrEmptyObject<E>
7373

7474
export interface HeadUtils {
7575
/**

‎packages/schema/src/tags.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
import type { RuntimeMode } from './head'
22
import type { Head } from './schema'
3-
import type { FalsyEntries } from './util'
3+
import type { ResolvableValues } from './util'
44

55
export interface ResolvesDuplicates {
66
/**
@@ -60,7 +60,7 @@ export interface TagPriority {
6060
tagPriority?: number | 'critical' | 'high' | 'low' | `before:${string}` | `after:${string}`
6161
}
6262

63-
export type TagUserProperties = FalsyEntries<TagPriority & TagPosition & InnerContent & ResolvesDuplicates & ProcessesTemplateParams>
63+
export type TagUserProperties = ResolvableValues<TagPriority & TagPosition & InnerContent & ResolvesDuplicates & ProcessesTemplateParams>
6464

6565
export type TagKey = keyof Head
6666

‎packages/schema/src/util.ts

+6-2
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,10 @@ export type Never<T> = {
22
[P in keyof T]?: never
33
}
44

5-
export type FalsyEntries<T> = {
6-
[key in keyof T]?: T[key] | null | false | undefined // false is soft deprecated
5+
export type MaybeFunction<T> = T | (() => T)
6+
7+
export type Falsey = false | null | undefined
8+
9+
export type ResolvableValues<T> = {
10+
[key in keyof T]?: MaybeFunction<T[key] | Falsey>
711
}

‎packages/shared/src/normalise.ts

+9
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,10 @@ export function normaliseProps<T extends HeadTag>(props: T['props'], virtual: bo
6565
}
6666

6767
if (!virtual && !TagConfigKeys.has(k as string)) {
68+
if (typeof props[k] === 'function' && !String(k).startsWith('on')) {
69+
// @ts-expect-error untyped
70+
props[k] = props[k]()
71+
}
6872
const v = String(props[k])
6973
// data keys get special treatment, we opt for more verbose syntax
7074
const isDataKey = (k as string).startsWith('data-')
@@ -105,6 +109,11 @@ export function normaliseEntryTags<T extends object = Head>(e: HeadEntry<T>): He
105109
}
106110
continue
107111
}
112+
else if (typeof v === 'function' && k !== 'titleTemplate') {
113+
// resolve titles that may be functions
114+
input[k] = v()
115+
continue
116+
}
108117
// @ts-expect-error untyped
109118
tags.push(normaliseTag(k as keyof Head, v, e))
110119
}

‎packages/unhead/src/optionalPlugins/inferSeoMetaPlugin.ts

+1
Original file line numberDiff line numberDiff line change
@@ -72,6 +72,7 @@ export function InferSeoMetaPlugin(options: InferSeoMetaPluginOptions = {}) {
7272
if (description && !hasOgDescription) {
7373
let newOgDescription = options?.ogDescription || description
7474
if (typeof newOgDescription === 'function')
75+
// @ts-expect-error untyped
7576
newOgDescription = newOgDescription(title)
7677

7778
if (newOgDescription) {

‎packages/vue/src/types/schema.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import type { Base as _Base, Link as _Link, Meta as _Meta, Noscript as _Noscript, Script as _Script, Style as _Style, Title as _Title, TitleTemplate as _TitleTemplate, BaseBodyAttr, BaseHtmlAttr, BodyEvents, DataKeys, DefinedValueOrEmptyObject, EntryAugmentation, HeadEntryOptions, MaybeArray, MaybeFunctionEntries, MergeHead, MetaFlatInput, SchemaAugmentations, Unhead } from '@unhead/schema'
1+
import type { Base as _Base, Link as _Link, Meta as _Meta, Noscript as _Noscript, Script as _Script, Style as _Style, Title as _Title, TitleTemplate as _TitleTemplate, BaseBodyAttr, BaseHtmlAttr, BodyEvents, DataKeys, DefinedValueOrEmptyObject, EntryAugmentation, HeadEntryOptions, MaybeArray, MaybeEventFnHandlers, MergeHead, MetaFlatInput, SchemaAugmentations, Unhead } from '@unhead/schema'
22
import type { Plugin, Ref } from 'vue'
33
import type { MaybeComputedRef, MaybeComputedRefEntries, MaybeComputedRefEntriesOnly } from './util'
44

@@ -35,7 +35,7 @@ export type Style<E extends EntryAugmentation = Record<string, any>> = MaybeComp
3535
export type Script<E extends EntryAugmentation = Record<string, any>> = MaybeComputedRefEntries<_Script<E>>
3636
export type Noscript<E extends EntryAugmentation = Record<string, any>> = MaybeComputedRefEntries<_Noscript<E>>
3737
export type HtmlAttributes<E extends EntryAugmentation = Record<string, any>> = MaybeComputedRef<MaybeComputedRefEntries<HtmlAttr & DataKeys & SchemaAugmentations['htmlAttrs'] & DefinedValueOrEmptyObject<E>>>
38-
export type BodyAttributes<E extends EntryAugmentation = Record<string, any>> = MaybeComputedRef<MaybeComputedRefEntries<BodyAttr & DataKeys & SchemaAugmentations['bodyAttrs'] & DefinedValueOrEmptyObject<E>> & MaybeFunctionEntries<BodyEvents>>
38+
export type BodyAttributes<E extends EntryAugmentation = Record<string, any>> = MaybeComputedRef<MaybeComputedRefEntries<BodyAttr & DataKeys & SchemaAugmentations['bodyAttrs'] & DefinedValueOrEmptyObject<E>> & MaybeEventFnHandlers<BodyEvents>>
3939

4040
export interface ReactiveHead<E extends MergeHead = MergeHead> {
4141
/**

‎packages/vue/src/types/util.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,9 @@
11
import type { ComputedRef, Ref } from 'vue'
22

33
// copied from @vueuse/shared
4-
export type MaybeReadonlyRef<T> = (() => T) | ComputedRef<T>
4+
export type MaybeReadonlyRef<T> = ComputedRef<T>
55
export type MaybeComputedRef<T> = T | MaybeReadonlyRef<T> | Ref<T>
6-
export type MaybeComputedRefOrFalsy<T> = undefined | false | null | T | MaybeReadonlyRef<T> | Ref<T>
6+
export type MaybeComputedRefOrFalsy<T> = T | MaybeReadonlyRef<T> | Ref<T>
77

88
/**
99
* @deprecated Use MaybeComputedRefOrFalsy

‎test/unhead/ssr/ssr.test.ts

+26-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import { renderSSRHead } from '@unhead/ssr'
2-
import { useSeoMeta } from 'unhead'
2+
import { useHead, useSeoMeta } from 'unhead'
33
import { describe, it } from 'vitest'
44
import { basicSchema } from '../../fixtures'
55
import { createHeadWithContext } from '../../util'
@@ -33,7 +33,6 @@ describe('ssr', () => {
3333
const head = createHeadWithContext()
3434

3535
head.push({
36-
// @ts-expect-error handle numbers
3736
title: 12345,
3837
})
3938

@@ -190,4 +189,29 @@ describe('ssr', () => {
190189
}
191190
`)
192191
})
192+
193+
it('title function', async () => {
194+
const head = createHeadWithContext()
195+
196+
useHead({
197+
title: 'my default title',
198+
})
199+
200+
useHead({
201+
title: () => {
202+
return undefined
203+
},
204+
})
205+
206+
const ctx = await renderSSRHead(head)
207+
expect(ctx).toMatchInlineSnapshot(`
208+
{
209+
"bodyAttrs": "",
210+
"bodyTags": "",
211+
"bodyTagsOpen": "",
212+
"headTags": "<title>my default title</title>",
213+
"htmlAttrs": "",
214+
}
215+
`)
216+
})
193217
})

0 commit comments

Comments
 (0)
Please sign in to comment.