import * as PropertySymbol from '../PropertySymbol.js';
import SVGLengthTypeEnum from './SVGLengthTypeEnum.js';
import BrowserWindow from '../window/BrowserWindow.js';

const ATTRIBUTE_REGEXP = /^(\d+|\d+\.\d+)(px|em|ex|cm|mm|in|pt|pc|%|)$/;

/**
 * SVG length.
 *
 * @see https://developer.mozilla.org/en-US/docs/Web/API/SVGLength
 */
export default class SVGLength {
	// Static properties
	public static SVG_LENGTHTYPE_UNKNOWN = SVGLengthTypeEnum.unknown;
	public static SVG_LENGTHTYPE_NUMBER = SVGLengthTypeEnum.number;
	public static SVG_LENGTHTYPE_PERCENTAGE = SVGLengthTypeEnum.percentage;
	public static SVG_LENGTHTYPE_EMS = SVGLengthTypeEnum.ems;
	public static SVG_LENGTHTYPE_EXS = SVGLengthTypeEnum.exs;
	public static SVG_LENGTHTYPE_PX = SVGLengthTypeEnum.px;
	public static SVG_LENGTHTYPE_CM = SVGLengthTypeEnum.cm;
	public static SVG_LENGTHTYPE_MM = SVGLengthTypeEnum.mm;
	public static SVG_LENGTHTYPE_IN = SVGLengthTypeEnum.in;
	public static SVG_LENGTHTYPE_PT = SVGLengthTypeEnum.pt;
	public static SVG_LENGTHTYPE_PC = SVGLengthTypeEnum.pc;

	// Internal properties
	public [PropertySymbol.window]: BrowserWindow;
	public [PropertySymbol.getAttribute]: () => string | null = null;
	public [PropertySymbol.setAttribute]: (value: string) => void | null = null;
	public [PropertySymbol.attributeValue]: string | null = null;
	public [PropertySymbol.readOnly]: boolean = false;

	/**
	 * Constructor.
	 *
	 * @param illegalConstructorSymbol Illegal constructor symbol.
	 * @param window Window.
	 * @param [options] Options.
	 * @param [options.readOnly] Read only.
	 * @param [options.getAttribute] Get attribute.
	 * @param [options.setAttribute] Set attribute.
	 */
	constructor(
		illegalConstructorSymbol: symbol,
		window: BrowserWindow,
		options?: {
			readOnly?: boolean;
			getAttribute?: () => string | null;
			setAttribute?: (value: string) => void;
		}
	) {
		if (illegalConstructorSymbol !== PropertySymbol.illegalConstructor) {
			throw new TypeError('Illegal constructor');
		}

		this[PropertySymbol.window] = window;

		if (options) {
			this[PropertySymbol.readOnly] = !!options.readOnly;
			this[PropertySymbol.getAttribute] = options.getAttribute || null;
			this[PropertySymbol.setAttribute] = options.setAttribute || null;
		}
	}

	/**
	 * Returns unit type.
	 *
	 * @returns Unit type.
	 */
	public get unitType(): SVGLengthTypeEnum {
		const attributeValue = this[PropertySymbol.getAttribute]
			? this[PropertySymbol.getAttribute]() || ''
			: this[PropertySymbol.attributeValue] || '';
		const match = attributeValue.match(ATTRIBUTE_REGEXP);

		if (!match) {
			return SVGLengthTypeEnum.unknown;
		}

		if (isNaN(parseFloat(match[1]))) {
			return SVGLengthTypeEnum.unknown;
		}

		switch (match[2]) {
			case '':
				return SVGLengthTypeEnum.number;
			case 'px':
				return SVGLengthTypeEnum.px;
			case 'cm':
				return SVGLengthTypeEnum.cm;
			case 'mm':
				return SVGLengthTypeEnum.mm;
			case 'in':
				return SVGLengthTypeEnum.in;
			case 'pt':
				return SVGLengthTypeEnum.pt;
			case 'pc':
				return SVGLengthTypeEnum.pc;
			case 'em':
			case 'ex':
			case '%':
				throw new this[PropertySymbol.window].TypeError(
					`Failed to execute 'value' on 'SVGLength': Could not resolve relative length.`
				);
			default:
				return SVGLengthTypeEnum.unknown;
		}
	}

	/**
	 * Returns value.
	 *
	 * @returns Value.
	 */
	public get value(): number {
		const attributeValue = this[PropertySymbol.getAttribute]
			? this[PropertySymbol.getAttribute]() || ''
			: this[PropertySymbol.attributeValue] || '';
		const match = attributeValue.match(ATTRIBUTE_REGEXP);

		if (!match) {
			return 0;
		}

		const parsedValue = parseFloat(match[1]);

		if (isNaN(parsedValue)) {
			return 0;
		}

		switch (match[2]) {
			case '':
				return parsedValue;
			case 'px':
				return parsedValue;
			case 'cm':
				return (parsedValue / 2.54) * 96;
			case 'mm':
				return (parsedValue / 25.4) * 96;
			case 'in':
				return parsedValue * 96;
			case 'pt':
				return parsedValue * 72;
			case 'pc':
				return parsedValue * 6;
			case 'em':
			case 'ex':
			case '%':
				throw new this[PropertySymbol.window].TypeError(
					`Failed to execute 'value' on 'SVGLength': Could not resolve relative length.`
				);
			default:
				return 0;
		}
	}

	/**
	 * Sets value.
	 *
	 * @param value Value in pixels.
	 */
	public set value(value: number) {
		if (this[PropertySymbol.readOnly]) {
			throw new this[PropertySymbol.window].TypeError(
				`Failed to set the 'value' property on 'SVGLength': The object is read-only.`
			);
		}

		// Value in pixels
		value = typeof value !== 'number' ? parseFloat(String(value)) : value;
		if (isNaN(value)) {
			throw new this[PropertySymbol.window].TypeError(
				`Failed to set the 'value' property on 'SVGLength': The provided float value is non-finite.`
			);
		}
		let unitType = '';
		let valueInSpecifiedUnits = value;
		switch (this.unitType) {
			case SVGLengthTypeEnum.number:
				valueInSpecifiedUnits = value;
				unitType = '';
				break;
			case SVGLengthTypeEnum.px:
				valueInSpecifiedUnits = value;
				unitType = 'px';
				break;
			case SVGLengthTypeEnum.cm:
				valueInSpecifiedUnits = (value / 96) * 2.54;
				unitType = 'cm';
				break;
			case SVGLengthTypeEnum.mm:
				valueInSpecifiedUnits = (value / 96) * 25.4;
				unitType = 'mm';
				break;
			case SVGLengthTypeEnum.in:
				valueInSpecifiedUnits = value / 96;
				unitType = 'in';
				break;
			case SVGLengthTypeEnum.pt:
				valueInSpecifiedUnits = value / 72;
				unitType = 'pt';
				break;
			case SVGLengthTypeEnum.pc:
				valueInSpecifiedUnits = value / 6;
				unitType = 'pc';
				break;
			case SVGLengthTypeEnum.percentage:
			case SVGLengthTypeEnum.ems:
			case SVGLengthTypeEnum.exs:
				throw new this[PropertySymbol.window].TypeError(
					`Failed to set the 'value' property on 'SVGLength': Could not resolve relative length.`
				);
			default:
				break;
		}

		this[PropertySymbol.attributeValue] = String(valueInSpecifiedUnits) + unitType;

		if (this[PropertySymbol.setAttribute]) {
			this[PropertySymbol.setAttribute](this[PropertySymbol.attributeValue]);
		}
	}

	/**
	 * Returns value as string.
	 *
	 * @returns Value as string.
	 */
	public get valueAsString(): string {
		return this[PropertySymbol.getAttribute]
			? this[PropertySymbol.getAttribute]() || '0'
			: this[PropertySymbol.attributeValue] || '0';
	}

	/**
	 * Returns value in specified units.
	 *
	 * @returns Value in specified units.
	 */
	public get valueInSpecifiedUnits(): number {
		const attributeValue = this.valueAsString;
		return parseFloat(attributeValue) || 0;
	}

	/**
	 * New value specific units.
	 *
	 * @param unitType
	 * @param value
	 */
	public newValueSpecifiedUnits(unitType: number, value: number): void {
		if (this[PropertySymbol.readOnly]) {
			throw new this[PropertySymbol.window].TypeError(
				`Failed to execute 'newValueSpecifiedUnits' on 'SVGLength': The object is read-only.`
			);
		}

		if (typeof unitType !== 'number') {
			throw new this[PropertySymbol.window].TypeError(
				`Failed to execute 'newValueSpecifiedUnits' on 'SVGLength': parameter 1 ('unitType') is not of type 'number'.`
			);
		}

		value = typeof value !== 'number' ? parseFloat(String(value)) : value;

		if (isNaN(value)) {
			throw new this[PropertySymbol.window].TypeError(
				`Failed to execute 'newValueSpecifiedUnits' on 'SVGLength': The provided float value is non-finite.`
			);
		}

		let unit = '';
		switch (unitType) {
			case SVGLengthTypeEnum.number:
				unit = '';
				break;
			case SVGLengthTypeEnum.px:
				unit = 'px';
				break;
			case SVGLengthTypeEnum.cm:
				unit = 'cm';
				break;
			case SVGLengthTypeEnum.mm:
				unit = 'mm';
				break;
			case SVGLengthTypeEnum.in:
				unit = 'in';
				break;
			case SVGLengthTypeEnum.pt:
				unit = 'pt';
				break;
			case SVGLengthTypeEnum.pc:
				unit = 'pc';
				break;
			case SVGLengthTypeEnum.ems:
			case SVGLengthTypeEnum.exs:
			case SVGLengthTypeEnum.percentage:
				throw new this[PropertySymbol.window].TypeError(
					`Failed to execute 'newValueSpecifiedUnits' on 'SVGLength': Could not resolve relative length.`
				);
			default:
				break;
		}

		this[PropertySymbol.attributeValue] = String(value) + unit;

		if (this[PropertySymbol.setAttribute]) {
			this[PropertySymbol.setAttribute](this[PropertySymbol.attributeValue]);
		}
	}

	/**
	 * Convert to specific units.
	 * @param unitType
	 */
	public convertToSpecifiedUnits(unitType: SVGLengthTypeEnum): void {
		if (this[PropertySymbol.readOnly]) {
			throw new this[PropertySymbol.window].TypeError(
				`Failed to execute 'convertToSpecifiedUnits' on 'SVGLength': The object is read-only.`
			);
		}

		if (typeof unitType !== 'number') {
			throw new this[PropertySymbol.window].TypeError(
				`Failed to execute 'convertToSpecifiedUnits' on 'SVGLength': parameter 1 ('unitType') is not of type 'number'.`
			);
		}

		let value = this.value;
		let unit = '';

		switch (unitType) {
			case SVGLengthTypeEnum.number:
				unit = '';
				break;
			case SVGLengthTypeEnum.px:
				unit = 'px';
				break;
			case SVGLengthTypeEnum.cm:
				value = (value / 96) * 2.54;
				unit = 'cm';
				break;
			case SVGLengthTypeEnum.mm:
				value = (value / 96) * 25.4;
				unit = 'mm';
				break;
			case SVGLengthTypeEnum.in:
				value = value / 96;
				unit = 'in';
				break;
			case SVGLengthTypeEnum.pt:
				value = value / 72;
				unit = 'pt';
				break;
			case SVGLengthTypeEnum.pc:
				value = value / 6;
				unit = 'pc';
				break;
			case SVGLengthTypeEnum.percentage:
			case SVGLengthTypeEnum.ems:
			case SVGLengthTypeEnum.exs:
				throw new this[PropertySymbol.window].TypeError(
					`Failed to execute 'convertToSpecifiedUnits' on 'SVGLength': Could not resolve relative length.`
				);
			default:
				break;
		}

		this[PropertySymbol.attributeValue] = String(value) + unit;

		if (this[PropertySymbol.setAttribute]) {
			this[PropertySymbol.setAttribute](this[PropertySymbol.attributeValue]);
		}
	}
}