Files
FHC-Core/public/js/components/Bootstrap/Modal.js
T
2026-02-10 11:02:20 +01:00

165 lines
4.2 KiB
JavaScript

//import Phrasen from '../../plugin/Phrasen.js';
export default {
name: 'BootstrapModal',
data: () => ({
modal: null,
fullscreen: false
}),
props: {
backdrop: {
type: [Boolean,String],
default: true,
validator(value) {
return ['static', true, false].includes(value);
}
},
focus: {
type: Boolean,
default: true
},
keyboard: {
type: Boolean,
default: true
},
noCloseBtn: Boolean,
dialogClass: [String,Array,Object],
headerClass: {
type: [String,Array,Object],
default: ''
},
bodyClass: {
type: [String,Array,Object],
default: 'px-4 py-5'
},
footerClass: {
type: [String,Array,Object],
default: ''
},
allowFullscreenExpand: {
type: Boolean,
default: false
}
},
emits: [
"hideBsModal",
"hiddenBsModal",
"hidePreventedBsModal",
"showBsModal",
"shownBsModal",
"toggleFullscreen"
],
methods: {
dispose() {
return this.modal.dispose();
},
handleUpdate() {
return this.modal.handleUpdate();
},
hide() {
return this.modal.hide();
},
show(relatedTarget) {
return this.modal.show(relatedTarget);
},
toggle() {
return this.modal.toggle();
},
toggleFullscreen() {
this.fullscreen = !this.fullscreen
this.$emit('toggleFullscreen')
}
},
mounted() {
if (this.$refs.modal)
this.modal = new bootstrap.Modal(this.$refs.modal, {
backdrop: this.backdrop,
focus: this.focus,
keyboard: this.keyboard
});
},
popup(body, options, title, footer) {
const BsModal = this,
slots = {};
if (body !== undefined)
slots.default = () => body;
if (title !== undefined)
slots.title = () => title;
if (footer !== undefined)
slots.footer = () => footer;
// little hack to check whether primevue is included in the app or not
let includedPrimevue = false;
if(typeof primevue !== 'undefined'){
includedPrimevue = true;
}
return new Promise((resolve,reject) => {
const instance = Vue.createApp({
name: 'ModalTmpApp',
setup() {
return () => Vue.h(BsModal, {...{
class: 'fade'
},...options, ...{
ref: 'modal',
'onHidden.bs.modal': instance.unmount
}}, slots);
},
mounted() {
this.$refs.modal.show();
},
beforeUnmount() {
if (this.$refs.modal)
this.$refs.modal.result !== false ? resolve(this.$refs.modal.result) : reject();
},
unmounted() {
wrapper.parentElement.removeChild(wrapper);
}
});
const wrapper = document.createElement("div");
// if(primevue) --> won't work because primevue is not defined in this scope and promise would be rejected
if (includedPrimevue){
instance.use(primevue.config.default, {zIndex: {overlay: 9999}})
}
//instance.use(Phrasen); // TODO(chris): find a more dynamic way
import('../../plugins/Phrasen.js').then((Phrasen) => {
instance.use(Phrasen.default);
instance.mount(wrapper);
document.body.appendChild(wrapper);
});
});
},
template: `<div ref="modal" class="bootstrap-modal modal" tabindex="-1" @[\`hide.bs.modal\`]="$emit('hideBsModal')" @[\`hidden.bs.modal\`]="$emit('hiddenBsModal')" @[\`hidePrevented.bs.modal\`]="$emit('hidePreventedBsModal')" @[\`show.bs.modal\`]="$emit('showBsModal')" @[\`shown.bs.modal\`]="$emit('shownBsModal')">
<div class="modal-dialog" :class="fullscreen ? 'modal-fullscreen' : dialogClass">
<div class="modal-content">
<div v-if="$slots.title" class="modal-header" :class="headerClass">
<h5 class="modal-title"><slot name="title"/></h5>
<div class="d-flex align-items-center ms-auto gap-2">
<button
type="button"
class="btn mb-1"
v-if="allowFullscreenExpand"
@click="toggleFullscreen"
:aria-label="fullscreen ? 'Exit Fullscreen' : 'Enter Fullscreen'"
>
<i v-if="!fullscreen" class="fa-solid fa-expand"></i>
<i v-else class="fa-solid fa-compress"></i>
</button>
<button v-if="!noCloseBtn" type="button" class="btn-close" data-bs-dismiss="modal" aria-label="Close"></button>
</div>
<slot name="modal-header-content"></slot>
</div>
<div class="modal-body" :class="bodyClass">
<slot></slot>
</div>
<div v-if="$slots.footer" class="modal-footer" :class="footerClass">
<slot name="footer"/>
</div>
</div>
</div>
</div>`
}