SBaseImageEdit.ts 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528
  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 { Marker } from "../type/Marker";
  32. import { ItemOrder } from '@persagy-web/big/lib';
  33. import { SMouseEvent } from '@persagy-web/base/lib';
  34. import { svgTobase64 } from "@persagy-web/big-edit/lib/until"
  35. /**
  36. * 图片编辑类
  37. *
  38. * @author 韩耀龙 <han_yao_long@163.com>
  39. */
  40. export class SBaseImageEdit extends SGraphEdit {
  41. //////////////////////////////////////////////////////////////////////////////////////////////////////////////////
  42. //属性
  43. /** item 数据*/
  44. _data: Marker | null = null;
  45. get data(): Marker | null {
  46. return this._data;
  47. } // Get data
  48. set data(v: Marker | null) {
  49. this._data = v;
  50. if (v) {
  51. this.inData(v);
  52. this.update();
  53. }
  54. } // Set data
  55. // default url key
  56. defaultUrl: string = '';
  57. /** 起始锚点 */
  58. startItem: SGraphItem | null = null;
  59. /** 结束锚点 */
  60. endItem: SGraphItem | null = null;
  61. /** 图片 */
  62. img: CanvasImageSource | undefined;
  63. /** 展示模式 */
  64. private _showType: SImageShowType = SImageShowType.Full;
  65. get showType(): SImageShowType {
  66. return this._showType;
  67. }
  68. set showType(v: SImageShowType) {
  69. this._showType = v;
  70. this.computeImgSize();
  71. this.update();
  72. }
  73. /** 边框色 */
  74. private _strokeColor: SColor = SColor.Transparent;
  75. get strokeColor(): SColor {
  76. return this._strokeColor;
  77. }
  78. set strokeColor(v: SColor) {
  79. this._strokeColor = v;
  80. this.update();
  81. }
  82. /** 边框宽度 */
  83. private _lineWidth: number = 1;
  84. get lineWidth(): number {
  85. return this._lineWidth;
  86. }
  87. set lineWidth(v: number) {
  88. this._lineWidth = v;
  89. this.update();
  90. }
  91. /** 原点位置 */
  92. private _originPosition: STextOrigin = STextOrigin.LeftTop;
  93. get originPosition(): STextOrigin {
  94. return this._originPosition;
  95. }
  96. set originPosition(v: STextOrigin) {
  97. this._originPosition = v;
  98. this.update();
  99. }
  100. /** 线条样式 */
  101. private _lineStyle: SLineStyle = SLineStyle.Solid;
  102. get lineStyle(): SLineStyle {
  103. return this._lineStyle;
  104. }
  105. set lineStyle(v: SLineStyle) {
  106. this._lineStyle = v;
  107. this.update();
  108. }
  109. /** 图片加载是否完成 */
  110. isLoadOver: boolean = false;
  111. /** 图片的宽度 */
  112. private imgWidth: number = this.width;
  113. /** 图片的高度 */
  114. private imgHeight: number = this.height;
  115. /** 图片地址 */
  116. private _url: string = "";
  117. get url(): string {
  118. return this._url;
  119. }
  120. set url(v: string) {
  121. // 处理svg
  122. if (v.endsWith('.svg')) {
  123. this.initSvg(v)
  124. } else {
  125. this.initUrl(v)
  126. }
  127. }
  128. /** 锚点 list */
  129. anchorList: SAnchorItem[] = [];
  130. /** 宽度 */
  131. private _width: number = 64;
  132. /** 原点 */
  133. origin = new SPoint();
  134. get width(): number {
  135. return this._width;
  136. }
  137. set width(v: number) {
  138. if (v >= 0) {
  139. if (v != this._width) {
  140. let w = this._width;
  141. this._width = v;
  142. this.onResize(
  143. new SSize(w, this._height),
  144. new SSize(this._width, this._height)
  145. );
  146. }
  147. }
  148. this.update();
  149. } //Set width
  150. /** 高度 */
  151. private _height: number = 64;
  152. get height(): number {
  153. return this._height;
  154. }
  155. set height(v: number) {
  156. if (v >= 0) {
  157. if (v != this._height) {
  158. let h = this._height;
  159. this._height = v;
  160. this.onResize(
  161. new SSize(this._width, h),
  162. new SSize(this._width, this._height)
  163. );
  164. }
  165. }
  166. this.update();
  167. } //Set height
  168. /**编辑状态 */
  169. protected _status: SItemStatus = SItemStatus.Normal;
  170. get status(): SItemStatus {
  171. return this._status;
  172. }
  173. set status(value: SItemStatus) {
  174. const oldStatus = this._status;
  175. const newStatus = value;
  176. this._status = value;
  177. //状态变更触发事件
  178. this.$emit('StatusChange', oldStatus, newStatus)
  179. this.update();
  180. }
  181. //////////////////////////////////////////////////////////////////////////////////////////////////////////////
  182. // 函数
  183. /**
  184. * 构造函数
  185. *
  186. * @param parent 指向父对象
  187. */
  188. constructor(parent: SGraphItem | null, data: Marker | null = null) {
  189. super(parent);
  190. if (data) {
  191. this.data = data;
  192. }
  193. }
  194. /**
  195. * 如果 data 设置;初始化data
  196. */
  197. inData(data: Marker) {
  198. this.zOrder = ItemOrder.imageOrder;
  199. // this.isTransform = false;
  200. this.url = "";
  201. this.name = data.name;
  202. this.moveTo(data.pos.x, data.pos.y);
  203. if (data.size) {
  204. this.width = data.size.width;
  205. this.height = data.size.height;
  206. }
  207. if (data.style && data.style.default) {
  208. if (data.style.default.zorder) {
  209. this.zOrder = data.style.default.zorder;
  210. }
  211. // 图片url
  212. if (data.style.default.url) {
  213. this.defaultUrl = data.style.default.url;
  214. }
  215. // 线宽
  216. if (data.style.default.lineWidth) {
  217. this.lineWidth = data.style.default.lineWidth;
  218. }
  219. // 线颜色
  220. if (data.style.default.strokeColor) {
  221. this.strokeColor = new SColor(data.style.default.strokeColor)
  222. }
  223. // 线样式
  224. if (data.style.default.lineStyle) {
  225. this.lineStyle = data.style.default.lineStyle
  226. }
  227. this.origin = new SPoint(this.width / 2, this.height / 2);
  228. }
  229. }// Function inData()
  230. /**
  231. * 初始化svgurl
  232. *
  233. * @param v url字符串
  234. */
  235. initSvg(v: any) {
  236. this._url = v;
  237. svgTobase64(this.url).then((res) => {
  238. this.initUrl(res)
  239. }).catch((res) => {
  240. this.initUrl(res)
  241. });
  242. }
  243. /**
  244. * 初始化url
  245. *
  246. * @param v url字符串
  247. */
  248. initUrl(v: any) {
  249. this._url = v;
  250. this.img = new Image();
  251. this.img.onload = (e): void => {
  252. const isiOS = !!navigator.userAgent.match(/\(i[^;]+;( U;)? CPU.+Mac OS X/);
  253. if (isiOS) {
  254. this.isLoadOver = true;
  255. this.computeImgSize();
  256. this.$emit("imgLoadOver");
  257. this.update();
  258. } else {
  259. // @ts-ignore
  260. const imgSrc = e.path[0].src;
  261. if (this.isUrlIdentical(imgSrc)) {
  262. this.isLoadOver = true;
  263. this.computeImgSize();
  264. this.$emit("imgLoadOver");
  265. this.update();
  266. }
  267. }
  268. };
  269. this.img.onerror = (e): void => {
  270. // @ts-ignore
  271. const imgSrc = e.path[0].src;
  272. if (this.isUrlIdentical(imgSrc)) {
  273. this.isLoadOver = false;
  274. this.$emit("imgLoadOver");
  275. this.update();
  276. console.log("加载图片错误!", e);
  277. }
  278. };
  279. this.img.src = v;
  280. }
  281. /**
  282. * 将类中得数据转换为方便存储格式的方法
  283. *
  284. * @return 针对 item 类型保持相应的格式
  285. */
  286. toData(): any {
  287. if (!this.data) {
  288. return
  289. }
  290. if (this.data.size) {
  291. this.data.size.width = this.width;
  292. this.data.size.height = this.height;
  293. } else {
  294. const width = this.width;
  295. const height = this.height;
  296. this.data.size = {
  297. width,
  298. height,
  299. }
  300. }
  301. this.data.pos.x = this.pos.x;
  302. this.data.pos.y = this.pos.y;
  303. this.data.style.default.zorder = this.zOrder;
  304. this.data.style.default.url = this.defaultUrl;
  305. this.data.style.default.lineWidth = this.lineWidth;
  306. this.data.style.default.strokeColor = this.strokeColor.value;
  307. this.data.style.default.lineStyle = this.lineStyle;
  308. return this.data
  309. }
  310. /**
  311. * 根据显示模式计算图片的宽高
  312. */
  313. computeImgSize(): void {
  314. if (this.isLoadOver) {
  315. // 要绘制图片的宽度
  316. let width = 0;
  317. // 要绘制图片的宽度
  318. let height = 0;
  319. // 图片 item 的宽高比
  320. let itemAspectRatio: number = this.width / this.height;
  321. // 原始图片的宽高比
  322. let imgAspectRatio: number =
  323. // @ts-ignore
  324. (this.img.width as number) / (this.img.height as number);
  325. // 原始图片的高宽比
  326. let imgHwRatio: number =
  327. // @ts-ignore
  328. (this.img.height as number) / (this.img.width as number);
  329. if (this.showType == SImageShowType.Full) {
  330. width = this.width;
  331. height = this.height;
  332. } else if (this.showType == SImageShowType.Equivalency) {
  333. if (itemAspectRatio > imgAspectRatio) {
  334. height = this.height;
  335. width = imgAspectRatio * height;
  336. } else if (itemAspectRatio < imgAspectRatio) {
  337. width = this.width;
  338. height = width * imgHwRatio;
  339. } else {
  340. width = this.width;
  341. height = this.height;
  342. }
  343. } else if (this.showType == SImageShowType.AutoFit) {
  344. // @ts-ignore
  345. this.width = this.img.width as number;
  346. // @ts-ignore
  347. this.height = this.img.height as number;
  348. width = this.width;
  349. height = this.height;
  350. }
  351. this.imgWidth = width;
  352. this.imgHeight = height;
  353. // 设置原点位置(默认左上角)
  354. if (this.originPosition == STextOrigin.Center) {
  355. this.origin = new SPoint(this.width / 2, this.height / 2);
  356. }
  357. }
  358. this.origin = new SPoint(this.width / 2, this.height / 2);
  359. this.update();
  360. }
  361. /**
  362. * 判断当前地址和回调地址是否相同
  363. *
  364. * @param imgUrl 图片回调地址
  365. * @return 当前地址和回调地址是否相同
  366. */
  367. private isUrlIdentical(imgUrl: string): boolean {
  368. if (this.url.indexOf("://") == -1) {
  369. const reg = /^\s*data:([a-z]+\/[a-z0-9-+.]+(;[a-z-]+=[a-z0-9-]+)?)?(;base64)?,([a-z0-9!$&',()*+;=\-._~:@\/?%\s]*?)\s*$/i;
  370. if (reg.test(this.url)) {
  371. return this.url == imgUrl;
  372. } else {
  373. return this.url == this.GetUrlRelativePath(imgUrl);
  374. }
  375. } else {
  376. return this.url == imgUrl;
  377. }
  378. }
  379. /**
  380. * 截取绝对路径中的相对路径
  381. *
  382. * @param url 绝对路径
  383. * @return 相对路径
  384. */
  385. private GetUrlRelativePath(url: string): string {
  386. const arrUrl = url.split("//");
  387. const start = arrUrl[1].indexOf("/");
  388. const relUrl = arrUrl[1].substring(start);
  389. return relUrl;
  390. }
  391. /**
  392. * 大小改变
  393. *
  394. * @param oldSize 旧数据
  395. * @param newSize 新数据
  396. */
  397. resize(oldSize: SRect, newSize: SRect): void {
  398. this.width = newSize.width;
  399. this.height = newSize.height;
  400. this.update()
  401. }
  402. /**
  403. * Item 对象边界区域
  404. *
  405. * @return 边界区域
  406. */
  407. boundingRect(): SRect {
  408. return new SRect(
  409. -this.origin.x,
  410. -this.origin.y,
  411. this.width * 1,
  412. this.height * 1
  413. );
  414. }
  415. /**
  416. * 宽高发发生变化
  417. *
  418. * @param oldSize 改之前的大小
  419. * @param newSize 改之后大小
  420. */
  421. protected onResize(oldSize: SSize, newSize: SSize): void {
  422. this.computeImgSize();
  423. this.update();
  424. }
  425. /**
  426. * 鼠标按下事件
  427. *
  428. * @param event 保存事件参数
  429. * @return 是否处理事件
  430. */
  431. onMouseDown(event: SMouseEvent): boolean {
  432. super.onMouseDown(event);
  433. return this.moveable;
  434. } // Function onMouseDown()
  435. /**
  436. * 释放鼠标事件
  437. *
  438. * @param event 保存事件参数
  439. * @return 是否处理事件
  440. */
  441. onMouseUp(event: SMouseEvent): boolean {
  442. super.onMouseUp(event);
  443. return this.moveable;
  444. } // Function onMouseUp()
  445. /**
  446. * Item 绘制操作
  447. *
  448. * @param painter 绘画类
  449. */
  450. onDraw(painter: SPainter): void {
  451. painter.translate(-this.origin.x, -this.origin.y);
  452. if (this.isLoadOver) {
  453. // @ts-ignore
  454. painter.drawImage(this.img, 0, 0, this.imgWidth, this.imgHeight);
  455. }
  456. // 是否选中
  457. if (this.selected) {
  458. painter.shadow.shadowBlur = 10;
  459. painter.shadow.shadowColor = new SColor(`#00000033`);
  460. painter.shadow.shadowOffsetX = 5;
  461. painter.shadow.shadowOffsetY = 5;
  462. } else {
  463. painter.shadow.shadowColor = SColor.Transparent;
  464. }
  465. if (this.lineStyle == SLineStyle.Dashed) {
  466. painter.pen.lineDash = [
  467. painter.toPx(this.lineWidth * 3),
  468. painter.toPx(this.lineWidth * 7)
  469. ];
  470. } else if (this.lineStyle == SLineStyle.Dotted) {
  471. painter.pen.lineDash = [
  472. painter.toPx(2 * this.lineWidth),
  473. painter.toPx(2 * this.lineWidth)
  474. ];
  475. }
  476. painter.pen.lineWidth = this.lineWidth;
  477. painter.pen.color = this.strokeColor;
  478. painter.brush.color = SColor.Transparent;
  479. painter.drawRect(0, 0, this.width, this.height);
  480. }
  481. }