import { Curve } from '../curve'
import { Point, PointOptions } from '../point'
import { Segment, SegmentOptions } from './segment'

export class CurveTo extends Segment {
  static create(
    x1: number,
    y1: number,
    x2: number,
    y2: number,
    x: number,
    y: number,
  ): CurveTo
  static create(
    x1: number,
    y1: number,
    x2: number,
    y2: number,
    x: number,
    y: number,
    ...coords: number[]
  ): CurveTo[]
  static create(c1: PointOptions, c2: PointOptions, p: PointOptions): CurveTo
  static create(
    c1: PointOptions,
    c2: PointOptions,
    p: PointOptions,
    ...points: PointOptions[]
  ): CurveTo[]
  static create(...args: any[]): CurveTo | CurveTo[] {
    const len = args.length
    const arg0 = args[0]

    // curve provided
    if (Curve.isCurve(arg0)) {
      return new CurveTo(arg0)
    }

    // points provided
    if (Point.isPointLike(arg0)) {
      if (len === 3) {
        return new CurveTo(args[0], args[1], args[2])
      }

      // this is a poly-bezier segment
      const segments: CurveTo[] = []
      for (let i = 0; i < len; i += 3) {
        segments.push(new CurveTo(args[i], args[i + 1], args[i + 2]))
      }
      return segments
    }

    // coordinates provided
    if (len === 6) {
      return new CurveTo(args[0], args[1], args[2], args[3], args[4], args[5])
    }

    // this is a poly-bezier segment
    const segments: CurveTo[] = []
    for (let i = 0; i < len; i += 6) {
      segments.push(
        new CurveTo(
          args[i],
          args[i + 1],
          args[i + 2],
          args[i + 3],
          args[i + 4],
          args[i + 5],
        ),
      )
    }
    return segments
  }
  controlPoint1: Point
  controlPoint2: Point

  constructor(curve: Curve)
  constructor(
    x1: number,
    y1: number,
    x2: number,
    y2: number,
    x: number,
    y: number,
  )
  constructor(p1: PointOptions, p2: PointOptions, p3: PointOptions)
  constructor(
    arg0: number | Curve | PointOptions,
    arg1?: number | PointOptions,
    arg2?: number | PointOptions,
    arg3?: number,
    arg4?: number,
    arg5?: number,
  ) {
    super()

    if (Curve.isCurve(arg0)) {
      this.controlPoint1 = arg0.controlPoint1.clone().round(2)
      this.controlPoint2 = arg0.controlPoint2.clone().round(2)
      this.endPoint = arg0.end.clone().round(2)
    } else if (typeof arg0 === 'number') {
      this.controlPoint1 = new Point(arg0, arg1 as number).round(2)
      this.controlPoint2 = new Point(arg2 as number, arg3).round(2)
      this.endPoint = new Point(arg4, arg5).round(2)
    } else {
      this.controlPoint1 = Point.create(arg0).round(2)
      this.controlPoint2 = Point.create(arg1).round(2)
      this.endPoint = Point.create(arg2).round(2)
    }
  }

  get type() {
    return 'C'
  }

  get curve() {
    return new Curve(
      this.start,
      this.controlPoint1,
      this.controlPoint2,
      this.end,
    )
  }

  bbox() {
    return this.curve.bbox()
  }

  closestPoint(p: PointOptions) {
    return this.curve.closestPoint(p)
  }

  closestPointLength(p: PointOptions) {
    return this.curve.closestPointLength(p)
  }

  closestPointNormalizedLength(p: PointOptions) {
    return this.curve.closestPointNormalizedLength(p)
  }

  closestPointTangent(p: PointOptions) {
    return this.curve.closestPointTangent(p)
  }

  length() {
    return this.curve.length()
  }

  divideAt(ratio: number, options: SegmentOptions = {}): [Segment, Segment] {
    // TODO: fix options
    const divided = this.curve.divideAt(ratio, options as any)
    return [new CurveTo(divided[0]), new CurveTo(divided[1])]
  }

  divideAtLength(
    length: number,
    options: SegmentOptions = {},
  ): [Segment, Segment] {
    // TODO: fix options
    const divided = this.curve.divideAtLength(length, options as any)
    return [new CurveTo(divided[0]), new CurveTo(divided[1])]
  }

  divideAtT(t: number): [Segment, Segment] {
    const divided = this.curve.divideAtT(t)
    return [new CurveTo(divided[0]), new CurveTo(divided[1])]
  }

  getSubdivisions() {
    return []
  }

  pointAt(ratio: number) {
    return this.curve.pointAt(ratio)
  }

  pointAtLength(length: number) {
    return this.curve.pointAtLength(length)
  }

  tangentAt(ratio: number) {
    return this.curve.tangentAt(ratio)
  }

  tangentAtLength(length: number) {
    return this.curve.tangentAtLength(length)
  }

  isDifferentiable() {
    if (!this.previousSegment) {
      return false
    }

    const start = this.start
    const control1 = this.controlPoint1
    const control2 = this.controlPoint2
    const end = this.end

    return !(
      start.equals(control1) &&
      control1.equals(control2) &&
      control2.equals(end)
    )
  }

  scale(sx: number, sy: number, origin?: PointOptions) {
    this.controlPoint1.scale(sx, sy, origin)
    this.controlPoint2.scale(sx, sy, origin)
    this.end.scale(sx, sy, origin)
    return this
  }

  rotate(angle: number, origin?: PointOptions) {
    this.controlPoint1.rotate(angle, origin)
    this.controlPoint2.rotate(angle, origin)
    this.end.rotate(angle, origin)
    return this
  }

  translate(tx: number, ty: number): this
  translate(p: PointOptions): this
  translate(tx: number | PointOptions, ty?: number): this {
    if (typeof tx === 'number') {
      this.controlPoint1.translate(tx, ty as number)
      this.controlPoint2.translate(tx, ty as number)
      this.end.translate(tx, ty as number)
    } else {
      this.controlPoint1.translate(tx)
      this.controlPoint2.translate(tx)
      this.end.translate(tx)
    }

    return this
  }

  equals(s: Segment) {
    return (
      this.start.equals(s.start) &&
      this.end.equals(s.end) &&
      this.controlPoint1.equals((s as CurveTo).controlPoint1) &&
      this.controlPoint2.equals((s as CurveTo).controlPoint2)
    )
  }

  clone() {
    return new CurveTo(this.controlPoint1, this.controlPoint2, this.end)
  }

  toJSON() {
    return {
      type: this.type,
      start: this.start.toJSON(),
      controlPoint1: this.controlPoint1.toJSON(),
      controlPoint2: this.controlPoint2.toJSON(),
      end: this.end.toJSON(),
    }
  }

  serialize() {
    const c1 = this.controlPoint1
    const c2 = this.controlPoint2
    const end = this.end
    return [this.type, c1.x, c1.y, c2.x, c2.y, end.x, end.y].join(' ')
  }
}
