// ==UserScript==
// @name         Jstris Tutor Maker
// @license      BSD-2-Clause
// @namespace    Jstris Tutor Maker
// @version      0.3.2
// @description  Helps you make a Jstris usermode for placing a queue of pieces in the right spots
// @author       TSTman
// @match        https://jstris.jezevec10.com/usermodes/create*
// @match        https://jstris.jezevec10.com/usermodes/*/edit*
// @icon         https://jstris.jezevec10.com/favicon.ico
// @grant        none
// ==/UserScript==
function setupTrainingMaker() {'use strict';
    // Uncomment an ExampleFumen below (only 1 at a time) to see how it works with Jstris Tutor Maker.
    // Thumbnail + 3 bags of fesh sprint
    let ExampleFumen = 'v115@ZfA8Bek0BeB8j0Aeh0AeB8g0A8k0AeB8g0B8j0AeA8?Bek0lfAglZfAABekHBeBAjHAehHAeBAgHAAkHAeBAgHBAjH?AeAABekHlf2uQ9BFLDmClcJSAVDEHBEooRBPoAVBKNUFDU+?DxCaeHgCzuPFDMXltCTXNPC0XEWCadNPCvuLuCMHmPCpyTx?CaeHgCsvTxCJnzPCJ3TWC6ujWCpvbgCU3jFDK+DxCzyKxCK?ezPCMdFgC6OOMCvintC6P9VCa3TxCsAAAAvhTzkBifB0sB9?tBXjBplBqrBFiBJmB+tBsrBnsBTpBOrBNqB6qBTpBxrBHtB?MtB';
    // Test single line clears:
    //let ExampleFumen = 'v115@vhJSSYtAFLDmClcJSAVDEHBEooRBMoAVBU3LMC6f/w?CpHLWCTO1LCJO1LCJO1LCpAAAA1wBTpB3qBxpBUsBWtBOmB?FqBctB';
    // Test perfect clears with O pieces:
    //let ExampleFumen = 'v115@vhJTJYdAFLDmClcJSAVDEHBEooRBPoAVBP3SgCP3Sg?CvAAAATqBTrBTsBTtBTpBTqBTrBTsBTtB';
    // 6 pieces
    //let ExampleFumen = 'v115@vhF1OY/BFLDmClcJSAVDEHBEooRBUoAVBpizPCP9aF?DpCmPCTentC6OMgCJnzPCzOUPCvS9wCad9VCU9aFDqCmPCP?ujFDqXMgCzeltCp/TFDv+TWCpXegCK+TFDPt/wCpXegCz/D?xCKtbMCviLuCq+KWCJtHgCpXExCJ3zBApmBckBWyBToBvqB?';
    // 10 pieces
    //let ExampleFumen = 'v115@vhJ1OY/BFLDmClcJSAVDEHBEooRBUoAVBpizPCP9aF?DpCmPCTentC6OMgCJnzPCzOUPCvS9wCad9VCU9aFDqCmPCP?ujFDqXMgCzeltCp/TFDv+TWCpXegCK+TFDPt/wCpXegCz/D?xCKtbMCviLuCq+KWCJtHgCpXExCJ3zBApmBckBWyBToBvqB?CjBcaBZmBSeB';
    // 20 pieces
    //let ExampleFumen = 'v115@vhT1OY/BFLDmClcJSAVDEHBEooRBUoAVBpizPCP9aF?DpCmPCTentC6OMgCJnzPCzOUPCvS9wCad9VCU9aFDqCmPCP?ujFDqXMgCzeltCp/TFDv+TWCpXegCK+TFDPt/wCpXegCz/D?xCKtbMCviLuCq+KWCJtHgCpXExCJ3zBApmBckBWyBToBvqB?CjBcaBZmBSeB7TBGPB/gBVcBWKB3RBcQBNMB6SBTAB';
    // 30 pieces
    //let ExampleFumen = 'v115@vhd1OY/BFLDmClcJSAVDEHBEooRBUoAVBpizPCP9aF?DpCmPCTentC6OMgCJnzPCzOUPCvS9wCad9VCU9aFDqCmPCP?ujFDqXMgCzeltCp/TFDv+TWCpXegCK+TFDPt/wCpXegCz/D?xCKtbMCviLuCq+KWCJtHgCpXExCJ3zBApmBckBWyBToBvqB?CjBcaBZmBSeB7TBGPB/gBVcBWKB3RBcQBNMB6SBTABZnBTX?BcaB+NB/WBNQBCPBWKBTABC7A';
    // 101 pieces:
    //let ExampleFumen = 'v115@vh/1OY/BFLDmClcJSAVDEHBEooRBUoAVBpizPCP9aF?DpCmPCTentC6OMgCJnzPCzOUPCvS9wCad9VCU9aFDqCmPCP?ujFDqXMgCzeltCp/TFDv+TWCpXegCK+TFDPt/wCpXegCz/D?xCKtbMCviLuCq+KWCJtHgCpXExCJ3zBApmBckBWyBToBvqB?CjBcaBZmBSeB7TBGPB/gBVcBWKB3RBcQBNMB6SBTABZnBTX?BcaB+NB/WBNQBCPBWKBTABC7AJnBlgBsaBXXBJnBCmBNpBX?gBTaBUcBGXBSeBdWBWSBTIBZnBXoBckB+SBcbBCjBZnBjbB?NpBZgB/kB/aBWyBScB0QBToB9RBXHBCjBvhkZnBcmBNaB5S?BWyBDoBCjBZXB0QB/RBWeBbUB9HBMIBF+ADGBZnB2VBCjBX?MBWeB3GBUIBNBBTUBCPBU9A/DBWKB95ATABC7AZnB/MBZnB?TaBZnB';
    // Holdless:
    //let ExampleFumen = 'v115@vhFRQYoAFLDmClcJSAVDEHBEooRBJoAVB6ybgCq+yt?C6/7LCUtzPCpOMgCUmBKpBvsBTtB+jf8gg0Ieg0Heh0deFq?QiAFLDmClcJSAVDEHBEooRBUoAVBPtzPCM+9tC6P9wCMHBA?AvhEzpBvrBMjBOkB6lf/ghlIeglIeglZexhQcAFLDmClcJS?AVDEHBEooRBJoAVBUtzPCpOMgCvhAlsfGhAPgHBegWwDCeA?PhHxSgWxDCegWAtxSgWQpxDAegWKe3sQaAFLDmClcJSAVDE?HBEooRBToAVB6P9wCMHBAAvhC0pBmkBxnfNhzhdelrQWAFL?DmClcJSAVDEHBEooRBUoAVBMHBAAvhBCrBTrB';
    // IsChallengeMode decides whether this is a Tutor mode or a Challenge mode.
    let IsChallengeMode = false;
    // HowManyBlocksPerSection decides how often to check if the player's field looks correct.
    let HowManyBlocksPerSection = 7;
    // PauseHowLongBetweenPieces is the number of seconds to pause between steps in tutor mode
    let PauseHowLongBetweenPieces = 0.7;
    // Set HowManyDemoSections if you want *only some* of the sections to have Demo Blocks.
    let HowManyDemoSections = 0;
    // BlockQueue starts out empty and is filled in after you load a Fumen. It adds an "H" before each piece that is held.
    let BlockQueue = '';
    // HowManyBlocks starts out at 0 and is set after you load a Fumen. It becomes the number of blocks the player must
    // use in order to complete your usermode
    let HowManyBlocks = 0;
    class Component {
        constructor(fieldType) {
            this.opts = {};
            this.id = ++Component.idCounter;
            this.field_type = fieldType;
        }
    }
    Component.idCounter = 0;
    class Condition extends Component {
        constructor(conditionType, conditionValue, doIfTrue, conditionDo) {
            super('cond');
            this.opts['do'] = conditionDo;
            this.opts['on'] = doIfTrue;
            this.opts['check'] = conditionType;
            this.opts['check2'] = conditionValue;
        }
    }
    class MapComponent extends Component {
        constructor(mapType) {
            super('map');
            this.opts['spawn'] = mapType;
        }
        updateContent(content) {
            if (typeof content !== 'string') {
                return;
            }
            this.opts['map'] = content;
        }
    }
    class QueueChange extends Component {
        constructor(queue, replace) {
            super('queue');
            this.opts['queue'] = queue;
            this.opts['wipe'] = replace;
        }
    }
    class RelativeTrigger extends Component {
        constructor(relativeTriggerType, amount, triggerID) {
            super('rtrig');
            this.opts['af'] = relativeTriggerType;
            this.opts['id'] = triggerID;
            this.opts['when'] = amount;
        }
    }
    class Ruleset extends Component {
        constructor(ruleset) {
            super('rule');
            this.opts['rule'] = ruleset;
        }
    }
    class Run extends Component {
        constructor(triggerID) {
            super('run');
            this.opts['id'] = triggerID;
        }
    }
    class Trigger extends Component {
        constructor(triggerType, triggerArg) {
            super('trig');
            this.opts['when'] = triggerType;
            if (typeof triggerArg === 'string') {
                this.opts['when2'] = triggerArg;
            }
        }
    }
    // Keeps the page from locking up while the components are generated, even though it sleeps for 0 seconds
    function sleep() {
        return new Promise(resolve => setTimeout(resolve, 0));
    }
    const TriggerTypeBeforeGame = 0;
    const TriggerTypeOnSpecificBlockNumber = 3;
    const TriggerTypeExternalConditional = 7;
    const TriggerTypeNever = 9;
    const TriggerTypeOnGameStart = 10;
    const TriggerIDTwoLinePC = '2_Line_PC';
    async function saveTextInput(element, value) {
        element.value = value;
        element.setAttribute('value', value);
        await saveInput(element);
    }
    async function saveInput(element) {
        element.dispatchEvent(new Event('input'));
        await sleep();
    }
    let componentList;
    async function addComponent(component) {
        componentList.push(component);
        await sleep();
    }
    // newTrigger creates a new Trigger component
    async function newTrigger(triggerType, triggerArg = null) {
        await addComponent(new Trigger(triggerType, triggerArg));
    }
    const QueueIPiece = 'I';
    const QueueHoldPiece = 'H';
    const QueueHoldPieceNone = 'NONE';
    const QUEUE_SIZE_LIMIT = 600;
    function buildQueue(holdPiece, queue) {
        const holdPrefix = !hasHold ? [] : [`h=${holdPiece}`];
        queue = holdPrefix.concat(queue.replace(new RegExp(QueueHoldPiece, 'g'), '').split('')).join(',');
        if (queue.length >= QUEUE_SIZE_LIMIT) {
            queue = queue.slice(0, QUEUE_SIZE_LIMIT);
            if (queue.slice(-1) === ',') {
                queue = queue.slice(0, -1);
            }
        }
        return queue;
    }
    // newQueueChange creates a new Queue Change component
    async function newQueueChange(queue, holdPiece, replace = true, repeat = false, prependPieceToQueue = false) {
        if (holdPiece === '') {
            holdPiece = QueueHoldPieceNone;
        }
        if (prependPieceToQueue) {
            queue = 'O' + queue;
        }
        queue = buildQueue(holdPiece, queue);
        await addComponent(new QueueChange(queue, replace));
    }
    const RelativeTriggerTypeTime = 0;
    const RelativeTriggerTypeBlocks = 1;
    const RelativeTriggerTypeLines = 2;
    // newRelativeTrigger creates a new Relative Trigger component
    async function newRelativeTrigger(relativeTriggerType, amount, triggerID) {
        await addComponent(new RelativeTrigger(relativeTriggerType, amount, triggerID));
    }
    const MapTypeReplaceBoard = 1;
    const MapTypeAddToCurrentBoardOnTop = 2;
    const MapTypeSubtractFromCurrentBoard = 3;
    // MapDataLineClear a place for an I piece to complete a 2-line PC. It is used to trigger a line clear after subtracting the expected field from the board, in order to make sure the whole board is clear (there was a PC)
    const MapDataLineClear = 'ERAAAREREREREQAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA==';
    async function updateSectionMapContent(maps, mapContent, mapToSkip = null) {
        if (!(maps instanceof Array)) {
            return;
        }
        for (const map of maps) {
            if (map === mapToSkip) {
                continue;
            }
            await updateMapContent(map, mapContent);
        }
    }
    async function updateMapContent(map, mapContent) {
        map.updateContent(mapContent);
    }
    // newMap creates a new Map component
    async function newMap(mapType, content = null) {
        const map = new MapComponent(mapType);
        if (mapType === MapTypeAddToCurrentBoardOnTop) {
            map.updateContent(MapDataLineClear);
        } else if (typeof content === 'string') {
            await updateMapContent(map, content);
        }
        await addComponent(map);
        return map;
    }
    // 28G and no lock delay
    const RulesetTypeFastDropLock = JSON.stringify({lockDelay: [0, 5000, 20000], gravityLvl: 28});
    let RulesetTypeDefault = JSON.stringify({});
    async function newRuleset(rulesetType) {
        await addComponent(new Ruleset(rulesetType));
    }
    async function newRun(triggerID) {
        await addComponent(new Run(triggerID));
    }
    const ConditionTypePCs = 7;
    const ConditionTypeHolds = 14;
    const ConditionTypeLines = 17;
    const ConditionResultTypeGameOver = 1;
    // newCondition creates a new Condition component.
    async function newCondition(conditionType, conditionValue, doIfTrue, conditionDo) {
        await addComponent(new Condition(conditionType, conditionValue, doIfTrue, conditionDo));
    }
    function saveAllButton() {
        return document.querySelector('#saveAll');
    }
    const saveButtonOriginalText = [...document.querySelectorAll('#saveAll')].map(el => el.textContent).join('');
    const ClassButtonSuccess = 'btn-success';
    const ClassButtonDanger = 'btn-danger';
    async function updateStatus(statusText) {
        const saveButton = saveAllButton();
        if (!(saveButton instanceof HTMLElement)) {
            return;
        }
        saveButton.classList.remove(ClassButtonSuccess);
        saveButton.classList.add(ClassButtonDanger);
        saveButton.textContent = statusText;
    }
    async function resetStatus() {
        const saveButton = saveAllButton();
        if (!(saveButton instanceof HTMLElement)) {
            return;
        }
        saveButton.classList.remove(ClassButtonDanger);
        saveButton.classList.add(ClassButtonSuccess);
        saveButton.textContent = saveButtonOriginalText;
        await sleep();
    }
    async function demoCycle(blockCount, demoTriggerID, queue, mapListForBlock) {
        await newRelativeTrigger(RelativeTriggerTypeTime, PauseHowLongBetweenPieces, demoTriggerID);
        await newTrigger(TriggerTypeExternalConditional, demoTriggerID);
        mapListForBlock.push(await newMap(MapTypeReplaceBoard));
        await newQueueChange(queue.slice(1), demoHoldPieces[blockCount], true, false);
    }
    // cycle goes through all the the steps to make sure the user placed a single piece correctly
    async function cycle(sectionCount, blockCount, queues, holdPieces, mapListForBlock) {
        await newTrigger(TriggerTypeOnSpecificBlockNumber, (blockCount + sectionCount - 1).toString());
        mapListForBlock.push(await newMap(MapTypeSubtractFromCurrentBoard));
        let nextQueue;
        let nextHoldPiece;
        if (blockCount < BlockQueue.length) {
            nextQueue = queues[blockCount + 1];
            nextHoldPiece = holdPieces[blockCount + 1];
        } else {
            nextQueue = '';
            nextHoldPiece = QueueHoldPieceNone;
        }
        await newQueueChange(QueueIPiece + nextQueue, nextHoldPiece, true, false);
        await newRun(TriggerIDTwoLinePC);
        const judgeTriggerID = `judge_stage${blockCount}`;
        await newRelativeTrigger(RelativeTriggerTypeLines, 1, judgeTriggerID);
        await newTrigger(TriggerTypeExternalConditional, judgeTriggerID);
        await newCondition(ConditionTypePCs, `=${sectionCount + actualPCCounts[blockCount]}`, false, ConditionResultTypeGameOver);
        await newCondition(ConditionTypeLines, `=${totalLinesCleared[blockCount] + 2 * sectionCount}`, false, ConditionResultTypeGameOver);
        mapListForBlock.push(await newMap(MapTypeReplaceBoard));
        await newRuleset(RulesetTypeDefault);
    }
    async function makeDemoCycles(blockCount, totalBlocks, queue, finalTriggerID, mapListsBySection) {
        let holdPiece = demoHoldPieces[blockCount];
        for (; blockCount <= totalBlocks; blockCount++) {
            let triggerSuffix = '';
            let triggerSection = 1;
            let demoTriggerID = `demo_block${blockCount}` + triggerSuffix;
            if (queue[0] === QueueHoldPiece) {
                let swap = queue[1];
                queue = holdPiece + queue.slice(2);
                holdPiece = swap;
            }
            if (!(mapListsBySection[blockCount] instanceof Array)) {
                mapListsBySection[blockCount] = new Array();
            }
            if (typeof fumenWithFullLines[blockCount] === 'string') {
                triggerSuffix = `_part_${triggerSection}`;
                let afterFullLinesTriggerID = `demo_block${blockCount}` + triggerSuffix;
                await newRelativeTrigger(RelativeTriggerTypeTime, PauseHowLongBetweenPieces, afterFullLinesTriggerID);
                await newTrigger(TriggerTypeExternalConditional, afterFullLinesTriggerID);
                triggerSection++;
                triggerSuffix = `_part_${triggerSection}`;
                demoTriggerID = `demo_block${blockCount}` + triggerSuffix;
                mapsWithFullLines[blockCount] = await newMap(MapTypeReplaceBoard);
                await updateMapContent(mapsWithFullLines[blockCount], fumenWithFullLines[blockCount]);
            }
            await demoCycle(blockCount, demoTriggerID, queue, mapListsBySection[blockCount]);
            queue = queue.slice(1);
        }
        await newRelativeTrigger(RelativeTriggerTypeTime, PauseHowLongBetweenPieces * 2, finalTriggerID);
    }
    // 1 cycle per block in BlockQueue
    async function makeCycles(sectionCount, blockCount, lastBlockInSection, mapListsBySection) {
        for (; blockCount <= lastBlockInSection; blockCount++) {
            if (!(mapListsBySection[blockCount] instanceof Array)) {
                mapListsBySection[blockCount] = new Array();
            }
            const lastCycleInStage = blockCount === lastBlockInSection;
            if (lastCycleInStage) {
                await cycle(sectionCount, blockCount, queues, holdPieces, mapListsBySection[blockCount]);
            }
        }
        if (!IsChallengeMode && blockCount <= HowManyBlocks && blockCount <= howManyDemoBlocks) {
            await newQueueChange(demoQueues[blockCount], demoHoldPieces[blockCount], true, false, true);
        }
    }
    async function initUserMode(queue) {
        await newTrigger(TriggerTypeBeforeGame, null);
        if (!hasHold) {
            await newRuleset(RulesetTypeDefault);
        }
        await newQueueChange(queue, QueueHoldPieceNone, true, false);
        await newTrigger(TriggerTypeOnGameStart, null);
        await newQueueChange(queue, QueueHoldPieceNone, true, false);
    }
    function setTotalSections() {
        totalSections = Math.ceil(HowManyBlocks / HowManyBlocksPerSection);
    }
    let totalSections;
    function fumenSaveButton() {
        return document.querySelector('div.fumen-section button.load-fumen');
    }
    function newDiv(parentElement, ...classes) {
        const row = document.createElement('div');
        row.classList.add(...classes);
        if (parentElement instanceof HTMLElement) {
            parentElement.appendChild(row);
        }
        return row;
    }
    const sectionID = 'blocks-per-section';
    const fumenInputID = 'fumen-input';
    const tutorModeID = 'usermode-type-tutor';
    const challengeModeID = 'usermode-type-challenge';
    const timePerPieceID = 'time-per-piece';
    async function fumenSection() {
        const fumenSection = newDiv(null, 'fumen-section', 'col-sm-10');
        const mainRow = newDiv(fumenSection, 'row');
        const buttonCol = newDiv(mainRow, 'col-sm-2');
        const loadFumenRow = newDiv(buttonCol, 'row');
        const loadFumenButton = document.createElement('button');
        loadFumenButton.classList.add('load-fumen');
        loadFumenButton.textContent = 'Load fumen';
        loadFumenButton.addEventListener('click', loadFumenToMaps);
        loadFumenButton.classList.add(...['col-sm-12', 'btn', 'btn-sm', 'btn-warning', 'control-label']);
        loadFumenRow.appendChild(loadFumenButton);
        const exampleFumenRow = newDiv(buttonCol, 'row');
        const exampleFumenButton = document.createElement('button');
        exampleFumenButton.classList.add('example-fumen');
        exampleFumenButton.textContent = 'Example fumen';
        exampleFumenButton.addEventListener('click', async () => await saveTextInput(fumenInput, ExampleFumen));
        exampleFumenButton.classList.add(...['col-sm-12', 'btn', 'btn-sm', 'btn-info', 'control-label']);
        exampleFumenRow.appendChild(exampleFumenButton);
        const inputsContainer = newDiv(mainRow, 'col-sm-10');
        const fumenContainer = newDiv(newDiv(newDiv(inputsContainer, 'form-group'), 'row'), 'col-sm-12');
        const fumenInput = document.createElement('input');
        const inputAttributes = {
            class: 'form-control',
            type: 'text',
            autocomplete: 'off',
            autocorrect: 'off',
            autocapitalize: 'off',
            spellcheck: 'false'
        };
        for (const attribute in inputAttributes) {
            fumenInput.setAttribute(attribute, inputAttributes[attribute]);
        }
        fumenInput.id = fumenInputID;
        fumenInput.placeholder = 'Enter fumen here';
        fumenContainer.appendChild(fumenInput);
        const settingsContainer = newDiv(newDiv(inputsContainer, 'form-group'), 'row');
        const sectionLabel = document.createElement('label');
        sectionLabel.textContent = 'Blocks per stage';
        sectionLabel.classList.add('col-sm-3', 'control-label');
        sectionLabel.htmlFor = sectionID;
        settingsContainer.appendChild(sectionLabel);
        const selectContainer = newDiv(settingsContainer, 'col-sm-3');
        const sectionSelect = document.createElement('select');
        sectionSelect.classList.add('form-control');
        sectionSelect.id = sectionID;
        for (let optionValue = 1; optionValue <= 28; optionValue++) {
            const option = document.createElement('option');
            option.value = optionValue.toString();
            option.textContent = optionValue.toString();
            if (optionValue === HowManyBlocksPerSection) {
                option.textContent += ' (default)';
                option.selected = true;
            }
            sectionSelect.appendChild(option);
        }
        settingsContainer.appendChild(selectContainer).appendChild(sectionSelect);
        const userModeTypeContainer = newDiv(newDiv(settingsContainer, 'col-sm-6'), 'row');
        const userModeTypeName = 'usermode-type';
        const tutorModeContainer = newDiv(userModeTypeContainer, 'form-check', 'col-sm-5');
        const tutorModeRadio = document.createElement('input');
        tutorModeRadio.classList.add('form-check-input');
        tutorModeRadio.type = 'radio';
        tutorModeRadio.name = userModeTypeName;
        tutorModeRadio.id = tutorModeID;
        tutorModeRadio.setAttribute('checked', '');
        tutorModeContainer.appendChild(tutorModeRadio);
        tutorModeContainer.innerHTML += ' ';
        const tutorModeLabel = document.createElement('label');
        tutorModeLabel.textContent = 'Tutor mode';
        tutorModeLabel.classList.add('form-check-label');
        tutorModeLabel.htmlFor = tutorModeID;
        tutorModeContainer.appendChild(tutorModeLabel);
        const challengeModeContainer = newDiv(userModeTypeContainer, 'form-check', 'col-sm-6');
        const challengeModeRadio = document.createElement('input');
        challengeModeRadio.classList.add('form-check-input');
        challengeModeRadio.type = 'radio';
        challengeModeRadio.name = userModeTypeName;
        challengeModeRadio.id = challengeModeID;
        challengeModeContainer.appendChild(challengeModeRadio);
        challengeModeContainer.innerHTML += ' ';
        const challengeModeLabel = document.createElement('label');
        challengeModeLabel.textContent = 'Challenge mode';
        challengeModeLabel.classList.add('form-check-label');
        challengeModeLabel.htmlFor = challengeModeID;
        challengeModeContainer.appendChild(challengeModeLabel);
        const timePerPieceContainer = newDiv(newDiv(inputsContainer, 'form-group'), 'row');
        const timePerPieceLabel = document.createElement('label');
        timePerPieceLabel.classList.add('col-sm-3', 'control-label');
        timePerPieceLabel.textContent = 'Demo time per piece';
        timePerPieceLabel.htmlFor = timePerPieceID;
        timePerPieceContainer.appendChild(timePerPieceLabel);
        const timePerPieceInput = document.createElement('input');
        for (const attribute in inputAttributes) {
            timePerPieceInput.setAttribute(attribute, inputAttributes[attribute]);
        }
        timePerPieceInput.placeholder = '(seconds)';
        timePerPieceInput.value = PauseHowLongBetweenPieces.toString();
        timePerPieceInput.id = timePerPieceID;
        newDiv(timePerPieceContainer, 'col-sm-3').appendChild(timePerPieceInput);
        fumenSection.appendChild(mainRow);
        const saveButtonDiv = saveAllButton().parentNode;
        saveButtonDiv.classList.remove('col-sm-12');
        saveButtonDiv.classList.add('col-sm-2');
        saveButtonDiv.parentNode.parentNode.insertBefore(fumenSection, saveButtonDiv.parentNode);
        fumenInput.focus();
        return loadFumenButton;
    }
    let mapListsByPieceIndex = {};
    let fumenWithFullLines = {};
    let mapsWithFullLines = {};
    function setDefaultRuleset() {
        const defaultRuleset = {};
        if (!hasHold) {
            defaultRuleset['hasHold'] = false;
        }
        RulesetTypeDefault = JSON.stringify(defaultRuleset);
    }
    let hasHold;
    function buildQueues(holdPiece) {
        let queue = BlockQueue;
        let blockCount = 1;
        while (queue.length > 0) {
            queues[blockCount] = queue;
            holdPieces[blockCount] = holdPiece;
            const shouldHold = queue[0] === QueueHoldPiece;
            if (shouldHold) {
                let swap = queue[1];
                queue = holdPiece + queue.slice(2);
                holdPiece = swap;
            }
            demoQueues[blockCount] = queue;
            demoHoldPieces[blockCount] = holdPiece;
            blockCount++;
            queue = queue.slice(1);
        }
    }
    let queues;
    let holdPieces;
    let demoQueues;
    let demoHoldPieces;
    async function loadComponents(shouldResetStatus = true, thumbnailContent = undefined) {
        if (typeof thumbnailContent === 'string') {
            await newTrigger(TriggerTypeNever);
            await newMap(MapTypeReplaceBoard, thumbnailContent);
        }
        await newTrigger(TriggerTypeExternalConditional, TriggerIDTwoLinePC);
        await newMap(MapTypeAddToCurrentBoardOnTop);
        await newRuleset(RulesetTypeFastDropLock);
        mapListsByPieceIndex = {};
        hasHold = BlockQueue.search(QueueHoldPiece) > -1;
        setDefaultRuleset();
        let fumenButton = fumenSaveButton();
        if (!(fumenButton instanceof HTMLButtonElement)) {
            fumenButton = await fumenSection();
        }
        setTotalSections();
        if (HowManyBlocks > 0) {
            fumenButton.classList.add('disabled');
            fumenButton.setAttribute('disabled', '');
            let firstSection = true;
            queues = [];
            holdPieces = [];
            demoQueues = [];
            demoHoldPieces = [];
            buildQueues('');
            await initUserMode(queues[1]);
            for (let section = 1; section <= totalSections; section++) {
                let sectionBeginningBlockCount = (section - 1) * HowManyBlocksPerSection + 1;
                let playTriggerID = `play_stage${section}`;
                let sectionFinalBlockCount = section * HowManyBlocksPerSection;
                if (sectionFinalBlockCount > HowManyBlocks) {
                    sectionFinalBlockCount = HowManyBlocks;
                }
                if (!IsChallengeMode && sectionBeginningBlockCount < howManyDemoBlocks) {
                    await makeDemoCycles(sectionBeginningBlockCount, sectionFinalBlockCount, demoQueues[sectionBeginningBlockCount], playTriggerID, mapListsByPieceIndex);
                    await newTrigger(TriggerTypeExternalConditional, playTriggerID);
                    let transitionMap = await newMap(MapTypeReplaceBoard);
                    if (firstSection) {
                        firstSection = false;
                    } else {
                        mapListsByPieceIndex[sectionBeginningBlockCount - 1].push(transitionMap);
                    }
                    await newQueueChange(queues[sectionBeginningBlockCount], holdPieces[sectionBeginningBlockCount], true, false);
                }
                await makeCycles(section, sectionBeginningBlockCount, sectionFinalBlockCount, mapListsByPieceIndex);
                if (sectionFinalBlockCount === HowManyBlocks) {
                    break;
                }
            }
            // Remove the relative trigger and trigger at the end, trigger ID block#_queue
            componentList.pop();
            componentList.pop();
            //const editButton: HTMLAnchorElement = mapListsByPieceIndex[newMapIndex][0].querySelector(':scope a.open-map-edit');
            //editButton.click();
            if (shouldResetStatus) {
                await resetStatus();
            }
            fumenButton.classList.remove('disabled');
            fumenButton.removeAttribute('disabled');
        }
    }
    (async function () {
        await fumenSection();
    })();
    const JstrisPiece = {
        'Empty': 0,
        'Z': 1,
        'L': 2,
        'O': 3,
        'S': 4,
        'I': 5,
        'J': 6,
        'T': 7,
        'Gray': 8,
    };
    const JstrisPieceByFumenPiece = {};
    function mapFumenPiecesToJstrisPieces(fumen) {
        for (let piece in JstrisPiece) {
            JstrisPieceByFumenPiece[fumen.Piece[piece]] = JstrisPiece[piece];
        }
    }
    function blockQueueFromPages(fumen, pages) {
        let blockQueue = '';
        let quiz;
        let usedNextPiece;
        let pieceIndex = 1;
        for (const page of pages) {
            if (HowManyBlocks > 0 && pieceIndex > HowManyBlocks) {
                break;
            }
            usedNextPiece = false;
            // Having an active mino is nice but not required
            const activeMino = page.operation;
            if (!page.flags.quiz) {
                if (activeMino instanceof fumen.Mino) {
                    blockQueue += activeMino.type;
                }
                continue;
            }
            quiz = new fumen.Quiz(page.comment);
            let quizPiece = quiz['current'];
            const holdPiece = quiz['hold'];
            if (activeMino instanceof fumen.Mino && activeMino.type !== quizPiece) {
                blockQueue += QueueHoldPiece;
                if (holdPiece === '') {
                    quizPiece += quiz['next'];
                    usedNextPiece = true;
                }
            }
            blockQueue += quizPiece;
            pieceIndex++;
        }
        if (quiz instanceof fumen.Quiz) {
            let nextPieces = quiz.getNextPieces().map(fumen.parsePieceName).join('');
            usedNextPiece = true;
            if (usedNextPiece) {
                nextPieces = nextPieces.slice(1);
            }
            blockQueue += nextPieces;
        }
        return blockQueue;
    }
    // howManyDemoBlocks is a computed field, you don't set it directly
    let howManyDemoBlocks;
    let progressInterval;
    let hasThumbnail = false;
    async function loadFumenToMaps() {
        componentList = [];
        const generateProgress = () => updateStatus(`Generated ${componentList.length} components`);
        const loadProgress = () => {
            console.log(document.querySelectorAll('span.cid-disp').length, 'so far');
            updateStatus(`Loaded ${document.querySelectorAll('span.cid-disp').length}/${componentList.length} components`);
        };
        // Periodic updates so you know if it's still busy generating stuff
        progressInterval = window.setInterval(generateProgress, 1000);
        await generateProgress();
        await sleep();
        const fumenButton = fumenSaveButton();
        fumenButton.classList.add('btn-warning');
        fumenButton.classList.remove('btn-success');
        fumenButton.textContent = 'Fumen loading...';
        let fumen = new Fumen();
        mapFumenPiecesToJstrisPieces(fumen);
        let inputElement = document.querySelector(`#${fumenInputID}`);
        const pages = fumen.decode(inputElement.value);
        let thumbnailContent;
        if (pages[0].flags.quiz === false && !(pages[0].operation instanceof fumen.Mino)) {
            hasThumbnail = true;
            thumbnailContent = fumenToMapData(fumen, pages.shift()['_field'].field['pieces']);
        }
        BlockQueue = blockQueueFromPages(fumen, pages);
        totalLinesCleared = Array(pages.length).fill(0);
        actualPCCounts = Array(pages.length).fill(0);
        fumenWithFullLines = {};
        let cumulativeLinesCleared = 0;
        let pieceIndex = 1;
        for (const page of pages) {
            if (HowManyBlocks > 0 && pieceIndex > HowManyBlocks) {
                break;
            }
            addMinoToField(fumen, page);
            const linesBeforeClearing = page['_field'].field.pieces.slice();
            const clearedThisPage = countLineClears(fumen, page, pieceIndex);
            if (clearedThisPage > 0) {
                fumenWithFullLines[pieceIndex] = fumenToMapData(fumen, linesBeforeClearing);
            }
            cumulativeLinesCleared += clearedThisPage;
            totalLinesCleared[pieceIndex] = cumulativeLinesCleared;
            pieceIndex++;
        }
        if (BlockQueue.length > 0 && HowManyBlocks === 0) {
            HowManyBlocks = pages.length;
        }
        HowManyBlocksPerSection = parseInt(document.querySelector(`#${sectionID}`).value);
        if (HowManyDemoSections > 0) {
            howManyDemoBlocks = HowManyBlocksPerSection * HowManyDemoSections;
        } else {
            howManyDemoBlocks = HowManyBlocks;
        }
        // BlockQueue length must be at least 1 greater HowManyBlocks
        if (BlockQueue.length === HowManyBlocks) {
            BlockQueue += 'I';
        }
        IsChallengeMode = document.querySelector(`#${challengeModeID}`).checked === true;
        PauseHowLongBetweenPieces = parseFloat(document.querySelector(`#${timePerPieceID}`).value);
        await loadComponents(false, thumbnailContent);
        pieceIndex = 1;
        for (const page of pages) {
            if (HowManyBlocks > 0 && pieceIndex > HowManyBlocks) {
                break;
            }
            await updateSectionMapContent(mapListsByPieceIndex[pieceIndex], fumenToMapData(fumen, page['_field'].field['pieces']));
            pieceIndex++;
        }
        clearInterval(progressInterval);
        loadProgress();
        // Progress for Backbone.js render components in the DOM. Backbone.js locks up the page until it's done, but if they ever don't,
        // this progress interval will work.
        progressInterval = window.setInterval(loadProgress, 1000);
        await sleep();
        // Load in a separate thread in an attempt to keep the page from locking up while it's loading
        setTimeout(loadUsermodeForm, 0);
    }
    async function loadingDone() {
        const fumenButton = fumenSaveButton();
        clearInterval(progressInterval);
        fumenButton.classList.add('btn-success');
        fumenButton.classList.remove('btn-warning');
        fumenButton.textContent = 'Fumen loaded!';
        await resetStatus();
    }
    async function loadUsermodeForm() {
        // @ts-ignore
        const fb = new Formbuilder({selector: '.components-main', bootstrapData: componentList});
        fb.on('save', function (payload) {
            // @ts-ignore
            $("#modeData").val(payload);
            // @ts-ignore
            $("#modeForm").submit();
        });
        // @ts-ignore
        $("#pubSection").hide();
        await sleep();
        await loadingDone();
    }
    let rowCount = 20;
    let columnCount = 10;
    let emptyRow = new Array(columnCount);
    let totalLinesCleared = [];
    function addMinoToField(fumen, page) {
        const field = page['_field'].field;
        const operation = page.operation;
        // Having an active mino is nice but not required
        if (operation instanceof Object) {
            const filledMino = page.mino();
            field.fill({
                type: fumen.parsePiece(operation.type),
                rotation: fumen.parseRotation(operation.rotation),
                x: operation.x,
                y: operation.y
            });
            for (const {x, y} of filledMino.positions()) {
                field.set(x, y, fumen.parsePiece(filledMino.type));
            }
        }
    }
    let actualPCCounts;
    function countLineClears(fumen, page, pieceIndex) {
        let linesCleared = 0;
        const pieces = page['_field'].field.pieces;
        let finalRow = rowCount;
        rowLoop: for (let row = 0; row < finalRow;) {
            let pieceIndex = row++ * columnCount;
            const nextPieceIndex = row * columnCount;
            for (; pieceIndex < nextPieceIndex; pieceIndex++) {
                if (pieces[pieceIndex] === fumen.Piece.Empty) {
                    continue rowLoop;
                }
            }
            linesCleared++;
            pieces.splice(--row * columnCount, columnCount);
            pieces.push(...emptyRow);
            --finalRow;
        }
        if (pieceIndex > 1) {
            actualPCCounts[pieceIndex] = actualPCCounts[pieceIndex - 1];
        }
        const maxPiece = Math.max(...pieces.slice(0, (rowCount - 1) * columnCount));
        if (maxPiece === 0) {
            actualPCCounts[pieceIndex]++;
        }
        return linesCleared;
    }
    function fumenToMapData(fumen, fieldPieces) {
        const fieldSize = rowCount * columnCount / 2; // 2 pieces per byte
        const mapDataBuffer = new ArrayBuffer(fieldSize);
        const mapDataField = new Uint8Array(mapDataBuffer);
        let fieldIndex = fieldSize;
        for (let row = 0; row < rowCount; row++) {
            for (let column = columnCount - 2; column >= 0; column -= 2) {
                mapDataField[--fieldIndex] = JstrisPieceByFumenPiece[fieldPieces[columnCount * row + column]] << 4 |
                    JstrisPieceByFumenPiece[fieldPieces[columnCount * row + column + 1]];
            }
        }
        return window.btoa(String.fromCharCode.apply(null, mapDataField));
    }
    function Fumen() {
        // From https://github.com/knewjade/tetris-fumen/tree/4a77d0dc52
        /**
         MIT License
         Copyright (c) 2019
         Permission is hereby granted, free of charge, to any person obtaining a copy
         of this software and associated documentation files (the "Software"), to deal
         in the Software without restriction, including without limitation the rights
         to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
         copies of the Software, and to permit persons to whom the Software is
         furnished to do so, subject to the following conditions:
         The above copyright notice and this permission notice shall be included in all
         copies or substantial portions of the Software.
         THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
         IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
         FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
         AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
         LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
         OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
         SOFTWARE.
         */
        function decodeBool(n) {
            return n !== 0;
        }
        const createActionDecoder = (width, fieldTop, garbageLine) => {
            const fieldMaxHeight = fieldTop + garbageLine;
            const numFieldBlocks = fieldMaxHeight * width;
            function decodePiece(n) {
                switch (n) {
                    case 0:
                        return Piece.Empty;
                    case 1:
                        return Piece.I;
                    case 2:
                        return Piece.L;
                    case 3:
                        return Piece.O;
                    case 4:
                        return Piece.Z;
                    case 5:
                        return Piece.T;
                    case 6:
                        return Piece.J;
                    case 7:
                        return Piece.S;
                    case 8:
                        return Piece.Gray;
                }
                throw new Error('Unexpected piece');
            }
            function decodeRotation(n) {
                switch (n) {
                    case 0:
                        return Rotation.Reverse;
                    case 1:
                        return Rotation.Right;
                    case 2:
                        return Rotation.Spawn;
                    case 3:
                        return Rotation.Left;
                }
                throw new Error('Unexpected rotation');
            }
            function decodeCoordinate(n, piece, rotation) {
                let x = n % width;
                const originY = Math.floor(n / 10);
                let y = fieldTop - originY - 1;
                if (piece === Piece.O && rotation === Rotation.Left) {
                    x += 1;
                    y -= 1;
                } else if (piece === Piece.O && rotation === Rotation.Reverse) {
                    x += 1;
                } else if (piece === Piece.O && rotation === Rotation.Spawn) {
                    y -= 1;
                } else if (piece === Piece.I && rotation === Rotation.Reverse) {
                    x += 1;
                } else if (piece === Piece.I && rotation === Rotation.Left) {
                    y -= 1;
                } else if (piece === Piece.S && rotation === Rotation.Spawn) {
                    y -= 1;
                } else if (piece === Piece.S && rotation === Rotation.Right) {
                    x -= 1;
                } else if (piece === Piece.Z && rotation === Rotation.Spawn) {
                    y -= 1;
                } else if (piece === Piece.Z && rotation === Rotation.Left) {
                    x += 1;
                }
                return {x, y};
            }
            return {
                decode: (v) => {
                    let value = v;
                    const type = decodePiece(value % 8);
                    value = Math.floor(value / 8);
                    const rotation = decodeRotation(value % 4);
                    value = Math.floor(value / 4);
                    const coordinate = decodeCoordinate(value % numFieldBlocks, type, rotation);
                    value = Math.floor(value / numFieldBlocks);
                    const isBlockUp = decodeBool(value % 2);
                    value = Math.floor(value / 2);
                    const isMirror = decodeBool(value % 2);
                    value = Math.floor(value / 2);
                    const isColor = decodeBool(value % 2);
                    value = Math.floor(value / 2);
                    const isComment = decodeBool(value % 2);
                    value = Math.floor(value / 2);
                    const isLock = !decodeBool(value % 2);
                    return {
                        rise: isBlockUp,
                        mirror: isMirror,
                        colorize: isColor,
                        comment: isComment,
                        lock: isLock,
                        piece: {
                            ...coordinate,
                            type,
                            rotation,
                        },
                    };
                },
            };
        };
        function encodeBool(flag) {
            return flag ? 1 : 0;
        }
        const createActionEncoder = (width, fieldTop, garbageLine) => {
            const fieldMaxHeight = fieldTop + garbageLine;
            const numFieldBlocks = fieldMaxHeight * width;
            function encodePosition(operation) {
                const {type, rotation} = operation;
                let x = operation.x;
                let y = operation.y;
                if (!isMinoPiece(type)) {
                    x = 0;
                    y = 22;
                } else if (type === Piece.O && rotation === Rotation.Left) {
                    x -= 1;
                    y += 1;
                } else if (type === Piece.O && rotation === Rotation.Reverse) {
                    x -= 1;
                } else if (type === Piece.O && rotation === Rotation.Spawn) {
                    y += 1;
                } else if (type === Piece.I && rotation === Rotation.Reverse) {
                    x -= 1;
                } else if (type === Piece.I && rotation === Rotation.Left) {
                    y += 1;
                } else if (type === Piece.S && rotation === Rotation.Spawn) {
                    y += 1;
                } else if (type === Piece.S && rotation === Rotation.Right) {
                    x += 1;
                } else if (type === Piece.Z && rotation === Rotation.Spawn) {
                    y += 1;
                } else if (type === Piece.Z && rotation === Rotation.Left) {
                    x -= 1;
                }
                return (fieldTop - y - 1) * width + x;
            }
            function encodeRotation({type, rotation}) {
                if (!isMinoPiece(type)) {
                    return 0;
                }
                switch (rotation) {
                    case Rotation.Reverse:
                        return 0;
                    case Rotation.Right:
                        return 1;
                    case Rotation.Spawn:
                        return 2;
                    case Rotation.Left:
                        return 3;
                }
                throw new Error('No reachable');
            }
            return {
                encode: (action) => {
                    const {lock, comment, colorize, mirror, rise, piece} = action;
                    let value = encodeBool(!lock);
                    value *= 2;
                    value += encodeBool(comment);
                    value *= 2;
                    value += (encodeBool(colorize));
                    value *= 2;
                    value += encodeBool(mirror);
                    value *= 2;
                    value += encodeBool(rise);
                    value *= numFieldBlocks;
                    value += encodePosition(piece);
                    value *= 4;
                    value += encodeRotation(piece);
                    value *= 8;
                    value += piece.type;
                    return value;
                },
            };
        };
        const ENCODE_TABLE = 'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/';
        class Buffer {
            constructor(data = '') {
                this.values = data.split('').map(decodeToValue);
            }
            poll(max) {
                let value = 0;
                for (let count = 0; count < max; count += 1) {
                    const v = this.values.shift();
                    if (v === undefined) {
                        throw new Error('Unexpected fumen');
                    }
                    value += v * Math.pow(Buffer.tableLength, count);
                }
                return value;
            }
            push(value, splitCount = 1) {
                let current = value;
                for (let count = 0; count < splitCount; count += 1) {
                    this.values.push(current % Buffer.tableLength);
                    current = Math.floor(current / Buffer.tableLength);
                }
            }
            merge(postBuffer) {
                for (const value of postBuffer.values) {
                    this.values.push(value);
                }
            }
            isEmpty() {
                return this.values.length === 0;
            }
            get length() {
                return this.values.length;
            }
            get(index) {
                return this.values[index];
            }
            set(index, value) {
                this.values[index] = value;
            }
            toString() {
                return this.values.map(encodeFromValue).join('');
            }
        }
        Buffer.tableLength = ENCODE_TABLE.length;
        function decodeToValue(v) {
            return ENCODE_TABLE.indexOf(v);
        }
        function encodeFromValue(index) {
            return ENCODE_TABLE[index];
        }
        const COMMENT_TABLE = ' !"#$%&\'()*+,-./0123456789:;<=>?@ABCDEFGHIJKLMNOPQRSTUVWXYZ[\\]^_`abcdefghijklmnopqrstuvwxyz{|}~';
        const MAX_COMMENT_CHAR_VALUE = COMMENT_TABLE.length + 1;
        const createCommentParser = () => {
            return {
                decode: (v) => {
                    let str = '';
                    let value = v;
                    for (let count = 0; count < 4; count += 1) {
                        const index = value % MAX_COMMENT_CHAR_VALUE;
                        str += COMMENT_TABLE[index];
                        value = Math.floor(value / MAX_COMMENT_CHAR_VALUE);
                    }
                    return str;
                },
                encode: (ch, count) => {
                    return COMMENT_TABLE.indexOf(ch) * Math.pow(MAX_COMMENT_CHAR_VALUE, count);
                },
            };
        };
        class Page {
            constructor(index, field, operation, comment, flags, refs) {
                this.index = index;
                this.operation = operation;
                this.comment = comment;
                this.flags = flags;
                this.refs = refs;
                this._field = field.copy();
            }
            get field() {
                return new Field(this._field.copy());
            }
            set field(field) {
                this._field = createInnerField(field);
            }
            mino() {
                return Mino.from(this.operation);
            }
        }
        function extract(str) {
            const format = (version, data) => {
                const trim = data.trim().replace(/[?\s]+/g, '');
                return {version, data: trim};
            };
            let data = str;
            // url parameters
            const paramIndex = data.indexOf('&');
            if (0 <= paramIndex) {
                data = data.substring(0, paramIndex);
            }
            // v115@~
            {
                const match = str.match(/[vmd]115@/);
                if (match !== undefined && match !== null && match.index !== undefined) {
                    const sub = data.substr(match.index + 5);
                    return format('115', sub);
                }
            }
            // v110@~
            {
                const match = str.match(/[vmd]110@/);
                if (match !== undefined && match !== null && match.index !== undefined) {
                    const sub = data.substr(match.index + 5);
                    return format('110', sub);
                }
            }
            throw new Error('Unsupported fumen version');
        }
        this.decode = function (fumen) {
            const {version, data} = extract(fumen);
            switch (version) {
                case '115':
                    return innerDecode(data, 23);
                case '110':
                    return innerDecode(data, 21);
            }
            throw new Error('Unsupported fumen version');
        };
        function innerDecode(data, fieldTop) {
            const fieldMaxHeight = fieldTop + FieldConstants.GarbageLine;
            const numFieldBlocks = fieldMaxHeight * FieldConstants.Width;
            const buffer = new Buffer(data);
            const updateField = (prev) => {
                const result = {
                    changed: true,
                    field: prev,
                };
                let index = 0;
                while (index < numFieldBlocks) {
                    const diffBlock = buffer.poll(2);
                    const diff = Math.floor(diffBlock / numFieldBlocks);
                    const numOfBlocks = diffBlock % numFieldBlocks;
                    if (diff === 8 && numOfBlocks === numFieldBlocks - 1) {
                        result.changed = false;
                    }
                    for (let block = 0; block < numOfBlocks + 1; block += 1) {
                        const x = index % FieldConstants.Width;
                        const y = fieldTop - Math.floor(index / FieldConstants.Width) - 1;
                        result.field.addNumber(x, y, diff - 8);
                        index += 1;
                    }
                }
                return result;
            };
            let pageIndex = 0;
            let prevField = createNewInnerField();
            const store = {
                repeatCount: -1,
                refIndex: {
                    comment: 0,
                    field: 0,
                },
                quiz: undefined,
                lastCommentText: '',
            };
            const pages = [];
            const actionDecoder = createActionDecoder(FieldConstants.Width, fieldTop, FieldConstants.GarbageLine);
            const commentDecoder = createCommentParser();
            while (!buffer.isEmpty()) {
                // Parse field
                let currentFieldObj;
                if (0 < store.repeatCount) {
                    currentFieldObj = {
                        field: prevField,
                        changed: false,
                    };
                    store.repeatCount -= 1;
                } else {
                    currentFieldObj = updateField(prevField.copy());
                    if (!currentFieldObj.changed) {
                        store.repeatCount = buffer.poll(1);
                    }
                }
                // Parse action
                const actionValue = buffer.poll(3);
                const action = actionDecoder.decode(actionValue);
                // Parse comment
                let comment;
                if (action.comment) {
                    // コメントに更新があるとき
                    const commentValues = [];
                    const commentLength = buffer.poll(2);
                    for (let commentCounter = 0; commentCounter < Math.floor((commentLength + 3) / 4); commentCounter += 1) {
                        const commentValue = buffer.poll(5);
                        commentValues.push(commentValue);
                    }
                    let flatten = '';
                    for (const value of commentValues) {
                        flatten += commentDecoder.decode(value);
                    }
                    const commentText = unescape(flatten.slice(0, commentLength));
                    store.lastCommentText = commentText;
                    comment = {text: commentText};
                    store.refIndex.comment = pageIndex;
                    const text = comment.text;
                    if (Quiz.isQuizComment(text)) {
                        try {
                            store.quiz = new Quiz(text);
                        } catch (e) {
                            store.quiz = undefined;
                        }
                    } else {
                        store.quiz = undefined;
                    }
                } else if (pageIndex === 0) {
                    // コメントに更新がないが、先頭のページのとき
                    comment = {text: ''};
                } else {
                    // コメントに更新がないとき
                    comment = {
                        text: store.quiz !== undefined ? store.quiz.format().toString() : undefined,
                        ref: store.refIndex.comment,
                    };
                }
                // Quiz用の操作を取得し、次ページ開始時点のQuizに1手進める
                let quiz = false;
                if (store.quiz !== undefined) {
                    quiz = true;
                    if (store.quiz.canOperate() && action.lock) {
                        if (isMinoPiece(action.piece.type)) {
                            try {
                                const nextQuiz = store.quiz.nextIfEnd();
                                const operation = nextQuiz.getOperation(action.piece.type);
                                store.quiz = nextQuiz.operate(operation);
                            } catch (e) {
                                // console.error(e.message);
                                // Not operate
                                store.quiz = store.quiz.format();
                            }
                        } else {
                            store.quiz = store.quiz.format();
                        }
                    }
                }
                // データ処理用に加工する
                let currentPiece;
                if (action.piece.type !== Piece.Empty) {
                    currentPiece = action.piece;
                }
                // pageの作成
                let field;
                if (currentFieldObj.changed || pageIndex === 0) {
                    // フィールドに変化があったとき
                    // フィールドに変化がなかったが、先頭のページだったとき
                    field = {};
                    store.refIndex.field = pageIndex;
                } else {
                    // フィールドに変化がないとき
                    field = {ref: store.refIndex.field};
                }
                pages.push(new Page(pageIndex, currentFieldObj.field, currentPiece !== undefined ? Mino.from({
                    type: parsePieceName(currentPiece.type),
                    rotation: parseRotationName(currentPiece.rotation),
                    x: currentPiece.x,
                    y: currentPiece.y,
                }) : undefined, comment.text !== undefined ? comment.text : store.lastCommentText, {
                    quiz,
                    lock: action.lock,
                    mirror: action.mirror,
                    colorize: action.colorize,
                    rise: action.rise,
                }, {
                    field: field.ref,
                    comment: comment.ref,
                }));
                // callback(
                //     currentFieldObj.field.copy()
                //     , currentPiece
                //     , store.quiz !== undefined ? store.quiz.format().toString() : store.lastCommentText,
                // );
                pageIndex += 1;
                if (action.lock) {
                    if (isMinoPiece(action.piece.type)) {
                        currentFieldObj.field.fill(action.piece);
                    }
                    currentFieldObj.field.clearLine();
                    if (action.rise) {
                        currentFieldObj.field.riseGarbage();
                    }
                    if (action.mirror) {
                        currentFieldObj.field.mirror();
                    }
                }
                prevField = currentFieldObj.field;
            }
            return pages;
        }
        let Piece;
        (function (Piece) {
            Piece[Piece["Empty"] = 0] = "Empty";
            Piece[Piece["I"] = 1] = "I";
            Piece[Piece["L"] = 2] = "L";
            Piece[Piece["O"] = 3] = "O";
            Piece[Piece["Z"] = 4] = "Z";
            Piece[Piece["T"] = 5] = "T";
            Piece[Piece["J"] = 6] = "J";
            Piece[Piece["S"] = 7] = "S";
            Piece[Piece["Gray"] = 8] = "Gray";
        })(Piece || (Piece = {}));
        this.Piece = Piece;
        function isMinoPiece(piece) {
            return piece !== Piece.Empty && piece !== Piece.Gray;
        }
        function parsePieceName(piece) {
            switch (piece) {
                case Piece.I:
                    return 'I';
                case Piece.L:
                    return 'L';
                case Piece.O:
                    return 'O';
                case Piece.Z:
                    return 'Z';
                case Piece.T:
                    return 'T';
                case Piece.J:
                    return 'J';
                case Piece.S:
                    return 'S';
                case Piece.Gray:
                    return 'X';
                case Piece.Empty:
                    return '_';
            }
            throw new Error(`Unknown piece: ${piece}`);
        }
        this.parsePieceName = parsePieceName;
        function parsePiece(piece) {
            switch (piece.toUpperCase()) {
                case 'I':
                    return Piece.I;
                case 'L':
                    return Piece.L;
                case 'O':
                    return Piece.O;
                case 'Z':
                    return Piece.Z;
                case 'T':
                    return Piece.T;
                case 'J':
                    return Piece.J;
                case 'S':
                    return Piece.S;
                case 'X':
                case 'GRAY':
                    return Piece.Gray;
                case ' ':
                case '_':
                case 'EMPTY':
                    return Piece.Empty;
            }
            throw new Error(`Unknown piece: ${piece}`);
        }
        this.parsePiece = parsePiece;
        let Rotation;
        (function (Rotation) {
            Rotation[Rotation["Spawn"] = 0] = "Spawn";
            Rotation[Rotation["Right"] = 1] = "Right";
            Rotation[Rotation["Reverse"] = 2] = "Reverse";
            Rotation[Rotation["Left"] = 3] = "Left";
        })(Rotation || (Rotation = {}));
        this.Rotation = Rotation;
        function parseRotationName(rotation) {
            switch (rotation) {
                case Rotation.Spawn:
                    return 'spawn';
                case Rotation.Left:
                    return 'left';
                case Rotation.Right:
                    return 'right';
                case Rotation.Reverse:
                    return 'reverse';
            }
            throw new Error(`Unknown rotation: ${rotation}`);
        }
        function parseRotation(rotation) {
            switch (rotation.toLowerCase()) {
                case 'spawn':
                    return Rotation.Spawn;
                case 'left':
                    return Rotation.Left;
                case 'right':
                    return Rotation.Right;
                case 'reverse':
                    return Rotation.Reverse;
            }
            throw new Error(`Unknown rotation: ${rotation}`);
        }
        this.parseRotation = parseRotation;
        function encode(pages) {
            const updateField = (prev, current) => {
                const {changed, values} = encodeField(prev, current);
                if (changed) {
                    // フィールドを記録して、リピートを終了する
                    buffer.merge(values);
                    lastRepeatIndex = -1;
                } else if (lastRepeatIndex < 0 || buffer.get(lastRepeatIndex) === Buffer.tableLength - 1) {
                    // フィールドを記録して、リピートを開始する
                    buffer.merge(values);
                    buffer.push(0);
                    lastRepeatIndex = buffer.length - 1;
                } else if (buffer.get(lastRepeatIndex) < (Buffer.tableLength - 1)) {
                    // フィールドは記録せず、リピートを進める
                    const currentRepeatValue = buffer.get(lastRepeatIndex);
                    buffer.set(lastRepeatIndex, currentRepeatValue + 1);
                }
            };
            let lastRepeatIndex = -1;
            const buffer = new Buffer();
            let prevField = createNewInnerField();
            const actionEncoder = createActionEncoder(FieldConstants.Width, 23, FieldConstants.GarbageLine);
            const commentParser = createCommentParser();
            let prevComment = '';
            let prevQuiz = undefined;
            const innerEncode = (index) => {
                const currentPage = pages[index];
                currentPage.flags = currentPage.flags ? currentPage.flags : {};
                const field = currentPage.field;
                const currentField = field !== undefined ? createInnerField(field) : prevField.copy();
                // フィールドの更新
                updateField(prevField, currentField);
                // アクションの更新
                const currentComment = currentPage.comment !== undefined
                    ? ((index !== 0 || currentPage.comment !== '') ? currentPage.comment : undefined)
                    : undefined;
                const piece = currentPage.operation !== undefined ? {
                    type: parsePiece(currentPage.operation.type),
                    rotation: parseRotation(currentPage.operation.rotation),
                    x: currentPage.operation.x,
                    y: currentPage.operation.y,
                } : {
                    type: Piece.Empty,
                    rotation: Rotation.Reverse,
                    x: 0,
                    y: 22,
                };
                let nextComment;
                if (currentComment !== undefined) {
                    if (currentComment.startsWith('#Q=')) {
                        // Quiz on
                        if (prevQuiz !== undefined && prevQuiz.format().toString() === currentComment) {
                            nextComment = undefined;
                        } else {
                            nextComment = currentComment;
                            prevComment = nextComment;
                            prevQuiz = new Quiz(currentComment);
                        }
                    } else {
                        // Quiz off
                        if (prevQuiz !== undefined && prevQuiz.format().toString() === currentComment) {
                            nextComment = undefined;
                            prevComment = currentComment;
                            prevQuiz = undefined;
                        } else {
                            nextComment = prevComment !== currentComment ? currentComment : undefined;
                            prevComment = prevComment !== currentComment ? nextComment : prevComment;
                            prevQuiz = undefined;
                        }
                    }
                } else {
                    nextComment = undefined;
                    prevQuiz = undefined;
                }
                if (prevQuiz !== undefined && prevQuiz.canOperate() && currentPage.flags.lock) {
                    if (isMinoPiece(piece.type)) {
                        try {
                            const nextQuiz = prevQuiz.nextIfEnd();
                            const operation = nextQuiz.getOperation(piece.type);
                            prevQuiz = nextQuiz.operate(operation);
                        } catch (e) {
                            // console.error(e.message);
                            // Not operate
                            prevQuiz = prevQuiz.format();
                        }
                    } else {
                        prevQuiz = prevQuiz.format();
                    }
                }
                const currentFlags = {
                    lock: true,
                    colorize: index === 0,
                    ...currentPage.flags,
                };
                const action = {
                    piece,
                    rise: !!currentFlags.rise,
                    mirror: !!currentFlags.mirror,
                    colorize: !!currentFlags.colorize,
                    lock: !!currentFlags.lock,
                    comment: nextComment !== undefined,
                };
                const actionNumber = actionEncoder.encode(action);
                buffer.push(actionNumber, 3);
                // コメントの更新
                if (nextComment !== undefined) {
                    const comment = escape(currentPage.comment);
                    const commentLength = Math.min(comment.length, 4095);
                    buffer.push(commentLength, 2);
                    // コメントを符号化
                    for (let index = 0; index < commentLength; index += 4) {
                        let value = 0;
                        for (let count = 0; count < 4; count += 1) {
                            const newIndex = index + count;
                            if (commentLength <= newIndex) {
                                break;
                            }
                            const ch = comment.charAt(newIndex);
                            value += commentParser.encode(ch, count);
                        }
                        buffer.push(value, 5);
                    }
                } else if (currentPage.comment === undefined) {
                    prevComment = undefined;
                }
                // 地形の更新
                if (action.lock) {
                    if (isMinoPiece(action.piece.type)) {
                        currentField.fill(action.piece);
                    }
                    currentField.clearLine();
                    if (action.rise) {
                        currentField.riseGarbage();
                    }
                    if (action.mirror) {
                        currentField.mirror();
                    }
                }
                prevField = currentField;
            };
            for (let index = 0; index < pages.length; index += 1) {
                innerEncode(index);
            }
            // テト譜が短いときはそのまま出力する
            // 47文字ごとに?が挿入されるが、実際は先頭にv115@が入るため、最初の?は42文字後になる
            const data = buffer.toString();
            if (data.length < 41) {
                return data;
            }
            // ?を挿入する
            const head = [data.substr(0, 42)];
            const tails = data.substring(42);
            const split = tails.match(/[\S]{1,47}/g) || [];
            return head.concat(split).join('?');
        }
        // フィールドをエンコードする
        // 前のフィールドがないときは空のフィールドを指定する
        // 入力フィールドの高さは23, 幅は10
        function encodeField(prev, current) {
            const FIELD_TOP = 23;
            const FIELD_MAX_HEIGHT = FIELD_TOP + 1;
            const FIELD_BLOCKS = FIELD_MAX_HEIGHT * FieldConstants.Width;
            const buffer = new Buffer();
            // 前のフィールドとの差を計算: 0〜16
            const getDiff = (xIndex, yIndex) => {
                const y = FIELD_TOP - yIndex - 1;
                return current.getNumberAt(xIndex, y) - prev.getNumberAt(xIndex, y) + 8;
            };
            // データの記録
            const recordBlockCounts = (diff, counter) => {
                const value = diff * FIELD_BLOCKS + counter;
                buffer.push(value, 2);
            };
            // フィールド値から連続したブロック数に変換
            let changed = true;
            let prev_diff = getDiff(0, 0);
            let counter = -1;
            for (let yIndex = 0; yIndex < FIELD_MAX_HEIGHT; yIndex += 1) {
                for (let xIndex = 0; xIndex < FieldConstants.Width; xIndex += 1) {
                    const diff = getDiff(xIndex, yIndex);
                    if (diff !== prev_diff) {
                        recordBlockCounts(prev_diff, counter);
                        counter = 0;
                        prev_diff = diff;
                    } else {
                        counter += 1;
                    }
                }
            }
            // 最後の連続ブロックを処理
            recordBlockCounts(prev_diff, counter);
            if (prev_diff === 8 && counter === FIELD_BLOCKS - 1) {
                changed = false;
            }
            return {
                changed,
                values: buffer,
            };
        }
        function toMino(operationOrMino) {
            return operationOrMino instanceof Mino ? operationOrMino.copy() : Mino.from(operationOrMino);
        }
        class Field {
            constructor(field) {
                this.field = field;
            }
            static create(field, garbage) {
                return new Field(new InnerField({
                    field: field !== undefined ? PlayField.load(field) : undefined,
                    garbage: garbage !== undefined ? PlayField.loadMinify(garbage) : undefined,
                }));
            }
            canFill(operation) {
                if (operation === undefined) {
                    return true;
                }
                const mino = toMino(operation);
                return this.field.canFillAll(mino.positions());
            }
            canLock(operation) {
                if (operation === undefined) {
                    return true;
                }
                if (!this.canFill(operation)) {
                    return false;
                }
                // Check on the ground
                return !this.canFill({...operation, y: operation.y - 1});
            }
            fill(operation, force = false) {
                if (operation === undefined) {
                    return undefined;
                }
                const mino = toMino(operation);
                if (!force && !this.canFill(mino)) {
                    throw Error('Cannot fill piece on field');
                }
                this.field.fillAll(mino.positions(), parsePiece(mino.type));
                return mino;
            }
            put(operation) {
                if (operation === undefined) {
                    return undefined;
                }
                const mino = toMino(operation);
                for (; 0 <= mino.y; mino.y -= 1) {
                    if (!this.canLock(mino)) {
                        continue;
                    }
                    this.fill(mino);
                    return mino;
                }
                throw Error('Cannot put piece on field');
            }
            clearLine() {
                this.field.clearLine();
            }
            at(x, y) {
                return parsePieceName(this.field.getNumberAt(x, y));
            }
            set(x, y, type) {
                this.field.setNumberAt(x, y, parsePiece(type));
            }
            copy() {
                return new Field(this.field.copy());
            }
            str(option = {}) {
                let skip = option.reduced !== undefined ? option.reduced : true;
                const separator = option.separator !== undefined ? option.separator : '\n';
                const minY = option.garbage === undefined || option.garbage ? -1 : 0;
                let output = '';
                for (let y = 22; minY <= y; y -= 1) {
                    let line = '';
                    for (let x = 0; x < 10; x += 1) {
                        line += this.at(x, y);
                    }
                    if (skip && line === '__________') {
                        continue;
                    }
                    skip = false;
                    output += line;
                    if (y !== minY) {
                        output += separator;
                    }
                }
                return output;
            }
        }
        class Mino {
            constructor(type, rotation, x, y) {
                this.type = type;
                this.rotation = rotation;
                this.x = x;
                this.y = y;
            }
            static from(operation) {
                return new Mino(operation.type, operation.rotation, operation.x, operation.y);
            }
            positions() {
                return getBlockXYs(parsePiece(this.type), parseRotation(this.rotation), this.x, this.y).sort((a, b) => {
                    if (a.y === b.y) {
                        return a.x - b.x;
                    }
                    return a.y - b.y;
                });
            }
            operation() {
                return {
                    type: this.type,
                    rotation: this.rotation,
                    x: this.x,
                    y: this.y,
                };
            }
            isValid() {
                try {
                    parsePiece(this.type);
                    parseRotation(this.rotation);
                } catch (e) {
                    return false;
                }
                return this.positions().every(({x, y}) => {
                    return 0 <= x && x < 10 && 0 <= y && y < 23;
                });
            }
            copy() {
                return new Mino(this.type, this.rotation, this.x, this.y);
            }
        }
        this.Mino = Mino;
        const FieldConstants = {
            GarbageLine: 1,
            Width: 10,
            Height: 23,
            PlayBlocks: 23 * 10, // Height * Width
        };
        function createNewInnerField() {
            return new InnerField({});
        }
        function createInnerField(field) {
            const innerField = new InnerField({});
            for (let y = -1; y < FieldConstants.Height; y += 1) {
                for (let x = 0; x < FieldConstants.Width; x += 1) {
                    const at = field.at(x, y);
                    innerField.setNumberAt(x, y, parsePiece(at));
                }
            }
            return innerField;
        }
        class InnerField {
            constructor({
                            field = InnerField.create(FieldConstants.PlayBlocks),
                            garbage = InnerField.create(FieldConstants.Width),
                        }) {
                this.field = field;
                this.garbage = garbage;
            }
            static create(length) {
                return new PlayField({length});
            }
            fill(operation) {
                this.field.fill(operation);
            }
            fillAll(positions, type) {
                this.field.fillAll(positions, type);
            }
            canFill(piece, rotation, x, y) {
                const positions = getBlockPositions(piece, rotation, x, y);
                return positions.every(([px, py]) => {
                    return 0 <= px && px < 10
                        && 0 <= py && py < FieldConstants.Height
                        && this.getNumberAt(px, py) === Piece.Empty;
                });
            }
            canFillAll(positions) {
                return positions.every(({x, y}) => {
                    return 0 <= x && x < 10
                        && 0 <= y && y < FieldConstants.Height
                        && this.getNumberAt(x, y) === Piece.Empty;
                });
            }
            isOnGround(piece, rotation, x, y) {
                return !this.canFill(piece, rotation, x, y - 1);
            }
            clearLine() {
                this.field.clearLine();
            }
            riseGarbage() {
                this.field.up(this.garbage);
                this.garbage.clearAll();
            }
            mirror() {
                this.field.mirror();
            }
            shiftToLeft() {
                this.field.shiftToLeft();
            }
            shiftToRight() {
                this.field.shiftToRight();
            }
            shiftToUp() {
                this.field.shiftToUp();
            }
            shiftToBottom() {
                this.field.shiftToBottom();
            }
            copy() {
                return new InnerField({field: this.field.copy(), garbage: this.garbage.copy()});
            }
            equals(other) {
                return this.field.equals(other.field) && this.garbage.equals(other.garbage);
            }
            addNumber(x, y, value) {
                if (0 <= y) {
                    this.field.addOffset(x, y, value);
                } else {
                    this.garbage.addOffset(x, -(y + 1), value);
                }
            }
            setNumberFieldAt(index, value) {
                this.field.setAt(index, value);
            }
            setNumberGarbageAt(index, value) {
                this.garbage.setAt(index, value);
            }
            setNumberAt(x, y, value) {
                return 0 <= y ? this.field.set(x, y, value) : this.garbage.set(x, -(y + 1), value);
            }
            getNumberAt(x, y) {
                return 0 <= y ? this.field.get(x, y) : this.garbage.get(x, -(y + 1));
            }
            getNumberAtIndex(index, isField) {
                if (isField) {
                    return this.getNumberAt(index % 10, Math.floor(index / 10));
                }
                return this.getNumberAt(index % 10, -(Math.floor(index / 10) + 1));
            }
            toFieldNumberArray() {
                return this.field.toArray();
            }
            toGarbageNumberArray() {
                return this.garbage.toArray();
            }
        }
        class PlayField {
            constructor({pieces, length = FieldConstants.PlayBlocks}) {
                if (pieces !== undefined) {
                    this.pieces = pieces;
                } else {
                    this.pieces = Array.from({length}).map(() => Piece.Empty);
                }
                this.length = length;
            }
            static load(...lines) {
                const blocks = lines.join('').trim();
                return PlayField.loadInner(blocks);
            }
            static loadMinify(...lines) {
                const blocks = lines.join('').trim();
                return PlayField.loadInner(blocks, blocks.length);
            }
            static loadInner(blocks, length) {
                const len = length !== undefined ? length : blocks.length;
                if (len % 10 !== 0) {
                    throw new Error('Num of blocks in field should be mod 10');
                }
                const field = length !== undefined ? new PlayField({length}) : new PlayField({});
                for (let index = 0; index < len; index += 1) {
                    const block = blocks[index];
                    field.set(index % 10, Math.floor((len - index - 1) / 10), parsePiece(block));
                }
                return field;
            }
            get(x, y) {
                return this.pieces[x + y * FieldConstants.Width];
            }
            addOffset(x, y, value) {
                this.pieces[x + y * FieldConstants.Width] += value;
            }
            set(x, y, piece) {
                this.setAt(x + y * FieldConstants.Width, piece);
            }
            setAt(index, piece) {
                this.pieces[index] = piece;
            }
            fill({type, rotation, x, y}) {
                const blocks = getBlocks(type, rotation);
                for (const block of blocks) {
                    const [nx, ny] = [x + block[0], y + block[1]];
                    this.set(nx, ny, type);
                }
            }
            fillAll(positions, type) {
                for (const {x, y} of positions) {
                    this.set(x, y, type);
                }
            }
            clearLine() {
                let newField = this.pieces.concat();
                const top = this.pieces.length / FieldConstants.Width - 1;
                for (let y = top; 0 <= y; y -= 1) {
                    const line = this.pieces.slice(y * FieldConstants.Width, (y + 1) * FieldConstants.Width);
                    const isFilled = line.every(value => value !== Piece.Empty);
                    if (isFilled) {
                        const bottom = newField.slice(0, y * FieldConstants.Width);
                        const over = newField.slice((y + 1) * FieldConstants.Width);
                        newField = bottom.concat(over, Array.from({length: FieldConstants.Width}).map(() => Piece.Empty));
                    }
                }
                this.pieces = newField;
            }
            up(blockUp) {
                this.pieces = blockUp.pieces.concat(this.pieces).slice(0, this.length);
            }
            mirror() {
                const newField = [];
                for (let y = 0; y < this.pieces.length; y += 1) {
                    const line = this.pieces.slice(y * FieldConstants.Width, (y + 1) * FieldConstants.Width);
                    line.reverse();
                    for (const obj of line) {
                        newField.push(obj);
                    }
                }
                this.pieces = newField;
            }
            shiftToLeft() {
                const height = this.pieces.length / 10;
                for (let y = 0; y < height; y += 1) {
                    for (let x = 0; x < FieldConstants.Width - 1; x += 1) {
                        this.pieces[x + y * FieldConstants.Width] = this.pieces[x + 1 + y * FieldConstants.Width];
                    }
                    this.pieces[9 + y * FieldConstants.Width] = Piece.Empty;
                }
            }
            shiftToRight() {
                const height = this.pieces.length / 10;
                for (let y = 0; y < height; y += 1) {
                    for (let x = FieldConstants.Width - 1; 1 <= x; x -= 1) {
                        this.pieces[x + y * FieldConstants.Width] = this.pieces[x - 1 + y * FieldConstants.Width];
                    }
                    this.pieces[y * FieldConstants.Width] = Piece.Empty;
                }
            }
            shiftToUp() {
                const blanks = Array.from({length: 10}).map(() => Piece.Empty);
                this.pieces = blanks.concat(this.pieces).slice(0, this.length);
            }
            shiftToBottom() {
                const blanks = Array.from({length: 10}).map(() => Piece.Empty);
                this.pieces = this.pieces.slice(10, this.length).concat(blanks);
            }
            toArray() {
                return this.pieces.concat();
            }
            get numOfBlocks() {
                return this.pieces.length;
            }
            copy() {
                return new PlayField({pieces: this.pieces.concat(), length: this.length});
            }
            toShallowArray() {
                return this.pieces;
            }
            clearAll() {
                this.pieces = this.pieces.map(() => Piece.Empty);
            }
            equals(other) {
                if (this.pieces.length !== other.pieces.length) {
                    return false;
                }
                for (let index = 0; index < this.pieces.length; index += 1) {
                    if (this.pieces[index] !== other.pieces[index]) {
                        return false;
                    }
                }
                return true;
            }
        }
        function getBlockPositions(piece, rotation, x, y) {
            return getBlocks(piece, rotation).map((position) => {
                position[0] += x;
                position[1] += y;
                return position;
            });
        }
        function getBlockXYs(piece, rotation, x, y) {
            return getBlocks(piece, rotation).map((position) => {
                return {x: position[0] + x, y: position[1] + y};
            });
        }
        function getBlocks(piece, rotation) {
            const blocks = getPieces(piece);
            switch (rotation) {
                case Rotation.Spawn:
                    return blocks;
                case Rotation.Left:
                    return rotateLeft(blocks);
                case Rotation.Reverse:
                    return rotateReverse(blocks);
                case Rotation.Right:
                    return rotateRight(blocks);
            }
            throw new Error('Unsupported block');
        }
        function getPieces(piece) {
            switch (piece) {
                case Piece.I:
                    return [[0, 0], [-1, 0], [1, 0], [2, 0]];
                case Piece.T:
                    return [[0, 0], [-1, 0], [1, 0], [0, 1]];
                case Piece.O:
                    return [[0, 0], [1, 0], [0, 1], [1, 1]];
                case Piece.L:
                    return [[0, 0], [-1, 0], [1, 0], [1, 1]];
                case Piece.J:
                    return [[0, 0], [-1, 0], [1, 0], [-1, 1]];
                case Piece.S:
                    return [[0, 0], [-1, 0], [0, 1], [1, 1]];
                case Piece.Z:
                    return [[0, 0], [1, 0], [0, 1], [-1, 1]];
            }
            throw new Error('Unsupported rotation');
        }
        function rotateRight(positions) {
            return positions.map(current => [current[1], -current[0]]);
        }
        function rotateLeft(positions) {
            return positions.map(current => [-current[1], current[0]]);
        }
        function rotateReverse(positions) {
            return positions.map(current => [-current[0], -current[1]]);
        }
        let QuizOperation;
        (function (QuizOperation) {
            QuizOperation["Direct"] = "direct";
            QuizOperation["Swap"] = "swap";
            QuizOperation["Stock"] = "stock";
        })(QuizOperation || (QuizOperation = {}));
        class Quiz {
            constructor(quiz) {
                this.quiz = Quiz.verify(quiz);
            }
            get next() {
                const index = this.quiz.indexOf(')') + 1;
                const name = this.quiz[index];
                if (name === undefined || name === ';') {
                    return '';
                }
                return name;
            }
            static isQuizComment(comment) {
                return comment.startsWith('#Q=');
            }
            static create(first, second) {
                const create = (hold, other) => {
                    const parse = (s) => s ? s : '';
                    return new Quiz(`#Q=[${parse(hold)}](${parse(other[0])})${parse(other.substring(1))}`);
                };
                return second !== undefined ? create(first, second) : create(undefined, first);
            }
            static trim(quiz) {
                return quiz.trim().replace(/\s+/g, '');
            }
            get least() {
                const index = this.quiz.indexOf(')');
                return this.quiz.substr(index + 1);
            }
            get current() {
                const index = this.quiz.indexOf('(') + 1;
                const name = this.quiz[index];
                if (name === ')') {
                    return '';
                }
                return name;
            }
            get hold() {
                const index = this.quiz.indexOf('[') + 1;
                const name = this.quiz[index];
                if (name === ']') {
                    return '';
                }
                return name;
            }
            get leastAfterNext2() {
                const index = this.quiz.indexOf(')');
                if (this.quiz[index + 1] === ';') {
                    return this.quiz.substr(index + 1);
                }
                return this.quiz.substr(index + 2);
            }
            getOperation(used) {
                const usedName = parsePieceName(used);
                const current = this.current;
                if (usedName === current) {
                    return QuizOperation.Direct;
                }
                const hold = this.hold;
                if (usedName === hold) {
                    return QuizOperation.Swap;
                }
                // 次のミノを利用できる
                if (hold === '') {
                    if (usedName === this.next) {
                        return QuizOperation.Stock;
                    }
                } else {
                    if (current === '' && usedName === this.next) {
                        return QuizOperation.Direct;
                    }
                }
                throw new Error(`Unexpected hold piece in quiz: ${this.quiz}`);
            }
            get leastInActiveBag() {
                const separateIndex = this.quiz.indexOf(';');
                const quiz = 0 <= separateIndex ? this.quiz.substring(0, separateIndex) : this.quiz;
                const index = quiz.indexOf(')');
                if (quiz[index + 1] === ';') {
                    return quiz.substr(index + 1);
                }
                return quiz.substr(index + 2);
            }
            static verify(quiz) {
                const replaced = this.trim(quiz);
                if (replaced.length === 0 || quiz === '#Q=[]()' || !quiz.startsWith('#Q=')) {
                    return quiz;
                }
                if (!replaced.match(/^#Q=\[[TIOSZJL]?]\([TIOSZJL]?\)[TIOSZJL]*;?.*$/i)) {
                    throw new Error(`Current piece doesn't exist, however next pieces exist: ${quiz}`);
                }
                return replaced;
            }
            direct() {
                if (this.current === '') {
                    const least = this.leastAfterNext2;
                    return new Quiz(`#Q=[${this.hold}](${least[0]})${least.substr(1)}`);
                }
                return new Quiz(`#Q=[${this.hold}](${this.next})${this.leastAfterNext2}`);
            }
            swap() {
                if (this.hold === '') {
                    throw new Error(`Cannot find hold piece: ${this.quiz}`);
                }
                const next = this.next;
                return new Quiz(`#Q=[${this.current}](${next})${this.leastAfterNext2}`);
            }
            stock() {
                if (this.hold !== '' || this.next === '') {
                    throw new Error(`Cannot stock: ${this.quiz}`);
                }
                const least = this.leastAfterNext2;
                const head = least[0] !== undefined ? least[0] : '';
                if (1 < least.length) {
                    return new Quiz(`#Q=[${this.current}](${head})${least.substr(1)}`);
                }
                return new Quiz(`#Q=[${this.current}](${head})`);
            }
            operate(operation) {
                switch (operation) {
                    case QuizOperation.Direct:
                        return this.direct();
                    case QuizOperation.Swap:
                        return this.swap();
                    case QuizOperation.Stock:
                        return this.stock();
                }
                throw new Error('Unexpected operation');
            }
            format() {
                const quiz = this.nextIfEnd();
                if (quiz.quiz === '#Q=[]()') {
                    return new Quiz('');
                }
                const current = quiz.current;
                const hold = quiz.hold;
                if (current === '' && hold !== '') {
                    return new Quiz(`#Q=[](${hold})${quiz.least}`);
                }
                if (current === '') {
                    const least = quiz.least;
                    const head = least[0];
                    if (head === undefined) {
                        return new Quiz('');
                    }
                    if (head === ';') {
                        return new Quiz(least.substr(1));
                    }
                    return new Quiz(`#Q=[](${head})${least.substr(1)}`);
                }
                return quiz;
            }
            getHoldPiece() {
                if (!this.canOperate()) {
                    return Piece.Empty;
                }
                const name = this.hold;
                if (name === undefined || name === '' || name === ';') {
                    return Piece.Empty;
                }
                return parsePiece(name);
            }
            getNextPieces(max) {
                if (!this.canOperate()) {
                    return max !== undefined ? Array.from({length: max}).map(() => Piece.Empty) : [];
                }
                let names = (this.current + this.next + this.leastInActiveBag).substr(0, max);
                if (max !== undefined && names.length < max) {
                    names += ' '.repeat(max - names.length);
                }
                return names.split('').map((name) => {
                    if (name === undefined || name === ' ' || name === ';') {
                        return Piece.Empty;
                    }
                    return parsePiece(name);
                });
            }
            toString() {
                return this.quiz;
            }
            canOperate() {
                let quiz = this.quiz;
                if (quiz.startsWith('#Q=[]();')) {
                    quiz = this.quiz.substr(8);
                }
                return quiz.startsWith('#Q=') && quiz !== '#Q=[]()';
            }
            nextIfEnd() {
                if (this.quiz.startsWith('#Q=[]();')) {
                    return new Quiz(this.quiz.substr(8));
                }
                return this;
            }
        }
        this.Quiz = Quiz;
    }
}
//document.body.appendChild(document.createElement('script')).textContent = setupTrainingMaker.toString().match(/^function setupTrainingMaker\(\) \{((.|[\n\r])*)}$/)[1]
setupTrainingMaker();