|
@@ -17,40 +17,30 @@ export class DrawingComponent implements OnInit, AfterViewInit {
|
|
|
private lastX = 0;
|
|
|
private lastY = 0;
|
|
|
|
|
|
- currentColor = '#000000';
|
|
|
- currentWidth = 5;
|
|
|
-
|
|
|
- customColor = '#000000';
|
|
|
- showGrid = false;
|
|
|
- opacity = 1;
|
|
|
- smoothing = true;
|
|
|
- showLayers = false;
|
|
|
- currentLayer = 'layer1';
|
|
|
-
|
|
|
- layers = [
|
|
|
- { id: 'layer1', name: '图层 1', visible: true },
|
|
|
- { id: 'layer2', name: '图层 2', visible: true }
|
|
|
- ];
|
|
|
-
|
|
|
- history: ImageData[] = [];
|
|
|
- historyIndex = -1;
|
|
|
-
|
|
|
+ // 简化工具列表
|
|
|
tools = [
|
|
|
{ id: 'brush', icon: 'brush', name: '画笔' },
|
|
|
{ id: 'eraser', icon: 'trash', name: '橡皮擦' },
|
|
|
- { id: 'line', icon: 'remove', name: '直线' },
|
|
|
- { id: 'rect', icon: 'square-outline', name: '矩形' },
|
|
|
- { id: 'circle', icon: 'ellipse-outline', name: '圆形' },
|
|
|
{ id: 'clear', icon: 'refresh', name: '清空' }
|
|
|
];
|
|
|
|
|
|
+ // 常用颜色
|
|
|
colors = [
|
|
|
'#000000', '#ffffff', '#ff0000', '#00ff00', '#0000ff',
|
|
|
- '#ffff00', '#00ffff', '#ff00ff', '#808080', '#800000'
|
|
|
+ '#ffff00', '#00ffff', '#ff00ff'
|
|
|
];
|
|
|
|
|
|
+ currentColor = '#000000';
|
|
|
+ currentWidth = 5;
|
|
|
+ customColor = '#000000';
|
|
|
selectedTool = 'brush';
|
|
|
|
|
|
+ // 撤销/重做功能
|
|
|
+ history: ImageData[] = [];
|
|
|
+ historyIndex = -1;
|
|
|
+ canUndo = false;
|
|
|
+ canRedo = false;
|
|
|
+
|
|
|
constructor() { }
|
|
|
|
|
|
ngOnInit() { }
|
|
@@ -60,6 +50,7 @@ export class DrawingComponent implements OnInit, AfterViewInit {
|
|
|
this.ctx = canvas.getContext('2d')!;
|
|
|
this.resizeCanvas();
|
|
|
this.initializeCanvas();
|
|
|
+ this.saveToHistory(); // 保存初始状态
|
|
|
|
|
|
window.addEventListener('resize', () => {
|
|
|
this.resizeCanvas();
|
|
@@ -69,12 +60,16 @@ export class DrawingComponent implements OnInit, AfterViewInit {
|
|
|
private resizeCanvas() {
|
|
|
const canvas = this.canvasRef.nativeElement;
|
|
|
canvas.width = window.innerWidth;
|
|
|
- canvas.height = window.innerHeight - 200; // 减去顶部工具栏的高度
|
|
|
+ canvas.height = window.innerHeight - 200;
|
|
|
+ this.ctx.fillStyle = '#ffffff';
|
|
|
+ this.ctx.fillRect(0, 0, canvas.width, canvas.height);
|
|
|
}
|
|
|
|
|
|
private initializeCanvas() {
|
|
|
this.ctx.fillStyle = '#ffffff';
|
|
|
this.ctx.fillRect(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height);
|
|
|
+ this.ctx.lineCap = 'round';
|
|
|
+ this.ctx.lineJoin = 'round';
|
|
|
}
|
|
|
|
|
|
startDrawing(event: TouchEvent | MouseEvent) {
|
|
@@ -89,93 +84,17 @@ export class DrawingComponent implements OnInit, AfterViewInit {
|
|
|
event.preventDefault();
|
|
|
|
|
|
const pos = this.getPosition(event);
|
|
|
-
|
|
|
- switch (this.selectedTool) {
|
|
|
- case 'brush':
|
|
|
- case 'eraser':
|
|
|
- this.drawFreehand(pos);
|
|
|
- break;
|
|
|
- case 'line':
|
|
|
- this.drawLine(pos);
|
|
|
- break;
|
|
|
- case 'rect':
|
|
|
- this.drawRect(pos);
|
|
|
- break;
|
|
|
- case 'circle':
|
|
|
- this.drawCircle(pos);
|
|
|
- break;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- private drawFreehand(pos: { x: number, y: number }) {
|
|
|
this.ctx.beginPath();
|
|
|
this.ctx.moveTo(this.lastX, this.lastY);
|
|
|
this.ctx.lineTo(pos.x, pos.y);
|
|
|
this.ctx.strokeStyle = this.selectedTool === 'eraser' ? '#ffffff' : this.currentColor;
|
|
|
this.ctx.lineWidth = this.currentWidth;
|
|
|
- this.ctx.lineCap = 'round';
|
|
|
this.ctx.stroke();
|
|
|
|
|
|
this.lastX = pos.x;
|
|
|
this.lastY = pos.y;
|
|
|
}
|
|
|
|
|
|
- private drawLine(pos: { x: number, y: number }) {
|
|
|
- const tempCanvas = document.createElement('canvas');
|
|
|
- tempCanvas.width = this.canvasRef.nativeElement.width;
|
|
|
- tempCanvas.height = this.canvasRef.nativeElement.height;
|
|
|
- const tempCtx = tempCanvas.getContext('2d')!;
|
|
|
-
|
|
|
- tempCtx.drawImage(this.canvasRef.nativeElement, 0, 0);
|
|
|
- tempCtx.beginPath();
|
|
|
- tempCtx.moveTo(this.lastX, this.lastY);
|
|
|
- tempCtx.lineTo(pos.x, pos.y);
|
|
|
- tempCtx.strokeStyle = this.currentColor;
|
|
|
- tempCtx.lineWidth = this.currentWidth;
|
|
|
- tempCtx.stroke();
|
|
|
-
|
|
|
- this.ctx.clearRect(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height);
|
|
|
- this.ctx.drawImage(tempCanvas, 0, 0);
|
|
|
- }
|
|
|
-
|
|
|
- private drawRect(pos: { x: number, y: number }) {
|
|
|
- const tempCanvas = document.createElement('canvas');
|
|
|
- tempCanvas.width = this.canvasRef.nativeElement.width;
|
|
|
- tempCanvas.height = this.canvasRef.nativeElement.height;
|
|
|
- const tempCtx = tempCanvas.getContext('2d')!;
|
|
|
-
|
|
|
- tempCtx.drawImage(this.canvasRef.nativeElement, 0, 0);
|
|
|
- tempCtx.beginPath();
|
|
|
- tempCtx.rect(this.lastX, this.lastY, pos.x - this.lastX, pos.y - this.lastY);
|
|
|
- tempCtx.strokeStyle = this.currentColor;
|
|
|
- tempCtx.lineWidth = this.currentWidth;
|
|
|
- tempCtx.stroke();
|
|
|
-
|
|
|
- this.ctx.clearRect(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height);
|
|
|
- this.ctx.drawImage(tempCanvas, 0, 0);
|
|
|
- }
|
|
|
-
|
|
|
- private drawCircle(pos: { x: number, y: number }) {
|
|
|
- const tempCanvas = document.createElement('canvas');
|
|
|
- tempCanvas.width = this.canvasRef.nativeElement.width;
|
|
|
- tempCanvas.height = this.canvasRef.nativeElement.height;
|
|
|
- const tempCtx = tempCanvas.getContext('2d')!;
|
|
|
-
|
|
|
- const radius = Math.sqrt(
|
|
|
- Math.pow(pos.x - this.lastX, 2) + Math.pow(pos.y - this.lastY, 2)
|
|
|
- );
|
|
|
-
|
|
|
- tempCtx.drawImage(this.canvasRef.nativeElement, 0, 0);
|
|
|
- tempCtx.beginPath();
|
|
|
- tempCtx.arc(this.lastX, this.lastY, radius, 0, Math.PI * 2);
|
|
|
- tempCtx.strokeStyle = this.currentColor;
|
|
|
- tempCtx.lineWidth = this.currentWidth;
|
|
|
- tempCtx.stroke();
|
|
|
-
|
|
|
- this.ctx.clearRect(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height);
|
|
|
- this.ctx.drawImage(tempCanvas, 0, 0);
|
|
|
- }
|
|
|
-
|
|
|
stopDrawing() {
|
|
|
if (this.isDrawing) {
|
|
|
this.isDrawing = false;
|
|
@@ -214,11 +133,12 @@ export class DrawingComponent implements OnInit, AfterViewInit {
|
|
|
clearCanvas() {
|
|
|
this.ctx.fillStyle = '#ffffff';
|
|
|
this.ctx.fillRect(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height);
|
|
|
+ this.saveToHistory();
|
|
|
}
|
|
|
|
|
|
async saveDrawing() {
|
|
|
- const canvas = this.canvasRef.nativeElement;
|
|
|
try {
|
|
|
+ const canvas = this.canvasRef.nativeElement;
|
|
|
const dataUrl = canvas.toDataURL('image/png');
|
|
|
const response = await fetch(dataUrl);
|
|
|
const blob = await response.blob();
|
|
@@ -236,71 +156,42 @@ export class DrawingComponent implements OnInit, AfterViewInit {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- private async showToast(message: string) {
|
|
|
- const toast = document.createElement('ion-toast');
|
|
|
- toast.message = message;
|
|
|
- toast.duration = 2000;
|
|
|
- toast.position = 'top';
|
|
|
- document.body.appendChild(toast);
|
|
|
- await toast.present();
|
|
|
+ private saveToHistory() {
|
|
|
+ this.historyIndex++;
|
|
|
+ this.history = this.history.slice(0, this.historyIndex);
|
|
|
+ this.history.push(this.ctx.getImageData(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height));
|
|
|
+ this.updateUndoRedoState();
|
|
|
+ }
|
|
|
+
|
|
|
+ private updateUndoRedoState() {
|
|
|
+ this.canUndo = this.historyIndex > 0;
|
|
|
+ this.canRedo = this.historyIndex < this.history.length - 1;
|
|
|
}
|
|
|
|
|
|
undo() {
|
|
|
- if (this.historyIndex > 0) {
|
|
|
+ if (this.canUndo) {
|
|
|
this.historyIndex--;
|
|
|
const imageData = this.history[this.historyIndex];
|
|
|
this.ctx.putImageData(imageData, 0, 0);
|
|
|
+ this.updateUndoRedoState();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
redo() {
|
|
|
- if (this.historyIndex < this.history.length - 1) {
|
|
|
+ if (this.canRedo) {
|
|
|
this.historyIndex++;
|
|
|
const imageData = this.history[this.historyIndex];
|
|
|
this.ctx.putImageData(imageData, 0, 0);
|
|
|
+ this.updateUndoRedoState();
|
|
|
}
|
|
|
}
|
|
|
|
|
|
- saveToHistory() {
|
|
|
- this.historyIndex++;
|
|
|
- this.history = this.history.slice(0, this.historyIndex);
|
|
|
- this.history.push(this.ctx.getImageData(0, 0, this.canvasRef.nativeElement.width, this.canvasRef.nativeElement.height));
|
|
|
- }
|
|
|
-
|
|
|
- toggleGrid() {
|
|
|
- this.showGrid = !this.showGrid;
|
|
|
- }
|
|
|
-
|
|
|
- toggleOpacity() {
|
|
|
- this.opacity = this.opacity === 1 ? 0.5 : 1;
|
|
|
- this.ctx.globalAlpha = this.opacity;
|
|
|
- }
|
|
|
-
|
|
|
- toggleSmooth() {
|
|
|
- this.smoothing = !this.smoothing;
|
|
|
- this.ctx.imageSmoothingEnabled = this.smoothing;
|
|
|
- }
|
|
|
-
|
|
|
- toggleLayers() {
|
|
|
- this.showLayers = !this.showLayers;
|
|
|
- }
|
|
|
-
|
|
|
- addLayer() {
|
|
|
- const newId = `layer${this.layers.length + 1}`;
|
|
|
- this.layers.push({
|
|
|
- id: newId,
|
|
|
- name: `图层 ${this.layers.length + 1}`,
|
|
|
- visible: true
|
|
|
- });
|
|
|
- this.currentLayer = newId;
|
|
|
- }
|
|
|
-
|
|
|
- deleteLayer(layerId: string) {
|
|
|
- if (this.layers.length > 1) {
|
|
|
- this.layers = this.layers.filter(layer => layer.id !== layerId);
|
|
|
- if (this.currentLayer === layerId) {
|
|
|
- this.currentLayer = this.layers[0].id;
|
|
|
- }
|
|
|
- }
|
|
|
+ private async showToast(message: string) {
|
|
|
+ const toast = document.createElement('ion-toast');
|
|
|
+ toast.message = message;
|
|
|
+ toast.duration = 2000;
|
|
|
+ toast.position = 'top';
|
|
|
+ document.body.appendChild(toast);
|
|
|
+ await toast.present();
|
|
|
}
|
|
|
}
|