Advanced League Customization in the Worker Console


What is the worker console?

Since ZenGM Baseball runs inside your web browser, you can edit pretty much any part of the game from within your web browser. You do this by using the JavaScript console in your browser to access the core of the game. There, you will find an API that you can use to run little scripts that edit your league.

This page explains how to access the console and shows examples of what you can do there.

How do I access the worker console?

You can easily access a simple version of the worker console at Tools > Danger Zone > Worker Console within any league. However, if you're doing a lot there, you may find it preferable to use your browser's built-in console. It's a little harder to access, but has a lot more functionality. How to access it depends on your browser:

Chrome:

  1. Open your league in ZGMB.
  2. Open a new tab, copy chrome://inspect/#workers into the address bar and hit enter.
  3. Click "Inspect" under the URL similar to https://baseball.zengm.com/gen/worker-2020.08.02.0001.js and it will pop up another window where you can run the code snippets below.

Firefox:

  1. Open your league in ZGMB.
  2. Open a new tab, copy about:debugging#/runtime/this-firefox into the address bar and hit enter.
  3. Scroll down (maybe way down) until you see the "Shared Workers" section. There should be an entry there similar to /gen/worker-2020.08.02.0001.js.
  4. Click "Inspect" next to it. That will open a new tab where you can run the code snippets below.

Only needed for Safari 14 or later, and very old versions of other browsers! If your code uses the keyword await (which is almost always true), you need to wrap it in an async function. That means you need to put (async () => { at the beginning of the code and })(); at the end.

Code snippets


Buff/nerf player ratings

var players = await bbgm.idb.cache.players.getAll();
for (const p of players) {
    const ratings = p.ratings.at(-1);

    // Increase speed by 20%
    ratings.spd = bbgm.player.limitRating(ratings.spd * 1.2);

    // Decrease height by 10
    ratings.hgt = bbgm.player.limitRating(ratings.hgt - 10);

    // Recompute ovr and pot
    await bbgm.player.develop(p, 0);

    // Recompute player value
    await bbgm.player.updateValues(p);

    await bbgm.idb.cache.players.put(p);
}

That will increase the speed rating by 20% and decrease the height rating by 10, while bounding them to the normal 0-100 range.

You can do other stuff to player ratings. These are the names of all the rating variables:

hgt
spd
hpw
con
eye
gnd
fly
thr
cat
ppw
ctl
mov
endu

The above code runs on all active players and draft prospects. If instead you want to run on some subset of players, you can. Generally, this code looks like:

var players = await bbgm.idb.cache.players.getAll();
for (const p of players) {
    if (SOME_CONDITION) {
        const ratings = p.ratings.at(-1);
        ratings.spd = bbgm.player.limitRating(ratings.spd * 1.2);
        ratings.hgt = bbgm.player.limitRating(ratings.hgt - 10);
        await bbgm.player.develop(p, 0);
        await bbgm.player.updateValues(p);
        await bbgm.idb.cache.players.put(p);
    }
}

That's saying if SOME_CONDITION is true, only then run the code to update the ratings. But what is SOME_CONDITION? Some examples:

You can combine these conditions too, like this is all draft prospects at one position: if (p.tid === bbgm.PLAYER.UNDRAFTED && p.ratings.at(-1).pos === "LF") {

Conditions like this can similarly be used for many of the code snippets below too.

Generate a random player

async function createPlayer(age, ovr, country) {
    const name = await bbgm.player.name(country);
    const p = bbgm.player.generate(bbgm.PLAYER.FREE_AGENT, age, bbgm.g.get("season") - 1, false, 34, name);
    await bbgm.player.develop(p, 0);
    const ratings = p.ratings.at(-1);
    ratings.season = bbgm.g.get("season");

    // Adjust ratings up/down
    const maxTries = 100;
    let i = 0;
    if (ratings.ovr > ovr) {
        while (ratings.ovr > ovr) {
            for (const key of bbgm.RATINGS) {
                ratings[key] = bbgm.player.limitRating(ratings[key] - 1);
                await bbgm.player.develop(p, 0);
                if (ratings.ovr <= ovr) {
                    break;
                }
            }
        }
        await bbgm.player.updateValues(p);
    } else if (ratings.ovr < ovr) {
        while (ratings.ovr < ovr) {
            for (const key of bbgm.RATINGS) {
                ratings[key] = bbgm.player.limitRating(ratings[key] + 1);
                await bbgm.player.develop(p, 0);
                if (ratings.ovr >= ovr) {
                    break;
                }
            }
        }
        await bbgm.player.updateValues(p);
    }

    return p;
}

var age = 25;
var ovr = 70;
var p = await createPlayer(age, ovr, "England");

await bbgm.idb.cache.players.put(p);
console.log(`Created ${p.firstName} ${p.lastName}`);

That part at the bottom is customizable. The above code creates a 25 year old 70 ovr player from England. You could also do something like this:

var age = bbgm.random.randInt(25, 30);
var ovr = bbgm.random.randInt(50, 70);
var p = await createPlayer(age, ovr);
p.tid = 5;
p.contract = {
    amount: 5000,
    exp: 2028,
};
p.college = "Nebraska";

await bbgm.idb.cache.players.put(p);
console.log(`Created ${p.firstName} ${p.lastName}`);

That creates a 25-30 year old player with 50-70 ovr from a random country, and puts him on team 5 (based on team ID number) with a $5M/year contract expiring in 2028. And it also sets his college to Nebraska.

"Lock ratings" for all active players

var players = await bbgm.idb.cache.players.getAll();
for (const p of players) {
    p.ratings.at(-1).locked = true;
    await bbgm.idb.cache.players.put(p);
}

List all the players who have died

for await (const { value: p } of bbgm.idb.league.transaction("players").store) {
    if (typeof p.diedYear === "number") {
        console.log(`${p.firstName} ${p.lastName} (${p.born.year}-${p.diedYear})`);
    }
}

Change GM History teams

On August 20, 2020, a feature was added to view your overall performance as a GM across multiple different past teams. But prior to this update, it did not track which teams you controlled in the past. If you remember, you may want to manually specify your past teams.

The numbers in the value part are team ID numbers, which you can see in URLs when you click on a team.

The numbers in the start part are the season when you took control of that team. The first one must be -Infinity, representing the team you started with in this league.

The last entry must be your current team. If you just took control of the team after controlling another team for the season and playoffs this year, then use next season for start.

var history = [
    { start: -Infinity, value: 5 },
    { start: 2025, value: 7 },
    { start: 2028, value: 9 },
];

bbgm.g.userTid = history;
await bbgm.idb.cache.gameAttributes.put({
    key: "userTid",
    value: history,
});

Delete all players at a certain position

var players = await bbgm.idb.cache.players.getAll();
var pids = []
for (const p of players) {
    if (p.ratings.at(-1).pos === "LF") {
        pids.push(p.pid);
    }
}
await bbgm.player.remove(pids);

Kill a player with a custom tragic death

Replace 155 with the player ID number of the player you want to kill.

async function customTragedy(pid, reason) {
    const p = await bbgm.idb.getCopy.players({ pid });
    const tid = p.tid;
    await bbgm.player.retire(p, undefined, {
        logRetiredEvent: false,
    });
    p.diedYear = bbgm.g.get("season");
    await bbgm.idb.cache.players.put(p);
    bbgm.logEvent(
        {
            type: "tragedy",
            text: `<a href="${bbgm.helpers.leagueUrl(["player", p.pid])}">${p.firstName} ${
                p.lastName
            }</a> ${reason}.`,
            showNotification: true,
            pids: [p.pid],
            tids: [tid],
            persistent: true,
            score: 20,
        },
    );
}

customTragedy(155, "died with whatever text you want");

Change the jersey number of a player

var player_id_number = 150;
var new_jeresy_number = "45";

var p = await bbgm.idb.getCopy.players({ pid: player_id_number });
if (p.stats.length > 0) {
    p.stats.at(-1).jerseyNumber = new_jeresy_number;
} else {
    p.jerseyNumber = new_jeresy_number;
}
await bbgm.idb.cache.players.put(p);

Add custom award to a player

This will show up only on the player profile page, not on any other pages in the game.

var player_id_number = 150;
var p = await bbgm.idb.getCopy.players({ pid: player_id_number });
p.awards.push({
    season: 2025,
    type: "My Custom Award",
});
await bbgm.idb.cache.players.put(p);

To remove that custom award:

var player_id_number = 150;
var p = await bbgm.idb.getCopy.players({ pid: player_id_number });
p.awards = p.awards.filter(award => {
    if (award.season === 2025 && award.type === "My Custom Award") {
        return false;
    }

    return true;
});
await bbgm.idb.cache.players.put(p);