kt-template-online-playground/src/SplitPane.vue
2026-05-08 14:30:56 +08:00

221 lines
4.4 KiB
Vue

<script setup lang="ts">
import { computed, inject, reactive, useTemplateRef } from 'vue'
import { injectKeyPreviewRef, injectKeyProps } from './types'
const props = defineProps<{ layout?: 'horizontal' | 'vertical' }>()
const isVertical = computed(() => props.layout === 'vertical')
const containerRef = useTemplateRef('container')
const previewRef = inject(injectKeyPreviewRef)!
// mobile only
const { store, layoutReverse, splitPaneOptions } = inject(injectKeyProps)!
const state = reactive({
dragging: false,
split: 50,
viewHeight: 0,
viewWidth: 0,
})
const boundSplit = computed(() => {
const { split } = state
return split < 20 ? 20 : split > 80 ? 80 : split
})
let startPosition = 0
let startSplit = 0
function dragStart(e: MouseEvent) {
state.dragging = true
startPosition = isVertical.value ? e.pageY : e.pageX
startSplit = boundSplit.value
changeViewSize()
}
function dragMove(e: MouseEvent) {
if (containerRef.value && state.dragging) {
const position = isVertical.value ? e.pageY : e.pageX
const totalSize = isVertical.value
? containerRef.value.offsetHeight
: containerRef.value.offsetWidth
const dp = position - startPosition
state.split = startSplit + +((dp / totalSize) * 100).toFixed(2)
changeViewSize()
}
}
function dragEnd() {
state.dragging = false
}
function changeViewSize() {
const el = previewRef.value
if (!el) return
state.viewHeight = el.offsetHeight
state.viewWidth = el.offsetWidth
}
</script>
<template>
<div
ref="container"
class="split-pane"
:class="{
dragging: state.dragging,
'show-output': store.showOutput,
reverse: layoutReverse,
vertical: isVertical,
}"
@mousemove="dragMove"
@mouseup="dragEnd"
@mouseleave="dragEnd"
>
<div
class="left"
:style="{ [isVertical ? 'height' : 'width']: boundSplit + '%' }"
>
<slot name="left" />
<div class="dragger" @mousedown.prevent="dragStart" />
</div>
<div
class="right"
:style="{ [isVertical ? 'height' : 'width']: 100 - boundSplit + '%' }"
>
<div v-show="state.dragging" class="view-size">
{{ `${state.viewWidth}px x ${state.viewHeight}px` }}
</div>
<slot name="right" />
</div>
<button class="toggler" @click="store.showOutput = !store.showOutput">
{{
store.showOutput
? splitPaneOptions?.codeTogglerText || '< Code'
: splitPaneOptions?.outputTogglerText || 'Output >'
}}
</button>
</div>
</template>
<style scoped>
.split-pane {
display: flex;
height: 100%;
position: relative;
}
.split-pane.dragging {
cursor: ew-resize;
}
.dragging .left,
.dragging .right {
pointer-events: none;
}
.left,
.right {
position: relative;
height: 100%;
}
.view-size {
position: absolute;
top: 40px;
left: 10px;
font-size: 12px;
color: var(--text-light);
z-index: 100;
}
.left {
border-right: 1px solid var(--border);
}
.dragger {
position: absolute;
z-index: 3;
top: 0;
bottom: 0;
right: -5px;
width: 10px;
cursor: ew-resize;
}
.toggler {
display: none;
z-index: 3;
font-family: var(--font-code);
color: var(--text-light);
position: absolute;
left: 50%;
bottom: 20px;
background-color: var(--bg);
padding: 8px 12px;
border-radius: 8px;
transform: translateX(-50%);
box-shadow: 0 3px 8px rgba(0, 0, 0, 0.25);
}
.dark .toggler {
background-color: var(--bg);
}
/* vertical */
@media (min-width: 721px) {
.split-pane.vertical {
display: block;
}
.split-pane.vertical.dragging {
cursor: ns-resize;
}
.vertical .dragger {
top: auto;
height: 10px;
width: 100%;
left: 0;
right: 0;
bottom: -5px;
cursor: ns-resize;
}
.vertical .left,
.vertical .right {
width: 100%;
}
.vertical .left {
border-right: none;
border-bottom: 1px solid var(--border);
}
}
/* mobile */
@media (max-width: 720px) {
.left,
.right {
position: absolute;
inset: 0;
width: auto !important;
height: auto !important;
}
.dragger {
display: none;
}
.split-pane .toggler {
display: block;
}
.split-pane .right,
.split-pane.show-output.reverse .right,
.split-pane.show-output .left,
.split-pane.reverse .left {
z-index: -1;
pointer-events: none;
}
.split-pane .left,
.split-pane.show-output.reverse .left,
.split-pane.show-output .right,
.split-pane.reverse .right {
z-index: 0;
pointer-events: all;
}
}
</style>