2022-09-06 12:03:37 +00:00
|
|
|
<template>
|
|
|
|
<Portal>
|
2023-01-09 12:58:56 +00:00
|
|
|
<transition name="p-confirm-popup" @enter="onEnter" @after-enter="onAfterEnter" @leave="onLeave" @after-leave="onAfterLeave">
|
2022-12-08 11:04:25 +00:00
|
|
|
<div v-if="visible" :ref="containerRef" v-focustrap role="alertdialog" :class="containerClass" :aria-modal="visible" @click="onOverlayClick" @keydown="onOverlayKeydown" v-bind="$attrs">
|
2022-09-06 12:03:37 +00:00
|
|
|
<template v-if="!$slots.message">
|
|
|
|
<div class="p-confirm-popup-content">
|
2023-04-18 11:50:19 +00:00
|
|
|
<slot name="icon">
|
|
|
|
<component v-if="$slots.icon" :is="$slots.icon" class="p-confirm-popup-icon" />
|
|
|
|
<span v-else-if="confirmation.icon" :class="iconClass" />
|
|
|
|
</slot>
|
2022-09-14 11:26:01 +00:00
|
|
|
<span class="p-confirm-popup-message">{{ confirmation.message }}</span>
|
2022-09-06 12:03:37 +00:00
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
<component v-else :is="$slots.message" :message="confirmation"></component>
|
|
|
|
<div class="p-confirm-popup-footer">
|
2023-04-18 11:50:19 +00:00
|
|
|
<CPButton :label="rejectLabel" :class="rejectClass" @click="reject()" @keydown="onRejectKeydown" :autofocus="autoFocusReject">
|
|
|
|
<template #icon="iconProps">
|
|
|
|
<slot name="rejecticon">
|
|
|
|
<span :class="[rejectIcon, iconProps.class]" />
|
|
|
|
</slot>
|
|
|
|
</template>
|
|
|
|
</CPButton>
|
|
|
|
<CPButton :label="acceptLabel" :class="acceptClass" @click="accept()" @keydown="onAcceptKeydown" :autofocus="autoFocusAccept">
|
|
|
|
<template #icon="iconProps">
|
|
|
|
<slot name="accepticon">
|
|
|
|
<span :class="[acceptIcon, iconProps.class]" />
|
|
|
|
</slot>
|
|
|
|
</template>
|
|
|
|
</CPButton>
|
2022-09-06 12:03:37 +00:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</transition>
|
|
|
|
</Portal>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
2022-12-08 11:04:25 +00:00
|
|
|
import Button from 'primevue/button';
|
2022-09-06 12:03:37 +00:00
|
|
|
import ConfirmationEventBus from 'primevue/confirmationeventbus';
|
2022-12-08 11:04:25 +00:00
|
|
|
import FocusTrap from 'primevue/focustrap';
|
2022-09-06 12:03:37 +00:00
|
|
|
import OverlayEventBus from 'primevue/overlayeventbus';
|
|
|
|
import Portal from 'primevue/portal';
|
2022-12-08 11:04:25 +00:00
|
|
|
import { ConnectedOverlayScrollHandler, DomHandler, ZIndexUtils } from 'primevue/utils';
|
2022-09-06 12:03:37 +00:00
|
|
|
|
|
|
|
export default {
|
|
|
|
name: 'ConfirmPopup',
|
|
|
|
inheritAttrs: false,
|
|
|
|
props: {
|
|
|
|
group: String
|
|
|
|
},
|
|
|
|
data() {
|
|
|
|
return {
|
|
|
|
visible: false,
|
2023-01-09 12:58:56 +00:00
|
|
|
confirmation: null,
|
|
|
|
autoFocusAccept: null,
|
|
|
|
autoFocusReject: null
|
2022-09-14 11:26:01 +00:00
|
|
|
};
|
2022-09-06 12:03:37 +00:00
|
|
|
},
|
|
|
|
target: null,
|
|
|
|
outsideClickListener: null,
|
|
|
|
scrollHandler: null,
|
|
|
|
resizeListener: null,
|
|
|
|
container: null,
|
|
|
|
confirmListener: null,
|
|
|
|
closeListener: null,
|
|
|
|
mounted() {
|
|
|
|
this.confirmListener = (options) => {
|
|
|
|
if (!options) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (options.group === this.group) {
|
|
|
|
this.confirmation = options;
|
|
|
|
this.target = options.target;
|
2022-12-08 11:04:25 +00:00
|
|
|
|
|
|
|
if (this.confirmation.onShow) {
|
|
|
|
this.confirmation.onShow();
|
|
|
|
}
|
|
|
|
|
2022-09-06 12:03:37 +00:00
|
|
|
this.visible = true;
|
|
|
|
}
|
|
|
|
};
|
2022-09-14 11:26:01 +00:00
|
|
|
|
2022-09-06 12:03:37 +00:00
|
|
|
this.closeListener = () => {
|
|
|
|
this.visible = false;
|
|
|
|
this.confirmation = null;
|
|
|
|
};
|
2022-09-14 11:26:01 +00:00
|
|
|
|
2022-09-06 12:03:37 +00:00
|
|
|
ConfirmationEventBus.on('confirm', this.confirmListener);
|
|
|
|
ConfirmationEventBus.on('close', this.closeListener);
|
|
|
|
},
|
|
|
|
beforeUnmount() {
|
|
|
|
ConfirmationEventBus.off('confirm', this.confirmListener);
|
|
|
|
ConfirmationEventBus.off('close', this.closeListener);
|
|
|
|
|
|
|
|
this.unbindOutsideClickListener();
|
2022-09-14 11:26:01 +00:00
|
|
|
|
2022-09-06 12:03:37 +00:00
|
|
|
if (this.scrollHandler) {
|
|
|
|
this.scrollHandler.destroy();
|
|
|
|
this.scrollHandler = null;
|
|
|
|
}
|
2022-09-14 11:26:01 +00:00
|
|
|
|
2022-09-06 12:03:37 +00:00
|
|
|
this.unbindResizeListener();
|
|
|
|
|
|
|
|
if (this.container) {
|
|
|
|
ZIndexUtils.clear(this.container);
|
|
|
|
this.container = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.target = null;
|
|
|
|
this.confirmation = null;
|
|
|
|
},
|
|
|
|
methods: {
|
2022-09-14 11:26:01 +00:00
|
|
|
accept() {
|
2022-09-06 12:03:37 +00:00
|
|
|
if (this.confirmation.accept) {
|
|
|
|
this.confirmation.accept();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.visible = false;
|
|
|
|
},
|
|
|
|
reject() {
|
|
|
|
if (this.confirmation.reject) {
|
|
|
|
this.confirmation.reject();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.visible = false;
|
|
|
|
},
|
2022-12-08 11:04:25 +00:00
|
|
|
onHide() {
|
|
|
|
if (this.confirmation.onHide) {
|
|
|
|
this.confirmation.onHide();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.visible = false;
|
|
|
|
},
|
|
|
|
onAcceptKeydown(event) {
|
|
|
|
if (event.code === 'Space' || event.code === 'Enter') {
|
|
|
|
this.accept();
|
|
|
|
DomHandler.focus(this.target);
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
onRejectKeydown(event) {
|
|
|
|
if (event.code === 'Space' || event.code === 'Enter') {
|
|
|
|
this.reject();
|
|
|
|
DomHandler.focus(this.target);
|
|
|
|
event.preventDefault();
|
|
|
|
}
|
|
|
|
},
|
2022-09-06 12:03:37 +00:00
|
|
|
onEnter(el) {
|
2023-01-09 12:58:56 +00:00
|
|
|
this.autoFocusAccept = this.confirmation.defaultFocus === undefined || this.confirmation.defaultFocus === 'accept' ? true : false;
|
|
|
|
this.autoFocusReject = this.confirmation.defaultFocus === 'reject' ? true : false;
|
|
|
|
|
2022-09-06 12:03:37 +00:00
|
|
|
this.bindOutsideClickListener();
|
|
|
|
this.bindScrollListener();
|
|
|
|
this.bindResizeListener();
|
|
|
|
|
|
|
|
ZIndexUtils.set('overlay', el, this.$primevue.config.zIndex.overlay);
|
|
|
|
},
|
2023-01-09 12:58:56 +00:00
|
|
|
onAfterEnter() {
|
|
|
|
this.focus();
|
|
|
|
},
|
2022-09-06 12:03:37 +00:00
|
|
|
onLeave() {
|
2023-03-10 11:14:16 +00:00
|
|
|
this.autoFocusAccept = null;
|
|
|
|
this.autoFocusReject = null;
|
|
|
|
|
2022-09-06 12:03:37 +00:00
|
|
|
this.unbindOutsideClickListener();
|
|
|
|
this.unbindScrollListener();
|
|
|
|
this.unbindResizeListener();
|
|
|
|
},
|
|
|
|
onAfterLeave(el) {
|
|
|
|
ZIndexUtils.clear(el);
|
|
|
|
},
|
|
|
|
alignOverlay() {
|
|
|
|
DomHandler.absolutePosition(this.container, this.target);
|
|
|
|
|
|
|
|
const containerOffset = DomHandler.getOffset(this.container);
|
|
|
|
const targetOffset = DomHandler.getOffset(this.target);
|
|
|
|
let arrowLeft = 0;
|
|
|
|
|
|
|
|
if (containerOffset.left < targetOffset.left) {
|
|
|
|
arrowLeft = targetOffset.left - containerOffset.left;
|
|
|
|
}
|
2022-09-14 11:26:01 +00:00
|
|
|
|
2022-09-06 12:03:37 +00:00
|
|
|
this.container.style.setProperty('--overlayArrowLeft', `${arrowLeft}px`);
|
|
|
|
|
|
|
|
if (containerOffset.top < targetOffset.top) {
|
|
|
|
DomHandler.addClass(this.container, 'p-confirm-popup-flipped');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
bindOutsideClickListener() {
|
|
|
|
if (!this.outsideClickListener) {
|
|
|
|
this.outsideClickListener = (event) => {
|
|
|
|
if (this.visible && this.container && !this.container.contains(event.target) && !this.isTargetClicked(event)) {
|
2022-12-08 11:04:25 +00:00
|
|
|
if (this.confirmation.onHide) {
|
|
|
|
this.confirmation.onHide();
|
|
|
|
}
|
|
|
|
|
2022-09-06 12:03:37 +00:00
|
|
|
this.visible = false;
|
|
|
|
} else {
|
|
|
|
this.alignOverlay();
|
|
|
|
}
|
|
|
|
};
|
2022-09-14 11:26:01 +00:00
|
|
|
|
2022-09-06 12:03:37 +00:00
|
|
|
document.addEventListener('click', this.outsideClickListener);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
unbindOutsideClickListener() {
|
|
|
|
if (this.outsideClickListener) {
|
|
|
|
document.removeEventListener('click', this.outsideClickListener);
|
|
|
|
this.outsideClickListener = null;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
bindScrollListener() {
|
|
|
|
if (!this.scrollHandler) {
|
|
|
|
this.scrollHandler = new ConnectedOverlayScrollHandler(this.target, () => {
|
|
|
|
if (this.visible) {
|
|
|
|
this.visible = false;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
this.scrollHandler.bindScrollListener();
|
|
|
|
},
|
|
|
|
unbindScrollListener() {
|
|
|
|
if (this.scrollHandler) {
|
|
|
|
this.scrollHandler.unbindScrollListener();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
bindResizeListener() {
|
|
|
|
if (!this.resizeListener) {
|
|
|
|
this.resizeListener = () => {
|
|
|
|
if (this.visible && !DomHandler.isTouchDevice()) {
|
|
|
|
this.visible = false;
|
|
|
|
}
|
|
|
|
};
|
2022-09-14 11:26:01 +00:00
|
|
|
|
2022-09-06 12:03:37 +00:00
|
|
|
window.addEventListener('resize', this.resizeListener);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
unbindResizeListener() {
|
|
|
|
if (this.resizeListener) {
|
|
|
|
window.removeEventListener('resize', this.resizeListener);
|
|
|
|
this.resizeListener = null;
|
|
|
|
}
|
|
|
|
},
|
2022-12-08 11:04:25 +00:00
|
|
|
focus() {
|
|
|
|
let focusTarget = this.container.querySelector('[autofocus]');
|
|
|
|
|
|
|
|
if (focusTarget) {
|
2023-01-09 12:58:56 +00:00
|
|
|
focusTarget.focus({ preventScroll: true }); // Firefox requires preventScroll
|
2022-12-08 11:04:25 +00:00
|
|
|
}
|
|
|
|
},
|
2022-09-06 12:03:37 +00:00
|
|
|
isTargetClicked(event) {
|
|
|
|
return this.target && (this.target === event.target || this.target.contains(event.target));
|
|
|
|
},
|
|
|
|
containerRef(el) {
|
|
|
|
this.container = el;
|
|
|
|
},
|
|
|
|
onOverlayClick(event) {
|
|
|
|
OverlayEventBus.emit('overlay-click', {
|
|
|
|
originalEvent: event,
|
|
|
|
target: this.target
|
|
|
|
});
|
2022-12-08 11:04:25 +00:00
|
|
|
},
|
|
|
|
onOverlayKeydown(event) {
|
|
|
|
if (event.code === 'Escape') {
|
|
|
|
ConfirmationEventBus.emit('close', this.closeListener);
|
|
|
|
DomHandler.focus(this.target);
|
|
|
|
}
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
computed: {
|
|
|
|
containerClass() {
|
2022-09-14 11:26:01 +00:00
|
|
|
return [
|
|
|
|
'p-confirm-popup p-component',
|
|
|
|
{
|
|
|
|
'p-input-filled': this.$primevue.config.inputStyle === 'filled',
|
|
|
|
'p-ripple-disabled': this.$primevue.config.ripple === false
|
|
|
|
}
|
|
|
|
];
|
2022-09-06 12:03:37 +00:00
|
|
|
},
|
|
|
|
message() {
|
|
|
|
return this.confirmation ? this.confirmation.message : null;
|
|
|
|
},
|
|
|
|
iconClass() {
|
|
|
|
return ['p-confirm-popup-icon', this.confirmation ? this.confirmation.icon : null];
|
|
|
|
},
|
|
|
|
acceptLabel() {
|
2022-09-14 11:26:01 +00:00
|
|
|
return this.confirmation ? this.confirmation.acceptLabel || this.$primevue.config.locale.accept : null;
|
2022-09-06 12:03:37 +00:00
|
|
|
},
|
|
|
|
rejectLabel() {
|
2022-09-14 11:26:01 +00:00
|
|
|
return this.confirmation ? this.confirmation.rejectLabel || this.$primevue.config.locale.reject : null;
|
2022-09-06 12:03:37 +00:00
|
|
|
},
|
|
|
|
acceptIcon() {
|
|
|
|
return this.confirmation ? this.confirmation.acceptIcon : null;
|
|
|
|
},
|
|
|
|
rejectIcon() {
|
|
|
|
return this.confirmation ? this.confirmation.rejectIcon : null;
|
|
|
|
},
|
|
|
|
acceptClass() {
|
|
|
|
return ['p-confirm-popup-accept p-button-sm', this.confirmation ? this.confirmation.acceptClass : null];
|
|
|
|
},
|
|
|
|
rejectClass() {
|
2022-09-14 11:26:01 +00:00
|
|
|
return ['p-confirm-popup-reject p-button-sm', this.confirmation ? this.confirmation.rejectClass || 'p-button-text' : null];
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
|
|
|
},
|
|
|
|
components: {
|
2022-09-14 11:26:01 +00:00
|
|
|
CPButton: Button,
|
|
|
|
Portal: Portal
|
2022-12-08 11:04:25 +00:00
|
|
|
},
|
|
|
|
directives: {
|
|
|
|
focustrap: FocusTrap
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
2022-09-14 11:26:01 +00:00
|
|
|
};
|
2022-09-06 12:03:37 +00:00
|
|
|
</script>
|
|
|
|
|
|
|
|
<style>
|
|
|
|
.p-confirm-popup {
|
|
|
|
position: absolute;
|
|
|
|
margin-top: 10px;
|
|
|
|
top: 0;
|
|
|
|
left: 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
.p-confirm-popup-flipped {
|
|
|
|
margin-top: 0;
|
|
|
|
margin-bottom: 10px;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Animation */
|
|
|
|
.p-confirm-popup-enter-from {
|
|
|
|
opacity: 0;
|
|
|
|
transform: scaleY(0.8);
|
|
|
|
}
|
|
|
|
|
|
|
|
.p-confirm-popup-leave-to {
|
|
|
|
opacity: 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
.p-confirm-popup-enter-active {
|
2022-09-14 11:26:01 +00:00
|
|
|
transition: transform 0.12s cubic-bezier(0, 0, 0.2, 1), opacity 0.12s cubic-bezier(0, 0, 0.2, 1);
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
.p-confirm-popup-leave-active {
|
2022-09-14 11:26:01 +00:00
|
|
|
transition: opacity 0.1s linear;
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
|
|
|
|
2022-09-14 11:26:01 +00:00
|
|
|
.p-confirm-popup:after,
|
|
|
|
.p-confirm-popup:before {
|
|
|
|
bottom: 100%;
|
|
|
|
left: calc(var(--overlayArrowLeft, 0) + 1.25rem);
|
|
|
|
content: ' ';
|
|
|
|
height: 0;
|
|
|
|
width: 0;
|
|
|
|
position: absolute;
|
|
|
|
pointer-events: none;
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
.p-confirm-popup:after {
|
2022-09-14 11:26:01 +00:00
|
|
|
border-width: 8px;
|
|
|
|
margin-left: -8px;
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
.p-confirm-popup:before {
|
2022-09-14 11:26:01 +00:00
|
|
|
border-width: 10px;
|
|
|
|
margin-left: -10px;
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
|
|
|
|
2022-09-14 11:26:01 +00:00
|
|
|
.p-confirm-popup-flipped:after,
|
|
|
|
.p-confirm-popup-flipped:before {
|
2022-09-06 12:03:37 +00:00
|
|
|
bottom: auto;
|
|
|
|
top: 100%;
|
|
|
|
}
|
|
|
|
|
|
|
|
.p-confirm-popup.p-confirm-popup-flipped:after {
|
|
|
|
border-bottom-color: transparent;
|
|
|
|
}
|
|
|
|
|
|
|
|
.p-confirm-popup.p-confirm-popup-flipped:before {
|
2022-09-14 11:26:01 +00:00
|
|
|
border-bottom-color: transparent;
|
2022-09-06 12:03:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
.p-confirm-popup .p-confirm-popup-content {
|
|
|
|
display: flex;
|
|
|
|
align-items: center;
|
|
|
|
}
|
|
|
|
</style>
|