DrawingBoardTool.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371
  1. <template>
  2. <div class="drawing-board-wrap"
  3. :style="{
  4. 'pointer-events':model==='mouse'?'none':'auto',
  5. }" >
  6. <!-- 画布 -->
  7. <canvas id="paint" ref="canvasRef"
  8. :style="{
  9. width:canvasWidth+'px',
  10. height:canvasHeight+'px',
  11. }"
  12. @mousedown.stop="startDraw"
  13. @mousemove="(e)=>{mouse.x=e.clientX;mouse.y=e.clientY}"
  14. />
  15. <!-- 工具栏 -->
  16. <!-- 鼠标跟随-画笔和橡皮擦图标 -->
  17. <div class="pen"
  18. :style="{
  19. left: mouse.x - 3+ 'px',
  20. top: mouse.y - 22 + 3 + 'px',
  21. color: choosedColor,
  22. }"
  23. v-if="model==='paint'&&activeTool === 'brush'"
  24. ><img src="~@/assets/icons/brush.svg" :style="brushStyle({iconName:'brush'})"></div>
  25. <div
  26. class="eraser"
  27. :style="{
  28. left: mouse.x - 18 + 'px',
  29. top: mouse.y - 18 + 'px',
  30. width: 36 + 'px',
  31. height: 36 + 'px',
  32. }"
  33. v-if="model==='paint'&&activeTool === 'eraser'"
  34. ></div>
  35. </div>
  36. </template>
  37. <script>
  38. import {boardTool,colorList} from '../utils/config.js';
  39. export default {
  40. components:{},
  41. props:{
  42. model:{
  43. type:String,
  44. default:'mouse'
  45. },
  46. pageList:{}
  47. },
  48. data() {
  49. return {
  50. ctx:null,//canvas上下文
  51. canvasWidth:0,//canvas宽
  52. canvasHeight:0,//canvas高
  53. toolEl:{left:5,top:5},//画笔工具栏位置
  54. paletteEl:{left:-235,top:77},//画笔工具-颜色选择器位置
  55. mouse:{x:0,y:0},//鼠标坐标
  56. isMouseDown:false,//画笔工具栏移动标识
  57. toolList:boardTool,//画笔工具栏
  58. colorList:colorList,//画笔工具栏-颜色选择
  59. activeTool:'brush',//选中的画笔工具
  60. isPalettleShow:false,//是否显示颜色选择
  61. choosedColor:'#000000',//画笔颜色
  62. movePoint:{},//画笔坐标
  63. isStartDraw:false,//是否在绘图中
  64. hasPainted:false,//当前画布是否有笔迹
  65. isMoved:false,//画笔是否移动过
  66. };
  67. },
  68. computed:{
  69. brushStyle(){
  70. return function(tool){
  71. if(tool.iconName==='brush'){
  72. return {'transform': 'translateX(44px)','filter':`drop-shadow(${this.choosedColor} -44px 0px 0px)`}
  73. }else if(tool.iconName==='palette'){
  74. return {'transform': 'translateX(44px)','filter':`drop-shadow(#000000 -44px 0px 0px)`}
  75. }else{
  76. return {'transform': 'translateX(44px)','filter':`drop-shadow(${this.hasPainted?'#000000':'#b5b5b5'} -44px 0px 0px)`}
  77. }
  78. }
  79. }
  80. },
  81. methods: {
  82. //初始化画布及工具栏
  83. initCanvas(){
  84. this.ctx = this.$refs.canvasRef.getContext('2d',{willReadFrequently: true})
  85. this.canvasWidth = window.screen.width
  86. this.canvasHeight = window.screen.height
  87. this.toolEl.left = window.screen.width-56-5
  88. //设置canvas宽高
  89. this.$refs.canvasRef.width=this.canvasWidth
  90. this.$refs.canvasRef.height=this.canvasHeight
  91. if(!this.ctx) return
  92. this.ctx.lineCap = 'round'
  93. this.ctx.lineJoin = 'round'
  94. this.ctx.lineWidth = 6
  95. this.ctx.strokeStyle = this.choosedColor
  96. },
  97. //移动画布工具栏
  98. handleMoveStart(event){
  99. this.isMouseDown = true
  100. this.isStartDraw = false
  101. let startX = event.clientX
  102. let startY = event.clientY
  103. document.onmousemove=(e)=>{
  104. if(!this.isMouseDown) return
  105. if(this.isPalettleShow) this.isPalettleShow = false
  106. let moveX = e.clientX - startX
  107. let moveY = e.clientY - startY
  108. const {left,top} = this.toolEl
  109. const toolWidth = 56
  110. const toolHeight = 242
  111. //移动边界:不超过屏幕
  112. this.toolEl.left = Math.min(Math.max(left+moveX,5),this.canvasWidth-toolWidth-5)
  113. this.toolEl.top = Math.min(Math.max(top+moveY,5),this.canvasHeight-toolHeight-5)
  114. //toolEl的位置决定了paletteEl的位置
  115. if(this.toolEl.left+220>this.canvasWidth-toolWidth){
  116. this.paletteEl.left=-235
  117. }else{
  118. this.paletteEl.left=57
  119. }
  120. startX = e.clientX
  121. startY = e.clientY
  122. }
  123. document.onmouseup = ()=>{
  124. this.isMouseDown = false
  125. document.onmousemove = null
  126. document.onmouseup = null
  127. }
  128. },
  129. //选择画布工具
  130. chooseBoardTool(tool){
  131. //在没有笔迹时,点击这两个按钮不生效
  132. if(['broom','eraser'].includes(tool)&&!this.hasPainted){
  133. return
  134. }
  135. if(tool==='palette') return
  136. this.activeTool = tool
  137. if(tool==='broom'){
  138. this.hasPainted&&this.clearCanvas()
  139. //区分清除画布的触发方式,如果是点击画布工具触发则需要将画布页面数据也一起清除
  140. this.$emit('salfClear')
  141. this.activeTool = 'brush'
  142. }
  143. },
  144. //是否显示颜色选择器
  145. checkPalettle(hoverTool,type){
  146. if(type==='enter'&&hoverTool!=='palette'){
  147. this.isPalettleShow = false
  148. }
  149. if(type==='enter'&&hoverTool==='palette'){
  150. this.isPalettleShow = true
  151. }
  152. },
  153. //点击调色盘-切换画笔颜色
  154. changeBrushColor(color){
  155. if(!this.ctx) return
  156. this.ctx.strokeStyle = color
  157. this.choosedColor = color
  158. this.isPalettleShow = false
  159. this.activeTool = 'brush'
  160. },
  161. //清空画布
  162. clearCanvas(){
  163. if (!this.ctx) return
  164. this.ctx.clearRect(0, 0, this.canvasWidth, this.canvasHeight)
  165. this.hasPainted = false
  166. },
  167. //保存画布
  168. saveCanvas(){
  169. if(!this.ctx) return
  170. return this.ctx.getImageData(0,0,this.canvasWidth,this.canvasHeight)
  171. },
  172. //重新绘制画布
  173. reDrawCanvas(imgData){
  174. if(!this.ctx) return
  175. this.ctx.putImageData(imgData,0,0)
  176. },
  177. //canvas绘制相关
  178. startDraw(e){
  179. //console.log('click',e)
  180. this.isStartDraw = true
  181. this.isMouseDown = false
  182. this.movePoint.startX = e.clientX
  183. this.movePoint.startY = e.clientY
  184. this.mouse.x = e.clientX
  185. this.mouse.y = e.clientY
  186. document.onmousemove = (event)=>{
  187. this.moveBrush(event)
  188. }
  189. document.onmouseup = (e)=>{
  190. this.endDraw(e)
  191. }
  192. },
  193. endDraw(e){
  194. this.isStartDraw = false
  195. //如果没有移动过,只是鼠标按下抬起,则在按下的地方绘制/擦除
  196. if(!this.isMoved&&e.which===1){
  197. this.movePoint.endX = this.movePoint.startX
  198. this.movePoint.endY = this.movePoint.startY
  199. this.activeTool==='brush'&&this.draw()
  200. this.activeTool==='eraser'&&this.erase()
  201. }
  202. this.isMoved = false
  203. if(this.isCanvasEmpty()){
  204. this.chooseBoardTool('broom')
  205. }
  206. document.onmousemove = null
  207. document.onmouseup = null
  208. },
  209. moveBrush(e){
  210. if(!this.isStartDraw) return
  211. this.movePoint.endX = e.clientX
  212. this.movePoint.endY = e.clientY
  213. this.activeTool==='brush'&&e.which===1&&this.draw()
  214. this.activeTool==='eraser'&&e.which===1&&this.erase()
  215. this.movePoint.startX = e.clientX
  216. this.movePoint.startY = e.clientY
  217. if(!this.isMoved) this.isMoved = true
  218. },
  219. //绘制笔迹
  220. draw(){
  221. if(!this.ctx) return
  222. const {startX,startY,endX,endY} = this.movePoint
  223. this.ctx.beginPath()
  224. this.ctx.moveTo(startX,startY)
  225. this.ctx.lineTo(endX,endY)
  226. this.ctx.stroke()
  227. this.ctx.closePath()
  228. this.hasPainted = true
  229. },
  230. //擦除笔迹
  231. erase(){
  232. if(!this.ctx) return
  233. const {startX,startY,endX,endY} = this.movePoint
  234. const radius = 18 //橡皮擦半径
  235. //让橡皮擦笔迹平滑 参考https://github.com/pipipi-pikachu/PPTist/src/components/WritingBoard.vue
  236. const sinRadius = radius * Math.sin(Math.atan((endY - startY) / (endX - startX)))
  237. const cosRadius = radius * Math.cos(Math.atan((endY - startY) / (endX - startX)))
  238. const rectPoint1 = [startX + sinRadius, startY - cosRadius]
  239. const rectPoint2 = [startX - sinRadius, startY + cosRadius]
  240. const rectPoint3 = [endX + sinRadius, endY - cosRadius]
  241. const rectPoint4 = [endX - sinRadius, endY + cosRadius]
  242. this.ctx.save() //保存画布区域
  243. this.ctx.beginPath()
  244. this.ctx.arc(endX, endY, radius, 0, Math.PI * 2)
  245. this.ctx.clip()
  246. this.ctx.clearRect(0, 0, this.canvasWidth, this.canvasHeight)
  247. this.ctx.restore() //恢复画布区域,clip的部分
  248. //修正橡皮擦的笔迹
  249. this.ctx.save()
  250. this.ctx.beginPath()
  251. this.ctx.moveTo(...rectPoint1)
  252. this.ctx.lineTo(...rectPoint3)
  253. this.ctx.lineTo(...rectPoint4)
  254. this.ctx.lineTo(...rectPoint2)
  255. this.ctx.closePath()
  256. this.ctx.clip()
  257. this.ctx.clearRect(0, 0, this.canvasWidth, this.canvasHeight)
  258. this.ctx.restore()
  259. },
  260. isCanvasEmpty(){
  261. const blank = document.createElement('canvas')
  262. blank.width = this.canvasWidth
  263. blank.height = this.canvasHeight
  264. return this.$refs.canvasRef.toDataURL() == blank.toDataURL()
  265. }
  266. },
  267. mounted(){
  268. this.initCanvas()
  269. }
  270. };
  271. </script>
  272. <style scoped lang="scss">
  273. .drawing-board-wrap{
  274. position: absolute;
  275. left:0;
  276. right:0;
  277. top:0;
  278. bottom:0;
  279. z-index: 999;
  280. /* background-color: pink; */
  281. canvas{
  282. position: absolute;
  283. left:0;
  284. top:0;
  285. cursor: none;
  286. }
  287. .tools{
  288. position:absolute;
  289. border: 1px solid #999999;
  290. background-color: white;
  291. width:56px;
  292. height:242px;
  293. padding:8px;
  294. border-radius: 8px;
  295. display: flex;
  296. flex-direction: column;
  297. justify-content: space-around;
  298. .move-area{
  299. position:absolute;
  300. top:0;
  301. left:0;
  302. right:0;
  303. height:8px;
  304. cursor:move;
  305. .move-line{
  306. width:23px;
  307. height:2px;
  308. background-color: #333333;
  309. cursor:move;
  310. }
  311. }
  312. .tool-item{
  313. width:40px;
  314. height:40px;
  315. cursor: pointer;
  316. position:relative;
  317. overflow: hidden;
  318. img{
  319. display: block;
  320. width: 22px;
  321. height: 22px;
  322. }
  323. &:hover,&.active{
  324. background-color: #E9F4FF;
  325. }
  326. }
  327. .color-box{
  328. position:absolute;
  329. width:230px;
  330. height:30px;
  331. background-color: white;
  332. border:1px solid #999999;
  333. padding:5px;
  334. border-radius: 2px;
  335. display: flex;
  336. justify-content: space-around;
  337. .color-item{
  338. display: inline-block;
  339. width:17px;
  340. height: 17px;
  341. border: 1px solid #DCDFE6;
  342. }
  343. }
  344. }
  345. .pen,.eraser{
  346. position:absolute;
  347. pointer-events: none;
  348. z-index: 3;
  349. overflow: hidden;
  350. user-select: none;
  351. img{
  352. display: block;
  353. width: 22px;
  354. height: 22px;
  355. }
  356. }
  357. .eraser {
  358. display: flex;
  359. justify-content: center;
  360. align-items: center;
  361. border-radius: 50%;
  362. border: 4px solid rgba(85, 85, 85, 0.685);
  363. }
  364. }
  365. </style>