const ACTION = /^\s*(\w+ ?\w*):\s+(.+)/;
const TEXT = /^\s*\[(-?\d+), (-?\d+)\]:\s+(.+)/;
const WINDOW_SIZE = /^\[(-?\d+), (-?\d+)\]$/;
const RECTANGLE = /^\[(-?\d+), (-?\d+), (-?\d+), (-?\d+)\]\s+(.+)$/;
const REPEAT = /^Image\s+\("(.+)"\)\s+\[(-?\d+), (-?\d+)\]\s+Size\s+\((-?\d+), (-?\d+)\)\s+(.+)$/;
const PATH = /^Path \((.+)\)$/;
const VECTOR = /^Vector\(x: (-?\d+), y: (-?\d+)\)$/;
const BEZIER_CURVE = /^BezierCurve\(x0: (-?\d+), y0: (-?\d+), x1: (-?\d+), y1: (-?\d+), cx0: (-?\d+), cy0: (-?\d+), cx1: (-?\d+), cy1: (-?\d+)\)$/;
const SHAPE = /^(rgba?\((:?.+)\)) (Path .+)$/;
const CIRCLE = /^(rgba?\((:?.+)\)) Circle\(x: (-?\d+), y: (-?\d+), r: (-?\d+)\)$/;
const IMAGE = /^Image\s+\("(.+)"\)\s+\(source:\s+\[(-?\d+), (-?\d+), (-?\d+), (-?\d+)\]\)\s+\(destination:\s+\[(-?\d+), (-?\d+), (-?\d+), (-?\d+)\]\)$/;
const CANVAS = /^(Canvas)\s+\(source:\s+\[(-?\d+), (-?\d+), (-?\d+), (-?\d+)\]\)\s+\(destination:\s+\[(-?\d+), (-?\d+), (-?\d+), (-?\d+)\]\)$/;
const GRADIENT = /^\[(-?\d+), (-?\d+), (-?\d+), (-?\d+)\]\s+linear-gradient\(x0: (-?\d+), x1: (-?\d+), y0: (-?\d+), y1: (-?\d+) (.+)\)$/;
const TRANSFORM = /^\((-?\d+), (-?\d+)\) \[(.+)\]$/;

function parsePath(path) {
    const parts = path.match(PATH)[1];
    return parts.split(' > ').map(p => {
        const vector = p.match(VECTOR);
        if (vector) {
            return {
                type: 'Vector',
                x: parseInt(vector[1], 10),
                y: parseInt(vector[2], 10)
            };
        } else {
            const bezier = p.match(BEZIER_CURVE);
            return {
                type: 'BezierCurve',
                x0: parseInt(bezier[1], 10),
                y0: parseInt(bezier[2], 10),
                x1: parseInt(bezier[3], 10),
                y1: parseInt(bezier[4], 10),
                cx0: parseInt(bezier[5], 10),
                cy0: parseInt(bezier[6], 10),
                cx1: parseInt(bezier[7], 10),
                cy1: parseInt(bezier[8], 10)
            };
        }
    });
}

function parseRefTest(txt) {
    return txt.split(/\n/g).filter(l => l.length > 0).map((l, i) => {
        const parseAction = l.match(ACTION);
        if (!parseAction) {
            const text = l.match(TEXT);
            return {
                action: 'T',
                x: parseInt(text[1], 10),
                y: parseInt(text[2], 10),
                text: text[3]
            };
        }
        const args = parseAction[2];

        const data = {
            action: parseAction[1],
            line: i + 1
        };

        switch (data.action) {
            case 'Opacity':
                data.opacity = parseFloat(args);
                break;
            case 'Fill':
                data.color = args;
                break;
            case 'Clip':
                data.path = args.split(' | ').map(path => parsePath(path));
                break;
            case 'Window':
                const windowSize = args.match(WINDOW_SIZE);
                data.width = parseInt(windowSize[1], 10);
                data.height = parseInt(windowSize[2], 10);
                break;
            case 'Rectangle':
                const rectangle = args.match(RECTANGLE);
                data.x = parseInt(rectangle[1], 10);
                data.y = parseInt(rectangle[2], 10);
                data.width = parseInt(rectangle[3], 10);
                data.height = parseInt(rectangle[4], 10);
                data.color = rectangle[5];
                break;
            case 'Repeat':
                const repeat = args.match(REPEAT);
                data.imageSrc = repeat[1];
                data.x = parseInt(repeat[2], 10);
                data.y = parseInt(repeat[3], 10);
                data.width = parseInt(repeat[4], 10);
                data.height = parseInt(repeat[5], 10);
                data.path = parsePath(repeat[6]);
                break;
            case 'Shape':
                const shape = args.match(SHAPE);
                if (!shape) {
                    const circle = args.match(CIRCLE);
                    data.color = circle[1];
                    data.path = [
                        {
                            type: 'Circle',
                            x: parseInt(circle[2], 10),
                            y: parseInt(circle[3], 10),
                            r: parseInt(circle[4], 10)
                        }
                    ];
                } else {
                    data.color = shape[1];
                    data.path = parsePath(shape[3]);
                }
                break;
            case 'Text':
                data.font = args;
                break;
            case 'Draw image':
                const image = args.match(IMAGE) ? args.match(IMAGE) : args.match(CANVAS);
                data.imageSrc = image[1];
                data.sx = parseInt(image[2], 10);
                data.xy = parseInt(image[3], 10);
                data.sw = parseInt(image[4], 10);
                data.sh = parseInt(image[5], 10);
                data.dx = parseInt(image[6], 10);
                data.dy = parseInt(image[7], 10);
                data.dw = parseInt(image[8], 10);
                data.dh = parseInt(image[9], 10);
                break;
            case 'Gradient':
                const gradient = args.match(GRADIENT);
                data.x = parseInt(gradient[1], 10);
                data.y = parseInt(gradient[2], 10);
                data.width = parseInt(gradient[3], 10);
                data.height = parseInt(gradient[4], 10);
                data.x0 = parseInt(gradient[5], 10);
                data.x1 = parseInt(gradient[6], 10);
                data.y0 = parseInt(gradient[7], 10);
                data.y1 = parseInt(gradient[8], 10);
                data.stops = gradient[9];
                break;
            case 'Transform':
                const transform = args.match(TRANSFORM);
                data.x = parseInt(transform[1], 10);
                data.y = parseInt(transform[2], 10);
                data.matrix = transform[3];
                break;
            default:
                console.log(args);
                throw new Error('Unhandled action ' + data.action);
        }

        return data;
    });
}

module.exports = parseRefTest;