1
0
Fork 0
mirror of https://gitlab.com/bramw/baserow.git synced 2025-04-04 13:15:24 +00:00

fixed element not found bug and fire less events in Context component

This commit is contained in:
Bram Wiepjes 2019-07-16 20:43:03 +02:00
parent 2139f3d9da
commit 3b1d94e23f
5 changed files with 62 additions and 79 deletions

View file

@ -1,10 +1,5 @@
<template>
<div
v-move-to-body
v-click-outside="hide"
class="context"
:class="{ 'visibility-hidden': !open }"
>
<div class="context" :class="{ 'visibility-hidden': !open }">
<slot></slot>
</div>
</template>
@ -26,15 +21,30 @@ export default {
* element and in the child element we need to register the child with their parent to
* prevent this.
*/
beforeMount() {
mounted() {
let $parent = this.$parent
while ($parent !== undefined) {
if ($parent.registerContextChild) {
$parent.registerContextChild(this.$el)
break
}
$parent = $parent.$parent
}
// Move the rendered element to the top of the body so it can be positioned over any
// other element.
const body = document.body
body.insertBefore(this.$el, body.firstChild)
},
/**
* Make sure the context menu is not open and all the events on the body are removed
* and that the element is removed from the body.
*/
destroyed() {
this.hide()
if (this.$el.parentNode) {
this.$el.parentNode.removeChild(this.$el)
}
},
methods: {
/**
@ -62,33 +72,56 @@ export default {
}
if (value) {
const css = this.calculatePosition(target, vertical, horizontal, offset)
this.show(target, vertical, horizontal, offset)
} else {
this.hide()
}
},
/**
* Calculate the position, show the context menu and register a click event on the
* body to check if the user has clicked outside the context.
*/
show(target, vertical, horizontal, offset) {
const css = this.calculatePosition(target, vertical, horizontal, offset)
// Set the calculated positions of the context.
for (const key in css) {
const value = css[key] !== null ? Math.ceil(css[key]) + 'px' : 'auto'
this.$el.style[key] = value
}
// Set the calculated positions of the context.
for (const key in css) {
const value = css[key] !== null ? Math.ceil(css[key]) + 'px' : 'auto'
this.$el.style[key] = value
}
// If we store the element who opened the context menu we can exclude the element
// when clicked outside of this element.
this.opener = value ? target : null
this.open = value
},
hide(event) {
// Checks if the click is inside one of our children. In that code the context
// must stay open.
const isChild = this.children.some(element =>
isElement(element, event.target)
)
this.opener = target
this.open = true
// Checks if the context is already opened, if the click was not on the opener
// because he can trigger the toggle method and if the click was not in one of
// our child contexts.
if (this.open && !isElement(this.opener, event.target) && !isChild) {
this.open = false
this.$el.clickOutsideEvent = event => {
if (
// Check if the context menu is still open
this.open &&
// If the click was outside the context element because we want to ignore
// clicks inside it.s
!isElement(this.$el, event.target) &&
// If the click was not on the opener because he can trigger the toggle
// method.
!isElement(this.opener, event.target) &&
// If the click was not inside one of the context children of this context
// menu.
!this.children.some(element => isElement(element, event.target))
) {
this.hide()
}
}
document.body.addEventListener('click', this.$el.clickOutsideEvent)
},
/**
* Hide the context menu and make sure the body event is removed.
*/
hide() {
this.opener = null
this.open = false
document.body.removeEventListener('click', this.$el.clickOutsideEvent)
},
/**
* Calculates the absolute position of the context based on the original clicked
@ -147,8 +180,6 @@ export default {
/**
* A child context can register itself with the parent to prevent closing of the
* parent when clicked inside the child.
*
* @param element HTMLElement
*/
registerContextChild(element) {
this.children.push(element)

View file

@ -25,11 +25,7 @@ export default {
/*
** Plugins to load before mounting the App
*/
plugins: [
{ src: '@/plugins/auth.js' },
{ src: '@/plugins/directives.js' },
{ src: '@/plugins/vuelidate.js' }
],
plugins: [{ src: '@/plugins/auth.js' }, { src: '@/plugins/vuelidate.js' }],
/*
** Nuxt.js modules

View file

@ -1,19 +0,0 @@
import { isElement } from '@/utils/dom'
/**
* This directive calls a custom method if the user clicks outside of the
* element.
*/
export default {
bind: (el, binding, vnode) => {
el.clickOutsideEvent = event => {
if (!isElement(el, event.target)) {
vnode.context[binding.expression](event)
}
}
document.body.addEventListener('click', el.clickOutsideEvent)
},
unbind: el => {
document.body.removeEventListener('click', el.clickOutsideEvent)
}
}

View file

@ -1,18 +0,0 @@
/**
* This directive moves the whole element to the document body so that it can be
* positioned over another element.
*/
export default {
inserted: el => {
const body = document.body
// The element is added as first child in the body so that child contexts
// are being shown on top of their parent.
body.insertBefore(el, body.firstChild)
},
unbind: el => {
if (el.parentNode) {
el.parentNode.removeChild(el)
}
}
}

View file

@ -1,7 +0,0 @@
import Vue from 'vue'
import moveToBody from '@/directives/moveToBody'
import clickOutside from '@/directives/clickOutside'
Vue.directive('move-to-body', moveToBody)
Vue.directive('click-outside', clickOutside)