Jstris Tutor Maker

Helps you make a Jstris usermode for placing a queue of pieces in the right spots

目前為 2023-01-15 提交的版本,檢視 最新版本

// ==UserScript==
// @name         Jstris Tutor Maker
// @license      BSD-2-Clause
// @namespace    Jstris Tutor Maker
// @version      0.4.1
// @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==
async function setupTutorMaker() {'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';

    // Thumbnail + first 3 bags of Qinghan's 2022-07-23 100-piece cheese PB (185 blocks): https://jstris.jezevec10.com/replay/57182729
    let ExampleFumen = 'v115@teilDeilSpglAehlVprlUphlQphlRphlQphlQphlRp?hlTpql3fAglteiWDeiWySgWAehW1SrW0ShWwShWxShWwShW?wShWxShWzSqWTeA8AeQ8AeH8AeE8AeK8AeF8AeJ8AeK8AeK?8AeA8Je/BQRDFLDmClcJSAVDEHBEooRBToAVBK+VWCaX9wC?v3/VCzijxCJ3jPCJ9aFDqOEWC6/9tCPOltCs/lFDPejxCM3?jPCsP9wCa9aFDvO8LCKdltCvXUPCUHExCKujFDqn9wCzyaF?DpyTxCvP9wCv/TFDT+VWCKnLuCUnzPCMujPCsHztCp+KWCq?eltC6/dgCpyjFDquLuCMuCMCq+aFDzSFgC6/9tCsyytCpXm?PCp+TWC6vaPCUHstCaebMCsvTxCvubMCzyaPCsAAAAvhEpF?BmOBNSBURBzTB+fRpwhQ4BtAewwBeRpwhF8AeF8AeJ8AeB8?xwQ4whBtA8QpA8AexwQ4FeA8AeAAMeAAAeA8IeAAAeA8IeA?AAeA8KepxAvhCdEB6HBTGBifwhIewhJeRpHeRphlBewwRpA?exwB8AeD8Q4xwh0B8QLxwQ4DeA8CeAADeAABeA8EeAAEeA8?DeA8AAGeA8CeAAFeAAA8YeMDBvhCaJBfPBKGBjfglHewhgl?HewhhlCehlQ4AexhQpCeAtglR4Aeg0B8AeE8Q4h0AeB8h0w?hA8Q4xwB8AeAtg0xhQ4CeA8BeAAEeA8AAJeA8AeAADeAADe?A8FeA8AAYe8CBvhCOHBJFBzGB1fwhAeglRpEexhglRpFewh?g0xwA8AeC8R4g0xwC8AeA8R4CeAAA8PeAAAeA8JeAAAeA8D?eA8DeAAEeAAAeA8FeAADeA8MefDBvhBFJBxNB';

    // IsChallengeMode decides whether this is a Tutor mode or a Challenge mode.
    let IsChallengeMode = false;

    // DemoStageRetries decides whether the player can try a stage again or gets a game over if they do it wrong
    let DemoStageRetries = true;

    // 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.8;

    // PauseHowLongBetweenPieces is the number of seconds to pause between steps in tutor mode
    let PauseHowLongOnLineClear = 0.2;

    // 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;

    // loadMapType is necessary because https://jstris.jezevec10.com/usermodes/create has a global JavaScript object named Map
    async function loadMapType() {
        const mapLoader = new Worker('data:application/javascript,onmessage = function () { postMessage(new%20Map()) }');
        return await new Promise(resolve => {
            mapLoader.onmessage = (event) => resolve(event.data.constructor);
            mapLoader.postMessage('Get Map type');
        });
    }

    const Map = await loadMapType();

    class Component {
        constructor(fieldType) {
            this.opts = {};
            this.on = true;
            this.id = ++Component.idCounter;
            this.field_type = fieldType;
        }

        toggle(state) {
            this.on = state;
        }
    }

    Component.idCounter = 0;

    class ComponentSwitch extends Component {
        constructor(...switchOptions) {
            super('switch');
            this.switchIndex = 0;
            this.add(...switchOptions);
        }

        add(...switchOptions) {
            for (const switchOption of switchOptions) {
                let suffix = this.switchIndex === 0 ? '' : this.switchIndex.toString();
                this.opts[`id${suffix}`] = switchOption.triggerID;
                this.opts[`on${suffix}`] = switchOption.on;
                ComponentSwitch.sharedSwitchIndex += 2;
                this.switchIndex = ComponentSwitch.sharedSwitchIndex;
            }
        }
    }

    ComponentSwitch.sharedSwitchIndex = 2;

    class Condition extends Component {
        constructor(conditionType, conditionValue, doIfTrue, conditionDo, conditionDo2 = null) {
            super('cond');
            this.opts['do'] = conditionDo;
            if (typeof conditionDo2 === 'string') {
                this.opts['do2'] = conditionDo2;
            }
            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;
        }
    }

    const TextPositionValueNextToBoard = 2;

    class Text extends Component {
        constructor(position, text) {
            super('text');
            this.opts['pos'] = position;
            this.opts['text'] = text;
        }
    }

    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 TriggerTypeOnEachBlock = 6;
    const TriggerTypeExternalConditional = 7;
    const TriggerTypeNever = 9;
    const TriggerTypeOnGameStart = 10;
    const TriggerIDTwoLinePC = '2_line_PC';
    const TriggerCheckBoard = 'CheckBoard';
    const TriggerCheckLoop = 'CheckLoop';
    const TriggerIDDefaultRuleset = 'default_ruleset';

    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();
    }

    async function newText(position, text) {
        await addComponent(new Text(position, text));
    }

    // 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 ConditionTypeCustomExpression = 0;
    const ConditionTypePCs = 7;
    const ConditionTypeHolds = 14;
    const ConditionTypeLines = 17;
    const ConditionResultTypeGameOver = 1;
    const ConditionResultTypeRunTriggerActions = 2;
    const ConditionResultTypeSuccessfulGameEnd = 3;

    async function newComponentSwitch(...switchOptions) {
        const componentSwitch = new ComponentSwitch(...switchOptions);
        await addComponent(componentSwitch);
        return componentSwitch;
    }

    // newCondition creates a new Condition component.
    async function newCondition(conditionType, conditionValue, doIfTrue, conditionDo, conditionDo2 = null) {
        const condition = new Condition(conditionType, conditionValue, doIfTrue, conditionDo, conditionDo2);
        await addComponent(condition);
        return condition;
    }

    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, pauseType) {
        await newRelativeTrigger(RelativeTriggerTypeTime, pauseType, demoTriggerID);
        await newTrigger(TriggerTypeExternalConditional, demoTriggerID);
        mapListForBlock.push(await newMap(MapTypeReplaceBoard));
    }

    // cycle goes through all the the steps to make sure the user placed a single piece correctly
    async function cycle(stageCount, stagePart, playTriggerID, placedStageTriggerIDs, judgeTriggerID, doneStageTriggerIDs, startingBlockCount, blockCount, lastCycleBlockCount, queues, holdPieces, mapListForBlock, lastCycleBeforeRise, startComponentsDisabled) {
        const placedTriggerID = placedStageTriggerIDs.get(stageCount, stagePart);
        const calculatedBlockCount = blockCount + stageCount - 1 + risesSoFar.get(blockCount) - risesAtStageEnd.get(blockCount);
        const stageFinalBlockCount = startingBlockCount + HowManyBlocksPerSection - 1;
        if (IsChallengeMode || blockCount > howManyDemoBlocks) {
            await newTrigger(TriggerTypeOnSpecificBlockNumber, calculatedBlockCount.toString());
        } else {
            const waitForBlocks = blockCount - lastCycleBlockCount;
            await newRelativeTrigger(RelativeTriggerTypeBlocks, waitForBlocks, placedTriggerID);
            if (blockCount === howManyDemoBlocks && DemoStageRetries && howManyDemoBlocks < HowManyBlocks && !hasRises) {
                await newComponentSwitch({triggerID: TriggerCheckLoop, on: true});
            }
            if (!hasRises) {
                await newRun(TriggerCheckBoard);
            }
            await newTrigger(TriggerTypeExternalConditional, placedTriggerID);
        }
        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);
        if ((!hasRises && blockCount === HowManyBlocks && blockCount % HowManyBlocksPerSection !== 0) || lastCycleBeforeRise || hasRises) {
            await newRun(TriggerIDTwoLinePC);
        }
        await newRelativeTrigger(RelativeTriggerTypeLines, 2, judgeTriggerID);
        await newTrigger(TriggerTypeExternalConditional, judgeTriggerID);
        const doneStageTriggerID = doneStageTriggerIDs.get(stageCount, stagePart);
        let enableComponentsBeforeJudge;
        if (!IsChallengeMode && DemoStageRetries && blockCount <= howManyDemoBlocks) {
            enableComponentsBeforeJudge = await newComponentSwitch({triggerID: doneStageTriggerID, on: true});
        }
        let conditionDo;
        let conditionDo2 = null;
        if (IsChallengeMode || blockCount > howManyDemoBlocks || !DemoStageRetries) {
            conditionDo = ConditionResultTypeGameOver;
        } else {
            conditionDo = ConditionResultTypeRunTriggerActions;
            conditionDo2 = doneStageTriggerIDs.getLastOfStage(stageCount - 1);
        }
        const relativeLinesCleared = totalLinesCleared[blockCount] + 2 * risesSoFar.get(blockCount) - totalLinesCleared[startingBlockCount - 1] - 2 * risesSoFar.get(startingBlockCount);
        await newCondition(ConditionTypeCustomExpression, `${playTriggerID}.lines=${relativeLinesCleared + 2}`, false, conditionDo, conditionDo2);
        const relativePCCount = actualPCCounts[blockCount] + risesSoFar.get(blockCount) - actualPCCounts[startingBlockCount - 1] - risesSoFar.get(startingBlockCount);
        const pcCondition = await newCondition(ConditionTypeCustomExpression, `${playTriggerID}.PC=${relativePCCount + 1}`, false, conditionDo, conditionDo2);
        if (!IsChallengeMode && DemoStageRetries && blockCount <= howManyDemoBlocks) {
            startComponentsDisabled.add({triggerID: `ID-${pcCondition.id}`, on: false});
            enableComponentsBeforeJudge.add({triggerID: `ID-${pcCondition.id}`, on: true});
        }
        let startComponentsDisabledNextStage;
        if (!IsChallengeMode && blockCount <= howManyDemoBlocks) {
            await newRun(doneStageTriggerID);
            await newTrigger(TriggerTypeExternalConditional, doneStageTriggerID);
            if (blockCount >= HowManyBlocks) {
                await newCondition(ConditionTypeCustomExpression, 'blocks>=0', true, ConditionResultTypeSuccessfulGameEnd);
            } else {
                if (DemoStageRetries && doneStageTriggerIDs.hasNext(doneStageTriggerID, stageCount) && blockCount === stageFinalBlockCount) {
                    startComponentsDisabledNextStage = await newComponentSwitch();
                    for (const [, doneStageTriggerID] of doneStageTriggerIDs.objects.get(stageCount + 1)) {
                        startComponentsDisabledNextStage.add({triggerID: doneStageTriggerID, on: false});
                    }
                }
            }
            if (lastCycleBeforeRise) {
                await updateMapContent(await newMap(MapTypeReplaceBoard), fumenWithRises.get(blockCount));
            } else {
                mapListForBlock.push(await newMap(MapTypeReplaceBoard));
            }
        } else if (blockCount === HowManyBlocks) {
            mapListForBlock.push(await newMap(MapTypeReplaceBoard));
        } else if (lastCycleBeforeRise && blockCount % HowManyBlocksPerSection !== 0) {
            await updateMapContent(await newMap(MapTypeReplaceBoard), fumenWithRises.get(blockCount));
        }
        return startComponentsDisabledNextStage;
    }

    async function makeDemoCycles(section, blockCount, totalBlocks, queue, finalTriggerID, mapListsByPieceIndex) {
        let holdPiece = demoHoldPieces[blockCount];
        await newText(TextPositionValueNextToBoard, `Stage ${section}`);
        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 (!mapListsByPieceIndex.has(blockCount)) {
                mapListsByPieceIndex.set(blockCount, new Array());
            }
            let pauseType = PauseHowLongBetweenPieces;
            if (fumenWithFullLines.has(blockCount)) {
                triggerSuffix = `_part_${triggerSection}`;
                let afterFullLinesTriggerID = `demo_block${blockCount}` + triggerSuffix;
                await newRelativeTrigger(RelativeTriggerTypeTime, pauseType, afterFullLinesTriggerID);
                pauseType = PauseHowLongOnLineClear;
                await newTrigger(TriggerTypeExternalConditional, afterFullLinesTriggerID);
                triggerSection++;
                triggerSuffix = `_part_${triggerSection}`;
                demoTriggerID = `demo_block${blockCount}` + triggerSuffix;
                await updateMapContent(await newMap(MapTypeReplaceBoard), fumenWithFullLines.get(blockCount));
            }
            if (fumenWithRises.has(blockCount)) {
                triggerSuffix = `_part_${triggerSection}`;
                pauseType = PauseHowLongBetweenPieces;
            }
            await demoCycle(blockCount, demoTriggerID, queue, mapListsByPieceIndex.get(blockCount), pauseType);
            pauseType = PauseHowLongBetweenPieces;
            if (fumenWithRises.has(blockCount)) {
                triggerSection++;
                let afterFullLinesTriggerID = `demo_block${blockCount}` + triggerSuffix;
                await newRelativeTrigger(RelativeTriggerTypeTime, pauseType, afterFullLinesTriggerID);
                await newTrigger(TriggerTypeExternalConditional, afterFullLinesTriggerID);
                triggerSection++;
                triggerSuffix = `_part_${triggerSection}`;
                demoTriggerID = `demo_block${blockCount}` + triggerSuffix;
                await updateMapContent(await newMap(MapTypeReplaceBoard), fumenWithRises.get(blockCount));
            }
            await newQueueChange(queue.slice(1), demoHoldPieces[blockCount], true, false);
            queue = queue.slice(1);
        }
        await newRelativeTrigger(RelativeTriggerTypeTime, PauseHowLongBetweenPieces * 2, finalTriggerID);
    }

    // 1 cycle for every HowManyBlocksPerSection blocks
    async function makeCycles(stageCount, playTriggerID, placedStageTriggerIDs, doneStageTriggerIDs, blockCount, lastBlockInStage, mapListsByPieceIndex, startComponentsDisabled) {
        let startingBlockCount = blockCount;
        let lastCycleBlockCount = blockCount - 1;
        let lastCycleBeforeRise = false;
        let stagePart = 1;
        let startComponentsDisabledNextStage;
        for (; blockCount <= lastBlockInStage; blockCount++) {
            if (!mapListsByPieceIndex.has(blockCount)) {
                mapListsByPieceIndex.set(blockCount, new Array());
            }
            const lastCycleInStage = blockCount === lastBlockInStage;
            lastCycleBeforeRise = fumenWithRises.has(blockCount);
            if (lastCycleInStage || lastCycleBeforeRise) {
                let judgeTriggerID = `judge_stage${stageCount}`;
                if (lastCycleBeforeRise || stagePart > 1) {
                    judgeTriggerID += `_p${stagePart}`;
                }
                startComponentsDisabledNextStage = await cycle(stageCount, stagePart, playTriggerID, placedStageTriggerIDs, judgeTriggerID, doneStageTriggerIDs, startingBlockCount, blockCount, lastCycleBlockCount, queues, holdPieces, mapListsByPieceIndex.get(blockCount), lastCycleBeforeRise, startComponentsDisabled);
                stagePart++;
                lastCycleBlockCount = blockCount;
            }
            if (lastCycleBeforeRise) {
                lastCycleBeforeRise = false;
            }
        }
        if (!IsChallengeMode && blockCount <= howManyDemoBlocks) {
            await newQueueChange(demoQueues[blockCount], demoHoldPieces[blockCount], true, false, true);
        }
        return startComponentsDisabledNextStage;
    }

    async function initUserMode(queue, placedStageTriggerIDs, doneStageTriggerIDs) {
        await newTrigger(TriggerTypeBeforeGame, null);
        await newQueueChange(queue, QueueHoldPieceNone, true, false);
        await newTrigger(TriggerTypeOnGameStart, null);
        if (!IsChallengeMode) {
            await newQueueChange(queue, QueueHoldPieceNone, true, false);
            if (DemoStageRetries && howManyDemoBlocks < HowManyBlocks) {
                await newComponentSwitch({triggerID: TriggerCheckLoop, on: false});
            }
        }
        const initDoneTriggerID = doneStageTriggerIDs.get(0, 1);
        await newRun(initDoneTriggerID);
        await newTrigger(TriggerTypeExternalConditional, initDoneTriggerID);
        let startComponentsDisabled;
        if (!IsChallengeMode && DemoStageRetries) {
            startComponentsDisabled = await newComponentSwitch();
            for (const [, doneStageTriggerID] of doneStageTriggerIDs.objects.get(1)) {
                startComponentsDisabled.add({triggerID: doneStageTriggerID, on: false});
            }
        }
        await newRun(TriggerIDDefaultRuleset);
        return startComponentsDisabled;
    }

    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 retriesID = 'demo-stage-retries';
    const fumenInputID = 'fumen-input';
    const tutorModeID = 'usermode-type-tutor';
    const challengeModeID = 'usermode-type-challenge';
    const timePerPieceID = 'time-per-piece';
    const lineClearDelayID = 'line-clear-delay';

    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 or replay 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 delaysContainer = 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;
        delaysContainer.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(delaysContainer, 'col-sm-3').appendChild(timePerPieceInput);
        const lineClearDelayLabel = document.createElement('label');
        lineClearDelayLabel.classList.add('col-sm-3', 'control-label');
        lineClearDelayLabel.textContent = 'Demo line clear delay';
        lineClearDelayLabel.htmlFor = lineClearDelayID;
        delaysContainer.appendChild(lineClearDelayLabel);
        const lineClearDelayInput = document.createElement('input');
        for (const attribute in inputAttributes) {
            lineClearDelayInput.setAttribute(attribute, inputAttributes[attribute]);
        }
        lineClearDelayInput.placeholder = '(seconds)';
        lineClearDelayInput.value = PauseHowLongOnLineClear.toString();
        lineClearDelayInput.id = lineClearDelayID;
        newDiv(delaysContainer, 'col-sm-3').appendChild(lineClearDelayInput);
        const retriesContainer = newDiv(newDiv(inputsContainer, 'form-group'), 'row');
        const retriesLabel = document.createElement('label');
        retriesLabel.textContent = 'Demo stage retries';
        retriesLabel.classList.add('col-sm-3', 'control-label');
        retriesLabel.htmlFor = retriesID;
        retriesContainer.appendChild(retriesLabel);
        const retriesSelectContainer = newDiv(retriesContainer, 'col-sm-3');
        const retriesSelect = document.createElement('select');
        retriesSelect.classList.add('form-control');
        retriesSelect.id = retriesID;
        const retryOptions = {'Yes': true, 'No': false};
        for (const retryText in retryOptions) {
            const option = document.createElement('option');
            const optionValue = retryOptions[retryText];
            option.value = optionValue.toString();
            option.textContent = retryText;
            if (optionValue === DemoStageRetries) {
                option.textContent += ' (default)';
                option.selected = true;
            }
            retriesSelect.appendChild(option);
        }
        retriesContainer.appendChild(retriesSelectContainer).appendChild(retriesSelect);
        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 = new Map();
    let fumenWithFullLines = new Map();
    let fumenWithRises = new Map();
    let risesSoFar = new Map();
    let risesAtStageEnd = new Map();
    let hasRises;

    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);
        }
    }

    class PartMapsByStage {
        constructor() {
            this.objects = new Map();
        }

        add(stage, value) {
            let mapForStage;
            if (!this.objects.has(stage)) {
                mapForStage = new Map();
                this.objects.set(stage, mapForStage);
            } else {
                mapForStage = this.objects.get(stage);
            }
            mapForStage.set(mapForStage.size + 1, value);
        }

        get(stage, part = 1) {
            return this.objects.get(stage).get(part);
        }

        getPrevious(entity, stage) {
            const mapForStage = this.objects.get(stage);
            let index;
            let value;
            for ([index, value] of mapForStage) {
                if (entity === value) {
                    break;
                }
            }
            if (mapForStage.has(index - 1)) {
                return mapForStage.get(index - 1);
            }
            const previousStage = this.objects.get(stage - 1);
            return previousStage.get(previousStage.size);
        }

        hasNext(entity, stage) {
            return this.hasNextInStage(entity, stage) || this.objects.has(stage + 1);
        }

        hasNextInStage(entity, stage) {
            const mapForStage = this.objects.get(stage);
            let index;
            let value;
            for ([index, value] of mapForStage) {
                if (entity === value) {
                    break;
                }
            }
            return mapForStage.has(index + 1);
        }

        getNext(entity, stage) {
            const mapForStage = this.objects.get(stage);
            let index;
            let value;
            for ([index, value] of mapForStage) {
                if (entity === value) {
                    break;
                }
            }
            if (mapForStage.has(index + 1)) {
                return mapForStage.get(index + 1);
            }
            return this.objects.get(stage + 1).get(1);
        }

        getLastOfStage(stage) {
            const mapForStage = this.objects.get(stage);
            return mapForStage.get(mapForStage.size);
        }

        isLastOfStage(entity, stage) {
            const mapForStage = this.objects.get(stage);
            return entity === mapForStage.get(mapForStage.size);
        }
    }

    let queues;
    let holdPieces;
    let demoQueues;
    let demoHoldPieces;

    async function loadComponents(thumbnailContent = undefined) {
        if (typeof thumbnailContent === 'string') {
            await newTrigger(TriggerTypeNever);
            await newMap(MapTypeReplaceBoard, thumbnailContent);
        }
        mapListsByPieceIndex = new Map();
        hasHold = BlockQueue.search(QueueHoldPiece) > -1;
        setDefaultRuleset();
        if (!(fumenSaveButton() instanceof HTMLButtonElement)) {
            await fumenSection();
        }
        setTotalSections();
        if (HowManyBlocks > 0) {
            let firstSection = true;
            queues = [];
            holdPieces = [];
            demoQueues = [];
            demoHoldPieces = [];
            buildQueues('');
            const doneStageTriggerIDs = new PartMapsByStage();
            doneStageTriggerIDs.add(0, 'DnSt0');
            let sectionCount = 1;
            let partCount = 1;
            for (let blockCount = 1; blockCount <= HowManyBlocks; blockCount++) {
                if (blockCount % HowManyBlocksPerSection === 0 || blockCount === HowManyBlocks) {
                    let triggerID = `DnSt${sectionCount}`;
                    if (partCount > 1) {
                        triggerID += `p${partCount}`;
                    }
                    doneStageTriggerIDs.add(sectionCount, triggerID);
                    sectionCount++;
                    partCount = 1;
                } else if (fumenWithRises.has(blockCount)) {
                    doneStageTriggerIDs.add(sectionCount, `DnSt${sectionCount}p${partCount}`);
                    partCount++;
                }
            }
            const placedStageTriggerIDs = new PartMapsByStage();
            sectionCount = 1;
            partCount = 1;
            for (let blockCount = 1; blockCount <= HowManyBlocks; blockCount++) {
                if (blockCount % HowManyBlocksPerSection === 0 || blockCount === HowManyBlocks) {
                    let triggerID = `plc${sectionCount}`;
                    if (partCount > 1) {
                        triggerID += `p${partCount}`;
                    }
                    placedStageTriggerIDs.add(sectionCount, triggerID);
                    sectionCount++;
                    partCount = 1;
                } else if (fumenWithRises.has(blockCount)) {
                    placedStageTriggerIDs.add(sectionCount, `plc${sectionCount}p${partCount}`);
                    partCount++;
                }
            }
            let startComponentsDisabled = await initUserMode(queues[1], placedStageTriggerIDs, doneStageTriggerIDs);
            mapListsByPieceIndex.set(0, Array(await newMap(MapTypeReplaceBoard)));
            for (let section = 1; section <= totalSections; section++) {
                const playTriggerID = `Stage${section}`;
                let sectionBeginningBlockCount = (section - 1) * HowManyBlocksPerSection + 1;
                let sectionFinalBlockCount = section * HowManyBlocksPerSection;
                if (sectionFinalBlockCount > HowManyBlocks) {
                    sectionFinalBlockCount = HowManyBlocks;
                }
                if (!IsChallengeMode && sectionBeginningBlockCount <= howManyDemoBlocks) {
                    await makeDemoCycles(section, sectionBeginningBlockCount, sectionFinalBlockCount, demoQueues[sectionBeginningBlockCount], playTriggerID, mapListsByPieceIndex);
                } else {
                    await newRun(playTriggerID);
                }
                await newTrigger(TriggerTypeExternalConditional, playTriggerID);
                const transitionMap = await newMap(MapTypeReplaceBoard);
                if (fumenWithRises.has(sectionBeginningBlockCount - 1)) {
                    await updateMapContent(transitionMap, fumenWithRises.get(sectionBeginningBlockCount - 1));
                } else {
                    mapListsByPieceIndex.get(sectionBeginningBlockCount - 1).push(transitionMap);
                }
                if (firstSection || (!IsChallengeMode && sectionBeginningBlockCount <= howManyDemoBlocks)) {
                    await newQueueChange(queues[sectionBeginningBlockCount], holdPieces[sectionBeginningBlockCount], true, false, (IsChallengeMode || sectionBeginningBlockCount > howManyDemoBlocks) && holdPieces[sectionBeginningBlockCount].length === 1);
                }
                if (firstSection) {
                    firstSection = false;
                }
                startComponentsDisabled = await makeCycles(section, playTriggerID, placedStageTriggerIDs, doneStageTriggerIDs, sectionBeginningBlockCount, sectionFinalBlockCount, mapListsByPieceIndex, startComponentsDisabled);
                if (sectionFinalBlockCount === HowManyBlocks) {
                    break;
                }
            }
            // conditional trigger 2_line_PC, which immediately clears 2 lines
            await newTrigger(TriggerTypeExternalConditional, TriggerIDTwoLinePC);
            await newMap(MapTypeAddToCurrentBoardOnTop);
            await newRuleset(RulesetTypeFastDropLock);
            await newRelativeTrigger(RelativeTriggerTypeLines, 2, TriggerIDDefaultRuleset);
            // conditional trigger default_ruleset, which reverts the ruleset, waits for
            // 7 more blocks to be placed, then loops back to 2_line_PC
            await newTrigger(TriggerTypeExternalConditional, TriggerIDDefaultRuleset);
            await newRuleset(RulesetTypeDefault);
            if (!IsChallengeMode) {
                if (howManyDemoBlocks < HowManyBlocks) {
                    await newRun(TriggerCheckLoop);
                    await newTrigger(TriggerTypeExternalConditional, TriggerCheckLoop);
                    await newRelativeTrigger(RelativeTriggerTypeBlocks, 7, TriggerIDTwoLinePC);
                }
                if (!hasRises) {
                    await newTrigger(TriggerTypeExternalConditional, TriggerCheckBoard);
                }
            }
            if (!hasRises) {
                await newRelativeTrigger(RelativeTriggerTypeBlocks, 7, TriggerIDTwoLinePC);
            }
        }
    }

    await (async function () {
        await fumenSection();
    })();
    const JstrisPiece = new Map(Object.entries({
        'Empty': 0,
        'Z': 1,
        'L': 2,
        'O': 3,
        'S': 4,
        'I': 5,
        'J': 6,
        'T': 7,
        'Gray': 8,
    }));
    const JstrisPieceByFumenPiece = new Map();

    function mapFumenPiecesToJstrisPieces(fumen) {
        for (const [piece, jstrisPiece] of JstrisPiece) {
            JstrisPieceByFumenPiece.set(fumen.Piece[piece], jstrisPiece);
        }
    }

    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;

    function piecesToCompareString(pieces) {
        return pieces.slice(0, columnCount * (rowCount - 1)).join();
    }

    function pagesAreEqual(pieces1, pieces2) {
        const firstCompareString = piecesToCompareString(pieces1);
        const secondCompareString = piecesToCompareString(pieces2);
        return firstCompareString === secondCompareString;
    }

    async function fumenFromReplay(fumen, input) {
        let xhr = new XMLHttpRequest();
        input = await new Promise((resolve, reject) => {
            xhr.onload = () => {
                if (xhr.status !== 200) {
                    return reject();
                }
                return resolve(JSON.parse(xhr.responseText).fumen);
            };
            xhr.open('POST', 'https://fumen.tstman.net/jstris', true);
            xhr.send(`replay=${input}`);
        });
        const pages = fumen.decode(input);
        pages['type'] = 'replay';
        return pages;
    }

    async function fumenFromInput(fumen, input) {
        input = input.trim();
        try {
            const pages = fumen.decode(input);
            pages['type'] = 'fumen';
            return pages;
        } catch (err) {
        }
        return await fumenFromReplay(fumen, input);
    }

    let capitalizedType;

    async function loadFumenToMaps() {
        capitalizedType = 'Fumen';
        componentList = [];
        const generateProgress = () => updateStatus(`Generated ${componentList.length} components`);
        const loadProgress = () => {
            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 = capitalizedType + ' loading...';
        fumenButton.classList.add('disabled');
        fumenButton.setAttribute('disabled', '');
        let fumen = new Fumen();
        mapFumenPiecesToJstrisPieces(fumen);
        let inputElement = document.querySelector(`#${fumenInputID}`);
        const pages = await fumenFromInput(fumen, inputElement.value);
        capitalizedType = pages['type'][0].toUpperCase() + pages['type'].slice(1);
        let loadingText = capitalizedType + ' loading...';
        if (fumenButton.textContent !== loadingText) {
            fumenButton.textContent = loadingText;
        }
        await sleep();
        let thumbnailContent;
        if (pages[0].flags.quiz === false && !(pages[0].operation instanceof fumen.Mino)) {
            hasThumbnail = true;
            thumbnailContent = fumenToMapData(fumen, pages.shift()['_field'].field['pieces']);
        }
        const fumenForFirstPage = fumenToMapData(fumen, pages[0]['_field'].field['pieces']);
        BlockQueue = blockQueueFromPages(fumen, pages);
        totalLinesCleared = Array(pages.length + 1).fill(0);
        actualPCCounts = Array(pages.length + 1).fill(0);
        fumenWithFullLines = new Map();
        fumenWithRises = new Map();
        risesSoFar = new Map();
        risesAtStageEnd = new Map();
        hasRises = false;
        let cumulativeRises = 0;
        let cumulativeRisesAtStageEnd = 0;
        let cumulativeLinesCleared = 0;
        let pieceIndex = 1;
        let previousPieces = pages[0]['_field'].field.pieces;
        for (const page of pages) {
            if (HowManyBlocks > 0 && pieceIndex > HowManyBlocks) {
                break;
            }
            risesSoFar.set(pieceIndex - 1, cumulativeRises);
            risesAtStageEnd.set(pieceIndex - 1, cumulativeRisesAtStageEnd);
            if (!pagesAreEqual(previousPieces, page['_field'].field.pieces)) {
                const riseIndex = pieceIndex - 1;
                cumulativeRises++;
                fumenWithRises.set(riseIndex, fumenToMapData(fumen, page['_field'].field.pieces));
                if (riseIndex % HowManyBlocksPerSection === 0) {
                    cumulativeRisesAtStageEnd++;
                }
            }
            addMinoToField(fumen, page);
            const linesBeforeClearing = page['_field'].field.pieces.slice();
            const clearedThisPage = countLineClears(fumen, page, pieceIndex);
            if (clearedThisPage > 0) {
                fumenWithFullLines.set(pieceIndex, fumenToMapData(fumen, linesBeforeClearing));
            }
            cumulativeLinesCleared += clearedThisPage;
            totalLinesCleared[pieceIndex] = cumulativeLinesCleared;
            pieceIndex++;
            previousPieces = page['_field'].field.pieces.slice();
        }
        risesSoFar.set(pieceIndex - 1, cumulativeRises);
        risesAtStageEnd.set(pieceIndex - 1, cumulativeRisesAtStageEnd);
        if (cumulativeRises > 0) {
            hasRises = true;
        }
        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;
        DemoStageRetries = document.querySelector(`#${retriesID}`).value === 'true';
        PauseHowLongBetweenPieces = parseFloat(document.querySelector(`#${timePerPieceID}`).value);
        PauseHowLongOnLineClear = parseFloat(document.querySelector(`#${lineClearDelayID}`).value);
        await loadComponents(thumbnailContent);
        pieceIndex = 1;
        await updateSectionMapContent(mapListsByPieceIndex.get(0), fumenForFirstPage);
        for (const page of pages) {
            if (HowManyBlocks > 0 && pieceIndex > HowManyBlocks) {
                break;
            }
            await updateSectionMapContent(mapListsByPieceIndex.get(pieceIndex), fumenToMapData(fumen, page['_field'].field['pieces']));
            pieceIndex++;
        }
        clearInterval(progressInterval);
        loadProgress();
        // Progress for Backbone.js rendering 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();
        await scrubComponents();
        // Load in a separate thread in an attempt to keep the page from locking up while it's loading
        setTimeout(loadUsermodeForm, 0);
    }

    async function scrubComponents() {
        for (const component of componentList) {
            if (component instanceof ComponentSwitch) {
                delete component.switchIndex;
            }
        }
    }

    async function loadingDone() {
        const fumenButton = fumenSaveButton();
        clearInterval(progressInterval);
        fumenButton.classList.add('btn-success');
        fumenButton.classList.remove('btn-warning');
        fumenButton.textContent = capitalizedType + ' loaded!';
        fumenButton.classList.remove('disabled');
        fumenButton.removeAttribute('disabled');
        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.get(fieldPieces[columnCount * row + column]) << 4 |
                    JstrisPieceByFumenPiece.get(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 {
            static create(field, garbage) {
                return new Field(new InnerField({
                    field: field !== undefined ? PlayField.load(field) : undefined,
                    garbage: garbage !== undefined ? PlayField.loadMinify(garbage) : undefined,
                }));
            }

            constructor(field) {
                this.field = field;
            }

            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 {
            static from(operation) {
                return new Mino(operation.type, operation.rotation, operation.x, operation.y);
            }

            constructor(type, rotation, x, y) {
                this.type = type;
                this.rotation = rotation;
                this.x = x;
                this.y = 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 {
            static create(length) {
                return new PlayField({length});
            }

            constructor({
                            field = InnerField.create(FieldConstants.PlayBlocks),
                            garbage = InnerField.create(FieldConstants.Width),
                        }) {
                this.field = field;
                this.garbage = garbage;
            }

            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 {
            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;
            }

            constructor({pieces, length = FieldConstants.PlayBlocks}) {
                if (pieces !== undefined) {
                    this.pieces = pieces;
                } else {
                    this.pieces = Array.from({length}).map(() => Piece.Empty);
                }
                this.length = length;
            }

            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 {
            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, '');
            }

            constructor(quiz) {
                this.quiz = Quiz.verify(quiz);
            }

            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 = setupTutorMaker.toString().match(/^function setupTutorMaker\(\) \{((.|[\n\r])*)}$/)[1]
setupTutorMaker();