2021-02-08 05:50:50 +01:00
|
|
|
/**
|
|
|
|
* Extracts the members of a constant array as a type. Used as:
|
2021-02-21 18:38:21 -08:00
|
|
|
*
|
2021-02-08 05:50:50 +01:00
|
|
|
* @example
|
|
|
|
* const SOME_VALUES = ['a', 'b', 1, 2] as const;
|
|
|
|
* type SomeValues = MemberOf<typeof SOME_VALUES>; // 'a' | 'b' | 1 | 2
|
|
|
|
*/
|
|
|
|
export type MemberOf<T extends ReadonlyArray<unknown>> =
|
|
|
|
T extends ReadonlyArray<infer E> ? E : never;
|
|
|
|
|
2021-03-29 01:39:18 +02:00
|
|
|
/**
|
|
|
|
* Recursively extracts all members of a constant array as a type. Used as:
|
2021-07-06 17:04:02 -07:00
|
|
|
*
|
2021-03-29 01:39:18 +02:00
|
|
|
* @example
|
|
|
|
* const SOME_ARRAYS = [['a'],['b', 2], 3] as const;
|
|
|
|
* type SomeArrayValues = DeepMemberOf<typeof SOME_ARRAYS>; // 'a' | 'b' | 2 | 3
|
|
|
|
*/
|
|
|
|
export type DeepMemberOf<T extends ReadonlyArray<unknown>> =
|
|
|
|
T extends ReadonlyArray<infer E>
|
|
|
|
? (E extends ReadonlyArray<unknown> ? DeepMemberOf<E> : E)
|
|
|
|
: never;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Extracts the declared keys of a type by removing `string` and `number`.
|
2021-07-06 17:04:02 -07:00
|
|
|
*
|
2021-03-29 01:39:18 +02:00
|
|
|
* Cribbed from the interwebs:
|
|
|
|
* https://github.com/microsoft/TypeScript/issues/25987#issuecomment-408339599
|
|
|
|
*/
|
|
|
|
export type KnownKeys<T> = {
|
|
|
|
[K in keyof T]: string extends K ? never : number extends K ? never : K
|
|
|
|
} extends { [_ in keyof T]: infer U } ? U : never;
|
|
|
|
|
2021-03-31 02:27:44 +02:00
|
|
|
/**
|
|
|
|
* Extracts the declared values of a constant object.
|
|
|
|
*/
|
|
|
|
export type KnownValues<T> = T extends {
|
|
|
|
[_ in keyof T]: infer U } ? U : never;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Replacement for `includes` on constant types that is also a type assertion.
|
2021-07-06 17:04:02 -07:00
|
|
|
*
|
2021-03-31 02:27:44 +02:00
|
|
|
* @example
|
|
|
|
* const SOME_VALUES = [1, 2, 'a'] as const;
|
|
|
|
* let n: number = 1;
|
|
|
|
* let r = includes(SOME_VALUES, n); // r === true, n is 1 | 2 | 'a'
|
|
|
|
* n = 5;
|
|
|
|
* r = includes(SOME_VALUES, n); // r === false, n is number
|
|
|
|
*/
|
|
|
|
export function includes<S extends T, T>(a: ReadonlyArray<S>, v: T): v is S {
|
|
|
|
return (a as ReadonlyArray<T>).includes(v);
|
|
|
|
}
|
2021-03-29 01:39:18 +02:00
|
|
|
|
2021-02-08 05:50:50 +01:00
|
|
|
/** A bit. */
|
|
|
|
export type bit = 0 | 1;
|
|
|
|
|
|
|
|
/** A nibble. */
|
|
|
|
export type nibble =
|
|
|
|
0x0 | 0x1 | 0x2 | 0x3 | 0x4 | 0x5 | 0x6 | 0x7 |
|
|
|
|
0x8 | 0x9 | 0xa | 0xb | 0xc | 0xd | 0xe | 0xf;
|
|
|
|
|
Typescript conversion of several files, including js/cpu6502 (#38)
* Convert `js/util.js` to Typescript and add tests
Besides converting `js/util.js` to Typescript, this change also adds
`js/types.ts` that defines common types used in apple2js. Some of
these types, like `byte` and `word` are for information only.
* Convert `js/base64.js` to Typescript
This also adds a new type, `memory`, that is either an array of
numbers, or a Uint8Array.
* Convert `js/ram.js` to Typescript
This change does not convert `RAM` to a class; it just introduces types.
* Basic typing of cpu6502
This is a really rough first pass. There are some problems that can't
be fixed until this is turned into a real class, but at least all of
the function arguments are now typed. This caught a few cases where
extra arguments were being passed in.
* Convert `js/cpu6502` to a class
In theory, idiomatic classes should be better than the previous
closure-based classes. However, this conversion shows that the
instruction table does not fit well with idiomatic classes as method
referenced in the table need to be called with the correct `this`
everywhere.
This should, at best, be considered a first attempt.
2020-11-01 17:43:48 +01:00
|
|
|
/** A byte (0..255). This is not enforced by the compiler. */
|
|
|
|
export type byte = number;
|
|
|
|
|
|
|
|
/** A word (0..65535). This is not enforced by the compiler. */
|
|
|
|
export type word = number;
|
|
|
|
|
2020-11-24 17:48:14 +01:00
|
|
|
/** A raw region of memory. */
|
2021-02-08 05:50:50 +01:00
|
|
|
export type memory = Uint8Array;
|
|
|
|
|
|
|
|
/** A raw region of memory. */
|
|
|
|
export type rom = ReadonlyUint8Array;
|
Typescript conversion of several files, including js/cpu6502 (#38)
* Convert `js/util.js` to Typescript and add tests
Besides converting `js/util.js` to Typescript, this change also adds
`js/types.ts` that defines common types used in apple2js. Some of
these types, like `byte` and `word` are for information only.
* Convert `js/base64.js` to Typescript
This also adds a new type, `memory`, that is either an array of
numbers, or a Uint8Array.
* Convert `js/ram.js` to Typescript
This change does not convert `RAM` to a class; it just introduces types.
* Basic typing of cpu6502
This is a really rough first pass. There are some problems that can't
be fixed until this is turned into a real class, but at least all of
the function arguments are now typed. This caught a few cases where
extra arguments were being passed in.
* Convert `js/cpu6502` to a class
In theory, idiomatic classes should be better than the previous
closure-based classes. However, this conversion shows that the
instruction table does not fit well with idiomatic classes as method
referenced in the table need to be called with the correct `this`
everywhere.
This should, at best, be considered a first attempt.
2020-11-01 17:43:48 +01:00
|
|
|
|
2020-11-24 17:48:14 +01:00
|
|
|
export interface Memory {
|
2021-02-08 05:50:50 +01:00
|
|
|
/** Read a byte. */
|
|
|
|
read(page: byte, offset: byte): byte;
|
|
|
|
/** Write a byte. */
|
|
|
|
write(page: byte, offset: byte, value: byte): void;
|
2020-11-24 17:48:14 +01:00
|
|
|
}
|
|
|
|
|
2021-03-06 15:04:13 -08:00
|
|
|
/** A mapped region of memory. */
|
|
|
|
export interface MemoryPages extends Memory {
|
|
|
|
/** Start page. */
|
|
|
|
start(): byte;
|
|
|
|
/** End page, inclusive. */
|
|
|
|
end(): byte;
|
|
|
|
}
|
|
|
|
|
2021-02-21 18:38:21 -08:00
|
|
|
/* An interface card */
|
2022-05-31 17:38:40 +02:00
|
|
|
export interface Card<StateT = unknown> extends Memory, Restorable<StateT> {
|
2021-02-21 18:38:21 -08:00
|
|
|
/* Reset the card */
|
2021-03-13 13:18:32 -08:00
|
|
|
reset?(): void;
|
2021-02-21 18:38:21 -08:00
|
|
|
|
|
|
|
/* Draw card to canvas */
|
2021-03-15 12:51:40 -07:00
|
|
|
blit?(): ImageData | undefined;
|
2021-02-21 18:38:21 -08:00
|
|
|
|
|
|
|
/* Process period events */
|
|
|
|
tick?(): void;
|
|
|
|
|
|
|
|
/* Read or Write an I/O switch */
|
|
|
|
ioSwitch(off: byte, val?: byte): byte | undefined;
|
|
|
|
}
|
|
|
|
|
|
|
|
export type TapeData = Array<[duration: number, high: boolean]>;
|
|
|
|
|
2022-05-17 19:08:28 -07:00
|
|
|
export interface Restorable<T = unknown> {
|
2021-02-08 05:50:50 +01:00
|
|
|
getState(): T;
|
|
|
|
setState(state: T): void;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Read-only typed arrays for constants
|
|
|
|
export type TypedArrayMutableProperties = 'copyWithin' | 'fill' | 'reverse' | 'set' | 'sort';
|
|
|
|
export interface ReadonlyUint8Array extends Omit<Uint8Array, TypedArrayMutableProperties> {
|
2022-05-10 08:04:20 -07:00
|
|
|
readonly [n: number]: number;
|
2021-02-21 18:38:21 -08:00
|
|
|
}
|
2021-03-15 12:51:40 -07:00
|
|
|
|
|
|
|
// Readonly RGB color value
|
|
|
|
export type Color = readonly [r: byte, g: byte, b: byte];
|