refactor: Switch to 'map-like object keyed by string and type vector' format

See https://code.studioinfinity.org/glen/nanomath/wiki/Item-Specifications.
  Also stubs out the TypeDispatcher, mocking the merge function, so we
  can see that all of the proper things will be added.

  Ready for initial implementation of the TypeDispatcher.
This commit is contained in:
Glen Whitney 2025-04-02 11:22:53 -07:00
parent 040ec377a1
commit 69ef928b6e
10 changed files with 84 additions and 15 deletions

5
src/core/Type.js Normal file
View file

@ -0,0 +1,5 @@
export class Type {
constructor(f) {
this.test = f
}
}

View file

@ -0,0 +1,37 @@
import {Type} from './Type.js'
import {Implementations, isPlainObject} from './helpers.js'
export class TypeDispatcher {
constructor(...specs) {
for (const spec of specs) this.merge(spec)
}
merge(spec) {
if (!spec) return
if (typeof spec != 'object') {
throw new TypeError(
`TypeDispatcher specifications must be objects, not '${spec}'.`)
}
for (const key in spec) {
const val = spec[key]
if (val instanceof Type) {
console.log(`Pretending to install type ${key}: ${val}`)
continue
}
if (val instanceof Implementations) {
console.log(`Pretending to install implementations for ${key}`)
console.log(` --> ${val}`)
continue
}
if (typeof val === 'function') {
console.log(`Pretend install of catchall implementation for ${key}`)
continue
}
if (isPlainObject(val)) {
this.merge(val)
continue
}
console.log(`Pretend install of catchall value for ${key}: ${val}`)
}
}
}

27
src/core/helpers.js Normal file
View file

@ -0,0 +1,27 @@
import {Type} from './Type.js'
export class Implementations {
constructor(imps) {
this.patterns = new Map()
for (let i = 0; i < imps.length; ++i) {
let pattern = imps[i++]
if (!Array.isArray(pattern)) pattern = [pattern]
if (!pattern.every(item => item instanceof Type)) {
throw new TypeError(
`Implementation pattern ${pattern} contains non-Type entry`)
}
this.patterns.set(pattern, imps[i])
}
}
}
export const onType = (...imps) => new Implementations(imps)
export const Returns = (type, f) => (f.returns = type, f)
export const isPlainObject = (obj) => {
if (typeof obj !== 'object') return false
if (!obj) return false // excludes null
const proto = Object.getPrototypeOf(obj)
return !proto || proto === Object.prototype
}

View file

@ -1,7 +1,6 @@
import {TypeDispatcher} from './core/TypeDispatcher.js'
import {numbers} from './numbers.js'
for (const key in numbers) {
for (const subkey in numbers[key]) {
console.log(`${key}.${subkey} =`, numbers[key][subkey])
}
}
const math = new TypeDispatcher(numbers)
export default math

View file

@ -1,3 +1,3 @@
export const Number = {
test: n => typeof n === 'number'
}
import {Type} from '../core/Type.js'
export const Number = new Type(n => typeof n === 'number')

View file

@ -1,4 +1,4 @@
import {plain} from './tools.js'
import {plain} from './helpers.js'
export const abs = plain(Math.abs)
export const absquare = plain(a => a*a)

5
src/number/helpers.js Normal file
View file

@ -0,0 +1,5 @@
import {Returns, onType} from '../core/helpers.js'
import {Number} from './Number.js'
export const plain = f => onType(
Array(f.length).fill(Number), Returns(Number, f))

View file

@ -1,4 +0,0 @@
import {Number} from "./Number.js"
export const plain = f =>
[Array(f.length).fill(Number), {returns: Number, behavior: f}]

View file

@ -1,4 +1,4 @@
import {plain} from './tools.js'
import {plain} from './helpers.js'
// Not much to do so far when there is only one type
export const number = plain(a => a)

View file

@ -1,3 +1,3 @@
import {plain} from './tools.js'
import {plain} from './helpers.js'
export const clone = plain(a => a)