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

Implement Typescript and ESM, and add Parcel for builds #27

Open
wants to merge 7 commits into
base: master
Choose a base branch
from

Conversation

ericfennis
Copy link

@ericfennis ericfennis commented Jan 4, 2024

Closes #15

What is changed

I've implemented a typescript for each package except yivi-console, yivi-dummy and yivi-css.
Typescript is implemented with the weakest setting strict: false in the tsconfig.json, meaning that there is almost no "type safety", but that can be improved later to keep this PR as minimal as possible.
Typescript with strict: false is almost the same as ESM.

To support different environments, like NodeJS and bundlers like Vite a bundler is needed to output script for each environment. I've added Parcel, which is a super simple bundler, that uses the package.json entry paths (module, main, browser, source, etc) to configure the bundle configuration.

In yivi-frontend I've replaced the package imports with relative paths since the bundler doesn't like "external" packages to be built in, It is possible but costs some effort to make that work. And since these packages are "DEV" only and bundled in using relative paths, they are bit easier. An alternative strategy is to make them "external", but then it is required to mark them as "dependencies" in package.json.

Future improvements

NPM workspaces

I would recommend switching this repo to NPM Workspaces, which is designed and made for frontend mono repo. For example, makes installing the package much easier, for example running npm install from the root.

Typescript typing

After this PR there are a lot of types of "any" which is not benefit from the power of typescript. Making things strictly typed will help prevent bugs in the future

Testing

I would recommend starting with writing some tests for these packages to make sure everything is working correctly in the future. You may not realizing that this library will grow in the coming years, since Yivi is getting some attention lately.

@ivard
Copy link
Member

ivard commented Jan 8, 2024

Currently, there are no developers on the Yivi project. Therefore, this code will not be reviewed anywhere soon. It is not clear yet how this will look like.

Copy link

@jappe999 jappe999 left a comment

Choose a reason for hiding this comment

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

Thanks for this PR! I can't seem to find anything faulty apart from a few small type improvements. In a future update we can replace the any types so that, like you wrote, bugs can be prevented.

Comment on lines +5 to +8
_stateMachine: any;
_options: any;
_session: any;
_onCancel: any;

Choose a reason for hiding this comment

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

Since these properties are already marked as private with an underscore, it might be nice to enforce it with the TypeScript private keyword.

@@ -28,7 +38,7 @@ module.exports = class DOMManipulations {
if (state.isFinal) {
this._detachEventHandlers();
// Make sure all restart buttons are hidden when being in a final state
this._element.querySelectorAll('.yivi-web-restart-button').forEach((e) => (e.style.display = 'none'));
this._element.querySelectorAll('.yivi-web-restart-button').forEach((e) => ((e as HTMLButtonElement).style.display = 'none'));

Choose a reason for hiding this comment

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

The querySelectorAll method is generic, so we can specify the type of the elements like so:

Suggested change
this._element.querySelectorAll('.yivi-web-restart-button').forEach((e) => ((e as HTMLButtonElement).style.display = 'none'));
this._element.querySelectorAll<HTMLButtonElement>('.yivi-web-restart-button').forEach((e) => (e.style.display = 'none'));

@@ -237,21 +248,21 @@ module.exports = class DOMManipulations {
}

_stateEnterPairingCode({ transition, payload }) {
const form = this._element.querySelector('.yivi-web-pairing-form');
const form = this._element.querySelector('.yivi-web-pairing-form') as HTMLFormElement;

Choose a reason for hiding this comment

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

The querySelector method is also generic, but in this case it might be more a matter of preference of how to write it.

_resolve: any;
_reject: any;

constructor(options: Record<string, any>) {

Choose a reason for hiding this comment

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

Oh cool! I never knew this type existed. 😅 I always use something like { [key: string]: any }, but the Record type seems way better :)

(This comment can be resolved. It's just something I found interesting.)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

ESM support for all Yivi packages
3 participants