scaffold-gateables
Performance Notes
- Take your time to understand the game before proposing features
- Quality is more important than speed — wrong gateables hurt the game permanently
- Do not skip the locked-path verification in Step 4
Scaffold Gateables
Add gateable features to an existing game. This skill produces hooks — skin pickers, continue-after-death flows, bonus modes, save slots, daily challenges — each wired through EventBus with a single capability-check seam (isEntitled(key)). It does NOT add any specific monetization SDK (Play.fun, sub.games, Stripe, etc.); it produces the scaffolding that any paywall, subscription, or entitlement layer can later switch on.
The default state is everything locked — isEntitled(key) returns false for all keys. The game must feel completely normal in that state. When a downstream monetization layer flips an entitlement on, the gateable feature lights up.
Core Principles
Use these to reason about what to propose and what to reject. They are rules, not guidelines.
- Never gate the core loop. If the game is about jumping, everyone jumps. Gate skins, not gravity.
- Prefer additive over subtractive. Add a bonus mode; don't shrink the free mode. Free players never feel punished.
- Match the game's rhythm. Short sessions → session-scoped perks (continue, daily skin, daily challenge seed). Long-form → persistence (save slots, unlock tree, bonus chapters).
- Propose 2–3 gateables at silver and gold only. Bronze is the default — every player gets the bronze experience by default, so no feature is scaffolded there. Aim for one or two silver conveniences and one gold spectacle. Numbers are targets — suggest what actually fits the game.
- Every gateable must degrade cleanly. With
isEntitledreturningfalse, the game must feel normal — not broken, not missing.
Tier Vocabulary
Use game-standard progression labels: bronze / silver / gold. This keeps the skill monetization-agnostic and reads naturally to players.
Bronze is the default tier — every player is bronze by default. The bronze experience IS the core game loop that everyone always gets. No feature is scaffolded at bronze; scaffold-gateables only produces features at silver and gold.
The downstream integration layer maps silver/gold to its own paid tier system:
| scaffold-gateables | sub.games | Play.fun (points-based) |
|---|---|---|
| silver | supporter ($3–8/mo) | mid-threshold points |
| gold | founder ($150/yr) | high-threshold points |
State this mapping in the Step 2 output so the handoff to /subgames or /monetize-game is explicit.
Instructions
The user wants to add gateables to the game at $ARGUMENTS (or the current directory if no path given). Optional hint tokens after the path (e.g., skins continue) bias the proposal.
Step 1: Understand the game
- Read
package.json— detect engine (phaser→ Phaser,three→ Three.js). - Read
src/core/Constants.js,src/core/EventBus.js,src/core/GameState.js. - Read
src/main.jsto findwindow.render_game_to_text()and the orchestrator entry point. - Read
progress.mdat the project root if it exists (written by the/make-gamepipeline). - Read the primary scene/system files to understand the core loop.
Then tell the creator one sentence:
The core loop is
<verb>— I will not gate it. Gateables will be additive features that layer on top.
Step 2: Propose features
Generate 2–3 candidate gateables at silver and gold tiers anchored in the five principles. Present as a Markdown table:
| Feature | Tier | Entitlement key | What it adds | Locked path | EventBus events | GameState additions |
|---|---|---|---|---|---|---|
| Skin picker | silver | premium_skins |
5 cosmetic variants via a picker on GameOverScene | Default skin only; button greyed out | skin:selected, skin:unlocked |
selectedSkin: 'default', unlockedSkins: ['default'] (persist through reset) |
| Continue after death | silver | continue |
One continue per session offered on death | No continue prompt; immediate restart | continue:offered, continue:accepted |
continueOffered: false, usedContinue: false (clear on reset) |
| Daily challenge mode | gold | daily_challenge |
Unique seeded run each day with its own leaderboard | No daily challenge menu visible | daily:opened, daily:completed |
dailyChallengeSeed: null, dailyBestScore: 0 (persist through reset) |
Anchor picks in the principles. Reject any suggestion that gates the core loop. Never propose bronze-tier features — bronze is the default everyone gets. Prefer cosmetic > convenience > bonus > persistence for short-session games; flip the order for long-form.
In interactive mode (user-invoked skill): wait for creator confirmation before implementing. Let them drop, swap, or tweak suggestions.
In pipeline mode (called from /make-game Step 1.25): auto-pick the top 2–3 and continue without prompting.
Step 3: Implement
Follow this order strictly:
3a. Create src/systems/Entitlements.js — the single capability-check seam:
// src/systems/Entitlements.js
// Single capability check for all gateable features.
// Returns false by default — every gateable must degrade cleanly when locked.
// Wire this to your monetization layer (sub.games, Play.fun, custom paywall)
// by replacing the body with real checks. See `/monetize-game` or `/subgames`.
export function isEntitled(key) {
// TODO: wire to monetization layer.
return false;
}
3b. Add events to src/core/EventBus.js (append-only, domain:action form).
3c. Add state fields to src/core/GameState.js. Update reset() to preserve persistent fields (e.g., unlockedSkins, bonusModeUnlocked) and clear transient ones (e.g., usedContinue, continueOffered).
3d. Add constants to src/core/Constants.js under a new GATEABLES section (skin list, continue cooldown, bonus mode config, etc.).
3e. Create gateable modules in the existing src/ subdirectories matching the game's layout:
src/ui/SkinPicker.js(Phaser scene or Three.js panel)src/systems/ContinueFlow.jssrc/systems/BonusMode.js- etc.
Each gateable's entry point calls isEntitled(key) and branches:
import { isEntitled } from '../systems/Entitlements.js';
function openSkinPicker() {
if (!isEntitled('premium_skins')) {
// Locked path: greyed-out button with lock icon, no picker opens.
// Default skin remains selected. Game continues normally.
return;
}
// Entitled path: open the picker scene.
scene.scene.launch('SkinPickerScene');
}
Never create a pre-gameplay title screen or skin-picker-first flow. The template boots directly into gameplay. Gateable UI opens from pause menu, settings, or GameOverScene only.
3f. Update window.render_game_to_text() in src/main.js additively — add new observable fields (selectedSkin, continueAvailable, bonusModeActive) without removing existing ones. This is load-bearing for AI test harnesses.
Step 4: Verify
npm run buildclean.npm run devand play withisEntitledreturningfalse(default). Core loop must feel identical to pre-skill state. If anything feels broken or missing, you gated the core loop — revert and revise.- Temporarily edit
Entitlements.jstoreturn true;for all keys. Confirm each feature activates correctly. Revert. - If
tests/exists and snapshotsrender_game_to_text()via exact equality (toEqual), update baselines — field additions are intentional. If tests usetoMatchObject, no update needed. - Append a
## Gateablessection toprogress.mdat the project root with:- Features added (name + tier + entitlement key)
- EventBus events added
- GameState fields added (marked as persistent or transient)
- Constants keys added
- Locked-path description (one line per feature)
Troubleshooting
Feature works when entitled but the game breaks when locked
Cause: You gated part of the core loop, or the locked path is incomplete.
Fix: Revert. The locked path must produce normal-feeling gameplay. Test by running with isEntitled returning false — if anything is missing or broken, revise.
Snapshot tests fail after render_game_to_text update
Cause: Tests use exact toEqual on output; you added new fields.
Fix: Regenerate baselines — the additions are intentional and backward-compatible for any consumer that tolerates extra keys.
Skin picker blocks first play
Cause: Picker launches pre-gameplay. Fix: Move entry point to a pause menu, settings screen, or GameOverScene. The template boots directly into gameplay and must continue to do so.
Entitlements.js already exists
Cause: The skill has been run before, or a monetization layer is already wired.
Fix: Read the file. If it's a stub (return false), add new keys and continue. If it's wired to a real monetization layer, leave it alone and only add the new entitlement keys as new function branches.
Proposal over-gates or feels punishing
Cause: You picked features that restrict the free experience. Fix: Principle 2 (additive over subtractive). Free players should never feel punished by the absence of a gateable. If removing a gateable makes the game feel worse for a free player, it's gating the wrong thing.
Output
Tell the user:
- What was added — list each gateable (name, tier, entitlement key)
- The single wiring seam — point at
src/systems/Entitlements.jsand explain: "FlipisEntitled(key)to returntruefor a key to activate that feature. Wire it to/monetize-game(Play.fun) or/subgames(subscription) next." - How to test — "Run
npm run devand play. Everything should feel normal (all gateables locked by default). EditEntitlements.jstoreturn true;temporarily to see features light up." - Next step — suggest
/monetize-gameor/subgamesas appropriate.
Example Usage
Default proposal in the current game directory
/scaffold-gateables
Result: reads the game, proposes 2–3 gateables, implements them after confirmation. Creates Entitlements.js, new UI modules, new events and state. Game feels unchanged until the entitlement seam is flipped.
Biased proposal with hints
/scaffold-gateables skins continue
Result: proposal biased toward a skin picker and a continue-after-death flow. Other suggestions still possible if they fit the game better.
Explicit project path
/scaffold-gateables ./examples/flappy-bird
Result: operates on the given directory rather than the cwd.
Tips
Run this once per game. If you later want to add more gateables, use
/add-featurefor specific additions or run this skill again — it detects existingEntitlements.jsand only appends new keys.Integration comes next:
/monetize-gamewires Play.fun points to gateables viaEntitlements.js;/subgames(from thesubdotgames/skillsrepo, separate install) wires subscription tiers. Either way,Entitlements.jsis the only file that needs to change to turn gateables on.