From 5a6ad8fe0080f73b161c0dd8767d1939dd0eb0f5 Mon Sep 17 00:00:00 2001 From: Jan Scheidegger Date: Wed, 23 Oct 2019 14:56:48 +0200 Subject: [PATCH] feat(#66): draw multiple circles Create Interface for drawables Abstract x/y to point Drawables know how to draw themselfes Save currently Drawing element at last position in array and change on move Clear Canvas and draw whole array at once --- .../src/app/components/app-draw/app-draw.tsx | 148 +++++++++++++----- 1 file changed, 112 insertions(+), 36 deletions(-) diff --git a/remote/src/app/components/app-draw/app-draw.tsx b/remote/src/app/components/app-draw/app-draw.tsx index 955b7e7dc..0ecc0208c 100644 --- a/remote/src/app/components/app-draw/app-draw.tsx +++ b/remote/src/app/components/app-draw/app-draw.tsx @@ -1,13 +1,74 @@ -import {Component, Element, Method, Prop, State, Watch, EventEmitter, Event, h} from '@stencil/core'; +import {Component, Element, Event, EventEmitter, h, Method, Prop, State, Watch} from '@stencil/core'; import {unifyEvent} from '@deckdeckgo/utils'; - // Types -import {DeckdeckgoDrawAction, DeckdeckgoEventType, DeckdeckgoEventEmitter} from '@deckdeckgo/types'; - +import {DeckdeckgoDrawAction, DeckdeckgoEventEmitter, DeckdeckgoEventType} from '@deckdeckgo/types'; // Services import {CommunicationService} from '../../services/communication/communication.service'; +interface Point { + x: number; + y: number; +} + +interface Drawable { + draw(ctx: CanvasRenderingContext2D) +} + +class Circle implements Drawable { + + private readonly from: Point; + private readonly to: Point; + private readonly color: string; + + constructor(from: Point, to: Point, color: string) { + this.from = from; + this.to = to; + this.color = color; + } + + draw(ctx: CanvasRenderingContext2D) { + ctx.beginPath(); + + ctx.moveTo(this.from.x, this.from.y + (this.to.y - this.from.y) / 2); + ctx.bezierCurveTo(this.from.x, this.from.y, this.to.x, this.from.y, this.to.x, this.from.y + (this.to.y - this.from.y) / 2); + ctx.bezierCurveTo(this.to.x, this.to.y, this.from.x, this.to.y, this.from.x, this.from.y + (this.to.y - this.from.y) / 2); + + ctx.strokeStyle = this.color; + ctx.lineWidth = 3; + + ctx.stroke(); + ctx.closePath(); + } +} + +class Pencil implements Drawable { + + private readonly from: Point; + private readonly to: Point; + private readonly color: string; + + constructor(from: Point, to: Point, color: string) { + this.from = from; + this.to = to; + this.color = color; + } + + draw(ctx: CanvasRenderingContext2D) { + ctx.beginPath(); + + ctx.moveTo(this.from.x, this.from.y); + ctx.lineTo(this.to.x, this.to.y); + ctx.strokeStyle = this.color; + ctx.lineWidth = 3; + + ctx.stroke(); + ctx.closePath(); + + } +} + + @Component({ tag: 'app-draw', styleUrl: 'app-draw.scss', @@ -29,6 +90,8 @@ export class AppDraw { private startX: number; private startY: number; + private drawables: Drawable[] = []; + private leftOffset: number = 0; private drawEvents: boolean = false; @@ -144,12 +207,30 @@ export class AppDraw { this.startX = unifyEvent(e).clientX - this.leftOffset; this.startY = unifyEvent(e).clientY - this.heightOffset; + if (this.action === DeckdeckgoDrawAction.CIRCLE) { + this.drawables.push(new Circle({x: this.startX, y: this.startY}, { + x: this.startX, + y: this.startY + }, this.color)) + } + this.drawEvents = true; }; private endEvent = (e: MouseEvent) => { this.emit(DeckdeckgoEventType.END_DRAWING, e); + const toX: number = unifyEvent(e).clientX - this.leftOffset; + const toY: number = unifyEvent(e).clientY - this.heightOffset; + + + if (this.action === DeckdeckgoDrawAction.CIRCLE) { + this.drawables[this.drawables.length - 1] = new Circle({x: this.startX, y: this.startY}, { + x: toX, + y: toY + }, this.color); + } + this.drawEvents = false; }; @@ -163,40 +244,29 @@ export class AppDraw { const toX: number = unifyEvent(e).clientX - this.leftOffset; const toY: number = unifyEvent(e).clientY - this.heightOffset; - this.draw(toX, toY); - }; - - private draw(toX: number, toY: number) { - this.ctx.beginPath(); + if (this.action === DeckdeckgoDrawAction.PENCIL) { + this.drawables.push(new Pencil({x: this.startX, y: this.startY}, {x: toX, y: toY}, this.color)); + this.startX = toX; + this.startY = toY; + } if (this.action === DeckdeckgoDrawAction.CIRCLE) { - this.drawCircle(toX, toY); - } else { - this.drawPencil(toX, toY); + this.drawables[this.drawables.length - 1] = new Circle({x: this.startX, y: this.startY}, { + x: toX, + y: toY + }, this.color); } + this.draw(); + }; - this.ctx.strokeStyle = this.color; - this.ctx.lineWidth = 3; - - this.ctx.stroke(); - this.ctx.closePath(); - } - - private drawPencil(toX: number, toY: number) { - this.ctx.moveTo(this.startX, this.startY); - this.ctx.lineTo(toX, toY); - - this.startX = toX; - this.startY = toY; - } - - private drawCircle(toX: number, toY: number) { + private draw() { this.ctx.clearRect(-1 * this.leftOffset, 0, this.width, this.height); - this.ctx.moveTo(this.startX, this.startY + (toY - this.startY) / 2); - this.ctx.bezierCurveTo(this.startX, this.startY, toX, this.startY, toX, this.startY + (toY - this.startY) / 2); - this.ctx.bezierCurveTo(toX, toY, this.startX, toY, this.startX, this.startY + (toY - this.startY) / 2); + for (const drawable of this.drawables) { + drawable.draw(this.ctx); + } } + private switchTool(e: UIEvent, action: DeckdeckgoDrawAction) { e.stopPropagation(); @@ -238,7 +308,10 @@ export class AppDraw { return new Promise((resolve) => { e.stopPropagation(); - this.communicationService.emit({type: DeckdeckgoEventType.CLEAR_SLIDE, emitter: DeckdeckgoEventEmitter.APP}); + this.communicationService.emit({ + type: DeckdeckgoEventType.CLEAR_SLIDE, + emitter: DeckdeckgoEventEmitter.APP + }); this.ctx.beginPath(); this.ctx.clearRect(-1 * this.leftOffset, 0, this.width, this.height); @@ -252,7 +325,7 @@ export class AppDraw { render() { const styleColorPicker = { - color: this.color === 'red' ? 'black' : 'red' + color: this.color === 'red' ? 'black' : 'red' }; return ([ @@ -265,7 +338,8 @@ export class AppDraw { {this.renderPencilRubber()} - this.switchColor(e)}> + this.switchColor(e)}> this.clear(e)}> @@ -279,13 +353,15 @@ export class AppDraw { private renderPencilRubber() { if (this.action !== DeckdeckgoDrawAction.PENCIL) { return ( - this.switchTool(e, DeckdeckgoDrawAction.PENCIL)}> + this.switchTool(e, DeckdeckgoDrawAction.PENCIL)}> ); } else { return ( - this.switchTool(e, DeckdeckgoDrawAction.CIRCLE)}> + this.switchTool(e, DeckdeckgoDrawAction.CIRCLE)}> );