mirror of
https://github.com/pipipi-pikachu/PPTist.git
synced 2025-04-15 02:20:00 +08:00
feat: 形状替换
This commit is contained in:
parent
5837e6f901
commit
568184ffd9
66
src/views/Editor/CanvasTool/ShapeItemThumbnail.vue
Normal file
66
src/views/Editor/CanvasTool/ShapeItemThumbnail.vue
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
<template>
|
||||||
|
<div class="shape-item-thumbnail">
|
||||||
|
<div class="shape-content">
|
||||||
|
<svg
|
||||||
|
overflow="visible"
|
||||||
|
width="18"
|
||||||
|
height="18"
|
||||||
|
>
|
||||||
|
<g
|
||||||
|
:transform="`scale(${18 / shape.viewBox[0]}, ${18 / shape.viewBox[1]}) translate(0,0) matrix(1,0,0,1,0,0)`"
|
||||||
|
>
|
||||||
|
<path
|
||||||
|
class="shape-path"
|
||||||
|
:class="{ 'outlined': shape.outlined }"
|
||||||
|
vector-effect="non-scaling-stroke"
|
||||||
|
stroke-linecap="butt"
|
||||||
|
stroke-miterlimit="8"
|
||||||
|
:fill="shape.outlined ? '#999' : 'transparent'"
|
||||||
|
:stroke="shape.outlined ? 'transparent' : '#999'"
|
||||||
|
stroke-width="2"
|
||||||
|
:d="shape.path"
|
||||||
|
></path>
|
||||||
|
</g>
|
||||||
|
</svg>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script lang="ts" setup>
|
||||||
|
import { PropType } from 'vue'
|
||||||
|
import { ShapePoolItem } from '@/configs/shapes'
|
||||||
|
|
||||||
|
defineProps({
|
||||||
|
shape: {
|
||||||
|
type: Object as PropType<ShapePoolItem>,
|
||||||
|
required: true,
|
||||||
|
},
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.shape-item-thumbnail {
|
||||||
|
position: relative;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
.shape-content {
|
||||||
|
@include absolute-0();
|
||||||
|
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
|
||||||
|
&:hover .shape-path {
|
||||||
|
&:not(.outlined) {
|
||||||
|
stroke: $themeColor;
|
||||||
|
}
|
||||||
|
&.outlined {
|
||||||
|
fill: $themeColor;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
svg:not(:root) {
|
||||||
|
overflow: visible;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
@ -3,31 +3,13 @@
|
|||||||
<div class="category" v-for="item in SHAPE_LIST" :key="item.type">
|
<div class="category" v-for="item in SHAPE_LIST" :key="item.type">
|
||||||
<div class="category-name">{{item.type}}</div>
|
<div class="category-name">{{item.type}}</div>
|
||||||
<div class="shape-list">
|
<div class="shape-list">
|
||||||
<div class="shape-item" v-for="(shape, index) in item.children" :key="index">
|
<ShapeItemThumbnail
|
||||||
<div class="shape-content" @click="selectShape(shape)">
|
class="shape-item"
|
||||||
<svg
|
v-for="(shape, index) in item.children"
|
||||||
overflow="visible"
|
:key="index"
|
||||||
width="18"
|
:shape="shape"
|
||||||
height="18"
|
@click="selectShape(shape)"
|
||||||
>
|
/>
|
||||||
<g
|
|
||||||
:transform="`scale(${18 / shape.viewBox[0]}, ${18 / shape.viewBox[1]}) translate(0,0) matrix(1,0,0,1,0,0)`"
|
|
||||||
>
|
|
||||||
<path
|
|
||||||
class="shape-path"
|
|
||||||
:class="{ 'outlined': shape.outlined }"
|
|
||||||
vector-effect="non-scaling-stroke"
|
|
||||||
stroke-linecap="butt"
|
|
||||||
stroke-miterlimit="8"
|
|
||||||
:fill="shape.outlined ? '#999' : 'transparent'"
|
|
||||||
:stroke="shape.outlined ? 'transparent' : '#999'"
|
|
||||||
stroke-width="2"
|
|
||||||
:d="shape.path"
|
|
||||||
></path>
|
|
||||||
</g>
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@ -35,6 +17,7 @@
|
|||||||
|
|
||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import { SHAPE_LIST, ShapePoolItem } from '@/configs/shapes'
|
import { SHAPE_LIST, ShapePoolItem } from '@/configs/shapes'
|
||||||
|
import ShapeItemThumbnail from './ShapeItemThumbnail.vue'
|
||||||
|
|
||||||
const emit = defineEmits<{
|
const emit = defineEmits<{
|
||||||
(event: 'select', payload: ShapePoolItem): void
|
(event: 'select', payload: ShapePoolItem): void
|
||||||
@ -75,27 +58,5 @@ const selectShape = (shape: ShapePoolItem) => {
|
|||||||
height: 0;
|
height: 0;
|
||||||
padding-bottom: 8%;
|
padding-bottom: 8%;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
position: relative;
|
|
||||||
cursor: pointer;
|
|
||||||
}
|
|
||||||
.shape-content {
|
|
||||||
@include absolute-0();
|
|
||||||
|
|
||||||
display: flex;
|
|
||||||
justify-content: center;
|
|
||||||
align-items: center;
|
|
||||||
|
|
||||||
&:hover .shape-path {
|
|
||||||
&:not(.outlined) {
|
|
||||||
stroke: $themeColor;
|
|
||||||
}
|
|
||||||
&.outlined {
|
|
||||||
fill: $themeColor;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
svg:not(:root) {
|
|
||||||
overflow: visible;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
@ -1,5 +1,23 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="shape-style-panel">
|
<div class="shape-style-panel">
|
||||||
|
<div class="title">
|
||||||
|
<span>点击替换形状</span>
|
||||||
|
<IconDown />
|
||||||
|
</div>
|
||||||
|
<div class="shape-pool">
|
||||||
|
<div class="category" v-for="item in SHAPE_LIST" :key="item.type">
|
||||||
|
<div class="shape-list">
|
||||||
|
<ShapeItemThumbnail
|
||||||
|
class="shape-item"
|
||||||
|
v-for="(shape, index) in item.children"
|
||||||
|
:key="index"
|
||||||
|
:shape="shape"
|
||||||
|
@click="changeShape(shape)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<Select
|
<Select
|
||||||
style="flex: 10;"
|
style="flex: 10;"
|
||||||
@ -69,6 +87,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<ElementFlip />
|
<ElementFlip />
|
||||||
|
|
||||||
<Divider />
|
<Divider />
|
||||||
|
|
||||||
<template v-if="handleShapeElement.text?.content">
|
<template v-if="handleShapeElement.text?.content">
|
||||||
@ -245,6 +264,7 @@ import { storeToRefs } from 'pinia'
|
|||||||
import { useMainStore, useSlidesStore } from '@/store'
|
import { useMainStore, useSlidesStore } from '@/store'
|
||||||
import { PPTShapeElement, ShapeGradient, ShapeText } from '@/types/slides'
|
import { PPTShapeElement, ShapeGradient, ShapeText } from '@/types/slides'
|
||||||
import { WEB_FONTS } from '@/configs/font'
|
import { WEB_FONTS } from '@/configs/font'
|
||||||
|
import { ShapePoolItem, SHAPE_LIST, SHAPE_PATH_FORMULAS } from '@/configs/shapes'
|
||||||
import emitter, { EmitterEvents } from '@/utils/emitter'
|
import emitter, { EmitterEvents } from '@/utils/emitter'
|
||||||
import useHistorySnapshot from '@/hooks/useHistorySnapshot'
|
import useHistorySnapshot from '@/hooks/useHistorySnapshot'
|
||||||
import useTextFormatPainter from '@/hooks/useTextFormatPainter'
|
import useTextFormatPainter from '@/hooks/useTextFormatPainter'
|
||||||
@ -258,6 +278,7 @@ import TextColorButton from '../common/TextColorButton.vue'
|
|||||||
import CheckboxButton from '@/components/CheckboxButton.vue'
|
import CheckboxButton from '@/components/CheckboxButton.vue'
|
||||||
import CheckboxButtonGroup from '@/components/CheckboxButtonGroup.vue'
|
import CheckboxButtonGroup from '@/components/CheckboxButtonGroup.vue'
|
||||||
import ColorPicker from '@/components/ColorPicker/index.vue'
|
import ColorPicker from '@/components/ColorPicker/index.vue'
|
||||||
|
import ShapeItemThumbnail from '@/views/Editor/CanvasTool/ShapeItemThumbnail.vue'
|
||||||
import {
|
import {
|
||||||
Divider,
|
Divider,
|
||||||
Button,
|
Button,
|
||||||
@ -326,6 +347,32 @@ const updateFill = (value: string) => {
|
|||||||
updateElement({ fill: value })
|
updateElement({ fill: value })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 修改形状
|
||||||
|
const changeShape = (shape: ShapePoolItem) => {
|
||||||
|
const { width, height } = handleElement.value as PPTShapeElement
|
||||||
|
const props: Partial<PPTShapeElement> = {
|
||||||
|
viewBox: shape.viewBox,
|
||||||
|
path: shape.path,
|
||||||
|
special: shape.special,
|
||||||
|
}
|
||||||
|
if (shape.pathFormula) {
|
||||||
|
props.pathFormula = shape.pathFormula
|
||||||
|
props.viewBox = [width, height]
|
||||||
|
|
||||||
|
const pathFormula = SHAPE_PATH_FORMULAS[shape.pathFormula]
|
||||||
|
if ('editable' in pathFormula) {
|
||||||
|
props.path = pathFormula.formula(width, height, pathFormula.defaultValue)
|
||||||
|
props.keypoint = pathFormula.defaultValue
|
||||||
|
}
|
||||||
|
else props.path = pathFormula.formula(width, height)
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
props.pathFormula = undefined
|
||||||
|
props.keypoint = undefined
|
||||||
|
}
|
||||||
|
updateElement(props)
|
||||||
|
}
|
||||||
|
|
||||||
const updateTextAlign = (align: 'top' | 'middle' | 'bottom') => {
|
const updateTextAlign = (align: 'top' | 'middle' | 'bottom') => {
|
||||||
const _handleElement = handleElement.value as PPTShapeElement
|
const _handleElement = handleElement.value as PPTShapeElement
|
||||||
|
|
||||||
@ -366,4 +413,28 @@ const emitRichTextCommand = (command: string, value?: string) => {
|
|||||||
.slider {
|
.slider {
|
||||||
flex: 3;
|
flex: 3;
|
||||||
}
|
}
|
||||||
|
.title {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
.shape-pool {
|
||||||
|
width: 235px;
|
||||||
|
height: 190px;
|
||||||
|
overflow: auto;
|
||||||
|
padding: 5px;
|
||||||
|
padding-right: 10px;
|
||||||
|
border: 1px solid $borderColor;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
}
|
||||||
|
.shape-list {
|
||||||
|
@include flex-grid-layout();
|
||||||
|
}
|
||||||
|
.shape-item {
|
||||||
|
@include flex-grid-layout-children(6, 14%);
|
||||||
|
|
||||||
|
height: 0;
|
||||||
|
padding-bottom: 14%;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
Loading…
x
Reference in New Issue
Block a user