|
@@ -1,29 +1,26 @@
|
|
|
<template>
|
|
|
- <div
|
|
|
+ <div
|
|
|
class="editable-element-shape"
|
|
|
:class="{
|
|
|
- 'lock': elementInfo.lock,
|
|
|
- 'format-painter': shapeFormatPainter,
|
|
|
+ lock: elementInfo.lock,
|
|
|
+ 'format-painter': shapeFormatPainter
|
|
|
}"
|
|
|
:style="{
|
|
|
top: elementInfo.top + 'px',
|
|
|
left: elementInfo.left + 'px',
|
|
|
width: elementInfo.width + 'px',
|
|
|
- height: elementInfo.height + 'px',
|
|
|
+ height: elementInfo.height + 'px'
|
|
|
}"
|
|
|
>
|
|
|
- <div
|
|
|
- class="rotate-wrapper"
|
|
|
- :style="{ transform: `rotate(${elementInfo.rotate}deg)` }"
|
|
|
- >
|
|
|
- <div
|
|
|
- class="element-content"
|
|
|
+ <div class="rotate-wrapper" :style="{ transform: `rotate(${elementInfo.rotate}deg)` }">
|
|
|
+ <div
|
|
|
+ class="element-content"
|
|
|
:style="{
|
|
|
opacity: elementInfo.opacity,
|
|
|
filter: shadowStyle ? `drop-shadow(${shadowStyle})` : '',
|
|
|
transform: flipStyle,
|
|
|
color: text.defaultColor,
|
|
|
- fontFamily: text.defaultFontName,
|
|
|
+ fontFamily: text.defaultFontName
|
|
|
}"
|
|
|
v-contextmenu="contextmenus"
|
|
|
@mousedown="$event => handleSelectElement($event)"
|
|
@@ -31,37 +28,33 @@
|
|
|
@touchstart="$event => handleSelectElement($event)"
|
|
|
@dblclick="startEdit()"
|
|
|
>
|
|
|
- <svg
|
|
|
- overflow="visible"
|
|
|
- :width="elementInfo.width"
|
|
|
- :height="elementInfo.height"
|
|
|
- >
|
|
|
+ <svg overflow="visible" :width="elementInfo.width" :height="elementInfo.height">
|
|
|
<defs v-if="elementInfo.gradient">
|
|
|
<GradientDefs
|
|
|
- :id="`editabel-gradient-${elementInfo.id}`"
|
|
|
+ :id="`editabel-gradient-${elementInfo.id}`"
|
|
|
:type="elementInfo.gradient.type"
|
|
|
:colors="elementInfo.gradient.colors"
|
|
|
:rotate="elementInfo.gradient.rotate"
|
|
|
/>
|
|
|
</defs>
|
|
|
- <g
|
|
|
- :transform="`scale(${elementInfo.width / elementInfo.viewBox[0]}, ${elementInfo.height / elementInfo.viewBox[1]}) translate(0,0) matrix(1,0,0,1,0,0)`"
|
|
|
+ <g
|
|
|
+ :transform="`scale(${elementInfo.width / (elementInfo.viewBox[0] || elementInfo.viewBox)}, ${elementInfo.height / (elementInfo.viewBox[1] || elementInfo.viewBox)}) translate(0,0) matrix(1,0,0,1,0,0)`"
|
|
|
>
|
|
|
- <path
|
|
|
+ <path
|
|
|
class="shape-path"
|
|
|
- vector-effect="non-scaling-stroke"
|
|
|
- stroke-linecap="butt"
|
|
|
+ vector-effect="non-scaling-stroke"
|
|
|
+ stroke-linecap="butt"
|
|
|
stroke-miterlimit="8"
|
|
|
- :d="elementInfo.path"
|
|
|
+ :d="elementInfo.path"
|
|
|
:fill="elementInfo.gradient ? `url(#editabel-gradient-${elementInfo.id})` : elementInfo.fill"
|
|
|
:stroke="outlineColor"
|
|
|
- :stroke-width="outlineWidth"
|
|
|
- :stroke-dasharray="strokeDashArray"
|
|
|
+ :stroke-width="outlineWidth"
|
|
|
+ :stroke-dasharray="strokeDashArray"
|
|
|
></path>
|
|
|
</g>
|
|
|
</svg>
|
|
|
|
|
|
- <div class="shape-text" :class="[text.align, { 'editable': editable || text.content }]">
|
|
|
+ <div class="shape-text" :class="[text.align, { editable: editable || text.content }]">
|
|
|
<ProsemirrorEditor
|
|
|
ref="prosemirrorEditorRef"
|
|
|
v-if="editable || text.content"
|
|
@@ -81,18 +74,18 @@
|
|
|
</template>
|
|
|
|
|
|
<script lang="ts" setup>
|
|
|
-import { computed, nextTick, ref, watch } from 'vue'
|
|
|
-import { storeToRefs } from 'pinia'
|
|
|
-import { useMainStore, useSlidesStore } from '@/store'
|
|
|
-import type { PPTShapeElement, ShapeText } from '@/types/slides'
|
|
|
-import type { ContextmenuItem } from '@/components/Contextmenu/types'
|
|
|
-import useElementOutline from '@/views/components/element/hooks/useElementOutline'
|
|
|
-import useElementShadow from '@/views/components/element/hooks/useElementShadow'
|
|
|
-import useElementFlip from '@/views/components/element/hooks/useElementFlip'
|
|
|
-import useHistorySnapshot from '@/hooks/useHistorySnapshot'
|
|
|
-
|
|
|
-import GradientDefs from './GradientDefs.vue'
|
|
|
-import ProsemirrorEditor from '@/views/components/element/ProsemirrorEditor.vue'
|
|
|
+import { computed, nextTick, ref, watch } from "vue"
|
|
|
+import { storeToRefs } from "pinia"
|
|
|
+import { useMainStore, useSlidesStore } from "@/store"
|
|
|
+import type { PPTShapeElement, ShapeText } from "@/types/slides"
|
|
|
+import type { ContextmenuItem } from "@/components/Contextmenu/types"
|
|
|
+import useElementOutline from "@/views/components/element/hooks/useElementOutline"
|
|
|
+import useElementShadow from "@/views/components/element/hooks/useElementShadow"
|
|
|
+import useElementFlip from "@/views/components/element/hooks/useElementFlip"
|
|
|
+import useHistorySnapshot from "@/hooks/useHistorySnapshot"
|
|
|
+
|
|
|
+import GradientDefs from "./GradientDefs.vue"
|
|
|
+import ProsemirrorEditor from "@/views/components/element/ProsemirrorEditor.vue"
|
|
|
|
|
|
const props = defineProps<{
|
|
|
elementInfo: PPTShapeElement
|
|
@@ -118,10 +111,10 @@ const execFormatPainter = () => {
|
|
|
const { keep, ...newProps } = shapeFormatPainter.value
|
|
|
|
|
|
slidesStore.updateElement({
|
|
|
- id: props.elementInfo.id,
|
|
|
- props: newProps,
|
|
|
+ id: props.elementInfo.id,
|
|
|
+ props: newProps
|
|
|
})
|
|
|
-
|
|
|
+
|
|
|
addHistorySnapshot()
|
|
|
if (!keep) mainStore.setShapeFormatPainter(null)
|
|
|
}
|
|
@@ -146,10 +139,10 @@ watch(handleElementId, () => {
|
|
|
|
|
|
const text = computed<ShapeText>(() => {
|
|
|
const defaultText: ShapeText = {
|
|
|
- content: '',
|
|
|
- defaultFontName: '微软雅黑',
|
|
|
- defaultColor: '#000',
|
|
|
- align: 'middle',
|
|
|
+ content: "",
|
|
|
+ defaultFontName: "微软雅黑",
|
|
|
+ defaultColor: "#000",
|
|
|
+ align: "middle"
|
|
|
}
|
|
|
if (!props.elementInfo.text) return defaultText
|
|
|
|
|
@@ -159,19 +152,19 @@ const text = computed<ShapeText>(() => {
|
|
|
const updateText = (content: string, ignore = false) => {
|
|
|
const _text = { ...text.value, content }
|
|
|
slidesStore.updateElement({
|
|
|
- id: props.elementInfo.id,
|
|
|
- props: { text: _text },
|
|
|
+ id: props.elementInfo.id,
|
|
|
+ props: { text: _text }
|
|
|
})
|
|
|
-
|
|
|
+
|
|
|
if (!ignore) addHistorySnapshot()
|
|
|
}
|
|
|
|
|
|
const checkEmptyText = () => {
|
|
|
if (!props.elementInfo.text) return
|
|
|
|
|
|
- const pureText = props.elementInfo.text.content.replace(/<[^>]+>/g, '')
|
|
|
+ const pureText = props.elementInfo.text.content.replace(/<[^>]+>/g, "")
|
|
|
if (!pureText) {
|
|
|
- slidesStore.removeElementProps({ id: props.elementInfo.id, propName: 'text' })
|
|
|
+ slidesStore.removeElementProps({ id: props.elementInfo.id, propName: "text" })
|
|
|
addHistorySnapshot()
|
|
|
}
|
|
|
}
|
|
@@ -192,7 +185,10 @@ const startEdit = () => {
|
|
|
cursor: default;
|
|
|
}
|
|
|
&.format-painter .element-content {
|
|
|
- cursor: url() 2 5, default !important;
|
|
|
+ cursor:
|
|
|
+ url()
|
|
|
+ 2 5,
|
|
|
+ default !important;
|
|
|
}
|
|
|
}
|
|
|
.rotate-wrapper {
|