Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Migrate PerseusWidgetsMap to use a merged interface to represent all available widgets #1936

Merged
merged 6 commits into from
Dec 4, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions .changeset/tender-planets-greet.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@khanacademy/perseus": minor
---

Changes the PerseusWidgetsMap to be extensible so that widgets can be registered outside of Perseus and still have full type safety.
2 changes: 2 additions & 0 deletions packages/perseus/src/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -223,7 +223,9 @@ export type {
PerseusRenderer,
PerseusWidget,
PerseusWidgetsMap,
PerseusWidgetTypes,
MultiItem,
WidgetOptions,
Comment on lines +226 to +228
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you know that these will be used in Webapp or are we preemptively exporting them? If not, I wonder if it would be better to keep them internal until we have a use for them externally.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

PerseusWidgetTypes needs to be exported so that it can be extended from anywhere outside of Perseus. @kevinb-khan has already had a use case for this while building a hackathon widget inside of webapp.

WidgetOptions is a very useful helper type for defining a widget's widget options structure.

Both are forward-looking, but it seemed reasonable to export them as part of this PR given the "widgets can be created outside of the @khanacademy/perseus" package nature.

} from "./perseus-types";
export type {UserInputMap} from "./validation.types";
export type {Coord} from "./interactive2/types";
Expand Down
162 changes: 95 additions & 67 deletions packages/perseus/src/perseus-types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -11,74 +11,102 @@ export type Size = [number, number];
export type CollinearTuple = [vec.Vector2, vec.Vector2];
export type ShowSolutions = "all" | "selected" | "none";

/**
* Our core set of Perseus widgets.
*
* This interface is the basis for "registering" all Perseus widget types.
* There should be one key/value pair for each supported widget. If you create
* a new widget, an entry should be added to this interface. Note that this
* only registers the widget options type, you'll also need to register the
* widget so that it's available at runtime (@see
* {@link file://./widgets.ts#registerWidget}).
*
* Importantly, the key should be the name that is used in widget IDs. For most
* widgets that is the same as the widget option's `type` field. In cases where
* a widget has been deprecated and replaced with the deprecated-standin
* widget, it should be the original widget type!
*
* If you define the widget outside of this package, you can still add the new
* widget to this interface by writing the following in that package that
* contains the widget. TypeScript will merge that definition of the
* `PerseusWidgets` with the one defined below.
*
* ```typescript
* declare module "@khanacademy/perseus" {
* interface PerseusWidgetTypes {
* // A new widget
* "new-awesomeness": MyAwesomeNewWidget;
*
* // A deprecated widget
* "super-old-widget": AutoCorrectWidget;
* }
* }
*
* // The new widget's options definition
* type MyAwesomeNewWidget = WidgetOptions<'new-awesomeness', MyAwesomeNewWidgetOptions>;
*
* // The deprecated widget's options definition
* type SuperOldWidget = WidgetOptions<'super-old-widget', object>;
* ```
*
* This interface can be extended through the magic of TypeScript "Declaration
* merging". Specifically, we augment this module and extend this interface.
*
* @see {@link https://www.typescriptlang.org/docs/handbook/declaration-merging.html#module-augmentation}
*/
export interface PerseusWidgetTypes {
Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Although this looks like the same hard-coded list of widgets as PerseusWidgetsMap is in the "before" state of this PR, the magic is that we can extend this interface from anywhere (even in our own Perseus unit tests if we want to work with a mock widget). Once we've added an entry to this interface, PerseusWidgetsMap automatically know's about that widget!

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this is a step in the right direction, I just wish we could get to a place where the Renderer-layer of Perseus is completely unaware of the Widget-layer data which would mean finding ways to eliminate these type of widget union/map things.

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agreed! I've spent a decent amount of time thinking about how we could make the lower level types opaque to the Renderers, I don't have a solution yet.

categorizer: CategorizerWidget;
"cs-program": CSProgramWidget;
definition: DefinitionWidget;
dropdown: DropdownWidget;
explanation: ExplanationWidget;
expression: ExpressionWidget;
grapher: GrapherWidget;
"graded-group-set": GradedGroupSetWidget;
"graded-group": GradedGroupWidget;
group: GroupWidget;
iframe: IFrameWidget;
image: ImageWidget;
"input-number": InputNumberWidget;
interaction: InteractionWidget;
"interactive-graph": InteractiveGraphWidget;
"label-image": LabelImageWidget;
matcher: MatcherWidget;
matrix: MatrixWidget;
measurer: MeasurerWidget;
"molecule-renderer": MoleculeRendererWidget;
"number-line": NumberLineWidget;
"numeric-input": NumericInputWidget;
orderer: OrdererWidget;
"passage-ref-target": RefTargetWidget;
"passage-ref": PassageRefWidget;
passage: PassageWidget;
"phet-simulation": PhetSimulationWidget;
"python-program": PythonProgramWidget;
plotter: PlotterWidget;
radio: RadioWidget;
sorter: SorterWidget;
table: TableWidget;
video: VideoWidget;

// Deprecated widgets
sequence: AutoCorrectWidget;
}

/**
* A map of widget IDs to widget options. This is most often used as the type
* for a set of widgets defined in a `PerseusItem` but can also be useful to
* represent a function parameter where only `widgets` from a `PerseusItem` are
* needed. Today Widget IDs are made up of the widget type and an incrementing
* integer (eg. `interactive-graph 1` or `radio 3`). It is suggested to avoid
* reading/parsing the widget id to derive any information from it, except in
* the case of this map.
*
* @see {@link PerseusWidgetTypes} additional widgets can be added to this map type
* by augmenting the PerseusWidgetTypes with new widget types!
*/
export type PerseusWidgetsMap = {
[key in `categorizer ${number}`]: CategorizerWidget;
} & {
[key in `cs-program ${number}`]: CSProgramWidget;
} & {
[key in `definition ${number}`]: DefinitionWidget;
} & {
[key in `dropdown ${number}`]: DropdownWidget;
} & {
[key in `explanation ${number}`]: ExplanationWidget;
} & {
[key in `expression ${number}`]: ExpressionWidget;
} & {
[key in `grapher ${number}`]: GrapherWidget;
} & {
[key in `group ${number}`]: GroupWidget;
} & {
[key in `graded-group ${number}`]: GradedGroupWidget;
} & {
[key in `graded-group-set ${number}`]: GradedGroupSetWidget;
} & {
[key in `iframe ${number}`]: IFrameWidget;
} & {
[key in `image ${number}`]: ImageWidget;
} & {
[key in `input-number ${number}`]: InputNumberWidget;
} & {
[key in `interaction ${number}`]: InteractionWidget;
} & {
[key in `interactive-graph ${number}`]: InteractiveGraphWidget;
} & {
[key in `label-image ${number}`]: LabelImageWidget;
} & {
[key in `matcher ${number}`]: MatcherWidget;
} & {
[key in `matrix ${number}`]: MatrixWidget;
} & {
[key in `measurer ${number}`]: MeasurerWidget;
} & {
[key in `molecule-renderer ${number}`]: MoleculeRendererWidget;
} & {
[key in `number-line ${number}`]: NumberLineWidget;
} & {
[key in `numeric-input ${number}`]: NumericInputWidget;
} & {
[key in `orderer ${number}`]: OrdererWidget;
} & {
[key in `passage ${number}`]: PassageWidget;
} & {
[key in `passage-ref ${number}`]: PassageRefWidget;
} & {
[key in `passage-ref-target ${number}`]: PassageRefWidget;
} & {
[key in `phet-simulation ${number}`]: PhetSimulationWidget;
} & {
[key in `plotter ${number}`]: PlotterWidget;
} & {
[key in `python-program ${number}`]: PythonProgramWidget;
} & {
[key in `radio ${number}`]: RadioWidget;
} & {
[key in `sorter ${number}`]: SorterWidget;
} & {
[key in `table ${number}`]: TableWidget;
} & {
[key in `video ${number}`]: VideoWidget;
} & {
[key in `sequence ${number}`]: AutoCorrectWidget;
[Property in keyof PerseusWidgetTypes as `${Property} ${number}`]: PerseusWidgetTypes[Property];
};

/**
Expand Down
Loading