@holoscript/core 1.0.0-alpha.1 → 2.0.0
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/package.json +10 -9
- package/src/HoloScript2DParser.js +227 -0
- package/src/HoloScript2DParser.ts +5 -0
- package/src/HoloScriptCodeParser.js +1102 -0
- package/src/HoloScriptCodeParser.ts +145 -20
- package/src/HoloScriptDebugger.js +458 -0
- package/src/HoloScriptParser.js +338 -0
- package/src/HoloScriptPlusParser.js +371 -0
- package/src/HoloScriptPlusParser.ts +543 -0
- package/src/HoloScriptRuntime.js +1399 -0
- package/src/HoloScriptRuntime.test.js +351 -0
- package/src/HoloScriptRuntime.ts +257 -3
- package/src/HoloScriptTypeChecker.js +356 -0
- package/src/__tests__/GraphicsServices.test.js +357 -0
- package/src/__tests__/GraphicsServices.test.ts +427 -0
- package/src/__tests__/HoloScriptPlusParser.test.js +317 -0
- package/src/__tests__/HoloScriptPlusParser.test.ts +392 -0
- package/src/__tests__/integration.test.js +336 -0
- package/src/__tests__/performance.bench.js +218 -0
- package/src/__tests__/type-checker.test.js +60 -0
- package/src/__tests__/type-checker.test.ts +73 -0
- package/src/index.js +217 -0
- package/src/index.ts +158 -18
- package/src/interop/Interoperability.js +413 -0
- package/src/interop/Interoperability.ts +494 -0
- package/src/logger.js +42 -0
- package/src/parser/EnhancedParser.js +205 -0
- package/src/parser/EnhancedParser.ts +251 -0
- package/src/parser/HoloScriptPlusParser.js +928 -0
- package/src/parser/HoloScriptPlusParser.ts +1089 -0
- package/src/runtime/HoloScriptPlusRuntime.js +674 -0
- package/src/runtime/HoloScriptPlusRuntime.ts +861 -0
- package/src/runtime/PerformanceTelemetry.js +323 -0
- package/src/runtime/PerformanceTelemetry.ts +467 -0
- package/src/runtime/RuntimeOptimization.js +361 -0
- package/src/runtime/RuntimeOptimization.ts +416 -0
- package/src/services/HololandGraphicsPipelineService.js +506 -0
- package/src/services/HololandGraphicsPipelineService.ts +662 -0
- package/src/services/PlatformPerformanceOptimizer.js +356 -0
- package/src/services/PlatformPerformanceOptimizer.ts +503 -0
- package/src/state/ReactiveState.js +427 -0
- package/src/state/ReactiveState.ts +572 -0
- package/src/tools/DeveloperExperience.js +376 -0
- package/src/tools/DeveloperExperience.ts +438 -0
- package/src/traits/AIDriverTrait.js +322 -0
- package/src/traits/AIDriverTrait.test.js +329 -0
- package/src/traits/AIDriverTrait.test.ts +357 -0
- package/src/traits/AIDriverTrait.ts +474 -0
- package/src/traits/LightingTrait.js +313 -0
- package/src/traits/LightingTrait.test.js +410 -0
- package/src/traits/LightingTrait.test.ts +462 -0
- package/src/traits/LightingTrait.ts +505 -0
- package/src/traits/MaterialTrait.js +194 -0
- package/src/traits/MaterialTrait.test.js +286 -0
- package/src/traits/MaterialTrait.test.ts +329 -0
- package/src/traits/MaterialTrait.ts +324 -0
- package/src/traits/RenderingTrait.js +356 -0
- package/src/traits/RenderingTrait.test.js +363 -0
- package/src/traits/RenderingTrait.test.ts +427 -0
- package/src/traits/RenderingTrait.ts +555 -0
- package/src/traits/VRTraitSystem.js +740 -0
- package/src/traits/VRTraitSystem.ts +1040 -0
- package/src/traits/VoiceInputTrait.js +284 -0
- package/src/traits/VoiceInputTrait.test.js +226 -0
- package/src/traits/VoiceInputTrait.test.ts +252 -0
- package/src/traits/VoiceInputTrait.ts +401 -0
- package/src/types/AdvancedTypeSystem.js +226 -0
- package/src/types/AdvancedTypeSystem.ts +494 -0
- package/src/types/HoloScriptPlus.d.ts +853 -0
- package/src/types.js +6 -0
- package/src/types.ts +96 -1
- package/tsconfig.json +1 -1
- package/tsup.config.d.ts +2 -0
- package/tsup.config.js +18 -0
- package/LICENSE +0 -21
- package/dist/chunk-3X2EGU7Z.cjs +0 -52
- package/dist/chunk-3X2EGU7Z.cjs.map +0 -1
- package/dist/chunk-723TPVHD.js +0 -1074
- package/dist/chunk-723TPVHD.js.map +0 -1
- package/dist/chunk-EOKNAVDO.cjs +0 -424
- package/dist/chunk-EOKNAVDO.cjs.map +0 -1
- package/dist/chunk-HQZ3HUMY.js +0 -1087
- package/dist/chunk-HQZ3HUMY.js.map +0 -1
- package/dist/chunk-KWYIVRIH.js +0 -344
- package/dist/chunk-KWYIVRIH.js.map +0 -1
- package/dist/chunk-LKH4ZAN6.js +0 -421
- package/dist/chunk-LKH4ZAN6.js.map +0 -1
- package/dist/chunk-SATNCODL.js +0 -45
- package/dist/chunk-SATNCODL.js.map +0 -1
- package/dist/chunk-VMZN4EVR.cjs +0 -347
- package/dist/chunk-VMZN4EVR.cjs.map +0 -1
- package/dist/chunk-VV3UUUYP.cjs +0 -1089
- package/dist/chunk-VV3UUUYP.cjs.map +0 -1
- package/dist/chunk-XRYTSQHZ.cjs +0 -1076
- package/dist/chunk-XRYTSQHZ.cjs.map +0 -1
- package/dist/debugger.cjs +0 -19
- package/dist/debugger.cjs.map +0 -1
- package/dist/debugger.d.cts +0 -171
- package/dist/debugger.d.ts +0 -171
- package/dist/debugger.js +0 -6
- package/dist/debugger.js.map +0 -1
- package/dist/index.cjs +0 -755
- package/dist/index.cjs.map +0 -1
- package/dist/index.d.cts +0 -169
- package/dist/index.d.ts +0 -169
- package/dist/index.js +0 -699
- package/dist/index.js.map +0 -1
- package/dist/parser.cjs +0 -13
- package/dist/parser.cjs.map +0 -1
- package/dist/parser.d.cts +0 -154
- package/dist/parser.d.ts +0 -154
- package/dist/parser.js +0 -4
- package/dist/parser.js.map +0 -1
- package/dist/runtime.cjs +0 -13
- package/dist/runtime.cjs.map +0 -1
- package/dist/runtime.d.cts +0 -147
- package/dist/runtime.d.ts +0 -147
- package/dist/runtime.js +0 -4
- package/dist/runtime.js.map +0 -1
- package/dist/type-checker.cjs +0 -16
- package/dist/type-checker.cjs.map +0 -1
- package/dist/type-checker.d.cts +0 -105
- package/dist/type-checker.d.ts +0 -105
- package/dist/type-checker.js +0 -3
- package/dist/type-checker.js.map +0 -1
- package/dist/types-WQSk1Qs2.d.cts +0 -238
- package/dist/types-WQSk1Qs2.d.ts +0 -238
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@holoscript/core",
|
|
3
|
-
"version": "
|
|
4
|
-
"description": "
|
|
3
|
+
"version": "2.0.0",
|
|
4
|
+
"description": "HoloScript+: VR language with declarative syntax, state management, reactive traits, and VR interactions. Full backward compatible with original HoloScript.",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"main": "./dist/index.js",
|
|
7
7
|
"module": "./dist/index.mjs",
|
|
@@ -34,6 +34,12 @@
|
|
|
34
34
|
"require": "./dist/debugger.js"
|
|
35
35
|
}
|
|
36
36
|
},
|
|
37
|
+
"scripts": {
|
|
38
|
+
"build": "tsup",
|
|
39
|
+
"dev": "tsup --watch",
|
|
40
|
+
"test": "vitest run",
|
|
41
|
+
"test:watch": "vitest"
|
|
42
|
+
},
|
|
37
43
|
"keywords": [
|
|
38
44
|
"holoscript",
|
|
39
45
|
"vr",
|
|
@@ -54,13 +60,8 @@
|
|
|
54
60
|
},
|
|
55
61
|
"devDependencies": {
|
|
56
62
|
"tsup": "^8.0.1",
|
|
63
|
+
"typedoc": "^0.28.16",
|
|
57
64
|
"typescript": "^5.3.3",
|
|
58
65
|
"vitest": "^1.0.0"
|
|
59
|
-
},
|
|
60
|
-
"scripts": {
|
|
61
|
-
"build": "tsup",
|
|
62
|
-
"dev": "tsup --watch",
|
|
63
|
-
"test": "vitest run",
|
|
64
|
-
"test:watch": "vitest"
|
|
65
66
|
}
|
|
66
|
-
}
|
|
67
|
+
}
|
|
@@ -0,0 +1,227 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* HoloScript 2D Parser Extension
|
|
3
|
+
*
|
|
4
|
+
* Adds support for 2D UI elements to HoloScript for desktop/mobile apps.
|
|
5
|
+
* Works alongside 3D VR syntax for hybrid applications.
|
|
6
|
+
*/
|
|
7
|
+
import { logger } from './logger';
|
|
8
|
+
const UI_SECURITY_CONFIG = {
|
|
9
|
+
maxUIElements: 500,
|
|
10
|
+
maxNestingDepth: 10,
|
|
11
|
+
maxPropertyLength: 500,
|
|
12
|
+
allowedEventHandlers: ['onClick', 'onChange', 'onSubmit', 'onFocus', 'onBlur', 'onHover'],
|
|
13
|
+
};
|
|
14
|
+
export class HoloScript2DParser {
|
|
15
|
+
constructor() {
|
|
16
|
+
this.uiElements = new Map();
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Parse 2D UI element from HoloScript code
|
|
20
|
+
*/
|
|
21
|
+
parse2DElement(code, depth = 0) {
|
|
22
|
+
if (depth > UI_SECURITY_CONFIG.maxNestingDepth) {
|
|
23
|
+
logger.warn('Max nesting depth exceeded', { depth });
|
|
24
|
+
return null;
|
|
25
|
+
}
|
|
26
|
+
const trimmedCode = code.trim();
|
|
27
|
+
const lines = trimmedCode.split('\n');
|
|
28
|
+
if (lines.length === 0)
|
|
29
|
+
return null;
|
|
30
|
+
const firstLine = lines[0].trim();
|
|
31
|
+
const headerMatch = firstLine.match(/^([\w-]+)\s+(\w+)\s*\{/);
|
|
32
|
+
if (!headerMatch) {
|
|
33
|
+
logger.warn('Invalid 2D element syntax', { line: firstLine });
|
|
34
|
+
return null;
|
|
35
|
+
}
|
|
36
|
+
const [, elementType, name] = headerMatch;
|
|
37
|
+
if (!this.isValidUIElementType(elementType)) {
|
|
38
|
+
logger.warn('Invalid UI element type', { elementType });
|
|
39
|
+
return null;
|
|
40
|
+
}
|
|
41
|
+
const startIndex = trimmedCode.indexOf('{');
|
|
42
|
+
const endIndex = trimmedCode.lastIndexOf('}');
|
|
43
|
+
if (startIndex === -1 || endIndex === -1 || endIndex <= startIndex) {
|
|
44
|
+
return null;
|
|
45
|
+
}
|
|
46
|
+
const innerContent = trimmedCode.slice(startIndex + 1, endIndex).trim();
|
|
47
|
+
const innerLines = this.splitIntoLogicalBlocks(innerContent);
|
|
48
|
+
const properties = {};
|
|
49
|
+
const events = {};
|
|
50
|
+
const children = [];
|
|
51
|
+
for (const block of innerLines) {
|
|
52
|
+
const line = block.trim();
|
|
53
|
+
if (!line)
|
|
54
|
+
continue;
|
|
55
|
+
if (line.includes('{')) {
|
|
56
|
+
const childNode = this.parse2DElement(line, depth + 1);
|
|
57
|
+
if (childNode)
|
|
58
|
+
children.push(childNode);
|
|
59
|
+
continue;
|
|
60
|
+
}
|
|
61
|
+
const propMatch = line.match(/^(\w+):\s*(.+)$/);
|
|
62
|
+
if (propMatch) {
|
|
63
|
+
const [, key, rawValue] = propMatch;
|
|
64
|
+
if (UI_SECURITY_CONFIG.allowedEventHandlers.includes(key)) {
|
|
65
|
+
events[key] = rawValue.trim();
|
|
66
|
+
}
|
|
67
|
+
else {
|
|
68
|
+
properties[key] = this.parsePropertyValue(rawValue);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
const node = {
|
|
73
|
+
type: '2d-element',
|
|
74
|
+
elementType: elementType,
|
|
75
|
+
name,
|
|
76
|
+
properties: { ...this.getDefaultProperties(elementType), ...properties },
|
|
77
|
+
events: Object.keys(events).length > 0 ? events : undefined,
|
|
78
|
+
children: children.length > 0 ? children : undefined,
|
|
79
|
+
};
|
|
80
|
+
if (depth === 0) {
|
|
81
|
+
if (this.uiElements.size >= UI_SECURITY_CONFIG.maxUIElements) {
|
|
82
|
+
logger.warn('Max UI elements limit reached');
|
|
83
|
+
return null;
|
|
84
|
+
}
|
|
85
|
+
this.uiElements.set(name, node);
|
|
86
|
+
}
|
|
87
|
+
return node;
|
|
88
|
+
}
|
|
89
|
+
splitIntoLogicalBlocks(content) {
|
|
90
|
+
const blocks = [];
|
|
91
|
+
let currentBlock = '';
|
|
92
|
+
let bracketDepth = 0;
|
|
93
|
+
for (let i = 0; i < content.length; i++) {
|
|
94
|
+
const char = content[i];
|
|
95
|
+
if (char === '{')
|
|
96
|
+
bracketDepth++;
|
|
97
|
+
if (char === '}')
|
|
98
|
+
bracketDepth--;
|
|
99
|
+
currentBlock += char;
|
|
100
|
+
if (bracketDepth === 0) {
|
|
101
|
+
if (char === '\n' || i === content.length - 1) {
|
|
102
|
+
const trimmed = currentBlock.trim();
|
|
103
|
+
if (trimmed)
|
|
104
|
+
blocks.push(trimmed);
|
|
105
|
+
currentBlock = '';
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
const finalTrimmed = currentBlock.trim();
|
|
110
|
+
if (finalTrimmed)
|
|
111
|
+
blocks.push(finalTrimmed);
|
|
112
|
+
return blocks;
|
|
113
|
+
}
|
|
114
|
+
/**
|
|
115
|
+
* Parse voice command for 2D UI creation
|
|
116
|
+
*/
|
|
117
|
+
parse2DVoiceCommand(command) {
|
|
118
|
+
const tokens = command.toLowerCase().trim().split(/\s+/);
|
|
119
|
+
if (tokens.length < 3)
|
|
120
|
+
return null;
|
|
121
|
+
const action = tokens[0];
|
|
122
|
+
const elementType = tokens[1];
|
|
123
|
+
const name = tokens[2];
|
|
124
|
+
if (action !== 'create' && action !== 'add')
|
|
125
|
+
return null;
|
|
126
|
+
if (!this.isValidUIElementType(elementType))
|
|
127
|
+
return null;
|
|
128
|
+
const node = {
|
|
129
|
+
type: '2d-element',
|
|
130
|
+
elementType: elementType,
|
|
131
|
+
name,
|
|
132
|
+
properties: this.getDefaultProperties(elementType),
|
|
133
|
+
};
|
|
134
|
+
this.uiElements.set(name, node);
|
|
135
|
+
return node;
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Parse gesture for 2D UI interaction
|
|
139
|
+
*/
|
|
140
|
+
parse2DGesture(gestureType, position) {
|
|
141
|
+
switch (gestureType) {
|
|
142
|
+
case 'tap':
|
|
143
|
+
return this.createQuick2DElement('button', `button_${Date.now()}`, position);
|
|
144
|
+
case 'double-tap':
|
|
145
|
+
return this.createQuick2DElement('textinput', `input_${Date.now()}`, position);
|
|
146
|
+
case 'long-press':
|
|
147
|
+
return this.createQuick2DElement('panel', `panel_${Date.now()}`, position);
|
|
148
|
+
default:
|
|
149
|
+
return null;
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
createQuick2DElement(elementType, name, position) {
|
|
153
|
+
const node = {
|
|
154
|
+
type: '2d-element',
|
|
155
|
+
elementType,
|
|
156
|
+
name,
|
|
157
|
+
properties: {
|
|
158
|
+
...this.getDefaultProperties(elementType),
|
|
159
|
+
x: position.x,
|
|
160
|
+
y: position.y,
|
|
161
|
+
},
|
|
162
|
+
};
|
|
163
|
+
this.uiElements.set(name, node);
|
|
164
|
+
return node;
|
|
165
|
+
}
|
|
166
|
+
isValidUIElementType(type) {
|
|
167
|
+
const validTypes = [
|
|
168
|
+
'canvas', 'button', 'textinput', 'panel', 'text', 'image',
|
|
169
|
+
'list', 'modal', 'slider', 'toggle', 'dropdown',
|
|
170
|
+
'flex-container', 'grid-container', 'scroll-view', 'tab-view'
|
|
171
|
+
];
|
|
172
|
+
return validTypes.includes(type);
|
|
173
|
+
}
|
|
174
|
+
parsePropertyValue(value) {
|
|
175
|
+
const trimmed = value.trim();
|
|
176
|
+
if ((trimmed.startsWith('"') && trimmed.endsWith('"')) ||
|
|
177
|
+
(trimmed.startsWith("'") && trimmed.endsWith("'"))) {
|
|
178
|
+
return trimmed.slice(1, -1);
|
|
179
|
+
}
|
|
180
|
+
if (!isNaN(parseFloat(trimmed)) && isFinite(parseFloat(trimmed))) {
|
|
181
|
+
return parseFloat(trimmed);
|
|
182
|
+
}
|
|
183
|
+
if (trimmed === 'true')
|
|
184
|
+
return true;
|
|
185
|
+
if (trimmed === 'false')
|
|
186
|
+
return false;
|
|
187
|
+
if (trimmed.startsWith('[') && trimmed.endsWith(']')) {
|
|
188
|
+
const items = trimmed.slice(1, -1).split(',').map(item => this.parsePropertyValue(item.trim()));
|
|
189
|
+
return items;
|
|
190
|
+
}
|
|
191
|
+
return trimmed;
|
|
192
|
+
}
|
|
193
|
+
getDefaultProperties(elementType) {
|
|
194
|
+
const defaults = {
|
|
195
|
+
'canvas': { width: 800, height: 600, backgroundColor: '#ffffff' },
|
|
196
|
+
'button': { text: 'Button', width: 120, height: 40, backgroundColor: '#007bff', color: '#ffffff', borderRadius: 4 },
|
|
197
|
+
'textinput': { placeholder: '', width: 200, height: 36, fontSize: 14, borderColor: '#cccccc', borderWidth: 1, borderRadius: 4 },
|
|
198
|
+
'panel': { width: 200, height: 200, backgroundColor: '#f0f0f0', borderRadius: 0 },
|
|
199
|
+
'text': { content: 'Text', fontSize: 16, color: '#000000', fontFamily: 'sans-serif' },
|
|
200
|
+
'image': { src: '', width: 100, height: 100, fit: 'cover' },
|
|
201
|
+
'list': { items: [], itemHeight: 40, width: 200, height: 300 },
|
|
202
|
+
'modal': { title: 'Modal', width: 400, height: 300, visible: false, backgroundColor: '#ffffff' },
|
|
203
|
+
'slider': { min: 0, max: 100, value: 50, width: 200 },
|
|
204
|
+
'toggle': { checked: false, width: 50, height: 24 },
|
|
205
|
+
'dropdown': { options: [], selected: null, width: 200 },
|
|
206
|
+
'flex-container': { direction: 'row', gap: 10, padding: 10 },
|
|
207
|
+
'grid-container': { columns: 3, gap: 10, padding: 10 },
|
|
208
|
+
'scroll-view': { width: 300, height: 400, scrollDirection: 'vertical' },
|
|
209
|
+
'tab-view': { tabs: [], activeTabId: null, tabPosition: 'top', width: 400, height: 300 },
|
|
210
|
+
'dashboard': { title: 'Dashboard', width: 1200, height: 800, layout: 'grid', columns: 4 },
|
|
211
|
+
'card': { title: '', width: 300, height: 200, backgroundColor: '#ffffff', elevation: 2 },
|
|
212
|
+
'metric': { label: '', value: 0, unit: '', fontSize: 24, color: '#000000' },
|
|
213
|
+
'row': { height: 'auto', gap: 10, alignItems: 'center' },
|
|
214
|
+
'col': { width: 'auto', gap: 10, alignItems: 'stretch' },
|
|
215
|
+
};
|
|
216
|
+
return { ...defaults[elementType] };
|
|
217
|
+
}
|
|
218
|
+
getUIElements() {
|
|
219
|
+
return new Map(this.uiElements);
|
|
220
|
+
}
|
|
221
|
+
findElement(name) {
|
|
222
|
+
return this.uiElements.get(name) || null;
|
|
223
|
+
}
|
|
224
|
+
clear() {
|
|
225
|
+
this.uiElements.clear();
|
|
226
|
+
}
|
|
227
|
+
}
|
|
@@ -236,6 +236,11 @@ export class HoloScript2DParser {
|
|
|
236
236
|
'grid-container': { columns: 3, gap: 10, padding: 10 },
|
|
237
237
|
'scroll-view': { width: 300, height: 400, scrollDirection: 'vertical' },
|
|
238
238
|
'tab-view': { tabs: [], activeTabId: null, tabPosition: 'top', width: 400, height: 300 },
|
|
239
|
+
'dashboard': { title: 'Dashboard', width: 1200, height: 800, layout: 'grid', columns: 4 },
|
|
240
|
+
'card': { title: '', width: 300, height: 200, backgroundColor: '#ffffff', elevation: 2 },
|
|
241
|
+
'metric': { label: '', value: 0, unit: '', fontSize: 24, color: '#000000' },
|
|
242
|
+
'row': { height: 'auto', gap: 10, alignItems: 'center' },
|
|
243
|
+
'col': { width: 'auto', gap: 10, alignItems: 'stretch' },
|
|
239
244
|
};
|
|
240
245
|
return { ...defaults[elementType] };
|
|
241
246
|
}
|