血透系统pad前端

ElectronicSignature.vue 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485
  1. <template>
  2. <div class="mainBox">
  3. <div class="mainContent">
  4. <h1 class="title">
  5. <div class="GoBack" @click="$router.go(-1)">
  6. <span class="iconfont">&#xe720;</span>返回
  7. </div>
  8. <span class="name">电子签名</span>
  9. <div class="GoBack"></div>
  10. </h1>
  11. <div class="information">
  12. <div class="blueBorder"></div>
  13. <div
  14. class="name"
  15. v-show="showpan == 'canvas'"
  16. style="display:fixed;width:100%;left:0;top:0;"
  17. >
  18. <canvas
  19. id="canvas"
  20. class="fl"
  21. width="600"
  22. height="400"
  23. disable-scroll="true"
  24. @mousedown="canvasDown($event)"
  25. @mouseup="canvasUp($event)"
  26. @mousemove="canvasMove($event)"
  27. @touchstart="canvasDown($event)"
  28. @touchend="canvasUp($event)"
  29. @touchmove="canvasMove($event)"
  30. ></canvas>
  31. </div>
  32. <div class="imagename" v-show="showpan == 'image'">
  33. <img :src="electronicsignature" alt srcset />
  34. </div>
  35. <div class="set">
  36. <el-row>
  37. <el-button @click="clearCanvas">重写签名</el-button>
  38. <el-button :disabled="nobeginAction" type="primary" @click="getImageAction">上传签名</el-button>
  39. </el-row>
  40. </div>
  41. </div>
  42. </div>
  43. <side-bar :active_index="3"></side-bar>
  44. <el-dialog
  45. title="提示"
  46. :visible.sync="centerDialogVisible"
  47. width="70%"
  48. v-loading="loading"
  49. element-loading-text="正在保存签名图片"
  50. element-loading-spinner="el-icon-loading"
  51. center
  52. >
  53. <vue-cropper ref="cropper" :src="imgUrl" alt="Source Image" :cropmove="cropImage"></vue-cropper>
  54. <span slot="footer" class="dialog-footer">
  55. <el-button @click="centerDialogVisible = false">取 消</el-button>
  56. <el-button type="primary" @click="uploadImage">确 定</el-button>
  57. </span>
  58. </el-dialog>
  59. </div>
  60. </template>
  61. <script>
  62. import SideBar from "@/pages/layout/SideBar";
  63. import { getToken } from "@/api/qiniu";
  64. import {
  65. GetElectronicSignature,
  66. SaveElectronicSignature
  67. } from "@/api/admin_user";
  68. import VueCropper from "vue-cropperjs";
  69. import { Toast } from "vant";
  70. export default {
  71. name: "ElectronicSignature",
  72. components: {
  73. VueCropper,
  74. SideBar
  75. },
  76. data() {
  77. return {
  78. showpan: "",
  79. nobeginAction: true,
  80. loading: false,
  81. electronicsignature: "",
  82. uptoken: "",
  83. avatarUrl2: null,
  84. centerDialogVisible: false,
  85. cropper: "",
  86. centerDialogVisible: false,
  87. colors: [
  88. "#fef4ac",
  89. "#0018ba",
  90. "#ffc200",
  91. "#f32f15",
  92. "#cccccc",
  93. "#5ab639"
  94. ],
  95. brushs: [
  96. {
  97. className: "small fa fa-paint-brush",
  98. lineWidth: 3
  99. },
  100. {
  101. className: "middle fa fa-paint-brush",
  102. lineWidth: 6
  103. },
  104. {
  105. className: "big fa fa-paint-brush",
  106. lineWidth: 12
  107. }
  108. ],
  109. context: {},
  110. imgUrl: "",
  111. uploadImgUrl: "",
  112. canvasMoveUse: false,
  113. // 存储当前表面状态数组-上一步
  114. preDrawAry: [],
  115. // 存储当前表面状态数组-下一步
  116. nextDrawAry: [],
  117. // 中间数组
  118. middleAry: [],
  119. // 配置参数
  120. config: {
  121. lineWidth: 3,
  122. lineColor: "#000",
  123. shadowBlur: 1,
  124. fillStyle: "#fff"
  125. }
  126. };
  127. },
  128. created() {
  129. this.getToken();
  130. this.GetElectronicSignature();
  131. document.body.addEventListener(
  132. "touchmove",
  133. function(e) {
  134. e.preventDefault();
  135. },
  136. { passive: false }
  137. );
  138. },
  139. mounted() {
  140. const canvas = document.querySelector("#canvas");
  141. this.context = canvas.getContext("2d");
  142. this.context.fillStyle = this.config.fillStyle;
  143. this.context.fillRect(0, 0, 600, 400);
  144. this.initDraw();
  145. this.setCanvasStyle();
  146. },
  147. destroyed() {},
  148. computed: {
  149. controls() {
  150. return [
  151. {
  152. title: "上一步",
  153. action: "prev",
  154. className: this.preDrawAry.length
  155. ? "active fa fa-reply"
  156. : "fa fa-reply"
  157. },
  158. {
  159. title: "下一步",
  160. action: "next",
  161. className: this.nextDrawAry.length
  162. ? "active fa fa-share"
  163. : "fa fa-share"
  164. },
  165. {
  166. title: "清除",
  167. action: "clear",
  168. className:
  169. this.preDrawAry.length || this.nextDrawAry.length
  170. ? "active fa fa-trash"
  171. : "fa fa-trash"
  172. }
  173. ];
  174. }
  175. },
  176. methods: {
  177. GetElectronicSignature() {
  178. GetElectronicSignature().then(response => {
  179. if (response.data.state == 1 && response.data.data.state == 1) {
  180. this.electronicsignature =
  181. response.data.data.electronic_signature.url;
  182. this.showpan = "image";
  183. } else {
  184. this.showpan = "canvas";
  185. }
  186. });
  187. },
  188. getToken() {
  189. getToken().then(response => {
  190. if (response.data.state == 1) {
  191. this.uptoken = response.data.data.uptoken;
  192. }
  193. });
  194. },
  195. cropImage(event) {
  196. var dataurl = this.$refs.cropper.getCroppedCanvas();
  197. var rectImage = dataurl.toDataURL("image/png");
  198. this.uploadImgUrl = rectImage;
  199. },
  200. isPc() {
  201. const userAgentInfo = navigator.userAgent;
  202. const Agents = [
  203. "Android",
  204. "iPhone",
  205. "SymbianOS",
  206. "Windows Phone",
  207. "iPad",
  208. "iPod"
  209. ];
  210. let flag = true;
  211. for (let v = 0; v < Agents.length; v++) {
  212. if (userAgentInfo.indexOf(Agents[v]) > 0) {
  213. flag = false;
  214. break;
  215. }
  216. }
  217. return flag;
  218. },
  219. removeImg(src) {
  220. this.imgUrl = "";
  221. },
  222. initDraw() {
  223. const preData = this.context.getImageData(0, 0, 600, 400);
  224. // 空绘图表面进栈
  225. this.middleAry.push(preData);
  226. },
  227. canvasMove(e) {
  228. if (this.canvasMoveUse) {
  229. const t = e.target;
  230. let canvasX;
  231. let canvasY;
  232. if (this.isPc()) {
  233. canvasX = e.clientX - t.parentNode.offsetLeft;
  234. canvasY = e.clientY - t.parentNode.offsetTop;
  235. } else {
  236. canvasX = e.changedTouches[0].clientX - t.parentNode.offsetLeft;
  237. canvasY = e.changedTouches[0].clientY - t.parentNode.offsetTop;
  238. }
  239. this.context.lineTo(canvasX, canvasY);
  240. this.context.stroke();
  241. }
  242. },
  243. beginPath(e) {
  244. const canvas = document.querySelector("#canvas");
  245. if (e.target !== canvas) {
  246. this.context.beginPath();
  247. }
  248. },
  249. // mouseup
  250. canvasUp(e) {
  251. const preData = this.context.getImageData(0, 0, 600, 400);
  252. if (!this.nextDrawAry.length) {
  253. // 当前绘图表面进栈
  254. this.middleAry.push(preData);
  255. } else {
  256. this.middleAry = [];
  257. this.middleAry = this.middleAry.concat(this.preDrawAry);
  258. this.middleAry.push(preData);
  259. this.nextDrawAry = [];
  260. }
  261. this.canvasMoveUse = false;
  262. this.nobeginAction = false;
  263. },
  264. // mousedown
  265. canvasDown(e) {
  266. this.canvasMoveUse = true;
  267. this.nobeginAction = false;
  268. // client是基于整个页面的坐标
  269. // offset是cavas距离顶部以及左边的距离
  270. const canvasX = e.clientX - e.target.parentNode.offsetLeft;
  271. const canvasY = e.clientY - e.target.parentNode.offsetTop;
  272. this.setCanvasStyle();
  273. // 清除子路径
  274. this.context.beginPath();
  275. this.context.moveTo(canvasX, canvasY);
  276. // 当前绘图表面状态
  277. const preData = this.context.getImageData(0, 0, 600, 400);
  278. // 当前绘图表面进栈
  279. this.preDrawAry.push(preData);
  280. },
  281. // 设置颜色
  282. setColor(color) {
  283. this.config.lineColor = color;
  284. },
  285. // 设置笔刷大小
  286. setBrush(type) {
  287. this.config.lineWidth = type;
  288. },
  289. // 操作
  290. clearCanvas() {
  291. this.context.clearRect(
  292. 0,
  293. 0,
  294. this.context.canvas.width,
  295. this.context.canvas.height
  296. );
  297. this.context.fillStyle = this.config.fillStyle;
  298. this.context.fillRect(0, 0, 600, 400);
  299. this.preDrawAry = [];
  300. this.nextDrawAry = [];
  301. this.middleAry = [this.middleAry[0]];
  302. this.nobeginAction = true;
  303. this.showpan = "canvas";
  304. },
  305. getImageAction() {
  306. this.getImage();
  307. this.centerDialogVisible = true;
  308. if (typeof this.$refs.cropper != "undefined") {
  309. this.$refs.cropper.replace(this.imgUrl);
  310. }
  311. },
  312. uploadImage() {
  313. var dataurl = this.$refs.cropper.getCroppedCanvas();
  314. var rectImage = dataurl.toDataURL("image/png");
  315. this.uploadImgUrl = rectImage;
  316. if (typeof this.uploadImgUrl == "undefined") {
  317. Toast.fail("获取截图资源失败");
  318. return false;
  319. }
  320. if (this.uptoken == "") {
  321. Toast.fail("七牛上传token无效,无法上传");
  322. return false;
  323. }
  324. this.loading = true;
  325. var _this = this;
  326. var pic = this.uploadImgUrl.split(",")[1];
  327. var url = "https://upload.qiniup.com/putb64/-1";
  328. var xhr = new XMLHttpRequest();
  329. xhr.onreadystatechange = function() {
  330. if (xhr.readyState == 4) {
  331. var returnData = JSON.parse(xhr.responseText);
  332. SaveElectronicSignature(returnData).then(response => {
  333. if (response.data.state == 1) {
  334. _this.electronicsignature =
  335. response.data.data.electronic_signature.url;
  336. _this.showpan = "image";
  337. _this.nobeginAction = true;
  338. _this.centerDialogVisible = false;
  339. Toast.success("成功");
  340. // _this.$router.push({path: "/my"})
  341. } else {
  342. Toast.fail("保存图片时失败");
  343. }
  344. });
  345. _this.loading = false;
  346. }
  347. };
  348. xhr.open("POST", url, true);
  349. xhr.setRequestHeader("Content-Type", "application/octet-stream");
  350. xhr.setRequestHeader("Authorization", "UpToken " + _this.uptoken);
  351. xhr.send(pic);
  352. },
  353. // 生成图片
  354. getImage() {
  355. const canvas = document.querySelector("#canvas");
  356. const src = canvas.toDataURL("image/png");
  357. this.imgUrl = src;
  358. // if (!this.isPc()) {
  359. // // window.open(`data:text/plain,${src}`)
  360. // window.location.href = src
  361. // }
  362. },
  363. // 设置绘画配置
  364. setCanvasStyle() {
  365. this.context.lineWidth = this.config.lineWidth;
  366. this.context.shadowBlur = this.config.shadowBlur;
  367. this.context.shadowColor = this.config.lineColor;
  368. this.context.strokeStyle = this.config.lineColor;
  369. }
  370. }
  371. };
  372. </script>
  373. <style style="stylesheet/scss" lang="scss" scoped>
  374. .showpanner {
  375. text-align: center;
  376. }
  377. .mainContent {
  378. .title {
  379. font-size: 0.45rem;
  380. padding: 0.3rem 0.37rem;
  381. color: $title-color;
  382. @include align-items-center;
  383. @include display-flex;
  384. @include justify-content-between;
  385. @include text-align;
  386. background: #fff;
  387. .GoBack {
  388. color: $main-color;
  389. font-size: 0.45rem;
  390. @include display-flex;
  391. .iconfont {
  392. color: $main-color;
  393. font-size: 0.5rem;
  394. margin-top: 1px;
  395. @media only screen and (min-width: 768px) {
  396. margin-top: 3px;
  397. }
  398. }
  399. }
  400. .name {
  401. margin-right: 1.3rem;
  402. }
  403. }
  404. .information {
  405. width: 100%;
  406. margin: 0 auto;
  407. .imagename {
  408. width: 100%;
  409. margin: 0 auto;
  410. text-align: center;
  411. padding: 10px 0;
  412. background-image: url("data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABAAAAAQAQMAAAAlPW0iAAAAA3NCSVQICAjb4U/gAAAABlBMVEXMzMz////TjRV2AAAACXBIWXMAAArrAAAK6wGCiw1aAAAAHHRFWHRTb2Z0d2FyZQBBZG9iZSBGaXJld29ya3MgQ1M26LyyjAAAABFJREFUCJlj+M/AgBVhF/0PAH6/D/HkDxOGAAAAAElFTkSuQmCC");
  413. }
  414. .name {
  415. background: #fff;
  416. width: 100%;
  417. height: 400px;
  418. margin: 0 auto;
  419. border-radius: 4px;
  420. // @media only screen and (max-width: 812px) {
  421. // width: 500px !important;
  422. // height: 200px !important;
  423. // }
  424. .item {
  425. padding: 0.37rem 0.4rem;
  426. @include align-items-center;
  427. @include display-flex;
  428. @include justify-content-between;
  429. @include text-align;
  430. height: 500px;
  431. .tx {
  432. @include display-flex;
  433. @include align-items-center;
  434. @include text-align;
  435. img {
  436. width: 1.24rem;
  437. height: 1.24rem;
  438. border-radius: 50%;
  439. display: inline-block;
  440. margin-right: 0.26rem;
  441. }
  442. }
  443. }
  444. h2 {
  445. color: #34495e;
  446. font-size: 0.34rem;
  447. }
  448. }
  449. .set {
  450. width: 70%;
  451. margin: 0 auto;
  452. border-radius: 4px;
  453. margin-top: 0.3rem;
  454. text-align: center;
  455. }
  456. }
  457. }
  458. .fl {
  459. @media only screen and (max-width: 812px) {
  460. width: 100% !important;
  461. height: 400px !important;
  462. }
  463. }
  464. .mainBox {
  465. height: 100%;
  466. display: flex;
  467. overflow: hidden;
  468. flex-direction: column;
  469. > :first-child {
  470. flex: 1;
  471. overflow: auto;
  472. }
  473. }
  474. </style>