Home Reference Source

src/divisions/Pall.js

/**
 * @file Houses the class properties and methods for the Pall division.
 */
import * as Utilities from '../utilities';
import settings from '../settings';
import Division from '../division';

/** Pall pattern.
 *
 * @class
 * @classdesc The Pall pattern describes a Y shape on the field, typically oriented so the top of the Y is on the left of the field.
 * @augments Division
 * @namespace Divisions.Pall
 */
export default class Pall extends Division {
    /**
     * Creates a Pall.
     *
     * @example
     * // Creates a horizontally oriented, white Pall with a black border.
     * const pall = new Pall('fesswise', '#ffffff', true, 20, '#000000');
     * @param {string} direction - The orientation of the Pall. One of: fesswise, palewise, fesswiseReversed, palewiseReverse.
     * @param {string} color - A hexadecimal color string.
     * @param {number} width - A width value for drawing the division.
     * @param {boolean} border - Whether or not to draw a border around the Pall.
     * @param {number} borderWidth - The width of the border.
     * @param {string} borderColor - A hexadecimal color string.
     */
  constructor(params = {seed, direction, color, width, border, borderWidth, borderColor}) {
    super({seed: params.seed, count: 1, limit: 1, color: params.color});
    this.border = params.border;
    this.width = params.width;
    this.direction = typeof params.direction !== 'undefined' ? params.direction : this.generateDirection();
    // this.borderWidth = borderWidth > 0 ? borderWidth : this.generateSaltireWidth((settings.seed * .1234));
    this.borderWidth = params.borderWidth || 50;
    this.borderColor = params.borderColor || Utilities.generateColor(undefined, .12345);
  }
    /**
     * Generate a direction value for the Pall.
     *
     * @example
     * // Returns 'palewiseReversed'
     * const newPale = new Pale();
     * newPale.generateDirection(.1337)
     * @param {number} seed - The seed number used for generated values.
     * @returns {string} - One of: 'pallwise', 'fesswise', 'pallwiseReversed', 'fesswiseReversed'.
     */
    generateDirection(seed = this.seed) {
        let generated;
        // let seedDigit = +(Math.round(this.seed * this.seedMultiplier).toString().substr(-1));
        const seedDigit = Utilities.getLastDigit(Utilities.modifySeed(this.seed, this.seedMultiplier));
        if (seedDigit >=1 && seedDigit <= 4) {
            generated = 'palewise';
        } else if (seedDigit === 5 || seedDigit === 6) {
            generated = 'fesswise';
        } else if (seedDigit === 7 || seedDigit === 8) {
            generated = 'palewiseReversed';
        } else if (seedDigit === 0 || seedDigit === 9) {
            generated = 'fesswiseReversed';
        }
        return generated;
    }

    /**
     * Returns the proper draw function instructions for a given direction.
     *
     * @example
     * // Returns drawInstructionsPalewise();
     * const pall = new Pall();
     * pall.drawInstructions('palewise');
     * @param {string} direction - One of: 'fesswise', 'palewise', 'fesswiseReversed', 'palewiseReversed'.
     * @returns {Function} The draw instruction function corresponding to the direction.
     */
    drawInstructions(direction) {
        let instructions;
        switch (direction) {
            case 'palewise':
                instructions = this.drawInstructionsPalewise();
                break;
            case 'palewiseReversed':
                instructions = this.drawInstructionsPalewise(true);
                break;
            case 'fesswise':
                instructions = this.drawInstructionsFesswise();
                break;
            case 'fesswiseReversed':
                instructions = this.drawInstructionsFesswise(true);
                break;
        }
        return instructions;
    }

    /**
     * Generates the draw instructions for the palewise and palewiseReversed directions.
     *
     * @example
     * // Returns an instruction set for the palewise direction based on the flag dimensions.
     * // [
     * //     {moveTo: [x, y]},
     * //     {lineTo: [x, y]},
     * //     {lineTo: [x, y]},
     * //     {moveTo: [x, y]},
     * //     {lineTo: [x, y]},
     * // ]
     * const pall = new Pall();
     * const instructions = pall.drawInstructionsPalewise();
     * @param {boolean} reversed - Whether or not to return the reverse draw instructions.
     * @returns {Array} An array of objects containing canvas drawing instructions.
     */
    drawInstructionsPalewise(reversed) {
        let instructions;
        if (!reversed) {
            instructions = [
                {moveTo: [0, 0]},
                {lineTo: [settings.flagWidth / 2, settings.flagHeight / 2]},
                {lineTo: [0, settings.flagHeight]},
                {moveTo: [settings.flagWidth / 2, settings.flagHeight / 2]},
                {lineTo: [settings.flagWidth, settings.flagHeight / 2]},
            ]
        } else {
            instructions = [
                {moveTo: [settings.flagWidth, 0]}, // start top-right
                {lineTo: [settings.flagWidth / 2, settings.flagHeight / 2]}, // draw to center
                {lineTo: [settings.flagWidth, settings.flagHeight]}, // draw to bottom right
                {moveTo: [settings.flagWidth / 2, settings.flagHeight / 2]}, // move to center
                {lineTo: [0, settings.flagHeight / 2]}, // Draw to center left side
            ]
        }

        return instructions;
    }
    /**
     * Generates the draw instructions for the fesswise and fesswiseReversed directions.
     *
     * @example
     * // Returns an instruction set for the fesswise direction based on the flag dimensions.
     * // [
     * //     {moveTo: [x, y]},
     * //     {lineTo: [x, y]},
     * //     {lineTo: [x, y]},
     * //     {moveTo: [x, y]},
     * //     {lineTo: [x, y]},
     * // ]
     * const pall = new Pall();
     * const instructions = pall.drawInstructionsFesswise();
     * @param {boolean} reversed - Whether or not to return the reverse draw instructions.
     * @returns {Array} An array of objects containing canvas drawing instructions.
     */
    drawInstructionsFesswise(reversed) {
        let instructions;
        if (!reversed) {
            instructions = [
                {moveTo: [0, 0]}, // start top left
                {lineTo: [settings.flagWidth / 2, settings.flagHeight / 2]}, // draw to center
                {lineTo: [settings.flagWidth, 0]}, // draw to top right
                {moveTo: [settings.flagWidth / 2, settings.flagHeight / 2]}, // move to center
                {lineTo: [settings.flagWidth / 2, settings.flagHeight]}, // draw to center bottom
            ];
        } else {
            instructions = [
                {moveTo: [0, settings.flagHeight]}, // start bottom left
                {lineTo: [settings.flagWidth / 2, settings.flagHeight / 2]}, // draw to center
                {lineTo: [settings.flagWidth, settings.flagHeight]}, // draw to bottom right
                {moveTo: [settings.flagWidth / 2, settings.flagHeight / 2]}, // move to center
                {lineTo: [settings.flagWidth / 2, settings.flagHeight - settings.flagHeight]}, // draw to center top
            ];
        }
        return instructions;
    }

    /**
     * Draws the Pall pattern on a canvas.
     *
     * @example
     * // Draws the Pall pattern.
     * pall.draw(ctx);
     * @param {object} ctx - An object containing a canvas context.
     */
    draw(ctx) {
        const drawSteps = this.drawInstructions(this.direction);
        const pallWidth = this.width || Math.round(this.seed * 100);
        ctx.beginPath();
        // If there's a border, draw it first with a larger width.
        if (this.border) {
            for (let i = 0, len = drawSteps.length; i < len; i++) {
                const step = Object.keys(drawSteps[i]);
                const stepParams = Object.values(drawSteps[i])[0];
                ctx[step](...stepParams);
            }
            ctx.lineWidth = pallWidth + this.borderWidth;
            ctx.strokeStyle = this.color.complement;
            ctx.stroke();
        }

        for (let i = 0, len = drawSteps.length; i < len; i++) {
            const step = Object.keys(drawSteps[i]);
            const stepParams = Object.values(drawSteps[i])[0];
            ctx[step](...stepParams);
        }
        ctx.strokeStyle = this.color.color;
        ctx.lineWidth = pallWidth;
        ctx.stroke();
    }
}