checkpoint
This commit is contained in:
parent
78797aa175
commit
958134419d
@ -38,7 +38,9 @@ export class TheAbstractionEngine {
|
|||||||
[
|
[
|
||||||
new RadialObserve(),
|
new RadialObserve(),
|
||||||
new Modal(),
|
new Modal(),
|
||||||
new Level(isDev ? LevelNames.CarCadr : LevelNames.LevelSelection),
|
new Level(
|
||||||
|
isDev ? LevelNames.ChurchNumeralsOne : LevelNames.LevelSelection,
|
||||||
|
),
|
||||||
inputSystem,
|
inputSystem,
|
||||||
facingDirectionSystem,
|
facingDirectionSystem,
|
||||||
new Grid(
|
new Grid(
|
||||||
|
@ -24,8 +24,9 @@ import { Game } from "..";
|
|||||||
import { Grid as GridSystem, SystemNames } from "../systems";
|
import { Grid as GridSystem, SystemNames } from "../systems";
|
||||||
import { colors, tryWrap } from "../utils";
|
import { colors, tryWrap } from "../utils";
|
||||||
import {
|
import {
|
||||||
InvalidLambdaTermError,
|
DebrujinIndex,
|
||||||
SymbolTable,
|
SymbolTable,
|
||||||
|
Visitors,
|
||||||
emitNamed,
|
emitNamed,
|
||||||
interpret,
|
interpret,
|
||||||
} from "../../interpreter";
|
} from "../../interpreter";
|
||||||
@ -41,16 +42,9 @@ const APPLICATION_RESULTS: Record<
|
|||||||
export class FunctionApplication extends Entity {
|
export class FunctionApplication extends Entity {
|
||||||
private static spriteSpec = SPRITE_SPECS.get(Sprites.BUBBLE) as SpriteSpec;
|
private static spriteSpec = SPRITE_SPECS.get(Sprites.BUBBLE) as SpriteSpec;
|
||||||
|
|
||||||
private symbolTable: SymbolTable;
|
|
||||||
|
|
||||||
constructor(gridPosition: Coord2D, lambdaTerm: string) {
|
constructor(gridPosition: Coord2D, lambdaTerm: string) {
|
||||||
super(EntityNames.FunctionApplication);
|
super(EntityNames.FunctionApplication);
|
||||||
|
|
||||||
this.symbolTable = new SymbolTable();
|
|
||||||
Object.keys(APPLICATION_RESULTS).forEach((key) => {
|
|
||||||
this.symbolTable.add(key);
|
|
||||||
});
|
|
||||||
|
|
||||||
const dimension = {
|
const dimension = {
|
||||||
width: FunctionApplication.spriteSpec.width,
|
width: FunctionApplication.spriteSpec.width,
|
||||||
height: FunctionApplication.spriteSpec.height,
|
height: FunctionApplication.spriteSpec.height,
|
||||||
@ -151,7 +145,10 @@ export class FunctionApplication extends Entity {
|
|||||||
);
|
);
|
||||||
const newCode = applicationTerm.code.replace("_INPUT", functionTerm.code);
|
const newCode = applicationTerm.code.replace("_INPUT", functionTerm.code);
|
||||||
|
|
||||||
const result = tryWrap(() => interpret(newCode, this.symbolTable, true));
|
const { symbolTable, visitors } = this.getVisitors(game);
|
||||||
|
const result = tryWrap(() =>
|
||||||
|
interpret(newCode, symbolTable, true, visitors),
|
||||||
|
);
|
||||||
applicationTerm.last = result;
|
applicationTerm.last = result;
|
||||||
if (result.error || !result.data) {
|
if (result.error || !result.data) {
|
||||||
console.error(result.error);
|
console.error(result.error);
|
||||||
@ -166,13 +163,6 @@ export class FunctionApplication extends Entity {
|
|||||||
|
|
||||||
let applicationResultingEntity: Entity | null = null; // this should be its own function
|
let applicationResultingEntity: Entity | null = null; // this should be its own function
|
||||||
const { data } = result;
|
const { data } = result;
|
||||||
if ("application" in data) {
|
|
||||||
// if we get an application that means we didn't interpret correctly.
|
|
||||||
// this should "not" happen and should be fatal.
|
|
||||||
throw new InvalidLambdaTermError(
|
|
||||||
"produced term should not be an application",
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if ("abstraction" in data) {
|
if ("abstraction" in data) {
|
||||||
const code = emitNamed(data);
|
const code = emitNamed(data);
|
||||||
applicationResultingEntity = new FunctionBox(grid.gridPosition, code);
|
applicationResultingEntity = new FunctionBox(grid.gridPosition, code);
|
||||||
@ -187,20 +177,100 @@ export class FunctionApplication extends Entity {
|
|||||||
}
|
}
|
||||||
|
|
||||||
game.removeEntity(entity.id);
|
game.removeEntity(entity.id);
|
||||||
if (applicationResultingEntity) {
|
if (!applicationResultingEntity) {
|
||||||
const grid = applicationResultingEntity.getComponent<Grid>(
|
return;
|
||||||
ComponentNames.Grid,
|
}
|
||||||
);
|
|
||||||
grid.movingDirection = entityGrid.previousDirection;
|
|
||||||
applicationResultingEntity.addComponent(grid);
|
|
||||||
|
|
||||||
|
applicationResultingEntity.getComponent<Grid>(
|
||||||
|
ComponentNames.Grid,
|
||||||
|
).movingDirection = entityGrid.previousDirection;
|
||||||
game.addEntity(applicationResultingEntity);
|
game.addEntity(applicationResultingEntity);
|
||||||
}
|
}
|
||||||
|
|
||||||
SOUNDS.get(LambdaTransformSound.name)!.play();
|
private getVisitors(game: Game): {
|
||||||
|
visitors: Visitors;
|
||||||
|
symbolTable: SymbolTable;
|
||||||
|
} {
|
||||||
|
const directionKeywords = {
|
||||||
|
_LEFT: Direction.LEFT,
|
||||||
|
_RIGHT: Direction.RIGHT,
|
||||||
|
_DOWN: Direction.DOWN,
|
||||||
|
_UP: Direction.UP,
|
||||||
|
};
|
||||||
|
const entityKeywords = {
|
||||||
|
_KEY: (pos: Coord2D) => new Key(pos),
|
||||||
|
};
|
||||||
|
|
||||||
|
const visitors: Visitors = new Map();
|
||||||
|
visitors.set("_SPAWN", (_term) => {
|
||||||
|
const position = this.getComponent<Grid>(
|
||||||
|
ComponentNames.Grid,
|
||||||
|
).gridPosition;
|
||||||
|
return {
|
||||||
|
abstraction: {
|
||||||
|
param: "_DIRECTION",
|
||||||
|
body: (direction) => {
|
||||||
|
const destinationDirection =
|
||||||
|
directionKeywords[
|
||||||
|
(direction as DebrujinIndex)
|
||||||
|
.name! as keyof typeof directionKeywords
|
||||||
|
];
|
||||||
|
const destination = game
|
||||||
|
.getSystem<GridSystem>(SystemNames.Grid)
|
||||||
|
.getNewGridPosition(position, destinationDirection);
|
||||||
|
return {
|
||||||
|
abstraction: {
|
||||||
|
param: "_ENTITY",
|
||||||
|
body: (entityType) => {
|
||||||
|
const entityFactory =
|
||||||
|
entityKeywords[
|
||||||
|
(entityType as DebrujinIndex)
|
||||||
|
.name! as keyof typeof entityKeywords
|
||||||
|
];
|
||||||
|
const newEntity = entityFactory(destination);
|
||||||
|
game.addEntity(newEntity);
|
||||||
|
return {
|
||||||
|
abstraction: {
|
||||||
|
param: "_x",
|
||||||
|
body: (_t) => {
|
||||||
|
return {
|
||||||
|
application: {
|
||||||
|
left: {
|
||||||
|
index: 1,
|
||||||
|
name: "_SPAWN",
|
||||||
|
},
|
||||||
|
args: [direction, entityType],
|
||||||
|
},
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
visitors,
|
||||||
|
symbolTable: SymbolTable.from(
|
||||||
|
Array.from(visitors.keys())
|
||||||
|
.concat(Object.keys(APPLICATION_RESULTS))
|
||||||
|
.concat(Object.keys(directionKeywords))
|
||||||
|
.concat(Object.keys(entityKeywords))
|
||||||
|
.concat(["_x"]),
|
||||||
|
),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
private addParticles(game: Game, position: Coord2D) {
|
||||||
|
const gridSystem = game.getSystem<GridSystem>(SystemNames.Grid);
|
||||||
const { dimension } = gridSystem;
|
const { dimension } = gridSystem;
|
||||||
|
SOUNDS.get(LambdaTransformSound.name)!.play();
|
||||||
const particles = new Particles({
|
const particles = new Particles({
|
||||||
center: gridSystem.gridToScreenPosition(nextPosition),
|
center: gridSystem.gridToScreenPosition(position),
|
||||||
spawnerDimensions: {
|
spawnerDimensions: {
|
||||||
width: dimension.width / 2,
|
width: dimension.width / 2,
|
||||||
height: dimension.height / 2,
|
height: dimension.height / 2,
|
||||||
|
@ -9,8 +9,6 @@ import {
|
|||||||
Player,
|
Player,
|
||||||
Wall,
|
Wall,
|
||||||
} from "../entities";
|
} from "../entities";
|
||||||
import { Piston } from "../entities/Piston";
|
|
||||||
import { Direction } from "../interfaces";
|
|
||||||
import { Grid, SystemNames } from "../systems";
|
import { Grid, SystemNames } from "../systems";
|
||||||
import { normalRandom } from "../utils";
|
import { normalRandom } from "../utils";
|
||||||
|
|
||||||
|
45
src/engine/levels/ChurchNumeralsOne.ts
Normal file
45
src/engine/levels/ChurchNumeralsOne.ts
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
import { FunctionApplication, Grass, LambdaFactory, Player } from "../entities";
|
||||||
|
import { Game } from "../Game";
|
||||||
|
import { Grid, SystemNames } from "../systems";
|
||||||
|
import { normalRandom } from "../utils";
|
||||||
|
import { Level } from "./Level";
|
||||||
|
import { LevelNames } from "./LevelNames";
|
||||||
|
|
||||||
|
export class ChurchNumeralsOne extends Level {
|
||||||
|
constructor() {
|
||||||
|
super(LevelNames.ChurchNumeralsOne);
|
||||||
|
}
|
||||||
|
|
||||||
|
public init(game: Game) {
|
||||||
|
const grid = game.getSystem<Grid>(SystemNames.Grid);
|
||||||
|
const dimensions = grid.getGridDimensions();
|
||||||
|
|
||||||
|
const grasses = Array.from({ length: dimensions.width })
|
||||||
|
.fill(0)
|
||||||
|
.map(() => {
|
||||||
|
// random grass
|
||||||
|
return new Grass({
|
||||||
|
x: Math.floor(
|
||||||
|
normalRandom(dimensions.width / 2, dimensions.width / 4, 1.5),
|
||||||
|
),
|
||||||
|
y: Math.floor(
|
||||||
|
normalRandom(dimensions.height / 2, dimensions.height / 4, 1.5),
|
||||||
|
),
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
[
|
||||||
|
...grasses,
|
||||||
|
new LambdaFactory({ x: 1, y: 1 }, "(\\ (f) . (\\ (x) . (f f x)))", 1),
|
||||||
|
new FunctionApplication(
|
||||||
|
{ x: 2, y: 2 },
|
||||||
|
"(_INPUT ((_SPAWN _RIGHT) _KEY))",
|
||||||
|
),
|
||||||
|
new FunctionApplication(
|
||||||
|
{ x: 3, y: 3 },
|
||||||
|
"(_INPUT _EMPTY)",
|
||||||
|
),
|
||||||
|
new Player({ x: 0, y: 0 }),
|
||||||
|
].forEach((e) => game.addEntity(e));
|
||||||
|
}
|
||||||
|
}
|
@ -1,5 +1,6 @@
|
|||||||
export namespace LevelNames {
|
export namespace LevelNames {
|
||||||
export const Tutorial = "0";
|
export const Tutorial = "0";
|
||||||
export const CarCadr = "1";
|
export const CarCadr = "1";
|
||||||
|
export const ChurchNumeralsOne = "2";
|
||||||
export const LevelSelection = "LevelSelection";
|
export const LevelSelection = "LevelSelection";
|
||||||
}
|
}
|
||||||
|
@ -36,6 +36,7 @@ export class Tutorial extends Level {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// TODO: new level which adds introductory syntax
|
||||||
const entities = [
|
const entities = [
|
||||||
...grasses,
|
...grasses,
|
||||||
new Sign(
|
new Sign(
|
||||||
@ -51,7 +52,7 @@ export class Tutorial extends Level {
|
|||||||
new Wall({ x: 11, y: 10 }),
|
new Wall({ x: 11, y: 10 }),
|
||||||
new Curry({ x: 10, y: 10 }),
|
new Curry({ x: 10, y: 10 }),
|
||||||
new LockedDoor({ x: 9, y: 10 }),
|
new LockedDoor({ x: 9, y: 10 }),
|
||||||
new LambdaFactory({ x: 6, y: 3 }, "// TODO: Remove line\n(λ (x) . x)", 3),
|
new LambdaFactory({ x: 6, y: 3 }, "// TODO: Remove this comment\n(λ (x) . x)", 3),
|
||||||
new FunctionApplication({ x: 6, y: 6 }, "(_INPUT _KEY)"),
|
new FunctionApplication({ x: 6, y: 6 }, "(_INPUT _KEY)"),
|
||||||
new Player({ x: 2, y: 2 }),
|
new Player({ x: 2, y: 2 }),
|
||||||
];
|
];
|
||||||
|
@ -6,13 +6,16 @@ export * from "./CarCadr";
|
|||||||
|
|
||||||
import { LevelNames } from ".";
|
import { LevelNames } from ".";
|
||||||
import { CarCadr, LevelSelection, Tutorial, Level } from ".";
|
import { CarCadr, LevelSelection, Tutorial, Level } from ".";
|
||||||
|
import { ChurchNumeralsOne } from "./ChurchNumeralsOne";
|
||||||
|
|
||||||
export const LEVELS: Level[] = [
|
export const LEVELS: Level[] = [
|
||||||
new LevelSelection(),
|
new LevelSelection(),
|
||||||
new Tutorial(),
|
new Tutorial(),
|
||||||
new CarCadr(),
|
new CarCadr(),
|
||||||
|
new ChurchNumeralsOne(),
|
||||||
];
|
];
|
||||||
export const LEVEL_PROGRESSION: Record<string, string[]> = {
|
export const LEVEL_PROGRESSION: Record<string, string[]> = {
|
||||||
[LevelNames.LevelSelection]: [LevelNames.Tutorial],
|
[LevelNames.LevelSelection]: [LevelNames.Tutorial],
|
||||||
[LevelNames.Tutorial]: [LevelNames.CarCadr],
|
[LevelNames.Tutorial]: [LevelNames.CarCadr],
|
||||||
|
[LevelNames.CarCadr]: [LevelNames.ChurchNumeralsOne],
|
||||||
};
|
};
|
||||||
|
@ -46,4 +46,10 @@ export class SymbolTable {
|
|||||||
public createChild(): SymbolTable {
|
public createChild(): SymbolTable {
|
||||||
return new SymbolTable(this);
|
return new SymbolTable(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static from(collection: Array<string> | Set<string>): SymbolTable {
|
||||||
|
const table = new SymbolTable();
|
||||||
|
collection.forEach((symbol) => table.add(symbol));
|
||||||
|
return table;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -11,6 +11,11 @@ export class InvalidLambdaTermError extends Error {}
|
|||||||
|
|
||||||
export class MaxRecursionDepthError extends Error {}
|
export class MaxRecursionDepthError extends Error {}
|
||||||
|
|
||||||
|
export type Visitors = Map<
|
||||||
|
string,
|
||||||
|
(term: DebrujinifiedLambdaTerm) => DebrujinifiedLambdaTerm
|
||||||
|
>;
|
||||||
|
|
||||||
export type DebrujinAbstraction = {
|
export type DebrujinAbstraction = {
|
||||||
abstraction: {
|
abstraction: {
|
||||||
param: string;
|
param: string;
|
||||||
@ -28,6 +33,7 @@ export type DebrujinApplication = {
|
|||||||
export type DebrujinIndex = { name: string; index: number };
|
export type DebrujinIndex = { name: string; index: number };
|
||||||
|
|
||||||
export type DebrujinifiedLambdaTerm =
|
export type DebrujinifiedLambdaTerm =
|
||||||
|
| ((t: DebrujinifiedLambdaTerm) => DebrujinifiedLambdaTerm)
|
||||||
| DebrujinAbstraction
|
| DebrujinAbstraction
|
||||||
| DebrujinApplication
|
| DebrujinApplication
|
||||||
| DebrujinIndex;
|
| DebrujinIndex;
|
||||||
@ -76,6 +82,10 @@ export const substitute = (
|
|||||||
index: number,
|
index: number,
|
||||||
withTerm: DebrujinifiedLambdaTerm,
|
withTerm: DebrujinifiedLambdaTerm,
|
||||||
): DebrujinifiedLambdaTerm => {
|
): DebrujinifiedLambdaTerm => {
|
||||||
|
if (typeof inTerm === "function") {
|
||||||
|
return inTerm(withTerm);
|
||||||
|
}
|
||||||
|
|
||||||
if ("index" in inTerm) {
|
if ("index" in inTerm) {
|
||||||
if (inTerm.index > index) {
|
if (inTerm.index > index) {
|
||||||
return adjustIndices(inTerm, -1);
|
return adjustIndices(inTerm, -1);
|
||||||
@ -154,51 +164,45 @@ export const adjustIndices = (
|
|||||||
|
|
||||||
export const betaReduce = (
|
export const betaReduce = (
|
||||||
term: DebrujinifiedLambdaTerm,
|
term: DebrujinifiedLambdaTerm,
|
||||||
|
visitors: Visitors,
|
||||||
maxDepth: number,
|
maxDepth: number,
|
||||||
): DebrujinifiedLambdaTerm => {
|
): DebrujinifiedLambdaTerm => {
|
||||||
if (maxDepth === 0) {
|
if (maxDepth === 0) {
|
||||||
throw new MaxRecursionDepthError("max recursion depth identified");
|
throw new MaxRecursionDepthError("max recursion depth identified");
|
||||||
}
|
}
|
||||||
if ("index" in term) {
|
if (typeof term === "function") {
|
||||||
return term;
|
return term;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ("index" in term) {
|
||||||
|
const replacement = visitors.get(term.name)?.apply(null, [term]);
|
||||||
|
return replacement ?? term;
|
||||||
|
}
|
||||||
|
|
||||||
if ("abstraction" in term) {
|
if ("abstraction" in term) {
|
||||||
const { body, param } = term.abstraction;
|
const { body, param } = term.abstraction;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
abstraction: {
|
abstraction: {
|
||||||
body: betaReduce(body, maxDepth - 1),
|
body: betaReduce(body, visitors, maxDepth - 1),
|
||||||
param,
|
param,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
if ("application" in term) {
|
if ("application" in term) {
|
||||||
const { left } = term.application;
|
const { left, args } = term.application;
|
||||||
const args = term.application.args.map((term) =>
|
const [reducedLeft, ...reducedArgs] = [left, ...args].map((term) =>
|
||||||
betaReduce(term, maxDepth - 1),
|
betaReduce(term, visitors, maxDepth - 1),
|
||||||
);
|
);
|
||||||
|
return reducedArgs.reduce((acc: DebrujinifiedLambdaTerm, x) => {
|
||||||
return args.reduce((acc: DebrujinifiedLambdaTerm, x) => {
|
|
||||||
if ("abstraction" in acc) {
|
if ("abstraction" in acc) {
|
||||||
const { body } = acc.abstraction;
|
const { body } = acc.abstraction;
|
||||||
const newBody = substitute(body, 1, x);
|
const substituted = substitute(body, 1, x);
|
||||||
return newBody;
|
return substituted;
|
||||||
}
|
}
|
||||||
if ("application" in acc) {
|
return acc;
|
||||||
const {
|
}, reducedLeft);
|
||||||
application: { left, args },
|
|
||||||
} = acc;
|
|
||||||
return {
|
|
||||||
application: {
|
|
||||||
left,
|
|
||||||
args: [...args, x],
|
|
||||||
},
|
|
||||||
};
|
|
||||||
}
|
|
||||||
return { application: { left: acc, args: [x] } };
|
|
||||||
}, left);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
throw new InvalidLambdaTermError(
|
throw new InvalidLambdaTermError(
|
||||||
@ -207,6 +211,10 @@ export const betaReduce = (
|
|||||||
};
|
};
|
||||||
|
|
||||||
export const emitDebrujin = (term: DebrujinifiedLambdaTerm): string => {
|
export const emitDebrujin = (term: DebrujinifiedLambdaTerm): string => {
|
||||||
|
if (typeof term === "function") {
|
||||||
|
return term.toString();
|
||||||
|
}
|
||||||
|
|
||||||
if ("index" in term) {
|
if ("index" in term) {
|
||||||
return term.index.toString();
|
return term.index.toString();
|
||||||
}
|
}
|
||||||
@ -250,18 +258,20 @@ export const interpret = (
|
|||||||
term: string,
|
term: string,
|
||||||
symbolTable = new SymbolTable(),
|
symbolTable = new SymbolTable(),
|
||||||
allowUnderscores = false, // in our world, underscores should be internal to the game.
|
allowUnderscores = false, // in our world, underscores should be internal to the game.
|
||||||
maxDepth = 15,
|
visitors: Visitors = new Map(),
|
||||||
|
maxDepth = 20,
|
||||||
): DebrujinifiedLambdaTerm => {
|
): DebrujinifiedLambdaTerm => {
|
||||||
const ast = parse(term, allowUnderscores);
|
const ast = parse(term, allowUnderscores);
|
||||||
const debrujined = debrujinify(ast, symbolTable);
|
const debrujined = debrujinify(ast, symbolTable);
|
||||||
|
|
||||||
let prev = debrujined;
|
let prev = debrujined;
|
||||||
let next = betaReduce(prev, maxDepth);
|
let next = betaReduce(prev, visitors, maxDepth);
|
||||||
|
|
||||||
while (emitDebrujin(prev) !== emitDebrujin(next)) {
|
while (emitDebrujin(prev) !== emitDebrujin(next)) {
|
||||||
// alpha equivalence
|
// alpha equivalence
|
||||||
prev = next;
|
prev = next;
|
||||||
next = betaReduce(prev, maxDepth);
|
next = betaReduce(prev, visitors, maxDepth);
|
||||||
}
|
}
|
||||||
|
console.log(next);
|
||||||
return next;
|
return next;
|
||||||
};
|
};
|
||||||
|
Loading…
x
Reference in New Issue
Block a user