Skip to content

Commit

Permalink
Sizes styles rendering
Browse files Browse the repository at this point in the history
  • Loading branch information
Fsss126 committed May 1, 2024
1 parent 6e0c2fa commit a56337e
Show file tree
Hide file tree
Showing 17 changed files with 218 additions and 53 deletions.
26 changes: 19 additions & 7 deletions packages/nuxt/src/runtime/plugin.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,28 +9,27 @@ import {
type PartialGlobalConfig
} from 'vuestic-ui'
import { markRaw, computed, watch, type Ref } from 'vue'
import { useHead, ReactiveHead, defineNuxtPlugin, useCookie } from '#imports'
import type { VuesticOptions } from '../types'
import { useHead, ReactiveHead, defineNuxtPlugin, useCookie } from '#imports'
import NuxtLink from '#app/components/nuxt-link'
import configFromFile from '#vuestic-config'

import type { VuesticOptions } from '../types'

function getGlobalProperty (app, key) {
return app.config.globalProperties[key]
}

export default defineNuxtPlugin(async (nuxtApp) => {
export default defineNuxtPlugin((nuxtApp) => {
const { vueApp: app } = nuxtApp

// It's important to use `, because TS will compile qoutes to " and JSON will not be parsed...
const moduleOptions: VuesticOptions = JSON.parse(`<%= options.value %>`)
const moduleOptions: VuesticOptions = JSON.parse('<%= options.value %>')
const themeCookie = useCookie(moduleOptions.themeCookieKey)
const userConfig = configFromFile || moduleOptions.config || {}
const configWithColors: PartialGlobalConfig = {
...userConfig,
colors: {
currentPresetName: themeCookie.value || userConfig.colors?.currentPresetName || 'light',
...userConfig.colors,
...userConfig.colors
}
}

Expand All @@ -44,7 +43,7 @@ export default defineNuxtPlugin(async (nuxtApp) => {
VaDropdownPlugin,
VaToastPlugin,
VaModalPlugin,
ColorsClassesPlugin,
ColorsClassesPlugin
},
/** Do not import any components. Nuxt will import them automatically */
components: {}
Expand Down Expand Up @@ -76,6 +75,19 @@ export default defineNuxtPlugin(async (nuxtApp) => {
}))
}

const componentConfig = getGlobalProperty(app, '$vaComponentConfig')
if (componentConfig) {
useHead(computed(() => {
return {
style: [
{
innerHTML: componentConfig.renderStyles()
}
]
} satisfies ReactiveHead
}))
}

// Watch for preset name change and update cookie
const { globalConfig } = getGlobalProperty(app, '$vaConfig') as { globalConfig: Ref<GlobalConfig> }
watch(() => globalConfig.value.colors.currentPresetName, (newTheme) => {
Expand Down
4 changes: 2 additions & 2 deletions packages/ui/.storybook/preview.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ import {
VaToastPlugin,
VaModalPlugin,
VaDropdownPlugin,
BreakpointConfigPlugin,
BreakpointConfigPlugin, ComponentConfigPlugin,
} from './../src/main'
import demoIconAliases from './vuestic-config/demo-icon-aliases'
import demoIconFonts from './vuestic-config/demo-icon-fonts'
Expand Down Expand Up @@ -49,7 +49,7 @@ setup((app) => {
},
},
},
plugins: { VaToastPlugin, VaDropdownPlugin, VaModalPlugin, BreakpointConfigPlugin },
plugins: { VaToastPlugin, VaDropdownPlugin, VaModalPlugin, BreakpointConfigPlugin, ComponentConfigPlugin },
}))
})

Expand Down
20 changes: 17 additions & 3 deletions packages/ui/src/components/va-config/VaConfig.demo.vue
Original file line number Diff line number Diff line change
Expand Up @@ -86,8 +86,19 @@
<VbCard title="Colors">
<va-config :colors="{ variables: {
primary: '#f0f',
} }">
<va-button>
} }" :components="isOn ? {
VaButton: {
size: 'xs',
sizesConfig: {
xs: {
variables: {
fontSize: '10px',
},
},
},
},
} : undefined">
<va-button @click="isOn = !isOn">
Button inside va-config
</va-button>

Expand All @@ -106,7 +117,7 @@
</template>

<script>
import { computed } from 'vue'
import { computed, ref } from 'vue'
import { useGlobalConfig } from '../../services/global-config/global-config'
import { useColors } from '../../composables'
import { VaButton } from '../va-button'
Expand Down Expand Up @@ -148,6 +159,8 @@ export default {
const { getColor } = useColors()
const isOn = ref(true)
const cardPreset = {
highlightTop: {
square: true,
Expand Down Expand Up @@ -198,6 +211,7 @@ export default {
buttonRoundConfigValue,
cardPreset,
buttonPreset,
isOn,
}
},
computed: {
Expand Down
76 changes: 52 additions & 24 deletions packages/ui/src/components/va-config/VaConfig.vue
Original file line number Diff line number Diff line change
@@ -1,52 +1,78 @@
<template>
<CssVarsRenderer v-if="doRenderCssVars" v-bind="$attrs">
<CssVarsRenderer v-if="doRenderCssVars" v-bind="{ ...props, ...$attrs }">
<slot />
</CssVarsRenderer>
<slot v-else />
</template>

<script lang="ts">
import { computed, PropType, h, Fragment, defineComponent } from 'vue'
import { computed, PropType, h, Fragment, defineComponent, useCssVars } from 'vue'
import { useComponentPresetProp } from '../../composables/useComponentPreset'
import { ComponentConfig } from '../../services/component-config'
import { provideLocalConfig, useLocalConfig } from '../../composables/useLocalConfig'
import {
provideLocalConfig,
useLocalConfig,
} from '../../composables/useLocalConfig'
import { useGlobalConfigProvider } from './hooks/useGlobalConfigProvider'
import { PartialGlobalConfig } from '../../services/global-config'
import { renderSlotNodes } from '../../utils/headless'
import { useColors } from '../../composables'
import { useColors, useCurrentComponentId } from '../../composables'
import { renderComponentsStyles } from '../../services/component-config/utils/render-styles-from-config'
const ConfigProps = {
components: {
type: Object as PropType<ComponentConfig>,
default: () => ({}),
},
colors: { type: Object as PropType<PartialGlobalConfig['colors']> },
i18n: { type: Object as PropType<PartialGlobalConfig['i18n']> },
}
const CssVarsRenderer = defineComponent({
name: 'VaCssVarsRenderer',
props: ConfigProps,
inheritAttrs: false,
setup (props, { slots, attrs }) {
const { colorsToCSSVariable, colors } = useColors()
const { colorsToCSSVariable, currentPresetName } = useColors()
const id = useCurrentComponentId()
const style = computed(() => {
return colorsToCSSVariable(colors)
const styleAttr = computed(() => {
return colorsToCSSVariable({
...props.colors?.variables,
...(props.colors?.presets || {})[currentPresetName.value],
})
})
return () => h(Fragment, attrs, renderSlotNodes(slots.default, {}, {
style: style.value,
}) || undefined)
const elementId = `va-config-${id}`
const componentStyles = computed(() => renderComponentsStyles(props.components, `#${elementId}`))
return () =>
h(
'div',
{ ...attrs, style: styleAttr.value, id: elementId },
[
componentStyles.value ? h('style', { innerHTML: componentStyles.value }) : undefined,
slots.default?.(),
],
)
},
})
</script>

<script lang="ts" setup>
defineOptions({
name: 'VaConfig',
inheritAttrs: false,
})
const props = defineProps({
...useComponentPresetProp,
components: { type: Object as PropType<ComponentConfig>, default: () => ({}) },
colors: { type: Object as PropType<PartialGlobalConfig['colors']> },
i18n: { type: Object as PropType<PartialGlobalConfig['i18n']> },
...ConfigProps,
})
const prevChain = useLocalConfig()
Expand All @@ -55,19 +81,21 @@ const nextChain = computed(() => [...prevChain.value, props.components])
provideLocalConfig(nextChain)
const newConfig = useGlobalConfigProvider(computed(() => {
const config = {} as any
const newConfig = useGlobalConfigProvider(
computed(() => {
const config = {} as any
if (props.colors) {
config.colors = props.colors
}
if (props.colors) {
config.colors = props.colors
}
if (props.i18n) {
config.i18n = props.i18n
}
if (props.i18n) {
config.i18n = props.i18n
}
return config
}))
return config
}),
)
const doRenderCssVars = computed(() => {
return Boolean(props.colors)
Expand Down
12 changes: 1 addition & 11 deletions packages/ui/src/composables/useComponentVariables.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,24 +3,14 @@ import { ReadonlyOrPlainArray } from '../utils/types/array'
import { computed, getCurrentInstance } from 'vue'
import { isCSSSizeValue, isCSSVariable } from '../utils/css'
import isNil from 'lodash/isNil'
import { SizeProps, SizesConfig } from '../services/size'
import { SizeProps, SizesConfig } from '../services/component-config/theme'
import { useBem } from './useBem'

const sizeToAbsolute = (size: unknown) => {
if (typeof size === 'number') { return `${size}px` }
return String(size)
}

export const renderVariablesFromConfig = <Variables extends string>(sizesConfig: SizesConfig<Variables, string>, componentName: string) => {
return Object.entries(sizesConfig).reduce<Record<string, string>>((acc, [size, { variables }]) => {
for (const property in variables) {
acc[cssVariableName({ componentName, size, property })] = variables[property]
}

return acc
}, {})
}

export const useComponentVariables = <Variables extends string>(props: SizeProps<SizesConfig<Variables, string>>, componentName = getCurrentInstance()?.type.name) => {
if (!componentName) {
throw new Error('Component name must be provided!')
Expand Down
2 changes: 1 addition & 1 deletion packages/ui/src/composables/useSize.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { PropType } from 'vue'
import { SizesConfig } from '../services/size'
import { SizesConfig } from '../services/component-config/theme'

/**
* You could add these props to any component by destructuring them inside props option.
Expand Down
1 change: 1 addition & 0 deletions packages/ui/src/services/component-config/index.ts
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
export * from './theme'
export * from './types'
export * from './config/default'
export * from './utils/use-component-config-props'
49 changes: 49 additions & 0 deletions packages/ui/src/services/component-config/plugin/index.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,49 @@
import { computed, watch } from 'vue'

import { isServer } from '../../../utils/ssr'
import { useGlobalConfig } from '../../../composables/useGlobalConfig'
import { defineGlobalProperty, defineVuesticPlugin } from '../../vue-plugin/utils'
import { addOrUpdateStyleElement } from '../../../utils/dom'

import { ComponentConfig } from '../types'
import { generateUniqueId } from '../../../utils/uuid'
import { renderComponentsStyles } from '../utils/render-styles-from-config'

const handleConfigUpdate = (config: Partial<ComponentConfig>, styleId: string) => {
addOrUpdateStyleElement(
`va-component-classes-${styleId}`,
() => renderComponentsStyles(config),
)
}

const createComponentConfigPlugin = () => {
if (isServer()) { return }

const { globalConfig } = useGlobalConfig()

const uniqueId = computed(generateUniqueId)

watch(() => globalConfig.value.components, (components) => {
if (components) {
handleConfigUpdate(components, uniqueId.value)
}
}, { immediate: true, deep: true })

return {
renderStyles: () => {
return renderComponentsStyles(globalConfig.value.components)
},
}
}

export const ComponentConfigPlugin = defineVuesticPlugin(() => ({
install (app) {
defineGlobalProperty(app, '$vaComponentConfig', createComponentConfigPlugin())
},
}))

declare module '@vue/runtime-core' {
export interface ComponentCustomProperties {
$vaComponentConfig: ReturnType<typeof createComponentConfigPlugin>
}
}
File renamed without changes.
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
import { SizesConfig } from '../theme'
import { cssVariableName } from '../../../utils/css-variables'
import { CSSProperties } from 'vue'
import { bemClassName } from '../../../utils/bem'
import { renderCSSRule } from '../../../utils/css'
import { ComponentConfig } from '../types'

const renderComponentStyles = (sizesConfig: SizesConfig<string, string>, componentName: string, scopeSelector?: string) => {
return Object.entries(sizesConfig).reduce((acc, [size, { variables }]) => {
const definitions: CSSProperties = {}

for (const property in variables) {
definitions[cssVariableName({ componentName, property })] = variables[property]
}

const classSelector = `.${bemClassName({ block: componentName, modifier: size })}`

const selector = scopeSelector ? `${scopeSelector} ${classSelector}` : classSelector

acc += renderCSSRule(selector, definitions)

return acc
}, '')
}

export const renderComponentsStyles = (config: ComponentConfig, scopeSelector?: string) => {
return Object.entries(config).reduce((styles, [componentName, componentConfig]) => {
const sizesConfig: SizesConfig<string, string> | undefined = 'sizesConfig' in componentConfig ? componentConfig.sizesConfig : undefined

if (sizesConfig) {
styles += renderComponentStyles(sizesConfig, componentName, scopeSelector)
}

return styles
}, '')
}
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import { useLocalConfig } from '../../../composables/useLocalConfig'
import { useGlobalConfig } from '../../global-config/global-config'
import { computed } from 'vue'
import { injectChildPresetPropFromParent } from '../../../composables/useChildComponents'
import omit from 'lodash/omit'

export const useComponentConfigProps = <T extends VuesticComponent>(component: T, originalProps: Props) => {
const localConfig = useLocalConfig()
Expand All @@ -27,6 +28,6 @@ export const useComponentConfigProps = <T extends VuesticComponent>(component: T
const presetName = parentPropPreset?.value || instancePreset.value || localConfigProps.preset || globalConfigProps.preset
const presetProps = presetName && getPresetProps(presetName)

return { ...globalConfigProps, ...localConfigProps, ...presetProps }
return omit({ ...globalConfigProps, ...localConfigProps, ...presetProps }, 'sizesConfig')
})
}
1 change: 0 additions & 1 deletion packages/ui/src/services/size/index.ts

This file was deleted.

Loading

0 comments on commit a56337e

Please sign in to comment.