From dc890600a2274aaec8da71f5e16be084a7f15614 Mon Sep 17 00:00:00 2001 From: quimt <126020181+quimt@users.noreply.github.com> Date: Wed, 16 Aug 2023 06:00:16 -0400 Subject: [PATCH] Happyx support (#212) * Update jsutils.nim create backend for happyx: API is analogous to karax backend but since happyx uses Jester-style routing syntax in the front-end DSL, we use happyxRoutes instead of happyxHtml * Update nimib.nim create block for embedding happyx code analagously to the one for embedding karax code * Update interactivity.nim Added example for happyX to parallel one for karax. Also gave a brief explanation of the problem with syntax like @click: ... and why @click() was used instead. * Update nimib.nimble **reminder**: this is planned to be moved in nimibex sometime in the future (along with nbKarax, nbPython, ...) --- docsrc/interactivity.nim | 51 +++++++++++++++++++++++++++++++++++++++- nimib.nimble | 2 +- src/nimib.nim | 6 +++++ src/nimib/jsutils.nim | 42 +++++++++++++++++++++++++++++++++ 4 files changed, 99 insertions(+), 2 deletions(-) diff --git a/docsrc/interactivity.nim b/docsrc/interactivity.nim index 07d7d309..77bc5265 100644 --- a/docsrc/interactivity.nim +++ b/docsrc/interactivity.nim @@ -253,4 +253,53 @@ And here the problem lies: `counter_2` is generated both times we compile the bl The solution is stated above: don't name multiple separate variables the same in a `nbKaraxCode` or `nbJsFromCodeOwnFile` block! This isn't a problem for the other nbJs blocks luckily. """ -nbSave \ No newline at end of file +nbText: hlMd""" +### nbHappyxCode +HappyX is an emerging alternative to Jester (on the back-end) and Karax(on the front end). It aims to streamline the syntax for writing full-stack applications and to allow more flexibility in its single page applications, which use a routing mechanism to switch between different pages for the app. It is being actively developed and some of the syntax for the DSL may change, so the introduction will be brief. + +The system for HappyX in nimib is analogous to the system for Karax. Note the parts of a typical Karax code block. + +```nim + template karaxExample = + let x = 3.14 + nbKaraxCode(x): + var message = "Pi is roughly " & $x + karaxHtml: + p: + text message + button: + text "Click me!" + proc onClick() = + message = "Poof! Gone!" +``` +Here's how it changes for HappyX: + +""" +nimibCode: + template happyxExample = + let x = 3.14 + nbHappyxCode(x): + var message = remember fmt"pi is roughly {x}" + happyxRoutes: + "/": + p: + {message} + tButton: + "Click me!" + @click( + message.set("Poof! Gone!")) + +nbText: "This is the output this code produces when called:" + +happyxExample() + +nbText: hlMd""" +There are many differences worth noticing, like use embedding of `fmt`'s `{}` to make a text node from data or the more prolific use of the prefix `t` before html tags (which is stylistic, as it's an optional disambiguator in happyX. The key thing to get you going, though, is that: + +`nbKaraxCode` becomes `nbHappyxCode` -- obviously + +`karaxHtml` becomes `happyxRoutes` -- this is due to differences in the DSLs. Karax uses a `buildHtml()` macro directly when creating VNodes and components. Happyx, on the other hand, enters into the front end DSL with an `appRoutes()` macro since the blocks beneath it like `"/":` define the different routes or 'subpages' of the app. So `happyxRoutes` imitates the `appRoutes` that it is meant to replace. + +There is **one other note for users of happyX**. The event handlers beginning in `@` must be called unambiguously. The more normal block declaration with `:` will not work in the current commit. HappyX does some massaging with its macros to make the syntax work in either case but plain Nim doesn't recognize `@click: ` as a call. The best strategy for resolving the inconsistency hasn't been decided yet, and a potential refactor as happyX continues development may resolve it spontaneously. +""" +nbSave diff --git a/nimib.nimble b/nimib.nimble index da1295f6..58a3e778 100644 --- a/nimib.nimble +++ b/nimib.nimble @@ -16,7 +16,7 @@ requires "parsetoml >= 0.7.0" requires "jsony >= 1.1.5" task docsdeps, "install dependendencies required for doc building": - exec "nimble -y install ggplotnim@0.5.9 numericalnim@0.8.8 nimoji nimpy karax@1.2.2" + exec "nimble -y install ggplotnim@0.5.9 numericalnim@0.8.8 nimoji nimpy karax@1.2.2 happyx@2.0.0" task test, "General tests": for file in ["tsources.nim", "tblocks.nim", "tnimib.nim", "trenders.nim"]: diff --git a/src/nimib.nim b/src/nimib.nim index 951e056e..0745658b 100644 --- a/src/nimib.nim +++ b/src/nimib.nim @@ -217,6 +217,12 @@ when moduleAvailable(karax/kbase): nbRawHtml: "
" nbKaraxCodeBackend(rootId, args) +when moduleAvailable(happyx): + template nbHappyxCode*(args: varargs[untyped]) = + let rootId = "happyx-" & $nb.newId() + nbRawHtml: "
" + nbHappyxCodeBackend(rootId, args) + template nbJsShowSource*(message: string = "") {.deprecated: "Use nbCodeDisplay instead".} = nb.blk.context["js_show_nim_source"] = true if message.len > 0: diff --git a/src/nimib/jsutils.nim b/src/nimib/jsutils.nim index 2a05ccb7..5e7d0321 100644 --- a/src/nimib/jsutils.nim +++ b/src/nimib/jsutils.nim @@ -137,6 +137,48 @@ macro nbKaraxCodeBackend*(rootId: untyped, args: varargs[untyped]) = result = call +macro nbHappyxCodeBackend*(rootId: untyped, args: varargs[untyped]) = + if args.len == 0: + error("nbHappyxCode needs a code block to be passed", args) + let body = args[^1] + let captureVars = + if args.len == 1: + @[] + else: + args[0 ..< ^1] + + let caller = body[^1][0] + if caller != ident"happyxRoutes": + error(fmt "{caller} != happyxRoutes: a happyxRoutes: block must terminate the nbHappyxCode block") + + let happyxRoutesBody = body[^1][1] + + let preHappyxRoutes = + if body.len == 1: + @[] + else: + body[0 ..< ^1] + + var preRoutesCode = newStmtList() + for i in preHappyxRoutes.items: + preRoutesCode.add(i) + + let imports = quote do: + import happyx, std/ strformat + + let newBody = quote do: + `imports` + `preRoutesCode` + appRoutes(`rootId`): + `happyxRoutesBody` + + var callArgs = @[rootId] + callArgs.add captureVars + callArgs.add newBody + + let call = newCall(ident"nbJsFromCodeOwnFile", callArgs) + result = call + proc compileNimToJs*(doc: var NbDoc, blk: var NbBlock) = let tempdir = getTempDir() / "nimib" createDir(tempdir)