temp-icon.vue 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. <template>
  2. <v-group :config="groupConfig" v-if="groupConfig && svg" ref="shape">
  3. <v-group :config="initDecMat" name="rep-position">
  4. <v-rect :config="rectConfig" name="repShape" />
  5. <v-path v-for="config in pathConfigs" :config="config" name="icon-path" />
  6. </v-group>
  7. </v-group>
  8. </template>
  9. <script lang="ts" setup>
  10. import { defaultStyle, IconData } from "./icon.ts";
  11. import { computed, ref, watch } from "vue";
  12. import { getSvgContent, parseSvgContent, SVGParseResult } from "@/utils/resource.ts";
  13. import { Group } from "konva/lib/Group";
  14. import { DC } from "@/deconstruction.js";
  15. import { Transform } from "konva/lib/Util";
  16. import { useViewerInvertTransform, useViewSize } from "@/core/hook/use-viewer.ts";
  17. import { getFixPosition } from "@/utils/bound.ts";
  18. const props = defineProps<{ data: IconData; addMode?: boolean }>();
  19. const svg = ref<SVGParseResult | null>(null);
  20. const shape = ref<DC<Group>>();
  21. const data = computed(() => ({ ...defaultStyle, ...props.data }));
  22. defineExpose({
  23. get shape() {
  24. return shape.value;
  25. },
  26. });
  27. watch(
  28. () => data.value.url,
  29. async (url) => {
  30. svg.value = null;
  31. const svgContent = await getSvgContent(url);
  32. const content = parseSvgContent(svgContent);
  33. if (content.paths.length === 0) {
  34. svg.value = null;
  35. console.error(props.data.url, content, "路径数据不正确不是svg");
  36. } else {
  37. svg.value = content;
  38. }
  39. },
  40. { immediate: true }
  41. );
  42. const scale = computed(() => {
  43. if (!svg.value) return null;
  44. let w = data.value.width;
  45. let h = data.value.height;
  46. w = w || svg.value.width || 0;
  47. h = h || svg.value.height || 0;
  48. const scale = {
  49. x: w / svg.value.width,
  50. y: h / svg.value.height,
  51. };
  52. return scale;
  53. });
  54. const pathConfigs = computed(() => {
  55. if (!svg.value) return [];
  56. return svg.value.paths.map((path) => ({
  57. ...path,
  58. ...data.value,
  59. id: undefined,
  60. lineWidth: 1000,
  61. zIndex: undefined,
  62. offset: { x: svg.value!.x, y: svg.value!.y },
  63. }));
  64. });
  65. const initDecMat = computed(() => {
  66. if (!svg.value || !scale.value) return;
  67. return new Transform()
  68. .scale(scale.value.x, scale.value.y)
  69. .multiply(new Transform().translate(-svg.value.width / 2, -svg.value.height / 2))
  70. .decompose();
  71. });
  72. const viewInvTransform = useViewerInvertTransform();
  73. const size = useViewSize();
  74. const groupConfig = computed(() => {
  75. let mat = new Transform(data.value.mat);
  76. if (data.value.fixScreen) {
  77. if (!size.value) return {};
  78. const pos = getFixPosition(data.value.fixScreen, data.value, size.value);
  79. pos.x += data.value.width / 2;
  80. pos.y += data.value.height / 2;
  81. mat = viewInvTransform.value.copy().translate(pos.x, pos.y).multiply(mat);
  82. }
  83. return {
  84. ...mat.decompose(),
  85. zIndex: undefined,
  86. listening: data.value.listening,
  87. id: data.value.id,
  88. opacity: props.addMode ? 0.3 : 1,
  89. };
  90. });
  91. const rectConfig = computed(() => {
  92. if (!svg.value) return null;
  93. return {
  94. fill: data.value.coverFill,
  95. id: "rep",
  96. stroke: data.value.coverStroke,
  97. opacity: data.value.coverOpcatiy,
  98. strokeWidth: data.value.coverStrokeWidth,
  99. width: svg.value.width,
  100. height: svg.value.height,
  101. };
  102. });
  103. </script>