mirror of
https://github.com/KwiTsukasa/kt-template-online-playground.git
synced 2026-05-28 00:47:42 +08:00
221 lines
4.4 KiB
Vue
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>
|