SBaseImageEdit.ts 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597
  1. /*
  2. * *********************************************************************************************************************
  3. *
  4. * !!
  5. * .F88X
  6. * X8888Y
  7. * .}888888N;
  8. * i888888N; .:! .I$WI:
  9. * R888888I .'N88~ i8}+8Y&8"l8i$8>8W~'>W8}8]KW+8IIN"8&
  10. * .R888888I .;N8888~ .X8' "8I.!,/8" !%NY8`"8I8~~8>,88I
  11. * +888888N; .8888888Y "&&8Y.}8,
  12. * ./888888N; .R888888Y .'}~ .>}'.`+> i}! "i' +/' .'i~ !11,.:">, .~]! .i}i
  13. * ~888888%: .I888888l .]88~`1/iY88Ii+1'.R$8$8]"888888888> Y8$ W8E X8E W8888'188Il}Y88$*
  14. * 18888888 E8888881 .]W%8$`R8X'&8%++N8i,8N%N8+l8%` .}8N:.R$RE%N88N%N$K$R 188,FE$8%~Y88I
  15. * .E888888I .i8888888' .:$8I;88+`E8R:/8N,.>881.`$8E/1/]N8X.Y8N`"KF&&FK!'88*."88K./$88%RN888+~
  16. * 8888888I .,N888888~ ~88i"8W,!N8*.I88.}888%F,i$88"F88" 888:E8X.>88!i88>`888*.}Fl1]*}1YKi'
  17. * i888888N' I888Y ]88;/EX*IFKFK88X K8R .l8W 88Y ~88}'88E&%8W.X8N``]88!.$8K .:W8I
  18. * .i888888N; I8Y .&8$ .X88! i881.:%888>I88 ;88] +88+.';;;;:.Y88X 18N.,88l .+88/
  19. * .:R888888I
  20. * .&888888I Copyright (c) 2016-2020. 博锐尚格科技股份有限公司
  21. * ~8888'
  22. * .!88~ All rights reserved.
  23. *
  24. * *********************************************************************************************************************
  25. */
  26. import { SPainter, SRect, SSize, SColor, SPoint } from "@persagy-web/draw";
  27. import { SImageShowType, STextOrigin } from "@persagy-web/graph";
  28. import { SGraphItem, SAnchorItem, SLineStyle } from "@persagy-web/graph";
  29. import { SItemStatus } from "@persagy-web/big";
  30. import { SGraphEdit } from "..";
  31. import {
  32. ItemOrder,
  33. Marker,
  34. svgTobase64,
  35. svgUrlTobase64
  36. } from "@persagy-web/big/lib";
  37. import { SMouseEvent } from "@persagy-web/base/lib";
  38. /**
  39. * 图片编辑类
  40. *
  41. * @author 韩耀龙 <han_yao_long@163.com>
  42. */
  43. export class SBaseImageEdit extends SGraphEdit {
  44. //////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  45. //属性
  46. /** item 数据*/
  47. get data(): Marker | null {
  48. return this._data;
  49. } // Get data
  50. set data(v: Marker | null) {
  51. this._data = v;
  52. // 数据存在,则初始化数据
  53. if (v) {
  54. this.initData(v);
  55. this.update();
  56. }
  57. } // Set data
  58. // default url key
  59. defaultUrl: string = "";
  60. /** 起始锚点 */
  61. startItem: SGraphItem | null = null;
  62. /** 结束锚点 */
  63. endItem: SGraphItem | null = null;
  64. /** 图片 */
  65. img: CanvasImageSource | undefined;
  66. /** 展示模式 */
  67. private _showType: SImageShowType = SImageShowType.Full;
  68. get showType(): SImageShowType {
  69. return this._showType;
  70. }
  71. set showType(v: SImageShowType) {
  72. this._showType = v;
  73. this.computeImgSize();
  74. this.update();
  75. }
  76. /** 边框色 */
  77. private _strokeColor: SColor = SColor.Transparent;
  78. get strokeColor(): SColor {
  79. return this._strokeColor;
  80. }
  81. set strokeColor(v: SColor) {
  82. this._strokeColor = v;
  83. this.update();
  84. }
  85. /** 边框宽度 */
  86. private _lineWidth: number = 1;
  87. get lineWidth(): number {
  88. return this._lineWidth;
  89. }
  90. set lineWidth(v: number) {
  91. this._lineWidth = v;
  92. this.update();
  93. }
  94. /** 原点位置 */
  95. private _originPosition: STextOrigin = STextOrigin.LeftTop;
  96. get originPosition(): STextOrigin {
  97. return this._originPosition;
  98. }
  99. set originPosition(v: STextOrigin) {
  100. this._originPosition = v;
  101. this.update();
  102. }
  103. /** 线条样式 */
  104. private _lineStyle: SLineStyle = SLineStyle.Solid;
  105. get lineStyle(): SLineStyle {
  106. return this._lineStyle;
  107. }
  108. set lineStyle(v: SLineStyle) {
  109. this._lineStyle = v;
  110. this.update();
  111. }
  112. // 新增测试属性
  113. _filterColor: string = "";
  114. get filterColor() {
  115. return this._filterColor;
  116. }
  117. set filterColor(v) {
  118. this._filterColor = v;
  119. this.changeSvgColor(v);
  120. }
  121. /** 图片加载是否完成 */
  122. isLoadOver: boolean = false;
  123. /** 图片的宽度 */
  124. private imgWidth: number = this.width;
  125. /** 图片的高度 */
  126. private imgHeight: number = this.height;
  127. /** 图片地址 */
  128. private _url: string = "";
  129. get url(): string {
  130. return this._url;
  131. }
  132. set url(v: string) {
  133. // 处理svg
  134. if (v.endsWith(".svg")) {
  135. this.initSvg(v);
  136. } else {
  137. this.initUrl(v);
  138. }
  139. }
  140. /** svgUrl 字符串 */
  141. svgUrl: string = "";
  142. /** 锚点 list */
  143. anchorList: SAnchorItem[] = [];
  144. /** 宽度 */
  145. private _width: number = 64;
  146. /** 原点 */
  147. origin = new SPoint();
  148. get width(): number {
  149. return this._width;
  150. }
  151. set width(v: number) {
  152. if (v >= 0) {
  153. // 宽度有效
  154. if (v != this._width) {
  155. // 宽度更新
  156. let w = this._width;
  157. this._width = v;
  158. this.onResize(
  159. new SSize(w, this._height),
  160. new SSize(this._width, this._height)
  161. );
  162. }
  163. }
  164. this.update();
  165. } //Set width
  166. /** 高度 */
  167. private _height: number = 64;
  168. get height(): number {
  169. return this._height;
  170. }
  171. set height(v: number) {
  172. if (v >= 0) {
  173. // 高度有效
  174. if (v != this._height) {
  175. // 高度更新
  176. let h = this._height;
  177. this._height = v;
  178. this.onResize(
  179. new SSize(this._width, h),
  180. new SSize(this._width, this._height)
  181. );
  182. }
  183. }
  184. this.update();
  185. } //Set height
  186. /**编辑状态 */
  187. protected _status: SItemStatus = SItemStatus.Normal;
  188. get status(): SItemStatus {
  189. return this._status;
  190. }
  191. set status(value: SItemStatus) {
  192. const oldStatus = this._status;
  193. const newStatus = value;
  194. this._status = value;
  195. //状态变更触发事件
  196. this.$emit("StatusChange", oldStatus, newStatus);
  197. this.update();
  198. }
  199. //////////////////////////////////////////////////////////////////////////////////////////////////////////////
  200. // 函数
  201. /**
  202. * 构造函数
  203. *
  204. * @param parent 指向父对象
  205. * @param data 数据
  206. */
  207. constructor(parent: SGraphItem | null, data: Marker | null = null) {
  208. super(parent);
  209. if (data) {
  210. this.data = data;
  211. }
  212. }
  213. /**
  214. * 如果 data 设置;初始化data
  215. */
  216. initData(data: Marker): void {
  217. this.zOrder = ItemOrder.imageOrder;
  218. // this.isTransform = false;
  219. this.url =
  220. // eslint-disable-next-line max-len
  221. "";
  222. this.name = data.name;
  223. this.moveTo(data.pos.x, data.pos.y);
  224. if (data.size) {
  225. // 设置 size 宽高
  226. this.width = data.size.width;
  227. this.height = data.size.height;
  228. }
  229. if (data.style && data.style.default) {
  230. // 存在默认 style
  231. if (data.style.default.zorder) {
  232. // 设置层级
  233. this.zOrder = data.style.default.zorder;
  234. }
  235. // 图片url
  236. if (data.style.default.url) {
  237. this.defaultUrl = data.style.default.url;
  238. }
  239. // 线宽
  240. if (data.style.default.lineWidth) {
  241. this.lineWidth = data.style.default.lineWidth;
  242. }
  243. // 线颜色
  244. if (data.style.default.strokeColor) {
  245. this.strokeColor = new SColor(data.style.default.strokeColor);
  246. }
  247. // 线样式
  248. if (data.style.default.lineStyle) {
  249. this.lineStyle = data.style.default.lineStyle;
  250. }
  251. this.origin = new SPoint(this.width / 2, this.height / 2);
  252. }
  253. } // Function initData()
  254. /**
  255. * 初始化svgurl
  256. *
  257. * @param v url字符串
  258. */
  259. initSvg(v: any): void {
  260. this._url = v;
  261. svgTobase64(this.url)
  262. .then(res => {
  263. this.initUrl(res);
  264. })
  265. .catch(res => {
  266. this.initUrl(res);
  267. });
  268. }
  269. /**
  270. * svg 添加滤镜
  271. *
  272. * @param {*} val 修改svg图对应的色值(矩阵)
  273. */
  274. changeSvgColor(val: string): void {
  275. const parser = new DOMParser();
  276. const doc = parser.parseFromString(this.svgUrl, "text/xml");
  277. const svgDom = doc.children[0];
  278. // @ts-ignore
  279. svgDom.style.filter = "url('#lighten')";
  280. const colorDefs = parser.parseFromString(
  281. `<defs>
  282. <filter id="lighten">
  283. <feColorMatrix
  284. type="matrix"
  285. values=${val}/>
  286. </filter>
  287. </defs>`,
  288. "text/xml"
  289. );
  290. svgDom.appendChild(colorDefs.children[0]);
  291. const a = svgUrlTobase64(svgDom);
  292. this.initUrl(a);
  293. }
  294. /**
  295. * 初始化url
  296. *
  297. * @param v url字符串
  298. */
  299. initUrl(v: any): void {
  300. this._url = v;
  301. this.img = new Image();
  302. this.img.onload = (e): void => {
  303. const isiOS = !!navigator.userAgent.match(
  304. /\(i[^;]+;( U;)? CPU.+Mac OS X/
  305. );
  306. if (isiOS) {
  307. this.isLoadOver = true;
  308. this.computeImgSize();
  309. this.$emit("imgLoadOver");
  310. // 如果有填充色则设置填充色
  311. if (this._filterColor) {
  312. this.changeSvgColor(this._filterColor);
  313. }
  314. this.update();
  315. } else {
  316. // @ts-ignore
  317. const imgSrc = e.path[0].src;
  318. if (this.isUrlIdentical(imgSrc)) {
  319. // 当前地址和回调地址相同
  320. this.isLoadOver = true;
  321. this.computeImgSize();
  322. this.$emit("imgLoadOver");
  323. this.update();
  324. }
  325. }
  326. };
  327. this.img.onerror = (e): void => {
  328. // @ts-ignore
  329. const imgSrc = e.path[0].src;
  330. if (this.isUrlIdentical(imgSrc)) {
  331. // 当前地址和回调地址相同
  332. this.isLoadOver = false;
  333. this.$emit("imgLoadOver");
  334. this.update();
  335. console.log("加载图片错误!", e);
  336. }
  337. };
  338. this.img.src = v;
  339. }
  340. /**
  341. * 将类中得数据转换为方便存储格式的方法
  342. *
  343. * @return 针对 item 类型保持相应的格式
  344. */
  345. toData(): any {
  346. // 数据不存在,不处理
  347. if (!this.data) {
  348. return;
  349. }
  350. if (this.data.size) {
  351. // data中存在 size, 则设置 size
  352. this.data.size.width = this.width;
  353. this.data.size.height = this.height;
  354. } else {
  355. // 否则设置默认
  356. const width = this.width;
  357. const height = this.height;
  358. this.data.size = {
  359. width,
  360. height
  361. };
  362. }
  363. this.data.pos.x = this.pos.x;
  364. this.data.pos.y = this.pos.y;
  365. this.data.style.default.zorder = this.zOrder;
  366. this.data.style.default.url = this.defaultUrl;
  367. this.data.style.default.lineWidth = this.lineWidth;
  368. this.data.style.default.strokeColor = this.strokeColor.value;
  369. this.data.style.default.lineStyle = this.lineStyle;
  370. return this.data;
  371. }
  372. /**
  373. * 根据显示模式计算图片的宽高
  374. */
  375. computeImgSize(): void {
  376. // 图片加载完成
  377. if (this.isLoadOver) {
  378. // 要绘制图片的宽度
  379. let width = 0;
  380. // 要绘制图片的宽度
  381. let height = 0;
  382. // 图片 item 的宽高比
  383. let itemAspectRatio: number = this.width / this.height;
  384. // 原始图片的宽高比
  385. let imgAspectRatio: number =
  386. // @ts-ignore
  387. (this.img.width as number) / (this.img.height as number);
  388. // 原始图片的高宽比
  389. let imgHwRatio: number =
  390. // @ts-ignore
  391. (this.img.height as number) / (this.img.width as number);
  392. if (this.showType == SImageShowType.Full) {
  393. width = this.width;
  394. height = this.height;
  395. } else if (this.showType == SImageShowType.Equivalency) {
  396. if (itemAspectRatio > imgAspectRatio) {
  397. height = this.height;
  398. width = imgAspectRatio * height;
  399. } else if (itemAspectRatio < imgAspectRatio) {
  400. width = this.width;
  401. height = width * imgHwRatio;
  402. } else {
  403. width = this.width;
  404. height = this.height;
  405. }
  406. } else if (this.showType == SImageShowType.AutoFit) {
  407. // @ts-ignore
  408. this.width = this.img.width as number;
  409. // @ts-ignore
  410. this.height = this.img.height as number;
  411. width = this.width;
  412. height = this.height;
  413. }
  414. this.imgWidth = width;
  415. this.imgHeight = height;
  416. // 设置原点位置(默认左上角)
  417. if (this.originPosition == STextOrigin.Center) {
  418. this.origin = new SPoint(this.width / 2, this.height / 2);
  419. }
  420. }
  421. this.origin = new SPoint(this.width / 2, this.height / 2);
  422. this.update();
  423. }
  424. /**
  425. * 判断当前地址和回调地址是否相同
  426. *
  427. * @param imgUrl 图片回调地址
  428. * @return 当前地址和回调地址是否相同
  429. */
  430. private isUrlIdentical(imgUrl: string): boolean {
  431. // url 中没有该片段
  432. if (this.url.indexOf("://") == -1) {
  433. // eslint-disable-next-line max-len
  434. const reg = /^\s*data:([a-z]+\/[a-z0-9-+.]+(;[a-z-]+=[a-z0-9-]+)?)?(;base64)?,([a-z0-9!$&',()*+;=\-._~:@\/?%\s]*?)\s*$/i;
  435. if (reg.test(this.url)) {
  436. // 通过正则检验
  437. return this.url == imgUrl;
  438. } else {
  439. // 否则
  440. return this.url == this.GetUrlRelativePath(imgUrl);
  441. }
  442. } else {
  443. // 否则
  444. return this.url == imgUrl;
  445. }
  446. }
  447. /**
  448. * 截取绝对路径中的相对路径
  449. *
  450. * @param url 绝对路径
  451. * @return 相对路径
  452. */
  453. private GetUrlRelativePath(url: string): string {
  454. const arrUrl = url.split("//");
  455. const start = arrUrl[1].indexOf("/");
  456. const relUrl = arrUrl[1].substring(start);
  457. return relUrl;
  458. }
  459. /**
  460. * 大小改变
  461. *
  462. * @param oldSize 旧数据
  463. * @param newSize 新数据
  464. */
  465. resize(oldSize: SRect, newSize: SRect): void {
  466. this.width = newSize.width;
  467. this.height = newSize.height;
  468. this.update();
  469. }
  470. /**
  471. * Item 对象边界区域
  472. *
  473. * @return 边界区域
  474. */
  475. boundingRect(): SRect {
  476. return new SRect(
  477. -this.origin.x,
  478. -this.origin.y,
  479. this.width * 1,
  480. this.height * 1
  481. );
  482. }
  483. /**
  484. * 宽高发发生变化
  485. *
  486. * @param oldSize 改之前的大小
  487. * @param newSize 改之后大小
  488. */
  489. protected onResize(oldSize: SSize, newSize: SSize): void {
  490. this.computeImgSize();
  491. this.update();
  492. }
  493. /**
  494. * 鼠标按下事件
  495. *
  496. * @param event 保存事件参数
  497. * @return 是否处理事件
  498. */
  499. onMouseDown(event: SMouseEvent): boolean {
  500. super.onMouseDown(event);
  501. return this.moveable;
  502. } // Function onMouseDown()
  503. /**
  504. * 释放鼠标事件
  505. *
  506. * @param event 保存事件参数
  507. * @return 是否处理事件
  508. */
  509. onMouseUp(event: SMouseEvent): boolean {
  510. super.onMouseUp(event);
  511. return this.moveable;
  512. } // Function onMouseUp()
  513. /**
  514. * Item 绘制操作
  515. *
  516. * @param painter 绘画类
  517. */
  518. onDraw(painter: SPainter): void {
  519. painter.translate(-this.origin.x, -this.origin.y);
  520. // 图片加载完成
  521. if (this.isLoadOver) {
  522. // @ts-ignore
  523. painter.drawImage(this.img, 0, 0, this.imgWidth, this.imgHeight);
  524. }
  525. if (this.selected) {
  526. // 选中
  527. painter.shadow.shadowBlur = 10;
  528. painter.shadow.shadowColor = new SColor(`#00000033`);
  529. painter.shadow.shadowOffsetX = 5;
  530. painter.shadow.shadowOffsetY = 5;
  531. } else {
  532. // 否则
  533. painter.shadow.shadowColor = SColor.Transparent;
  534. }
  535. if (this.lineStyle == SLineStyle.Dashed) {
  536. // 线类型为虚线
  537. painter.pen.lineDash = [
  538. painter.toPx(this.lineWidth * 3),
  539. painter.toPx(this.lineWidth * 7)
  540. ];
  541. } else if (this.lineStyle == SLineStyle.Dotted) {
  542. // 线类型是点线
  543. painter.pen.lineDash = [
  544. painter.toPx(2 * this.lineWidth),
  545. painter.toPx(2 * this.lineWidth)
  546. ];
  547. }
  548. painter.pen.lineWidth = this.lineWidth;
  549. painter.pen.color = this.strokeColor;
  550. painter.brush.color = SColor.Transparent;
  551. painter.drawRect(0, 0, this.width, this.height);
  552. }
  553. }