mirror of
https://github.com/pipipi-pikachu/PPTist.git
synced 2025-04-15 02:20:00 +08:00
103 lines
2.2 KiB
Vue
103 lines
2.2 KiB
Vue
<template>
|
|
<div
|
|
class="mask"
|
|
@contextmenu.prevent="removeContextmenu()"
|
|
@mousedown="removeContextmenu()"
|
|
></div>
|
|
|
|
<div
|
|
class="contextmenu"
|
|
:style="{
|
|
left: style.left + 'px',
|
|
top: style.top + 'px',
|
|
}"
|
|
@contextmenu.prevent
|
|
>
|
|
<MenuContent
|
|
:menus="menus"
|
|
:handleClickMenuItem="handleClickMenuItem"
|
|
/>
|
|
</div>
|
|
</template>
|
|
|
|
<script lang="ts">
|
|
import { computed, defineComponent, PropType } from 'vue'
|
|
import { ContextmenuItem, Axis } from './types'
|
|
|
|
import MenuContent from './MenuContent.vue'
|
|
|
|
export default defineComponent({
|
|
name: 'contextmenu',
|
|
components: {
|
|
MenuContent,
|
|
},
|
|
props: {
|
|
axis: {
|
|
type: Object as PropType<Axis>,
|
|
required: true,
|
|
},
|
|
el: {
|
|
type: Object as PropType<HTMLElement>,
|
|
required: true,
|
|
},
|
|
menus: {
|
|
type: Array as PropType<ContextmenuItem[]>,
|
|
required: true,
|
|
},
|
|
removeContextmenu: {
|
|
type: Function,
|
|
required: true,
|
|
},
|
|
},
|
|
setup(props) {
|
|
const style = computed(() => {
|
|
const MENU_WIDTH = 170
|
|
const MENU_HEIGHT = 30
|
|
const DIVIDER_HEIGHT = 11
|
|
const PADDING = 5
|
|
|
|
const { x, y } = props.axis
|
|
const menuCount = props.menus.filter(menu => !(menu.divider || menu.hide)).length
|
|
const dividerCount = props.menus.filter(menu => menu.divider).length
|
|
|
|
const menuWidth = MENU_WIDTH
|
|
const menuHeight = menuCount * MENU_HEIGHT + dividerCount * DIVIDER_HEIGHT + PADDING * 2
|
|
|
|
const screenWidth = document.body.clientWidth
|
|
const screenHeight = document.body.clientHeight
|
|
|
|
return {
|
|
left: screenWidth <= x + menuWidth ? x - menuWidth : x,
|
|
top: screenHeight <= y + menuHeight ? y - menuHeight : y,
|
|
}
|
|
})
|
|
|
|
const handleClickMenuItem = (item: ContextmenuItem) => {
|
|
if (item.disable || item.children) return
|
|
if (item.handler) item.handler(props.el)
|
|
props.removeContextmenu()
|
|
}
|
|
|
|
return {
|
|
style,
|
|
handleClickMenuItem,
|
|
}
|
|
},
|
|
})
|
|
</script>
|
|
|
|
<style lang="scss">
|
|
.mask {
|
|
position: fixed;
|
|
left: 0;
|
|
top: 0;
|
|
width: 100vw;
|
|
height: 100vh;
|
|
z-index: 9998;
|
|
}
|
|
.contextmenu {
|
|
position: fixed;
|
|
z-index: 9999;
|
|
user-select: none;
|
|
}
|
|
</style> |