pxt-ev3/sim/dalboard.ts

199 lines
6.4 KiB
TypeScript
Raw Normal View History

2017-07-11 10:15:17 +02:00
/// <reference path="../node_modules/pxt-core/built/pxtsim.d.ts"/>
/// <reference path="../node_modules/pxt-core/localtypings/pxtarget.d.ts"/>
/// <reference path="../built/common-sim.d.ts"/>
namespace pxsim {
2017-12-18 22:04:17 +01:00
export class EV3Board extends CoreBoard {
viewHost: visuals.BoardHost;
2017-07-11 10:15:17 +02:00
view: SVGSVGElement;
2017-12-18 22:04:17 +01:00
outputState: EV3OutputState;
analogState: EV3AnalogState;
uartState: EV3UArtState;
motorState: EV3MotorState;
screenState: ScreenState;
2017-12-18 22:04:17 +01:00
audioState: AudioState;
remoteState: RemoteState;
2017-12-18 22:04:17 +01:00
inputNodes: SensorNode[] = [];
brickNode: BrickNode;
outputNodes: MotorNode[] = [];
highcontrastMode?: boolean;
lightMode?: boolean;
public motorMap: pxt.Map<number> = {
2017-12-18 22:04:17 +01:00
0x01: 0,
0x02: 1,
0x04: 2,
0x08: 3
}
2017-07-11 10:15:17 +02:00
constructor() {
super()
this.bus.setNotify(DAL.DEVICE_ID_NOTIFY, DAL.DEVICE_ID_NOTIFY_ONE);
2017-12-18 22:04:17 +01:00
this.brickNode = new BrickNode();
this.outputState = new EV3OutputState();
this.analogState = new EV3AnalogState();
this.uartState = new EV3UArtState();
this.motorState = new EV3MotorState();
this.screenState = new ScreenState(["#97b5a6", "#000000"], visuals.SCREEN_WIDTH, visuals.SCREEN_HEIGHT);
2017-12-18 22:04:17 +01:00
this.audioState = new AudioState();
this.remoteState = new RemoteState();
2017-07-11 10:15:17 +02:00
}
receiveMessage(msg: SimulatorMessage) {
if (!runtime || runtime.dead) return;
switch (msg.type || "") {
case "eventbus": {
let ev = <SimulatorEventBusMessage>msg;
this.bus.queue(ev.id, ev.eventid, ev.value);
break;
}
case "serial": {
let data = (<SimulatorSerialMessage>msg).data || "";
// TODO
break;
}
}
}
initAsync(msg: SimulatorRunMessage): Promise<void> {
super.initAsync(msg);
const options = (msg.options || {}) as pxt.RuntimeOptions;
const boardDef = msg.boardDefinition;
const cmpsList = msg.parts;
const cmpDefs = msg.partDefinitions || {};
const fnArgs = msg.fnArgs;
const opts: visuals.BoardHostOpts = {
state: this,
boardDef: boardDef,
partsList: cmpsList,
partDefs: cmpDefs,
fnArgs: fnArgs,
maxWidth: "100%",
maxHeight: "100%",
highContrast: msg.highContrast,
light: msg.light
2017-07-11 10:15:17 +02:00
};
this.viewHost = new visuals.BoardHost(pxsim.visuals.mkBoardView({
boardDef,
visual: boardDef.visual,
highContrast: msg.highContrast,
light: msg.light
2017-07-11 10:15:17 +02:00
}), opts);
document.body.innerHTML = ""; // clear children
document.body.className = msg.light ? "light" : "";
document.body.appendChild(this.view = this.viewHost.getView() as SVGSVGElement);
2017-07-11 10:15:17 +02:00
this.inputNodes = [];
this.outputNodes = [];
this.highcontrastMode = msg.highContrast;
this.lightMode = msg.light;
2017-07-11 10:15:17 +02:00
return Promise.resolve();
}
screenshotAsync(width?: number): Promise<ImageData> {
return this.viewHost.screenshotAsync(width);
2017-07-11 10:15:17 +02:00
}
2017-12-18 22:04:17 +01:00
getBrickNode() {
return this.brickNode;
}
motorUsed(ports: number, large: boolean): boolean {
for (let i = 0; i < DAL.NUM_OUTPUTS; ++i) {
const p = 1 << i;
if (ports & p) {
const motorPort = this.motorMap[p];
const outputNode = this.outputNodes[motorPort];
if (!outputNode) {
this.outputNodes[motorPort] = new MotorNode(motorPort, large);
continue;
}
if (outputNode && outputNode.isLarge() != large)
return false;
}
}
return true;
}
2017-12-18 22:04:17 +01:00
getMotor(port: number, large?: boolean): MotorNode[] {
const r = [];
for (let i = 0; i < DAL.NUM_OUTPUTS; ++i) {
const p = 1 << i;
if (port & p) {
const motorPort = this.motorMap[p];
const outputNode = this.outputNodes[motorPort];
if (outputNode)
r.push(outputNode);
}
}
return r;
2017-12-18 22:04:17 +01:00
}
getMotors() {
return this.outputNodes;
}
hasSensor(port: number) {
return !!this.inputNodes[port];
}
getSensor(port: number, type: number): SensorNode | SensorExtendedNode {
if (!this.inputNodes[port]) {
2017-12-18 22:04:17 +01:00
switch (type) {
case DAL.DEVICE_TYPE_GYRO: this.inputNodes[port] = new GyroSensorNode(port); break;
case DAL.DEVICE_TYPE_COLOR: this.inputNodes[port] = new ColorSensorNode(port); break;
case DAL.DEVICE_TYPE_TOUCH: this.inputNodes[port] = new TouchSensorNode(port); break;
case DAL.DEVICE_TYPE_ULTRASONIC: this.inputNodes[port] = new UltrasonicSensorNode(port); break;
case DAL.DEVICE_TYPE_IR: this.inputNodes[port] = new InfraredSensorNode(port); break;
2017-12-18 22:04:17 +01:00
}
}
return this.inputNodes[port];
}
getInputNodes() {
return this.inputNodes;
}
2017-07-11 10:15:17 +02:00
}
export function initRuntimeWithDalBoard() {
U.assert(!runtime.board);
2017-12-18 22:04:17 +01:00
let b = new EV3Board();
2017-07-11 10:15:17 +02:00
runtime.board = b;
runtime.postError = (e) => {
// TODO
runtime.updateDisplay();
console.log('runtime error: ' + e);
2017-07-11 10:15:17 +02:00
}
}
2017-12-18 22:04:17 +01:00
export function ev3board(): EV3Board {
return runtime.board as EV3Board;
}
export function inLightMode(): boolean {
return /light=1/i.test(window.location.href) || ev3board().lightMode;
}
export function inHighcontrastMode(): boolean {
///
return ev3board().highcontrastMode;
}
2017-07-11 10:15:17 +02:00
if (!pxsim.initCurrentRuntime) {
pxsim.initCurrentRuntime = initRuntimeWithDalBoard;
}
}