Merge pull request #116 from assasin0076/fix_accordion

fix: fixed accordion doc build
This commit is contained in:
Ilya Artamonov
2022-12-13 18:51:39 +03:00
committed by GitHub
6 changed files with 103 additions and 82 deletions

View File

@@ -1,10 +1,9 @@
<template> <template>
<div class="vp-raw"> <div class="vp-raw" :data-accordion-id="accordionId">
<slot :accordionId="accordionId"/> <slot />
</div> </div>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { provide } from 'vue'
import { nanoid } from 'nanoid' import { nanoid } from 'nanoid'
import { useAccordionState } from '@/components/Accordion/composables/useAccordionState' import { useAccordionState } from '@/components/Accordion/composables/useAccordionState'
@@ -20,6 +19,5 @@ const props = defineProps({
}) })
const accordionId = nanoid() const accordionId = nanoid()
provide('accordionId', accordionId)
useAccordionState(accordionId, { ...props }) useAccordionState(accordionId, { ...props })
</script> </script>

View File

@@ -1,13 +1,26 @@
<template> <template>
<div <div ref="content">
:class="contentClasses" <div
> v-if="isLoaded"
<slot /> :class="contentClasses"
>
<slot />
</div>
</div> </div>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { useAccordionContentClasses } from '@/components/Accordion/composables/useAccordionContentClasses' import { useAccordionContentClasses } from '@/components/Accordion/composables/useAccordionContentClasses'
import { onMounted, ref } from 'vue'
const { contentClasses } = useAccordionContentClasses() const isLoaded = ref(false)
const content = ref()
let contentClasses
onMounted(() => {
const accordionHeaderClasses = useAccordionContentClasses(content)
contentClasses = accordionHeaderClasses.contentClasses
isLoaded.value = true
})
</script> </script>

View File

@@ -1,40 +1,45 @@
<template> <template>
<button <div ref="header">
type="button" <button
@click="toggleItem" v-if="isLoaded"
:class="headerClasses" type="button"
> @click="toggleItem"
<span class="w-full"><slot /></span> :class="headerClasses"
<svg
data-accordion-icon
:class="arrowClasses"
fill="currentColor"
viewBox="0 0 20 20"
xmlns="http://www.w3.org/2000/svg"
> >
<path fill-rule="evenodd" d="M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a1 1 0 01-1.414 0l-4-4a1 1 0 010-1.414z"></path> <span class="w-full"><slot /></span>
</svg> <svg
</button> data-accordion-icon
:class="arrowClasses"
fill="currentColor"
viewBox="0 0 20 20"
xmlns="http://www.w3.org/2000/svg"
>
<path fill-rule="evenodd" d="M5.293 7.293a1 1 0 011.414 0L10 10.586l3.293-3.293a1 1 0 111.414 1.414l-4 4a1 1 0 01-1.414 0l-4-4a1 1 0 010-1.414z"></path>
</svg>
</button>
</div>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { useAccordionState } from '@/components/Accordion/composables/useAccordionState' import { useAccordionState } from '@/components/Accordion/composables/useAccordionState'
import { computed, inject } from 'vue' import { computed, onMounted, ref } from 'vue'
import { useAccordionHeaderClasses } from '@/components/Accordion/composables/useAccordionHeaderClasses' import { useAccordionHeaderClasses } from '@/components/Accordion/composables/useAccordionHeaderClasses'
const accordionId: string = inject('accordionId') ?? '' const isLoaded = ref(false)
const panelId: string = inject('panelId') ?? '' const header = ref()
const accordionId = computed(() => header.value.parentElement.parentElement.dataset.accordionId)
const panelId = computed(() => header.value.parentElement.dataset.panelId)
const { accordionsStates } = useAccordionState() const { accordionsStates } = useAccordionState()
const accordionState = computed(() => accordionsStates[accordionId]) const accordionState = computed(() => accordionsStates[accordionId.value])
const panelState = computed(() => accordionState.value.panels[panelId]) const panelState = computed(() => accordionState.value.panels[panelId.value])
const { headerClasses, arrowClasses } = useAccordionHeaderClasses() let headerClasses, arrowClasses
function commonToggleItem() { function commonToggleItem() {
const isSelectedVisible = panelState.value.isVisible const isSelectedVisible = panelState.value.isVisible
for (const panelIndex in accordionState.value.panels) { for (const panelIndex in accordionState.value.panels) {
const panel = accordionState.value.panels[panelIndex] const panel = accordionState.value.panels[panelIndex]
if (panel.id !== panelId) panel.isVisible = false if (panel.id !== panelId.value) panel.isVisible = false
else panel.isVisible = !isSelectedVisible else panel.isVisible = !isSelectedVisible
} }
} }
@@ -46,4 +51,10 @@ function toggleItem() {
commonToggleItem() commonToggleItem()
} }
onMounted(() => {
const accordionHeaderClasses = useAccordionHeaderClasses(header)
headerClasses = accordionHeaderClasses.headerClasses
arrowClasses = accordionHeaderClasses.arrowClasses
isLoaded.value = true
})
</script> </script>

View File

@@ -1,26 +1,30 @@
<template> <template>
<div> <div :data-panel-id="panelId" ref="panel">
<slot></slot> <slot v-if="accordionId"></slot>
</div> </div>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { computed, inject, onBeforeMount, provide } from 'vue' import { computed, onMounted, ref } from 'vue'
import { useAccordionState } from '@/components/Accordion/composables/useAccordionState' import { useAccordionState } from '@/components/Accordion/composables/useAccordionState'
import { nanoid } from 'nanoid' import { nanoid } from 'nanoid'
const { accordionsStates } = useAccordionState() const { accordionsStates } = useAccordionState()
const panelId = nanoid()
const accordionId: string = inject('accordionId') ?? ''
provide('panelId', panelId)
const accordionState = computed(() => { const panelId = nanoid()
return accordionsStates[accordionId] const panel = ref()
const accordionId = computed(() => {
if (panel.value) return panel.value.parentElement.dataset.accordionId
return null
}) })
onBeforeMount(() => { const accordionState = computed(() => {
const panelsCount = Object.keys(accordionState.value.panels).length return accordionsStates[accordionId.value]
})
onMounted(() => {
const panelsCount = Object.keys(accordionState?.value?.panels)?.length
accordionState.value.panels[panelId] = { accordionState.value.panels[panelId] = {
id: panelId, id: panelId,
order: panelsCount, order: panelsCount,

View File

@@ -1,29 +1,25 @@
import { computed, inject } from 'vue' import { computed } from 'vue'
import { useAccordionState } from '@/components/Accordion/composables/useAccordionState' import { useAccordionState } from '@/components/Accordion/composables/useAccordionState'
import classNames from 'classnames' import classNames from 'classnames'
const baseContentClasses = 'p-5 border border-gray-200 dark:border-gray-700 dark:bg-gray-900' const baseContentClasses = 'p-5 border border-gray-200 dark:border-gray-700 dark:bg-gray-900'
export function useAccordionContentClasses() { export function useAccordionContentClasses(contentRef: any) {
const accordionId: string = inject('accordionId') ?? '' const accordionId = computed(() => contentRef.value.parentElement.parentElement.dataset.accordionId)
const panelId: string = inject('panelId') ?? '' const panelId = computed(() => contentRef.value.parentElement.dataset.panelId)
const { accordionsStates } = useAccordionState() const { accordionsStates } = useAccordionState()
const accordionState = computed(() => accordionsStates[accordionId.value])
const panelState = computed(() => accordionsStates[accordionId.value].panels[panelId.value])
const panelsCount = computed(() => Object.keys(accordionsStates[accordionId.value].panels[panelId.value]).length)
const accordionState = computed(() => accordionsStates[accordionId]) const contentClasses = computed(() => {
const panelState = computed(() => accordionsStates[accordionId].panels[panelId]) return classNames(baseContentClasses, {
const panelsCount = computed(() => Object.keys(accordionsStates[accordionId].panels[panelId]).length) hidden: !panelState.value.isVisible,
'border-b-0': panelState.value.order !== panelsCount.value - 1 || accordionState.value.flush,
const contentClasses = computed(() => classNames(baseContentClasses, { 'border-t-0': panelState.value.order === panelsCount.value - 1,
hidden: !panelState.value.isVisible, 'border-x-0': accordionState.value.flush,
'border-b-0': panelState.value.order !== panelsCount.value - 1 || accordionState.value.flush, })
'border-t-0': panelState.value.order === panelsCount.value - 1, })
'border-x-0': accordionState.value.flush,
}))
return { return {
contentClasses, contentClasses,
} }
} }

View File

@@ -1,35 +1,34 @@
import { computed, inject } from 'vue' import { computed } from 'vue'
import { useAccordionState } from '@/components/Accordion/composables/useAccordionState' import { useAccordionState } from '@/components/Accordion/composables/useAccordionState'
import classNames from 'classnames' import classNames from 'classnames'
const baseHeaderClasses =
const baseHeaderClasses = 'flex items-center p-5 w-full font-medium text-left text-gray-500 border border-gray-200 focus:ring-4 focus:ring-gray-200 dark:focus:ring-gray-800 dark:border-gray-700 dark:text-gray-400 hover:bg-gray-100 dark:hover:bg-gray-800' 'flex items-center p-5 w-full font-medium text-left text-gray-500 border border-gray-200 focus:ring-4 focus:ring-gray-200 dark:focus:ring-gray-800 dark:border-gray-700 dark:text-gray-400 hover:bg-gray-100 dark:hover:bg-gray-800'
const baseArrowClasses = 'w-6 h-6 shrink-0' const baseArrowClasses = 'w-6 h-6 shrink-0'
export function useAccordionHeaderClasses() { export function useAccordionHeaderClasses(headerRef: any) {
const accordionId: string = inject('accordionId') ?? '' const accordionId = computed(() => headerRef.value.parentElement.parentElement.dataset.accordionId)
const panelId: string = inject('panelId') ?? '' const panelId = computed(() => headerRef.value.parentElement.dataset.panelId)
const { accordionsStates } = useAccordionState() const { accordionsStates } = useAccordionState()
const accordionState = computed(() => accordionsStates[accordionId]) const accordionState = computed(() => accordionsStates[accordionId.value])
const panelState = computed(() => accordionState.value.panels[panelId]) const panelState = computed(() => accordionState.value.panels[panelId.value])
const panelsCount = computed(() => Object.keys(panelState.value).length) const panelsCount = computed(() => Object.keys(panelState.value).length)
const isPanelLast = computed(() => panelState.value.order !== panelsCount.value - 1) const isPanelLast = computed(() => panelState.value.order !== panelsCount.value - 1)
const isBottomBorderVisibleForFlush = computed(() => const isBottomBorderVisibleForFlush = computed(() => isPanelLast.value || (accordionState.value.flush && panelState.value.order === panelsCount.value - 1 && !panelState.value.isVisible))
isPanelLast.value ||
(accordionState.value.flush && panelState.value.order === panelsCount.value - 1 && !panelState.value.isVisible))
const headerClasses = computed(() => classNames(baseHeaderClasses, { const headerClasses = computed(() => {
'bg-gray-100 dark:bg-gray-800': panelState.value.isVisible, return classNames(baseHeaderClasses, {
'rounded-t-xl': panelState.value.order === 0 && !accordionState.value.flush, 'bg-gray-100 dark:bg-gray-800': panelState.value.isVisible,
'border-t-0': panelState.value.order === 0 && accordionState.value.flush, 'rounded-t-xl': panelState.value.order === 0 && !accordionState.value.flush,
'border-b-0': isBottomBorderVisibleForFlush.value, 'border-t-0': panelState.value.order === 0 && accordionState.value.flush,
'border-x-0': accordionState.value.flush, 'border-b-0': isBottomBorderVisibleForFlush.value,
})) 'border-x-0': accordionState.value.flush,
const arrowClasses = computed(() => classNames(baseArrowClasses,{ 'rotate-180': panelState.value.isVisible })) })
})
const arrowClasses = computed(() => {
return classNames(baseArrowClasses, { 'rotate-180': panelState.value.isVisible })
})
return { return {
headerClasses, headerClasses,
arrowClasses, arrowClasses,
} }
} }