276 lines
7.9 KiB
JavaScript
276 lines
7.9 KiB
JavaScript
'use strict';
|
|
|
|
Object.defineProperty(exports, '__esModule', { value: true });
|
|
|
|
var vue = require('vue');
|
|
var lodashUnified = require('lodash-unified');
|
|
var core = require('@vueuse/core');
|
|
var constants = require('./constants.js');
|
|
var index = require('../../../hooks/use-ordered-children/index.js');
|
|
var shared = require('@vue/shared');
|
|
var vnode = require('../../../utils/vue/vnode.js');
|
|
var event = require('../../../constants/event.js');
|
|
|
|
const THROTTLE_TIME = 300;
|
|
const useCarousel = (props, emit, componentName) => {
|
|
const {
|
|
children: items,
|
|
addChild: addItem,
|
|
removeChild: removeItem,
|
|
ChildrenSorter: ItemsSorter
|
|
} = index.useOrderedChildren(vue.getCurrentInstance(), constants.CAROUSEL_ITEM_NAME);
|
|
const slots = vue.useSlots();
|
|
const activeIndex = vue.ref(-1);
|
|
const timer = vue.ref(null);
|
|
const hover = vue.ref(false);
|
|
const root = vue.ref();
|
|
const containerHeight = vue.ref(0);
|
|
const isItemsTwoLength = vue.ref(true);
|
|
const arrowDisplay = vue.computed(() => props.arrow !== "never" && !vue.unref(isVertical));
|
|
const hasLabel = vue.computed(() => {
|
|
return items.value.some((item) => item.props.label.toString().length > 0);
|
|
});
|
|
const isCardType = vue.computed(() => props.type === "card");
|
|
const isVertical = vue.computed(() => props.direction === "vertical");
|
|
const containerStyle = vue.computed(() => {
|
|
if (props.height !== "auto") {
|
|
return {
|
|
height: props.height
|
|
};
|
|
}
|
|
return {
|
|
height: `${containerHeight.value}px`,
|
|
overflow: "hidden"
|
|
};
|
|
});
|
|
const throttledArrowClick = lodashUnified.throttle((index) => {
|
|
setActiveItem(index);
|
|
}, THROTTLE_TIME, { trailing: true });
|
|
const throttledIndicatorHover = lodashUnified.throttle((index) => {
|
|
handleIndicatorHover(index);
|
|
}, THROTTLE_TIME);
|
|
const isTwoLengthShow = (index) => {
|
|
if (!isItemsTwoLength.value)
|
|
return true;
|
|
return activeIndex.value <= 1 ? index <= 1 : index > 1;
|
|
};
|
|
function pauseTimer() {
|
|
if (timer.value) {
|
|
clearInterval(timer.value);
|
|
timer.value = null;
|
|
}
|
|
}
|
|
function startTimer() {
|
|
if (props.interval <= 0 || !props.autoplay || timer.value)
|
|
return;
|
|
timer.value = setInterval(() => playSlides(), props.interval);
|
|
}
|
|
const playSlides = () => {
|
|
if (activeIndex.value < items.value.length - 1) {
|
|
activeIndex.value = activeIndex.value + 1;
|
|
} else if (props.loop) {
|
|
activeIndex.value = 0;
|
|
}
|
|
};
|
|
function setActiveItem(index) {
|
|
if (shared.isString(index)) {
|
|
const filteredItems = items.value.filter((item) => item.props.name === index);
|
|
if (filteredItems.length > 0) {
|
|
index = items.value.indexOf(filteredItems[0]);
|
|
}
|
|
}
|
|
index = Number(index);
|
|
if (Number.isNaN(index) || index !== Math.floor(index)) {
|
|
return;
|
|
}
|
|
const itemCount = items.value.length;
|
|
const oldIndex = activeIndex.value;
|
|
if (index < 0) {
|
|
activeIndex.value = props.loop ? itemCount - 1 : 0;
|
|
} else if (index >= itemCount) {
|
|
activeIndex.value = props.loop ? 0 : itemCount - 1;
|
|
} else {
|
|
activeIndex.value = index;
|
|
}
|
|
if (oldIndex === activeIndex.value) {
|
|
resetItemPosition(oldIndex);
|
|
}
|
|
resetTimer();
|
|
}
|
|
function resetItemPosition(oldIndex) {
|
|
items.value.forEach((item, index) => {
|
|
item.translateItem(index, activeIndex.value, oldIndex);
|
|
});
|
|
}
|
|
function itemInStage(item, index) {
|
|
var _a, _b, _c, _d;
|
|
const _items = vue.unref(items);
|
|
const itemCount = _items.length;
|
|
if (itemCount === 0 || !item.states.inStage)
|
|
return false;
|
|
const nextItemIndex = index + 1;
|
|
const prevItemIndex = index - 1;
|
|
const lastItemIndex = itemCount - 1;
|
|
const isLastItemActive = _items[lastItemIndex].states.active;
|
|
const isFirstItemActive = _items[0].states.active;
|
|
const isNextItemActive = (_b = (_a = _items[nextItemIndex]) == null ? void 0 : _a.states) == null ? void 0 : _b.active;
|
|
const isPrevItemActive = (_d = (_c = _items[prevItemIndex]) == null ? void 0 : _c.states) == null ? void 0 : _d.active;
|
|
if (index === lastItemIndex && isFirstItemActive || isNextItemActive) {
|
|
return "left";
|
|
} else if (index === 0 && isLastItemActive || isPrevItemActive) {
|
|
return "right";
|
|
}
|
|
return false;
|
|
}
|
|
function handleMouseEnter() {
|
|
hover.value = true;
|
|
if (props.pauseOnHover) {
|
|
pauseTimer();
|
|
}
|
|
}
|
|
function handleMouseLeave() {
|
|
hover.value = false;
|
|
startTimer();
|
|
}
|
|
function handleButtonEnter(arrow) {
|
|
if (vue.unref(isVertical))
|
|
return;
|
|
items.value.forEach((item, index) => {
|
|
if (arrow === itemInStage(item, index)) {
|
|
item.states.hover = true;
|
|
}
|
|
});
|
|
}
|
|
function handleButtonLeave() {
|
|
if (vue.unref(isVertical))
|
|
return;
|
|
items.value.forEach((item) => {
|
|
item.states.hover = false;
|
|
});
|
|
}
|
|
function handleIndicatorClick(index) {
|
|
activeIndex.value = index;
|
|
}
|
|
function handleIndicatorHover(index) {
|
|
if (props.trigger === "hover" && index !== activeIndex.value) {
|
|
activeIndex.value = index;
|
|
}
|
|
}
|
|
function prev() {
|
|
setActiveItem(activeIndex.value - 1);
|
|
}
|
|
function next() {
|
|
setActiveItem(activeIndex.value + 1);
|
|
}
|
|
function resetTimer() {
|
|
pauseTimer();
|
|
if (!props.pauseOnHover)
|
|
startTimer();
|
|
}
|
|
function setContainerHeight(height) {
|
|
if (props.height !== "auto")
|
|
return;
|
|
containerHeight.value = height;
|
|
}
|
|
function PlaceholderItem() {
|
|
var _a;
|
|
const defaultSlots = (_a = slots.default) == null ? void 0 : _a.call(slots);
|
|
if (!defaultSlots)
|
|
return null;
|
|
const flatSlots = vnode.flattedChildren(defaultSlots);
|
|
const normalizeSlots = flatSlots.filter((slot) => {
|
|
return vue.isVNode(slot) && slot.type.name === constants.CAROUSEL_ITEM_NAME;
|
|
});
|
|
if ((normalizeSlots == null ? void 0 : normalizeSlots.length) === 2 && props.loop && !isCardType.value) {
|
|
isItemsTwoLength.value = true;
|
|
return normalizeSlots;
|
|
}
|
|
isItemsTwoLength.value = false;
|
|
return null;
|
|
}
|
|
vue.watch(() => activeIndex.value, (current, prev2) => {
|
|
resetItemPosition(prev2);
|
|
if (isItemsTwoLength.value) {
|
|
current = current % 2;
|
|
prev2 = prev2 % 2;
|
|
}
|
|
if (prev2 > -1) {
|
|
emit(event.CHANGE_EVENT, current, prev2);
|
|
}
|
|
});
|
|
const exposeActiveIndex = vue.computed({
|
|
get: () => {
|
|
return isItemsTwoLength.value ? activeIndex.value % 2 : activeIndex.value;
|
|
},
|
|
set: (value) => activeIndex.value = value
|
|
});
|
|
vue.watch(() => props.autoplay, (autoplay) => {
|
|
autoplay ? startTimer() : pauseTimer();
|
|
});
|
|
vue.watch(() => props.loop, () => {
|
|
setActiveItem(activeIndex.value);
|
|
});
|
|
vue.watch(() => props.interval, () => {
|
|
resetTimer();
|
|
});
|
|
const resizeObserver = vue.shallowRef();
|
|
vue.onMounted(() => {
|
|
vue.watch(() => items.value, () => {
|
|
if (items.value.length > 0)
|
|
setActiveItem(props.initialIndex);
|
|
}, {
|
|
immediate: true
|
|
});
|
|
resizeObserver.value = core.useResizeObserver(root.value, () => {
|
|
resetItemPosition();
|
|
});
|
|
startTimer();
|
|
});
|
|
vue.onBeforeUnmount(() => {
|
|
pauseTimer();
|
|
if (root.value && resizeObserver.value)
|
|
resizeObserver.value.stop();
|
|
});
|
|
vue.provide(constants.carouselContextKey, {
|
|
root,
|
|
isCardType,
|
|
isVertical,
|
|
items,
|
|
loop: props.loop,
|
|
cardScale: props.cardScale,
|
|
addItem,
|
|
removeItem,
|
|
setActiveItem,
|
|
setContainerHeight
|
|
});
|
|
return {
|
|
root,
|
|
activeIndex,
|
|
exposeActiveIndex,
|
|
arrowDisplay,
|
|
hasLabel,
|
|
hover,
|
|
isCardType,
|
|
items,
|
|
isVertical,
|
|
containerStyle,
|
|
isItemsTwoLength,
|
|
handleButtonEnter,
|
|
handleButtonLeave,
|
|
handleIndicatorClick,
|
|
handleMouseEnter,
|
|
handleMouseLeave,
|
|
setActiveItem,
|
|
prev,
|
|
next,
|
|
PlaceholderItem,
|
|
isTwoLengthShow,
|
|
ItemsSorter,
|
|
throttledArrowClick,
|
|
throttledIndicatorHover
|
|
};
|
|
};
|
|
|
|
exports.useCarousel = useCarousel;
|
|
//# sourceMappingURL=use-carousel.js.map
|