@sharpee/sharpee 1.0.0 → 1.0.8
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/index.d.ts +16 -16
- package/index.d.ts.map +1 -1
- package/index.js +14 -14
- package/package.json +21 -21
- package/runtime-surface.d.ts +17 -17
- package/runtime-surface.d.ts.map +1 -1
- package/runtime-surface.js +13 -13
- package/cli/build-browser.d.ts +0 -10
- package/cli/build-browser.d.ts.map +0 -1
- package/cli/build-browser.js +0 -208
- package/cli/build-browser.js.map +0 -1
- package/cli/build.d.ts +0 -8
- package/cli/build.d.ts.map +0 -1
- package/cli/build.js +0 -318
- package/cli/build.js.map +0 -1
- package/cli/ifid.d.ts +0 -2
- package/cli/ifid.d.ts.map +0 -1
- package/cli/ifid.js +0 -71
- package/cli/ifid.js.map +0 -1
- package/cli/index.d.ts +0 -8
- package/cli/index.d.ts.map +0 -1
- package/cli/index.js +0 -84
- package/cli/index.js.map +0 -1
- package/cli/init-browser.d.ts +0 -10
- package/cli/init-browser.d.ts.map +0 -1
- package/cli/init-browser.js +0 -194
- package/cli/init-browser.js.map +0 -1
- package/cli/init.d.ts +0 -10
- package/cli/init.d.ts.map +0 -1
- package/cli/init.js +0 -183
- package/cli/init.js.map +0 -1
- package/sharpee-sharpee-0.9.92.tgz +0 -0
- package/templates/browser/browser-entry.ts.template +0 -275
- package/templates/browser/index.html +0 -75
- package/templates/browser/styles.css +0 -495
- package/templates/story/index.ts.template +0 -77
- package/templates/story/package.json.template +0 -25
- package/templates/story/tsconfig.json.template +0 -14
package/cli/init-browser.js
DELETED
|
@@ -1,194 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
/**
|
|
3
|
-
* CLI: sharpee init-browser
|
|
4
|
-
*
|
|
5
|
-
* Adds browser client files to an existing Sharpee story project.
|
|
6
|
-
*/
|
|
7
|
-
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
8
|
-
if (k2 === undefined) k2 = k;
|
|
9
|
-
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
10
|
-
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
11
|
-
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
12
|
-
}
|
|
13
|
-
Object.defineProperty(o, k2, desc);
|
|
14
|
-
}) : (function(o, m, k, k2) {
|
|
15
|
-
if (k2 === undefined) k2 = k;
|
|
16
|
-
o[k2] = m[k];
|
|
17
|
-
}));
|
|
18
|
-
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
19
|
-
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
20
|
-
}) : function(o, v) {
|
|
21
|
-
o["default"] = v;
|
|
22
|
-
});
|
|
23
|
-
var __importStar = (this && this.__importStar) || (function () {
|
|
24
|
-
var ownKeys = function(o) {
|
|
25
|
-
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
26
|
-
var ar = [];
|
|
27
|
-
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
28
|
-
return ar;
|
|
29
|
-
};
|
|
30
|
-
return ownKeys(o);
|
|
31
|
-
};
|
|
32
|
-
return function (mod) {
|
|
33
|
-
if (mod && mod.__esModule) return mod;
|
|
34
|
-
var result = {};
|
|
35
|
-
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
36
|
-
__setModuleDefault(result, mod);
|
|
37
|
-
return result;
|
|
38
|
-
};
|
|
39
|
-
})();
|
|
40
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
41
|
-
exports.runInitBrowserCommand = runInitBrowserCommand;
|
|
42
|
-
const fs = __importStar(require("fs"));
|
|
43
|
-
const path = __importStar(require("path"));
|
|
44
|
-
// In source: src/cli/ → ../../templates. In npm publish: cli/ → ../templates.
|
|
45
|
-
const TEMPLATES_DIR = fs.existsSync(path.join(__dirname, '..', 'templates', 'browser'))
|
|
46
|
-
? path.join(__dirname, '..', 'templates', 'browser')
|
|
47
|
-
: path.join(__dirname, '..', '..', 'templates', 'browser');
|
|
48
|
-
/**
|
|
49
|
-
* Read story info from package.json or index.ts
|
|
50
|
-
*/
|
|
51
|
-
function getProjectInfo(projectDir) {
|
|
52
|
-
// Try package.json first
|
|
53
|
-
const packagePath = path.join(projectDir, 'package.json');
|
|
54
|
-
if (fs.existsSync(packagePath)) {
|
|
55
|
-
try {
|
|
56
|
-
const pkg = JSON.parse(fs.readFileSync(packagePath, 'utf-8'));
|
|
57
|
-
return {
|
|
58
|
-
storyId: pkg.name || 'my-story',
|
|
59
|
-
storyTitle: pkg.description || pkg.name || 'My Story',
|
|
60
|
-
};
|
|
61
|
-
}
|
|
62
|
-
catch {
|
|
63
|
-
// Fall through
|
|
64
|
-
}
|
|
65
|
-
}
|
|
66
|
-
// Try reading from src/index.ts
|
|
67
|
-
const indexPath = path.join(projectDir, 'src', 'index.ts');
|
|
68
|
-
if (fs.existsSync(indexPath)) {
|
|
69
|
-
const content = fs.readFileSync(indexPath, 'utf-8');
|
|
70
|
-
const idMatch = content.match(/id:\s*['"]([^'"]+)['"]/);
|
|
71
|
-
const titleMatch = content.match(/title:\s*['"]([^'"]+)['"]/);
|
|
72
|
-
if (idMatch || titleMatch) {
|
|
73
|
-
return {
|
|
74
|
-
storyId: idMatch?.[1] || 'my-story',
|
|
75
|
-
storyTitle: titleMatch?.[1] || 'My Story',
|
|
76
|
-
};
|
|
77
|
-
}
|
|
78
|
-
}
|
|
79
|
-
return null;
|
|
80
|
-
}
|
|
81
|
-
/**
|
|
82
|
-
* Read template file and replace placeholders
|
|
83
|
-
*/
|
|
84
|
-
function processTemplate(templatePath, info) {
|
|
85
|
-
const content = fs.readFileSync(templatePath, 'utf-8');
|
|
86
|
-
return content
|
|
87
|
-
.replace(/\{\{STORY_ID\}\}/g, info.storyId)
|
|
88
|
-
.replace(/\{\{STORY_TITLE\}\}/g, info.storyTitle);
|
|
89
|
-
}
|
|
90
|
-
/**
|
|
91
|
-
* Run the init-browser command
|
|
92
|
-
*/
|
|
93
|
-
async function runInitBrowserCommand(args) {
|
|
94
|
-
// Check for help
|
|
95
|
-
if (args.includes('--help') || args.includes('-h')) {
|
|
96
|
-
showHelp();
|
|
97
|
-
return;
|
|
98
|
-
}
|
|
99
|
-
const projectDir = process.cwd();
|
|
100
|
-
console.log('\n🌐 Adding browser client to your Sharpee project\n');
|
|
101
|
-
// Check if this is a Sharpee project
|
|
102
|
-
const info = getProjectInfo(projectDir);
|
|
103
|
-
if (!info) {
|
|
104
|
-
console.error('Error: This does not appear to be a Sharpee project.');
|
|
105
|
-
console.error('Make sure you have a package.json or src/index.ts with story config.');
|
|
106
|
-
console.error('\nRun "sharpee init" first to create a project.');
|
|
107
|
-
process.exit(1);
|
|
108
|
-
}
|
|
109
|
-
console.log(` Story: ${info.storyTitle} (${info.storyId})`);
|
|
110
|
-
// Check if browser-entry.ts already exists
|
|
111
|
-
const browserEntryPath = path.join(projectDir, 'src', 'browser-entry.ts');
|
|
112
|
-
if (fs.existsSync(browserEntryPath)) {
|
|
113
|
-
console.error('\nError: src/browser-entry.ts already exists.');
|
|
114
|
-
console.error('Remove it first if you want to regenerate.');
|
|
115
|
-
process.exit(1);
|
|
116
|
-
}
|
|
117
|
-
// Create browser directory for templates (optional, for customization)
|
|
118
|
-
const browserDir = path.join(projectDir, 'browser');
|
|
119
|
-
fs.mkdirSync(browserDir, { recursive: true });
|
|
120
|
-
// Copy browser-entry.ts template
|
|
121
|
-
const browserEntryTemplate = path.join(TEMPLATES_DIR, 'browser-entry.ts.template');
|
|
122
|
-
if (fs.existsSync(browserEntryTemplate)) {
|
|
123
|
-
const content = processTemplate(browserEntryTemplate, info);
|
|
124
|
-
fs.writeFileSync(browserEntryPath, content);
|
|
125
|
-
console.log(' ✓ Created src/browser-entry.ts');
|
|
126
|
-
}
|
|
127
|
-
else {
|
|
128
|
-
console.error(' ✗ Template not found: browser-entry.ts.template');
|
|
129
|
-
process.exit(1);
|
|
130
|
-
}
|
|
131
|
-
// Copy HTML template
|
|
132
|
-
const htmlTemplate = path.join(TEMPLATES_DIR, 'index.html');
|
|
133
|
-
if (fs.existsSync(htmlTemplate)) {
|
|
134
|
-
const content = processTemplate(htmlTemplate, info);
|
|
135
|
-
fs.writeFileSync(path.join(browserDir, 'index.html'), content);
|
|
136
|
-
console.log(' ✓ Created browser/index.html');
|
|
137
|
-
}
|
|
138
|
-
// Copy CSS
|
|
139
|
-
const cssTemplate = path.join(TEMPLATES_DIR, 'styles.css');
|
|
140
|
-
if (fs.existsSync(cssTemplate)) {
|
|
141
|
-
fs.copyFileSync(cssTemplate, path.join(browserDir, 'styles.css'));
|
|
142
|
-
console.log(' ✓ Created browser/styles.css');
|
|
143
|
-
}
|
|
144
|
-
// Update package.json to add esbuild dev dependency and build script
|
|
145
|
-
const packagePath = path.join(projectDir, 'package.json');
|
|
146
|
-
if (fs.existsSync(packagePath)) {
|
|
147
|
-
try {
|
|
148
|
-
const pkg = JSON.parse(fs.readFileSync(packagePath, 'utf-8'));
|
|
149
|
-
// Add esbuild to devDependencies
|
|
150
|
-
pkg.devDependencies = pkg.devDependencies || {};
|
|
151
|
-
if (!pkg.devDependencies.esbuild) {
|
|
152
|
-
pkg.devDependencies.esbuild = '^0.20.0';
|
|
153
|
-
}
|
|
154
|
-
// Add build:browser script
|
|
155
|
-
pkg.scripts = pkg.scripts || {};
|
|
156
|
-
if (!pkg.scripts['build:browser']) {
|
|
157
|
-
pkg.scripts['build:browser'] = 'npx sharpee build-browser';
|
|
158
|
-
}
|
|
159
|
-
fs.writeFileSync(packagePath, JSON.stringify(pkg, null, 2) + '\n');
|
|
160
|
-
console.log(' ✓ Updated package.json');
|
|
161
|
-
}
|
|
162
|
-
catch (e) {
|
|
163
|
-
console.warn(' ⚠ Could not update package.json');
|
|
164
|
-
}
|
|
165
|
-
}
|
|
166
|
-
console.log('\n✅ Browser client added!\n');
|
|
167
|
-
console.log('Next steps:');
|
|
168
|
-
console.log(' npm install # Install esbuild');
|
|
169
|
-
console.log(' npm run build:browser # Build web bundle');
|
|
170
|
-
console.log('');
|
|
171
|
-
console.log('Output will be in dist/web/');
|
|
172
|
-
console.log('');
|
|
173
|
-
console.log('Customize the UI:');
|
|
174
|
-
console.log(' browser/index.html # HTML template');
|
|
175
|
-
console.log(' browser/styles.css # Styles');
|
|
176
|
-
console.log('');
|
|
177
|
-
}
|
|
178
|
-
function showHelp() {
|
|
179
|
-
console.log(`
|
|
180
|
-
sharpee init-browser - Add browser client to a Sharpee project
|
|
181
|
-
|
|
182
|
-
Usage: sharpee init-browser
|
|
183
|
-
|
|
184
|
-
This command adds the files needed to build a web browser version
|
|
185
|
-
of your interactive fiction game:
|
|
186
|
-
|
|
187
|
-
src/browser-entry.ts Entry point for browser bundle
|
|
188
|
-
browser/index.html HTML template
|
|
189
|
-
browser/styles.css Infocom-style CSS
|
|
190
|
-
|
|
191
|
-
Run this in the root of your Sharpee project directory.
|
|
192
|
-
`);
|
|
193
|
-
}
|
|
194
|
-
//# sourceMappingURL=init-browser.js.map
|
package/cli/init-browser.js.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"init-browser.js","sourceRoot":"","sources":["../../../../../../../repos/sharpee/packages/sharpee/src/cli/init-browser.ts"],"names":[],"mappings":"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"}
|
package/cli/init.d.ts
DELETED
package/cli/init.d.ts.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"init.d.ts","sourceRoot":"","sources":["../../../../../../repos/sharpee/packages/sharpee/src/cli/init.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AA2DH;;GAEG;AACH,wBAAsB,cAAc,CAAC,IAAI,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAuFlE"}
|
package/cli/init.js
DELETED
|
@@ -1,183 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
/**
|
|
3
|
-
* CLI: sharpee init
|
|
4
|
-
*
|
|
5
|
-
* Creates a new Sharpee story project with the basic structure.
|
|
6
|
-
*/
|
|
7
|
-
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
8
|
-
if (k2 === undefined) k2 = k;
|
|
9
|
-
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
10
|
-
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
11
|
-
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
12
|
-
}
|
|
13
|
-
Object.defineProperty(o, k2, desc);
|
|
14
|
-
}) : (function(o, m, k, k2) {
|
|
15
|
-
if (k2 === undefined) k2 = k;
|
|
16
|
-
o[k2] = m[k];
|
|
17
|
-
}));
|
|
18
|
-
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
19
|
-
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
20
|
-
}) : function(o, v) {
|
|
21
|
-
o["default"] = v;
|
|
22
|
-
});
|
|
23
|
-
var __importStar = (this && this.__importStar) || (function () {
|
|
24
|
-
var ownKeys = function(o) {
|
|
25
|
-
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
26
|
-
var ar = [];
|
|
27
|
-
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
28
|
-
return ar;
|
|
29
|
-
};
|
|
30
|
-
return ownKeys(o);
|
|
31
|
-
};
|
|
32
|
-
return function (mod) {
|
|
33
|
-
if (mod && mod.__esModule) return mod;
|
|
34
|
-
var result = {};
|
|
35
|
-
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
36
|
-
__setModuleDefault(result, mod);
|
|
37
|
-
return result;
|
|
38
|
-
};
|
|
39
|
-
})();
|
|
40
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
41
|
-
exports.runInitCommand = runInitCommand;
|
|
42
|
-
const fs = __importStar(require("fs"));
|
|
43
|
-
const path = __importStar(require("path"));
|
|
44
|
-
const readline = __importStar(require("readline"));
|
|
45
|
-
// Template directory relative to this file.
|
|
46
|
-
// In source: src/cli/ → ../../templates. In npm publish: cli/ → ../templates.
|
|
47
|
-
const TEMPLATES_DIR = fs.existsSync(path.join(__dirname, '..', 'templates', 'story'))
|
|
48
|
-
? path.join(__dirname, '..', 'templates', 'story')
|
|
49
|
-
: path.join(__dirname, '..', '..', 'templates', 'story');
|
|
50
|
-
/**
|
|
51
|
-
* Prompt user for input
|
|
52
|
-
*/
|
|
53
|
-
async function prompt(question, defaultValue) {
|
|
54
|
-
const rl = readline.createInterface({
|
|
55
|
-
input: process.stdin,
|
|
56
|
-
output: process.stdout,
|
|
57
|
-
});
|
|
58
|
-
return new Promise((resolve) => {
|
|
59
|
-
const defaultHint = defaultValue ? ` (${defaultValue})` : '';
|
|
60
|
-
rl.question(`${question}${defaultHint}: `, (answer) => {
|
|
61
|
-
rl.close();
|
|
62
|
-
resolve(answer.trim() || defaultValue || '');
|
|
63
|
-
});
|
|
64
|
-
});
|
|
65
|
-
}
|
|
66
|
-
/**
|
|
67
|
-
* Convert title to kebab-case ID
|
|
68
|
-
*/
|
|
69
|
-
function toStoryId(title) {
|
|
70
|
-
return title
|
|
71
|
-
.toLowerCase()
|
|
72
|
-
.replace(/[^a-z0-9]+/g, '-')
|
|
73
|
-
.replace(/^-|-$/g, '');
|
|
74
|
-
}
|
|
75
|
-
/**
|
|
76
|
-
* Read template file and replace placeholders
|
|
77
|
-
*/
|
|
78
|
-
function processTemplate(templatePath, options) {
|
|
79
|
-
const content = fs.readFileSync(templatePath, 'utf-8');
|
|
80
|
-
return content
|
|
81
|
-
.replace(/\{\{STORY_ID\}\}/g, options.storyId)
|
|
82
|
-
.replace(/\{\{STORY_TITLE\}\}/g, options.storyTitle)
|
|
83
|
-
.replace(/\{\{AUTHOR\}\}/g, options.author)
|
|
84
|
-
.replace(/\{\{DESCRIPTION\}\}/g, options.description);
|
|
85
|
-
}
|
|
86
|
-
/**
|
|
87
|
-
* Run the init command
|
|
88
|
-
*/
|
|
89
|
-
async function runInitCommand(args) {
|
|
90
|
-
// Check for help
|
|
91
|
-
if (args.includes('--help') || args.includes('-h')) {
|
|
92
|
-
showHelp();
|
|
93
|
-
return;
|
|
94
|
-
}
|
|
95
|
-
// Check for non-interactive mode
|
|
96
|
-
const useDefaults = args.includes('-y') || args.includes('--yes');
|
|
97
|
-
const filteredArgs = args.filter(a => a !== '-y' && a !== '--yes');
|
|
98
|
-
// Get target directory
|
|
99
|
-
const targetDir = filteredArgs[0] || '.';
|
|
100
|
-
const absoluteTarget = path.resolve(process.cwd(), targetDir);
|
|
101
|
-
console.log('\n📖 Create a new Sharpee story\n');
|
|
102
|
-
// Check if directory exists and is not empty
|
|
103
|
-
if (fs.existsSync(absoluteTarget)) {
|
|
104
|
-
const files = fs.readdirSync(absoluteTarget);
|
|
105
|
-
if (files.length > 0 && !files.every(f => f.startsWith('.'))) {
|
|
106
|
-
console.error(`Error: Directory "${targetDir}" is not empty.`);
|
|
107
|
-
console.error('Please use an empty directory or specify a new one.');
|
|
108
|
-
process.exit(1);
|
|
109
|
-
}
|
|
110
|
-
}
|
|
111
|
-
// Gather project info (use defaults if -y flag)
|
|
112
|
-
const defaultTitle = path.basename(absoluteTarget) || 'My Adventure';
|
|
113
|
-
const storyTitle = useDefaults ? defaultTitle : await prompt('Story title', defaultTitle);
|
|
114
|
-
const storyId = useDefaults ? toStoryId(storyTitle) : await prompt('Story ID (package name)', toStoryId(storyTitle));
|
|
115
|
-
const author = useDefaults ? (process.env.USER || 'Anonymous') : await prompt('Author name', process.env.USER || 'Anonymous');
|
|
116
|
-
const description = useDefaults ? 'An interactive fiction adventure' : await prompt('Description', 'An interactive fiction adventure');
|
|
117
|
-
const options = {
|
|
118
|
-
storyId,
|
|
119
|
-
storyTitle,
|
|
120
|
-
author,
|
|
121
|
-
description,
|
|
122
|
-
};
|
|
123
|
-
console.log('\nCreating project...\n');
|
|
124
|
-
// Create directory structure
|
|
125
|
-
fs.mkdirSync(absoluteTarget, { recursive: true });
|
|
126
|
-
fs.mkdirSync(path.join(absoluteTarget, 'src'), { recursive: true });
|
|
127
|
-
// Copy and process templates
|
|
128
|
-
const templates = [
|
|
129
|
-
{ src: 'index.ts.template', dest: 'src/index.ts' },
|
|
130
|
-
{ src: 'package.json.template', dest: 'package.json' },
|
|
131
|
-
{ src: 'tsconfig.json.template', dest: 'tsconfig.json' },
|
|
132
|
-
];
|
|
133
|
-
for (const template of templates) {
|
|
134
|
-
const srcPath = path.join(TEMPLATES_DIR, template.src);
|
|
135
|
-
const destPath = path.join(absoluteTarget, template.dest);
|
|
136
|
-
if (fs.existsSync(srcPath)) {
|
|
137
|
-
const content = processTemplate(srcPath, options);
|
|
138
|
-
fs.writeFileSync(destPath, content);
|
|
139
|
-
console.log(` ✓ Created ${template.dest}`);
|
|
140
|
-
}
|
|
141
|
-
else {
|
|
142
|
-
console.warn(` ⚠ Template not found: ${template.src}`);
|
|
143
|
-
}
|
|
144
|
-
}
|
|
145
|
-
// Create .gitignore
|
|
146
|
-
const gitignore = `node_modules/
|
|
147
|
-
dist/
|
|
148
|
-
*.log
|
|
149
|
-
.DS_Store
|
|
150
|
-
`;
|
|
151
|
-
fs.writeFileSync(path.join(absoluteTarget, '.gitignore'), gitignore);
|
|
152
|
-
console.log(' ✓ Created .gitignore');
|
|
153
|
-
console.log('\n✅ Project created!\n');
|
|
154
|
-
console.log('Next steps:');
|
|
155
|
-
if (targetDir !== '.') {
|
|
156
|
-
console.log(` cd ${targetDir}`);
|
|
157
|
-
}
|
|
158
|
-
console.log(' npm install');
|
|
159
|
-
console.log(' npm run build');
|
|
160
|
-
console.log('');
|
|
161
|
-
console.log('To add a browser client:');
|
|
162
|
-
console.log(' npx sharpee init-browser');
|
|
163
|
-
console.log('');
|
|
164
|
-
}
|
|
165
|
-
function showHelp() {
|
|
166
|
-
console.log(`
|
|
167
|
-
sharpee init - Create a new Sharpee story project
|
|
168
|
-
|
|
169
|
-
Usage: sharpee init [directory] [options]
|
|
170
|
-
|
|
171
|
-
Arguments:
|
|
172
|
-
directory Target directory (default: current directory)
|
|
173
|
-
|
|
174
|
-
Options:
|
|
175
|
-
-y, --yes Use defaults without prompting
|
|
176
|
-
|
|
177
|
-
Examples:
|
|
178
|
-
sharpee init Create in current directory (interactive)
|
|
179
|
-
sharpee init my-adventure Create in ./my-adventure/ (interactive)
|
|
180
|
-
sharpee init my-adventure -y Create with defaults (non-interactive)
|
|
181
|
-
`);
|
|
182
|
-
}
|
|
183
|
-
//# sourceMappingURL=init.js.map
|
package/cli/init.js.map
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
{"version":3,"file":"init.js","sourceRoot":"","sources":["../../../../../../../repos/sharpee/packages/sharpee/src/cli/init.ts"],"names":[],"mappings":"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"}
|
|
Binary file
|
|
@@ -1,275 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* Browser Entry Point for {{STORY_TITLE}}
|
|
3
|
-
*
|
|
4
|
-
* This file connects your story to the browser UI.
|
|
5
|
-
* Generated by: npx sharpee init-browser
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
import { GameEngine } from '@sharpee/engine';
|
|
9
|
-
import { WorldModel, EntityType } from '@sharpee/world-model';
|
|
10
|
-
import { Parser } from '@sharpee/parser-en-us';
|
|
11
|
-
import { LanguageProvider } from '@sharpee/lang-en-us';
|
|
12
|
-
import { PerceptionService } from '@sharpee/stdlib';
|
|
13
|
-
import { renderToString } from '@sharpee/text-service';
|
|
14
|
-
import { story, config } from './index.js';
|
|
15
|
-
|
|
16
|
-
// DOM elements
|
|
17
|
-
let statusLocation: HTMLElement | null;
|
|
18
|
-
let statusScore: HTMLElement | null;
|
|
19
|
-
let textContent: HTMLElement | null;
|
|
20
|
-
let mainWindow: HTMLElement | null;
|
|
21
|
-
let commandInput: HTMLInputElement | null;
|
|
22
|
-
|
|
23
|
-
// Game state
|
|
24
|
-
let engine: GameEngine;
|
|
25
|
-
let world: WorldModel;
|
|
26
|
-
let commandHistory: string[] = [];
|
|
27
|
-
let historyIndex = -1;
|
|
28
|
-
let currentTurn = 0;
|
|
29
|
-
let currentScore = 0;
|
|
30
|
-
|
|
31
|
-
/**
|
|
32
|
-
* Initialize the game
|
|
33
|
-
*/
|
|
34
|
-
function initializeGame(): void {
|
|
35
|
-
// Create world and player
|
|
36
|
-
world = new WorldModel();
|
|
37
|
-
const player = world.createEntity('player', EntityType.ACTOR);
|
|
38
|
-
world.setPlayer(player.id);
|
|
39
|
-
|
|
40
|
-
// Create parser and language
|
|
41
|
-
const language = new LanguageProvider();
|
|
42
|
-
const parser = new Parser(language);
|
|
43
|
-
|
|
44
|
-
// Extend parser and language with story-specific vocabulary
|
|
45
|
-
if (story.extendParser) {
|
|
46
|
-
story.extendParser(parser);
|
|
47
|
-
}
|
|
48
|
-
if (story.extendLanguage) {
|
|
49
|
-
story.extendLanguage(language);
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
// Create perception service
|
|
53
|
-
const perceptionService = new PerceptionService();
|
|
54
|
-
|
|
55
|
-
// Create engine
|
|
56
|
-
engine = new GameEngine({
|
|
57
|
-
world,
|
|
58
|
-
player,
|
|
59
|
-
parser,
|
|
60
|
-
language,
|
|
61
|
-
perceptionService,
|
|
62
|
-
});
|
|
63
|
-
|
|
64
|
-
// Set up event handlers
|
|
65
|
-
engine.on('text:output', (blocks, turn) => {
|
|
66
|
-
displayText(renderToString(blocks));
|
|
67
|
-
currentTurn = turn;
|
|
68
|
-
updateStatusLine();
|
|
69
|
-
});
|
|
70
|
-
|
|
71
|
-
engine.on('event', (event: any) => {
|
|
72
|
-
// Track score changes
|
|
73
|
-
if (event.type === 'game.score_changed' && event.data) {
|
|
74
|
-
currentScore = event.data.newScore ?? currentScore;
|
|
75
|
-
updateStatusLine();
|
|
76
|
-
}
|
|
77
|
-
});
|
|
78
|
-
|
|
79
|
-
// Set the story
|
|
80
|
-
engine.setStory(story);
|
|
81
|
-
}
|
|
82
|
-
|
|
83
|
-
/**
|
|
84
|
-
* Set up DOM elements and event handlers
|
|
85
|
-
*/
|
|
86
|
-
function setupDOM(): void {
|
|
87
|
-
statusLocation = document.getElementById('location-name');
|
|
88
|
-
statusScore = document.getElementById('score-turns');
|
|
89
|
-
textContent = document.getElementById('text-content');
|
|
90
|
-
mainWindow = document.getElementById('main-window');
|
|
91
|
-
commandInput = document.getElementById('command-input') as HTMLInputElement;
|
|
92
|
-
|
|
93
|
-
if (!commandInput) {
|
|
94
|
-
console.error('Command input element not found');
|
|
95
|
-
return;
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
// Handle keyboard input
|
|
99
|
-
commandInput.addEventListener('keydown', (e: KeyboardEvent) => {
|
|
100
|
-
if (e.key === 'Enter') {
|
|
101
|
-
handleCommand();
|
|
102
|
-
} else if (e.key === 'ArrowUp') {
|
|
103
|
-
e.preventDefault();
|
|
104
|
-
navigateHistory(-1);
|
|
105
|
-
} else if (e.key === 'ArrowDown') {
|
|
106
|
-
e.preventDefault();
|
|
107
|
-
navigateHistory(1);
|
|
108
|
-
}
|
|
109
|
-
});
|
|
110
|
-
|
|
111
|
-
// Keep focus on input
|
|
112
|
-
document.addEventListener('click', () => {
|
|
113
|
-
if (commandInput && !commandInput.disabled) {
|
|
114
|
-
commandInput.focus();
|
|
115
|
-
}
|
|
116
|
-
});
|
|
117
|
-
}
|
|
118
|
-
|
|
119
|
-
/**
|
|
120
|
-
* Handle command submission
|
|
121
|
-
*/
|
|
122
|
-
async function handleCommand(): Promise<void> {
|
|
123
|
-
if (!commandInput) return;
|
|
124
|
-
|
|
125
|
-
const command = commandInput.value.trim();
|
|
126
|
-
if (!command) return;
|
|
127
|
-
|
|
128
|
-
// Add to history
|
|
129
|
-
commandHistory.push(command);
|
|
130
|
-
historyIndex = commandHistory.length;
|
|
131
|
-
|
|
132
|
-
// Clear input
|
|
133
|
-
commandInput.value = '';
|
|
134
|
-
|
|
135
|
-
// Display command echo
|
|
136
|
-
displayCommand(command);
|
|
137
|
-
|
|
138
|
-
// Execute command
|
|
139
|
-
try {
|
|
140
|
-
await engine.executeTurn(command);
|
|
141
|
-
} catch (error) {
|
|
142
|
-
const message = error instanceof Error ? error.message : String(error);
|
|
143
|
-
displayText(`[Error: ${message}]`);
|
|
144
|
-
}
|
|
145
|
-
}
|
|
146
|
-
|
|
147
|
-
/**
|
|
148
|
-
* Navigate command history
|
|
149
|
-
*/
|
|
150
|
-
function navigateHistory(direction: number): void {
|
|
151
|
-
if (!commandInput) return;
|
|
152
|
-
|
|
153
|
-
const newIndex = historyIndex + direction;
|
|
154
|
-
|
|
155
|
-
if (newIndex < 0) return;
|
|
156
|
-
|
|
157
|
-
if (newIndex >= commandHistory.length) {
|
|
158
|
-
historyIndex = commandHistory.length;
|
|
159
|
-
commandInput.value = '';
|
|
160
|
-
return;
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
historyIndex = newIndex;
|
|
164
|
-
commandInput.value = commandHistory[historyIndex];
|
|
165
|
-
|
|
166
|
-
// Move cursor to end
|
|
167
|
-
commandInput.setSelectionRange(
|
|
168
|
-
commandInput.value.length,
|
|
169
|
-
commandInput.value.length
|
|
170
|
-
);
|
|
171
|
-
}
|
|
172
|
-
|
|
173
|
-
/**
|
|
174
|
-
* Display text in the main window
|
|
175
|
-
*/
|
|
176
|
-
function displayText(text: string): void {
|
|
177
|
-
if (!textContent) return;
|
|
178
|
-
|
|
179
|
-
// Split on double newlines to get paragraphs
|
|
180
|
-
const paragraphs = text.split(/\n\n+/);
|
|
181
|
-
|
|
182
|
-
for (const para of paragraphs) {
|
|
183
|
-
const trimmed = para.trim();
|
|
184
|
-
if (trimmed) {
|
|
185
|
-
const p = document.createElement('p');
|
|
186
|
-
p.style.whiteSpace = 'pre-line';
|
|
187
|
-
p.textContent = trimmed;
|
|
188
|
-
textContent.appendChild(p);
|
|
189
|
-
}
|
|
190
|
-
}
|
|
191
|
-
|
|
192
|
-
scrollToBottom();
|
|
193
|
-
}
|
|
194
|
-
|
|
195
|
-
/**
|
|
196
|
-
* Display command echo
|
|
197
|
-
*/
|
|
198
|
-
function displayCommand(command: string): void {
|
|
199
|
-
if (!textContent) return;
|
|
200
|
-
|
|
201
|
-
const div = document.createElement('div');
|
|
202
|
-
div.className = 'command-echo';
|
|
203
|
-
div.textContent = `> ${command}`;
|
|
204
|
-
textContent.appendChild(div);
|
|
205
|
-
|
|
206
|
-
scrollToBottom();
|
|
207
|
-
}
|
|
208
|
-
|
|
209
|
-
/**
|
|
210
|
-
* Update the status line
|
|
211
|
-
*/
|
|
212
|
-
function updateStatusLine(): void {
|
|
213
|
-
const player = world.getPlayer();
|
|
214
|
-
let locationName = '';
|
|
215
|
-
|
|
216
|
-
if (player) {
|
|
217
|
-
const locationId = world.getLocation(player.id);
|
|
218
|
-
if (locationId) {
|
|
219
|
-
const room = world.getEntity(locationId);
|
|
220
|
-
if (room) {
|
|
221
|
-
locationName = room.name || 'Unknown';
|
|
222
|
-
}
|
|
223
|
-
}
|
|
224
|
-
}
|
|
225
|
-
|
|
226
|
-
if (statusLocation) {
|
|
227
|
-
statusLocation.textContent = locationName;
|
|
228
|
-
}
|
|
229
|
-
|
|
230
|
-
if (statusScore) {
|
|
231
|
-
statusScore.textContent = `Score: ${currentScore} | Turns: ${currentTurn}`;
|
|
232
|
-
}
|
|
233
|
-
}
|
|
234
|
-
|
|
235
|
-
/**
|
|
236
|
-
* Scroll main window to bottom
|
|
237
|
-
*/
|
|
238
|
-
function scrollToBottom(): void {
|
|
239
|
-
if (mainWindow) {
|
|
240
|
-
mainWindow.scrollTop = mainWindow.scrollHeight;
|
|
241
|
-
}
|
|
242
|
-
}
|
|
243
|
-
|
|
244
|
-
/**
|
|
245
|
-
* Start the game
|
|
246
|
-
*/
|
|
247
|
-
async function start(): Promise<void> {
|
|
248
|
-
console.log('=== {{STORY_TITLE}} BROWSER START ===');
|
|
249
|
-
|
|
250
|
-
try {
|
|
251
|
-
setupDOM();
|
|
252
|
-
initializeGame();
|
|
253
|
-
|
|
254
|
-
// Start the engine
|
|
255
|
-
await engine.start();
|
|
256
|
-
|
|
257
|
-
// Initial look
|
|
258
|
-
await engine.executeTurn('look');
|
|
259
|
-
|
|
260
|
-
// Focus input
|
|
261
|
-
if (commandInput) {
|
|
262
|
-
commandInput.focus();
|
|
263
|
-
}
|
|
264
|
-
} catch (error) {
|
|
265
|
-
console.error('=== STARTUP ERROR ===', error);
|
|
266
|
-
displayText(`[Startup Error: ${error}]`);
|
|
267
|
-
}
|
|
268
|
-
}
|
|
269
|
-
|
|
270
|
-
// Start when DOM is ready
|
|
271
|
-
if (document.readyState === 'loading') {
|
|
272
|
-
document.addEventListener('DOMContentLoaded', start);
|
|
273
|
-
} else {
|
|
274
|
-
start();
|
|
275
|
-
}
|