2022-09-06 12:03:37 +00:00
|
|
|
<template>
|
2024-04-30 09:39:18 +00:00
|
|
|
<div :class="cx('itemsContainer')" v-bind="ptm('itemsContainer')">
|
|
|
|
<div :class="cx('items')" v-bind="ptm('items')">
|
2024-07-18 07:04:59 +00:00
|
|
|
<button v-if="showItemNavigators" v-ripple type="button" :class="cx('prevButton')" @click="navBackward($event)" :disabled="isNavBackwardDisabled()" v-bind="ptm('prevButton')" data-pc-group-section="itemnavigator">
|
|
|
|
<component :is="templates.previousitemicon || 'ChevronLeftIcon'" :class="cx('prevIcon')" v-bind="ptm('prevIcon')" />
|
2022-09-14 11:26:01 +00:00
|
|
|
</button>
|
2023-05-31 22:28:41 +00:00
|
|
|
<div :id="id + '_item_' + activeIndex" :class="cx('item')" role="group" :aria-label="ariaSlideNumber(activeIndex + 1)" :aria-roledescription="ariaSlideLabel" v-bind="ptm('item')">
|
2022-09-14 11:26:01 +00:00
|
|
|
<component v-if="templates.item" :is="templates.item" :item="activeItem" />
|
|
|
|
</div>
|
2024-07-18 07:04:59 +00:00
|
|
|
<button v-if="showItemNavigators" v-ripple type="button" :class="cx('nextButton')" @click="navForward($event)" :disabled="isNavForwardDisabled()" v-bind="ptm('nextButton')" data-pc-group-section="itemnavigator">
|
|
|
|
<component :is="templates.nextitemicon || 'ChevronRightIcon'" :class="cx('nextIcon')" v-bind="ptm('nextIcon')" />
|
2022-09-14 11:26:01 +00:00
|
|
|
</button>
|
2023-05-31 22:28:41 +00:00
|
|
|
<div v-if="templates['caption']" :class="cx('caption')" v-bind="ptm('caption')">
|
2022-09-14 11:26:01 +00:00
|
|
|
<component v-if="templates.caption" :is="templates.caption" :item="activeItem" />
|
|
|
|
</div>
|
2022-09-06 12:03:37 +00:00
|
|
|
</div>
|
2024-04-30 09:39:18 +00:00
|
|
|
<ul v-if="showIndicators" ref="indicatorContent" :class="cx('indicatorList')" v-bind="ptm('indicatorList')">
|
2022-09-14 11:26:01 +00:00
|
|
|
<li
|
|
|
|
v-for="(item, index) of value"
|
|
|
|
:key="`p-galleria-indicator-${index}`"
|
2023-06-09 11:49:47 +00:00
|
|
|
:class="cx('indicator', { index })"
|
2022-12-08 11:04:25 +00:00
|
|
|
:aria-label="ariaPageLabel(index + 1)"
|
|
|
|
:aria-selected="activeIndex === index"
|
|
|
|
:aria-controls="id + '_item_' + index"
|
2022-09-14 11:26:01 +00:00
|
|
|
@click="onIndicatorClick(index)"
|
|
|
|
@mouseenter="onIndicatorMouseEnter(index)"
|
2022-12-08 11:04:25 +00:00
|
|
|
@keydown="onIndicatorKeyDown($event, index)"
|
2023-08-01 07:40:55 +00:00
|
|
|
v-bind="ptm('indicator', getIndicatorPTOptions(index))"
|
2024-05-24 11:07:49 +00:00
|
|
|
:data-p-active="isIndicatorItemActive(index)"
|
2022-09-14 11:26:01 +00:00
|
|
|
>
|
2023-12-02 07:02:01 +00:00
|
|
|
<button v-if="!templates['indicator']" type="button" :tabindex="activeIndex === index ? '0' : '-1'" :class="cx('indicatorButton')" v-bind="ptm('indicatorButton', getIndicatorPTOptions(index))"></button>
|
2022-09-14 11:26:01 +00:00
|
|
|
<component v-if="templates.indicator" :is="templates.indicator" :index="index" />
|
|
|
|
</li>
|
|
|
|
</ul>
|
2022-09-06 12:03:37 +00:00
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
2024-07-18 07:04:59 +00:00
|
|
|
import { find, findSingle, getAttribute } from '@primeuix/utils/dom';
|
2024-06-11 12:21:12 +00:00
|
|
|
import BaseComponent from '@primevue/core/basecomponent';
|
|
|
|
import ChevronLeftIcon from '@primevue/icons/chevronleft';
|
|
|
|
import ChevronRightIcon from '@primevue/icons/chevronright';
|
2022-09-06 12:03:37 +00:00
|
|
|
import Ripple from 'primevue/ripple';
|
|
|
|
|
|
|
|
export default {
|
|
|
|
name: 'GalleriaItem',
|
2023-07-19 12:46:57 +00:00
|
|
|
hostName: 'Galleria',
|
2023-05-04 08:29:52 +00:00
|
|
|
extends: BaseComponent,
|
2022-09-06 12:03:37 +00:00
|
|
|
emits: ['start-slideshow', 'stop-slideshow', 'update:activeIndex'],
|
|
|
|
props: {
|
|
|
|
circular: {
|
|
|
|
type: Boolean,
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
activeIndex: {
|
|
|
|
type: Number,
|
|
|
|
default: 0
|
|
|
|
},
|
|
|
|
value: {
|
|
|
|
type: Array,
|
|
|
|
default: null
|
|
|
|
},
|
|
|
|
showItemNavigators: {
|
|
|
|
type: Boolean,
|
|
|
|
default: true
|
|
|
|
},
|
|
|
|
showIndicators: {
|
|
|
|
type: Boolean,
|
|
|
|
default: true
|
|
|
|
},
|
|
|
|
slideShowActive: {
|
|
|
|
type: Boolean,
|
|
|
|
default: true
|
|
|
|
},
|
|
|
|
changeItemOnIndicatorHover: {
|
|
|
|
type: Boolean,
|
|
|
|
default: true
|
|
|
|
},
|
|
|
|
autoPlay: {
|
|
|
|
type: Boolean,
|
|
|
|
default: false
|
|
|
|
},
|
|
|
|
templates: {
|
|
|
|
type: null,
|
|
|
|
default: null
|
2022-12-08 11:04:25 +00:00
|
|
|
},
|
|
|
|
id: {
|
|
|
|
type: String,
|
|
|
|
default: null
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
mounted() {
|
|
|
|
if (this.autoPlay) {
|
|
|
|
this.$emit('start-slideshow');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
methods: {
|
2023-08-01 07:40:55 +00:00
|
|
|
getIndicatorPTOptions(index) {
|
|
|
|
return {
|
|
|
|
context: {
|
|
|
|
highlighted: this.activeIndex === index
|
|
|
|
}
|
|
|
|
};
|
|
|
|
},
|
2022-09-06 12:03:37 +00:00
|
|
|
next() {
|
|
|
|
let nextItemIndex = this.activeIndex + 1;
|
2022-09-14 11:26:01 +00:00
|
|
|
let activeIndex = this.circular && this.value.length - 1 === this.activeIndex ? 0 : nextItemIndex;
|
2022-09-06 12:03:37 +00:00
|
|
|
|
|
|
|
this.$emit('update:activeIndex', activeIndex);
|
|
|
|
},
|
|
|
|
prev() {
|
|
|
|
let prevItemIndex = this.activeIndex !== 0 ? this.activeIndex - 1 : 0;
|
2022-09-14 11:26:01 +00:00
|
|
|
let activeIndex = this.circular && this.activeIndex === 0 ? this.value.length - 1 : prevItemIndex;
|
2022-09-06 12:03:37 +00:00
|
|
|
|
|
|
|
this.$emit('update:activeIndex', activeIndex);
|
|
|
|
},
|
|
|
|
stopSlideShow() {
|
|
|
|
if (this.slideShowActive && this.stopSlideShow) {
|
|
|
|
this.$emit('stop-slideshow');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
navBackward(e) {
|
|
|
|
this.stopSlideShow();
|
|
|
|
this.prev();
|
|
|
|
|
|
|
|
if (e && e.cancelable) {
|
|
|
|
e.preventDefault();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
navForward(e) {
|
|
|
|
this.stopSlideShow();
|
|
|
|
this.next();
|
|
|
|
|
|
|
|
if (e && e.cancelable) {
|
|
|
|
e.preventDefault();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
onIndicatorClick(index) {
|
|
|
|
this.stopSlideShow();
|
|
|
|
this.$emit('update:activeIndex', index);
|
|
|
|
},
|
|
|
|
onIndicatorMouseEnter(index) {
|
|
|
|
if (this.changeItemOnIndicatorHover) {
|
|
|
|
this.stopSlideShow();
|
|
|
|
|
|
|
|
this.$emit('update:activeIndex', index);
|
|
|
|
}
|
|
|
|
},
|
2022-12-08 11:04:25 +00:00
|
|
|
onIndicatorKeyDown(event, index) {
|
|
|
|
switch (event.code) {
|
|
|
|
case 'Enter':
|
2023-12-20 10:45:43 +00:00
|
|
|
case 'NumpadEnter':
|
2022-12-08 11:04:25 +00:00
|
|
|
case 'Space':
|
|
|
|
this.stopSlideShow();
|
2022-09-06 12:03:37 +00:00
|
|
|
|
2022-12-08 11:04:25 +00:00
|
|
|
this.$emit('update:activeIndex', index);
|
|
|
|
event.preventDefault();
|
|
|
|
break;
|
|
|
|
|
2023-12-02 07:02:01 +00:00
|
|
|
case 'ArrowRight':
|
|
|
|
this.onRightKey();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'ArrowLeft':
|
|
|
|
this.onLeftKey();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'Home':
|
|
|
|
this.onHomeKey();
|
|
|
|
event.preventDefault();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'End':
|
|
|
|
this.onEndKey();
|
|
|
|
event.preventDefault();
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'Tab':
|
|
|
|
this.onTabKey();
|
|
|
|
break;
|
|
|
|
|
2022-12-08 11:04:25 +00:00
|
|
|
case 'ArrowDown':
|
|
|
|
case 'ArrowUp':
|
2023-12-02 07:02:01 +00:00
|
|
|
case 'PageUp':
|
|
|
|
case 'PageDown':
|
2022-12-08 11:04:25 +00:00
|
|
|
event.preventDefault();
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2022-09-06 12:03:37 +00:00
|
|
|
},
|
2023-12-02 07:02:01 +00:00
|
|
|
onRightKey() {
|
2024-06-26 07:46:26 +00:00
|
|
|
const indicators = [...find(this.$refs.indicatorContent, '[data-pc-section="indicator"]')];
|
2023-12-02 07:02:01 +00:00
|
|
|
const activeIndex = this.findFocusedIndicatorIndex();
|
|
|
|
|
|
|
|
this.changedFocusedIndicator(activeIndex, activeIndex + 1 === indicators.length ? indicators.length - 1 : activeIndex + 1);
|
|
|
|
},
|
|
|
|
onLeftKey() {
|
|
|
|
const activeIndex = this.findFocusedIndicatorIndex();
|
|
|
|
|
|
|
|
this.changedFocusedIndicator(activeIndex, activeIndex - 1 <= 0 ? 0 : activeIndex - 1);
|
|
|
|
},
|
|
|
|
onHomeKey() {
|
|
|
|
const activeIndex = this.findFocusedIndicatorIndex();
|
|
|
|
|
|
|
|
this.changedFocusedIndicator(activeIndex, 0);
|
|
|
|
},
|
|
|
|
onEndKey() {
|
2024-06-26 07:46:26 +00:00
|
|
|
const indicators = [...find(this.$refs.indicatorContent, '[data-pc-section="indicator"]')];
|
2023-12-02 07:02:01 +00:00
|
|
|
const activeIndex = this.findFocusedIndicatorIndex();
|
|
|
|
|
|
|
|
this.changedFocusedIndicator(activeIndex, indicators.length - 1);
|
|
|
|
},
|
|
|
|
onTabKey() {
|
2024-06-26 07:46:26 +00:00
|
|
|
const indicators = [...find(this.$refs.indicatorContent, '[data-pc-section="indicator"]')];
|
|
|
|
const highlightedIndex = indicators.findIndex((ind) => getAttribute(ind, 'data-p-active') === true);
|
2023-12-02 07:02:01 +00:00
|
|
|
|
2024-06-26 07:46:26 +00:00
|
|
|
const activeIndicator = findSingle(this.$refs.indicatorContent, '[data-pc-section="indicator"] > button[tabindex="0"]');
|
2023-12-02 07:02:01 +00:00
|
|
|
const activeIndex = indicators.findIndex((ind) => ind === activeIndicator.parentElement);
|
|
|
|
|
|
|
|
indicators[activeIndex].children[0].tabIndex = '-1';
|
|
|
|
indicators[highlightedIndex].children[0].tabIndex = '0';
|
|
|
|
},
|
|
|
|
findFocusedIndicatorIndex() {
|
2024-06-26 07:46:26 +00:00
|
|
|
const indicators = [...find(this.$refs.indicatorContent, '[data-pc-section="indicator"]')];
|
|
|
|
const activeIndicator = findSingle(this.$refs.indicatorContent, '[data-pc-section="indicator"] > button[tabindex="0"]');
|
2023-12-02 07:02:01 +00:00
|
|
|
|
|
|
|
return indicators.findIndex((ind) => ind === activeIndicator.parentElement);
|
|
|
|
},
|
|
|
|
changedFocusedIndicator(prevInd, nextInd) {
|
2024-06-26 07:46:26 +00:00
|
|
|
const indicators = [...find(this.$refs.indicatorContent, '[data-pc-section="indicator"]')];
|
2023-12-02 07:02:01 +00:00
|
|
|
|
|
|
|
indicators[prevInd].children[0].tabIndex = '-1';
|
|
|
|
indicators[nextInd].children[0].tabIndex = '0';
|
|
|
|
indicators[nextInd].children[0].focus();
|
|
|
|
},
|
2022-09-06 12:03:37 +00:00
|
|
|
isIndicatorItemActive(index) {
|
|
|
|
return this.activeIndex === index;
|
|
|
|
},
|
|
|
|
isNavBackwardDisabled() {
|
|
|
|
return !this.circular && this.activeIndex === 0;
|
|
|
|
},
|
|
|
|
isNavForwardDisabled() {
|
2022-09-14 11:26:01 +00:00
|
|
|
return !this.circular && this.activeIndex === this.value.length - 1;
|
2022-12-08 11:04:25 +00:00
|
|
|
},
|
|
|
|
ariaSlideNumber(value) {
|
|
|
|
return this.$primevue.config.locale.aria ? this.$primevue.config.locale.aria.slideNumber.replace(/{slideNumber}/g, value) : undefined;
|
|
|
|
},
|
|
|
|
ariaPageLabel(value) {
|
|
|
|
return this.$primevue.config.locale.aria ? this.$primevue.config.locale.aria.pageLabel.replace(/{page}/g, value) : undefined;
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
computed: {
|
|
|
|
activeItem() {
|
|
|
|
return this.value[this.activeIndex];
|
|
|
|
},
|
2023-05-31 22:28:41 +00:00
|
|
|
|
2022-12-08 11:04:25 +00:00
|
|
|
ariaSlideLabel() {
|
|
|
|
return this.$primevue.config.locale.aria ? this.$primevue.config.locale.aria.slide : undefined;
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
|
|
|
},
|
2023-04-06 08:21:23 +00:00
|
|
|
components: {
|
|
|
|
ChevronLeftIcon: ChevronLeftIcon,
|
|
|
|
ChevronRightIcon: ChevronRightIcon
|
|
|
|
},
|
2022-09-06 12:03:37 +00:00
|
|
|
directives: {
|
2022-09-14 11:26:01 +00:00
|
|
|
ripple: Ripple
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
</script>
|