import Point from './Point';

const shadowLength = 2000; // TODO: should be calculated

export default class Box {
    public position: Point;
    private angle: number;
    public color: string;
    private readonly shadowColor: string;

    private _size: number;
    private _halfSize: number;

    constructor(size: number, position: Point, startRotate: number, color: string, shadowColor: string) {
        this.size = size;
        this.position = position;
        this.color = color;
        this.angle = startRotate;
        this.shadowColor = shadowColor;
    }

    public get size(): number {
        return this._size;
    }

    public set size(size: number) {
        this._size = size;
        this._halfSize = Math.floor(size / 2);
    }

    public get halfSize(): number {
        return this._halfSize;
    }

    get points(): { p1: Point; p2: Point; p3: Point; p4: Point } {
        const full = (Math.PI * 2) / 4;

        const p1: Point = {
            x: this.position.x + this._halfSize * Math.sin(this.angle),
            y: this.position.y + this._halfSize * Math.cos(this.angle),
        };

        const p2: Point = {
            x: this.position.x + this._halfSize * Math.sin(this.angle + full),
            y: this.position.y + this._halfSize * Math.cos(this.angle + full),
        };

        const p3: Point = {
            x: this.position.x + this._halfSize * Math.sin(this.angle + full * 2),
            y: this.position.y + this._halfSize * Math.cos(this.angle + full * 2),
        };

        const p4: Point = {
            x: this.position.x + this._halfSize * Math.sin(this.angle + full * 3),
            y: this.position.y + this._halfSize * Math.cos(this.angle + full * 3),
        };

        return { p1, p2, p3, p4 };
    }

    rotate(): void {
        const speed = (60 - this._halfSize) / 20;
        this.angle += speed * 0.002;
        this.position.x += speed;
        this.position.y += speed;
    }

    draw(ctx: CanvasRenderingContext2D): void {
        const { points } = this;
        ctx.beginPath();
        ctx.moveTo(points.p1.x, points.p1.y);
        ctx.lineTo(points.p2.x, points.p2.y);
        ctx.lineTo(points.p3.x, points.p3.y);
        ctx.lineTo(points.p4.x, points.p4.y);
        ctx.fillStyle = this.color;
        ctx.fill();
    }

    drawShadow(ctx: CanvasRenderingContext2D, light: Point): void {
        const boxPoints = this.points;
        const points: Array<{
            startX: number;
            startY: number;
            endX: number;
            endY: number;
        }> = [];

        Object.values(boxPoints).forEach((point) => {
            const angle = Math.atan2(light.y - point.y, light.x - point.x);
            const endX = point.x + shadowLength * Math.sin(-angle - Math.PI / 2);
            const endY = point.y + shadowLength * Math.cos(-angle - Math.PI / 2);
            points.push({
                startX: point.x,
                startY: point.y,
                endX,
                endY,
            });
        });

        for (let i = points.length - 1; i >= 0; i--) {
            const n = i === 3 ? 0 : i + 1;
            ctx.beginPath();
            ctx.moveTo(points[i].startX, points[i].startY);
            ctx.lineTo(points[n].startX, points[n].startY);
            ctx.lineTo(points[n].endX, points[n].endY);
            ctx.lineTo(points[i].endX, points[i].endY);
            ctx.fillStyle = this.shadowColor;
            ctx.fill();
        }
    }
}