- Uses a modified version of hxgenjs that incorporates hxtsdgen.
- The introduction of hxgenjs provides more flexibility but adds a lot of complexity;
- Some edge-cases don't work well - i.e hxtsdgen doesn't seem to play well with the module approach from hxgenjs when using third-party libs with complex macros, such as tink.
- Current status is: It could be used as an example of TS<>Haxe interoperability or a base for a better boilerplate, but since hxdtsgen doesn't work well in all cases. It's still a viable alternative for more natural Haxe<>TS interoperability if TS type info from Haxe is not required or agumented or written manually (see the @:tsextern tag to selectively enable/disable generation the d.ts for single classes at a time).
This is an ongoing experiment. At the time of this writing, the actual app is a sample/exercise websocket app, so not much to see there in terms of features. The actual app doesn't really use Haxe, the experiment with Haxe came later, read on.
The actual client is served by Express (dev/prod). On dev it proxies the webpack dev server, on prod it serves the static files. A self-contained self-servable SPA like this makes it easier to deploy it in PaaS solutions like Heroku. It's also easy enough to decouple the serving of the client from the server, if needed, in the future.
The idea is to apply the paretto principle to Haxe and use what works and is easiest 80% of the time and Haxe 20% of the time for the use-cases where it is really good for or when you feel like it. The focus for this version of the experiment is to:
- Find a way to - as much as possible - transparently integrate Haxe in a TS/Babel project;
- Make the Haxe class - as much as possible - 1st class TS citizens.
For #1, we create a hx
folder under src
that mirrors the structure of the ts
source folder. hxgenjs
will then generate the files in the
correct places, providing the package name for the Haxe class matches the destination folder, and you can then require it from ts.
For #2 we use the TS definition files that are generated by hxgenjs
. Check the contents of build.hxml
to understand how it's done.
Because TS is nice and Haxe is awesome. It's much easier to start with ES/TS since most of the docs and examples out there are based off it. Also, most of the client-side frameworks and libraries are available for TS with type-definitions. It's much easier to build your app infrastructure with ES/TS and only then researching where Haxe could help. If it can, then the goal is to make it as smooth as possible to switch between and use both from each other, as needed.
Tested with Haxe 4.0RC2.
- Install Lix https://github.com/lix-pm/lix.client#downloading-all-dependencies;
- Install Haxe 4.0RC with Lix, if necessary:
lix install haxe 4.0.0-rc.2
; - Download haxelib dependencies with Lix:
lix download
; - haxe build.hxml;
- yarn;
- yarn dev.
Then:
- Search for the output in the shell:
[SERVER] src/hx/server/SuperServerComponent.hx:7: Hello from Haxe running on nodejs from a js file compiled by tsc!
This it the Haxe class being called from a TS class on the server. - Access
localhost:3000
in a browser, open the console and search for a similar message. This is a Haxe calss being called from TS on the client. - Play with the app if you want, although the rest doesn't have to do with Haxe at this point :)
- The definition files generated by
hxgenjs
seem to generate very poor type data. Everything seems to beAny
'fied. - Since there's no Haxe source for an entry-point (the entry-point being a regualar js file that already exists), the
main.js
generated byhxgenjs
is not needed, but is still generated (in blank) and I did not find a way to prevent it from being generated yet. - Haxe compilation is still manual and not part of the dev / build pipeline yet, but this is easy to solve (will do in the next iteration).
hxgenjs
automatically generates the TS "externs" (d.ts) for each Haxe class, but if there's a need to use TS from Haxe, Haxe externs still need to be written manually, it'd be nice to have a process part of the build pipeline that would go through each .ts file and generated an extern so that the communication would be 100% bidierectional and transparent.- I did not find a way to tell the Haxe compiler to include all the packages available, so I had to specify one-by-one in the
build.hxml
with theinclude
macro.