firefly-compiler 0.5.35 → 0.5.37

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (225) hide show
  1. package/.hintrc +4 -4
  2. package/.vscode/settings.json +4 -4
  3. package/bin/Release.ff +157 -154
  4. package/bin/firefly.mjs +1 -1
  5. package/compiler/Builder.ff +275 -277
  6. package/compiler/Compiler.ff +234 -233
  7. package/compiler/Dependencies.ff +186 -187
  8. package/compiler/DependencyLock.ff +17 -17
  9. package/compiler/Deriver.ff +23 -31
  10. package/compiler/Dictionaries.ff +1 -1
  11. package/compiler/Inference.ff +43 -20
  12. package/compiler/JsEmitter.ff +1437 -1282
  13. package/compiler/LspHook.ff +202 -202
  14. package/compiler/Main.ff +25 -24
  15. package/compiler/ModuleCache.ff +178 -178
  16. package/compiler/Parser.ff +36 -109
  17. package/compiler/Resolver.ff +5 -8
  18. package/compiler/Substitution.ff +1 -1
  19. package/compiler/Syntax.ff +1 -16
  20. package/compiler/Token.ff +9 -0
  21. package/compiler/Tokenizer.ff +4 -0
  22. package/compiler/Workspace.ff +88 -88
  23. package/core/.firefly/include/package.json +5 -5
  24. package/core/.firefly/package.ff +2 -2
  25. package/core/Any.ff +26 -30
  26. package/core/Array.ff +298 -265
  27. package/core/Atomic.ff +63 -64
  28. package/core/Box.ff +7 -7
  29. package/core/BrowserSystem.ff +40 -40
  30. package/core/Buffer.ff +185 -152
  31. package/core/BuildSystem.ff +156 -148
  32. package/core/Channel.ff +95 -92
  33. package/core/Char.ff +3 -2
  34. package/core/Core.ff +16 -23
  35. package/core/Crypto.ff +94 -96
  36. package/core/Equal.ff +41 -36
  37. package/core/Error.ff +15 -10
  38. package/core/FileHandle.ff +45 -37
  39. package/core/Float.ff +176 -200
  40. package/core/HttpClient.ff +142 -148
  41. package/core/Instant.ff +6 -8
  42. package/core/Int.ff +40 -24
  43. package/core/IntMap.ff +61 -39
  44. package/core/Js.ff +305 -0
  45. package/core/JsSystem.ff +135 -114
  46. package/core/JsValue.ff +303 -159
  47. package/core/Json.ff +423 -443
  48. package/core/List.ff +482 -486
  49. package/core/Lock.ff +108 -144
  50. package/core/Log.ff +25 -14
  51. package/core/NodeSystem.ff +198 -191
  52. package/core/Ordering.ff +160 -161
  53. package/core/Path.ff +377 -409
  54. package/core/Queue.ff +90 -0
  55. package/core/Random.ff +140 -134
  56. package/core/RbMap.ff +216 -216
  57. package/core/Serializable.ff +16 -13
  58. package/core/Show.ff +44 -43
  59. package/core/SourceLocation.ff +68 -68
  60. package/core/Stream.ff +1 -1
  61. package/core/String.ff +224 -202
  62. package/core/StringMap.ff +58 -36
  63. package/core/Task.ff +165 -149
  64. package/experimental/benchmarks/ListGrab.ff +23 -23
  65. package/experimental/benchmarks/ListGrab.java +55 -55
  66. package/experimental/benchmarks/Pyrotek45.ff +30 -30
  67. package/experimental/benchmarks/Pyrotek45.java +64 -64
  68. package/experimental/bidirectional/Bidi.ff +88 -88
  69. package/experimental/lines/Main.ff +40 -0
  70. package/experimental/random/Index.ff +53 -53
  71. package/experimental/random/Process.ff +120 -120
  72. package/experimental/random/RunLength.ff +65 -65
  73. package/experimental/random/Scrape.ff +51 -51
  74. package/experimental/random/Symbols.ff +73 -73
  75. package/experimental/random/Tensor.ff +52 -52
  76. package/experimental/random/Units.ff +36 -36
  77. package/experimental/s3/S3TestAuthorizationHeader.ff +39 -39
  78. package/experimental/s3/S3TestPut.ff +16 -16
  79. package/experimental/tests/TestJson.ff +26 -26
  80. package/firefly.sh +0 -0
  81. package/fireflysite/.firefly/package.ff +4 -4
  82. package/fireflysite/CommunityOverview.ff +20 -20
  83. package/fireflysite/CountingButtonDemo.ff +58 -58
  84. package/fireflysite/DocumentParser.ff +325 -331
  85. package/fireflysite/ExamplesOverview.ff +40 -40
  86. package/fireflysite/FrontPage.ff +344 -344
  87. package/fireflysite/GettingStarted.ff +45 -45
  88. package/fireflysite/Guide.ff +456 -456
  89. package/fireflysite/Main.ff +163 -152
  90. package/fireflysite/MatchingPasswordsDemo.ff +82 -82
  91. package/fireflysite/PackagesOverview.ff +49 -49
  92. package/fireflysite/PostgresqlDemo.ff +34 -34
  93. package/fireflysite/ReferenceAll.ff +18 -18
  94. package/fireflysite/ReferenceIntroduction.ff +11 -11
  95. package/fireflysite/Styles.ff +567 -567
  96. package/fireflysite/Test.ff +121 -62
  97. package/fireflysite/assets/markdown/reference/BaseTypes.md +209 -209
  98. package/fireflysite/assets/markdown/reference/EmittedJavascript.md +65 -65
  99. package/fireflysite/assets/markdown/reference/Exceptions.md +101 -101
  100. package/fireflysite/assets/markdown/reference/FunctionsAndMethods.md +364 -364
  101. package/fireflysite/assets/markdown/reference/JavascriptInterop.md +235 -172
  102. package/fireflysite/assets/markdown/reference/ModulesAndPackages.md +162 -162
  103. package/fireflysite/assets/markdown/reference/OldStructuredConcurrency.md +48 -48
  104. package/fireflysite/assets/markdown/reference/PatternMatching.md +224 -224
  105. package/fireflysite/assets/markdown/reference/StatementsAndExpressions.md +86 -86
  106. package/fireflysite/assets/markdown/reference/StructuredConcurrency.md +99 -99
  107. package/fireflysite/assets/markdown/reference/TraitsAndInstances.md +100 -100
  108. package/fireflysite/assets/markdown/reference/UserDefinedTypes.md +184 -184
  109. package/fireflysite/assets/markdown/scratch/ControlFlow.md +136 -136
  110. package/fireflysite/assets/markdown/scratch/Toc.md +40 -40
  111. package/lsp/.firefly/package.ff +1 -1
  112. package/lsp/CompletionHandler.ff +827 -827
  113. package/lsp/Handler.ff +714 -714
  114. package/lsp/HoverHandler.ff +79 -79
  115. package/lsp/LanguageServer.ff +272 -272
  116. package/lsp/SignatureHelpHandler.ff +55 -55
  117. package/lsp/SymbolHandler.ff +181 -181
  118. package/lsp/TestReferences.ff +17 -17
  119. package/lsp/TestReferencesCase.ff +7 -7
  120. package/lsp/stderr.txt +1 -1
  121. package/lsp/stdout.txt +34 -34
  122. package/lux/.firefly/package.ff +1 -1
  123. package/lux/Css.ff +648 -648
  124. package/lux/CssTest.ff +48 -48
  125. package/lux/Lux.ff +608 -617
  126. package/lux/LuxEvent.ff +79 -116
  127. package/lux/Main.ff +123 -123
  128. package/lux/Main2.ff +143 -143
  129. package/lux/TestDry.ff +28 -28
  130. package/output/js/ff/compiler/Builder.mjs +72 -71
  131. package/output/js/ff/compiler/Compiler.mjs +19 -13
  132. package/output/js/ff/compiler/Dependencies.mjs +8 -7
  133. package/output/js/ff/compiler/DependencyLock.mjs +6 -4
  134. package/output/js/ff/compiler/Deriver.mjs +26 -24
  135. package/output/js/ff/compiler/Dictionaries.mjs +14 -18
  136. package/output/js/ff/compiler/Environment.mjs +6 -4
  137. package/output/js/ff/compiler/Inference.mjs +238 -164
  138. package/output/js/ff/compiler/JsEmitter.mjs +1160 -350
  139. package/output/js/ff/compiler/JsImporter.mjs +20 -18
  140. package/output/js/ff/compiler/LspHook.mjs +12 -10
  141. package/output/js/ff/compiler/Main.mjs +61 -41
  142. package/output/js/ff/compiler/ModuleCache.mjs +10 -8
  143. package/output/js/ff/compiler/Parser.mjs +153 -669
  144. package/output/js/ff/compiler/Patterns.mjs +12 -10
  145. package/output/js/ff/compiler/Resolver.mjs +52 -78
  146. package/output/js/ff/compiler/Substitution.mjs +12 -16
  147. package/output/js/ff/compiler/Syntax.mjs +50 -341
  148. package/output/js/ff/compiler/Token.mjs +126 -4
  149. package/output/js/ff/compiler/Tokenizer.mjs +62 -52
  150. package/output/js/ff/compiler/Unification.mjs +74 -90
  151. package/output/js/ff/compiler/Wildcards.mjs +4 -2
  152. package/output/js/ff/compiler/Workspace.mjs +26 -20
  153. package/output/js/ff/core/Any.mjs +20 -20
  154. package/output/js/ff/core/Array.mjs +268 -175
  155. package/output/js/ff/core/AssetSystem.mjs +8 -6
  156. package/output/js/ff/core/Atomic.mjs +84 -52
  157. package/output/js/ff/core/Bool.mjs +6 -4
  158. package/output/js/ff/core/BrowserSystem.mjs +38 -29
  159. package/output/js/ff/core/Buffer.mjs +285 -133
  160. package/output/js/ff/core/BuildSystem.mjs +36 -56
  161. package/output/js/ff/core/Channel.mjs +250 -97
  162. package/output/js/ff/core/Char.mjs +5 -3
  163. package/output/js/ff/core/Core.mjs +28 -34
  164. package/output/js/ff/core/Crypto.mjs +30 -52
  165. package/output/js/ff/core/Duration.mjs +4 -2
  166. package/output/js/ff/core/Equal.mjs +14 -12
  167. package/output/js/ff/core/Error.mjs +17 -11
  168. package/output/js/ff/core/FileHandle.mjs +76 -38
  169. package/output/js/ff/core/Float.mjs +92 -160
  170. package/output/js/ff/core/HttpClient.mjs +208 -76
  171. package/output/js/ff/core/Instant.mjs +8 -10
  172. package/output/js/ff/core/Int.mjs +36 -26
  173. package/output/js/ff/core/IntMap.mjs +79 -33
  174. package/output/js/ff/core/Js.mjs +751 -0
  175. package/output/js/ff/core/JsSystem.mjs +54 -60
  176. package/output/js/ff/core/JsValue.mjs +294 -143
  177. package/output/js/ff/core/Json.mjs +443 -253
  178. package/output/js/ff/core/List.mjs +262 -214
  179. package/output/js/ff/core/Lock.mjs +156 -125
  180. package/output/js/ff/core/Log.mjs +20 -10
  181. package/output/js/ff/core/Map.mjs +10 -8
  182. package/output/js/ff/core/NodeSystem.mjs +189 -123
  183. package/output/js/ff/core/Nothing.mjs +4 -2
  184. package/output/js/ff/core/Option.mjs +40 -38
  185. package/output/js/ff/core/Ordering.mjs +26 -20
  186. package/output/js/ff/core/Pair.mjs +4 -2
  187. package/output/js/ff/core/Path.mjs +517 -315
  188. package/output/js/ff/core/Queue.mjs +306 -0
  189. package/output/js/ff/core/Random.mjs +141 -77
  190. package/output/js/ff/core/RbMap.mjs +36 -34
  191. package/output/js/ff/core/Serializable.mjs +44 -28
  192. package/output/js/ff/core/Set.mjs +6 -4
  193. package/output/js/ff/core/Show.mjs +8 -6
  194. package/output/js/ff/core/SourceLocation.mjs +4 -2
  195. package/output/js/ff/core/Stream.mjs +30 -50
  196. package/output/js/ff/core/String.mjs +263 -172
  197. package/output/js/ff/core/StringMap.mjs +77 -31
  198. package/output/js/ff/core/Task.mjs +91 -76
  199. package/output/js/ff/core/Try.mjs +20 -18
  200. package/output/js/ff/core/Unit.mjs +4 -2
  201. package/package.json +1 -1
  202. package/postgresql/Pg.ff +53 -59
  203. package/rpc/.firefly/package.ff +1 -1
  204. package/rpc/Rpc.ff +70 -70
  205. package/s3/.firefly/package.ff +1 -1
  206. package/s3/S3.ff +92 -94
  207. package/vscode/LICENSE.txt +21 -21
  208. package/vscode/Prepublish.ff +15 -15
  209. package/vscode/README.md +16 -16
  210. package/vscode/client/package-lock.json +544 -544
  211. package/vscode/client/package.json +22 -22
  212. package/vscode/client/src/extension.ts +104 -104
  213. package/vscode/icons/firefly-icon.svg +10 -10
  214. package/vscode/language-configuration.json +61 -61
  215. package/vscode/package-lock.json +3623 -3623
  216. package/vscode/package.json +1 -1
  217. package/vscode/snippets.json +241 -241
  218. package/vscode/syntaxes/firefly-markdown-injection.json +45 -45
  219. package/webserver/.firefly/include/package.json +5 -5
  220. package/webserver/.firefly/package.ff +2 -2
  221. package/webserver/WebServer.ff +647 -685
  222. package/websocket/.firefly/package.ff +1 -1
  223. package/websocket/WebSocket.ff +100 -131
  224. package/core/UnsafeJs.ff +0 -42
  225. package/output/js/ff/core/UnsafeJs.mjs +0 -191
@@ -1,152 +1,163 @@
1
- import WebServer from ff:webserver
2
- import Lux from ff:lux
3
- import Css from ff:lux
4
- import Guide
5
- import GettingStarted
6
- import ReferenceAll
7
- import ExamplesOverview
8
- import PackagesOverview
9
- import CommunityOverview
10
- import FrontPage
11
- import Styles
12
-
13
- nodeMain(system: NodeSystem): Unit {
14
- let host = system.arguments().grab(0)
15
- let port = system.arguments().grab(1).grabInt()
16
- let cacheSalt = system.crypto().randomBuffer(16)
17
- system.writeLine("Listening on " + host + ":" + port)
18
- WebServer.new(system, host, port).listen {request =>
19
- let path = request.readPath()
20
- let segments = path.split('/').filter {s => s != "" && s != "." && s != ".."}
21
- segments.{
22
- | ["js", ...] =>
23
- let asset = if(path == "/js/ff/fireflysite/Main.mjs" && system.assets().exists("/js/Main.bundle.js")) {
24
- "/js/Main.bundle.js"
25
- } else {
26
- path
27
- }
28
- serveAsset(system, cacheSalt, request, "text/javascript; charset=UTF-8", asset)
29
- | ["favicon.ico"] =>
30
- serveAsset(system, cacheSalt, request, "image/png", "/assets/image/firefly-logo-notext.png")
31
- | ["assets", "font", "firefly-mono.woff2"] =>
32
- serveAsset(system, cacheSalt, request, "font/woff2", "/assets/font/gf-noto-sans-mono-latin.woff2")
33
- | ["assets", "font", "firefly-sans.woff2"] =>
34
- serveAsset(system, cacheSalt, request, "font/woff2", "/assets/font/gf-nunito-sans-latin.woff2")
35
- | ["assets", ...rest] => serveAssets(system, cacheSalt, request, rest)
36
- | _ {serveGuide(system, request)} =>
37
- | _ =>
38
- let parameters = if(request.readRawQueryString().size() == 0) {""} else {
39
- "?" + request.readRawQueryString()
40
- }
41
- request.writeHeader("Location", "https://www.firefly-lang.org" + path + parameters)
42
- request.writeStatus("302 Found")
43
- }
44
- }
45
- }
46
-
47
- guides = [
48
- Guide("/front/", [FrontPage.new()])
49
- Guide("/getting-started/", [GettingStarted.new()])
50
- Guide("/examples/", ExamplesOverview.mock())
51
- Guide("/reference/", ReferenceAll.mock())
52
- Guide("/packages/", [PackagesOverview.new()])
53
- Guide("/community/", [CommunityOverview.new()])
54
- ]
55
-
56
- serveGuide(system: NodeSystem, request: WebRequest[WebResponse]): Bool {
57
- let demos = ExamplesOverview.demos()
58
- mutable served = False
59
- guides.each {guide =>
60
- if(request.readPath().startsWith(guide.prefix)):
61
- let kebab = request.readPath().dropFirst(guide.prefix.size())
62
- let htmlAndCss = Lux.renderToString(system) {lux =>
63
- Guide.render(lux, system.httpClient(), guide.prefix, kebab, guides, demos)
64
- }
65
- let title = guide.documents.find {d =>
66
- Guide.kebabCase(d.heading()) == kebab
67
- }.map {_.title(guide)}.else {guide.title()}
68
- serveGuideHtml(title, htmlAndCss.first, htmlAndCss.second, request)
69
- served = True
70
- }
71
- served
72
- }
73
-
74
- browserMain(system: BrowserSystem): Unit {
75
- let demos = ExamplesOverview.demos()
76
- guides.collect {guide =>
77
- if(system.urlPath().startsWith(guide.prefix)):
78
- Lux.renderById(system, "main") {lux =>
79
- let kebab = system.urlPath().dropFirst(guide.prefix.size())
80
- Guide.render(lux, system.httpClient(), guide.prefix, kebab, guides, demos)
81
- }
82
- }
83
- }
84
-
85
- buildMain(system: BuildSystem) {
86
- let browser = system.compileForBrowser("Main.ff").bundle()
87
- let assets = AssetSystem.create()
88
- .addAssets("/js/", browser.assets())
89
- .addAssets("/assets/", system.packageAssets().assets("/assets/"))
90
- system.setAssets(assets)
91
- }
92
-
93
- serveAssets(system: NodeSystem, salt: Buffer, request: WebRequest[WebResponse], segments: List[String]): Unit {
94
- let asset = "/assets/" + segments.join("/")
95
- if(!system.assets().exists(asset)) {
96
- request.writeStatus("404 Not Found")
97
- } else:
98
-
99
- guessContentType(asset).or {
100
- request.writeStatus("404 Not Found")
101
- }: contentType =>
102
-
103
- serveAsset(system, salt, request, contentType, asset)
104
- }
105
-
106
- guessContentType(path: String): Option[String] {
107
- path.split('.').last().flatMap {
108
- | "md" => Some("text/markdown")
109
- | "png" => Some("image/png")
110
- | "webp" => Some("image/webp")
111
- | "ttf" => Some("font/ttf")
112
- | _ => None
113
- }
114
- }
115
-
116
- serveAsset(system: NodeSystem, salt: Buffer, request: WebRequest[WebResponse], contentType: String, asset: String): Unit {
117
- let etag = "\"" + system.crypto().hmacSha256(salt, asset.toBuffer()).toHex() + "\""
118
- request.writeHeader("Content-Type", contentType)
119
- if(request.readHeader("host") == Some("localhost:8080")) {
120
- request.writeHeader("Cache-Control", "no-cache")
121
- } else {
122
- request.writeHeader("Cache-Control", "public, max-age=60, must-revalidate")
123
- request.writeHeader("ETag", etag)
124
- }
125
- if(request.readHeader("if-none-match").any {_.trim() == etag}) {
126
- request.writeStatus("304 Not Modified")
127
- } else {
128
- request.writeStream(system.assets().readStream(asset))
129
- }
130
- }
131
-
132
- serveGuideHtml(title: String, contentHtml: String, styleTags: String, request: WebRequest[WebResponse]): Unit {
133
- request.writeHeader("Content-Type", "text/html; charset=UTF-8")
134
- request.writeText("<!doctype html>")
135
- request.writeText("<html lang='en' style='background-color: #ffffff; color: #333333; width: 100%; height: 100%; color-scheme: light;'>")
136
- request.writeText("<head>")
137
- request.writeText("<title>" + title + "</title>")
138
- request.writeText("<meta name='viewport' content='width=device-width, initial-scale=1.0'>")
139
- request.writeText("<meta name='theme-color' content='#ecc45e'>")
140
- request.writeText("<script type='module' src='/js/ff/fireflysite/Main.mjs'></script>")
141
- request.writeText("<link rel='preload' href='/assets/font/firefly-mono.woff2' as='font' crossorigin='anonymous'>")
142
- request.writeText("<link rel='preload' href='/assets/font/firefly-sans.woff2' as='font' crossorigin='anonymous'>")
143
- request.writeText("<link rel='preload' href='/assets/image/firefly-logo-yellow.webp' as='image'>")
144
- request.writeText("<style>@font-face { font-family: 'Firefly Mono'; font-display: fallback; src: url('/assets/font/firefly-mono.woff2'); }</style>")
145
- request.writeText("<style>@font-face { font-family: 'Firefly Sans'; font-display: fallback; src: url('/assets/font/firefly-sans.woff2'); }</style>")
146
- request.writeText(styleTags)
147
- request.writeText("</head>")
148
- request.writeText("<body style='margin: 0; padding: 0; width: 100%; height: 100%; touch-action: manipulation;'>")
149
- request.writeText("<div id='main'>" + contentHtml + "</div>")
150
- request.writeText("</body>")
151
- request.writeText("</html>")
152
- }
1
+ import WebServer from ff:webserver
2
+ import Lux from ff:lux
3
+ import Css from ff:lux
4
+ import Guide
5
+ import GettingStarted
6
+ import ReferenceAll
7
+ import ExamplesOverview
8
+ import PackagesOverview
9
+ import CommunityOverview
10
+ import FrontPage
11
+ import Styles
12
+
13
+ nodeMain(system: NodeSystem): Unit {
14
+ let host = system.arguments().grab(0)
15
+ let port = system.arguments().grab(1).grabInt()
16
+ let cacheSalt = system.crypto().randomBuffer(16)
17
+ system.writeLine("Listening on " + host + ":" + port)
18
+ WebServer.new(system, host, port).listen {request =>
19
+ let path = request.readPath()
20
+ let segments = path.split('/').filter {s => s != "" && s != "." && s != ".."}
21
+ segments.{
22
+ | _ {
23
+ !request.readHeader("host").contains("www.firefly-lang.org") &&
24
+ !request.readHeader("host").any {_.startsWith("localhost")}
25
+ } =>
26
+ let parameters = if(request.readRawQueryString().size() == 0) {""} else {
27
+ "?" + request.readRawQueryString()
28
+ }
29
+ request.writeHeader("Location", "https://www.firefly-lang.org" + path + parameters)
30
+ request.writeStatus("302 Found")
31
+ | ["js", ...] =>
32
+ let asset = if(path == "/js/ff/fireflysite/Main.mjs" && system.assets().exists("/js/Main.bundle.js")) {
33
+ "/js/Main.bundle.js"
34
+ } else {
35
+ path
36
+ }
37
+ serveAsset(system, cacheSalt, request, "text/javascript; charset=UTF-8", asset)
38
+ | ["favicon.ico"] =>
39
+ serveAsset(system, cacheSalt, request, "image/png", "/assets/image/firefly-logo-notext.png")
40
+ | ["assets", "font", "firefly-mono.woff2"] =>
41
+ serveAsset(system, cacheSalt, request, "font/woff2", "/assets/font/gf-noto-sans-mono-latin.woff2")
42
+ | ["assets", "font", "firefly-sans.woff2"] =>
43
+ serveAsset(system, cacheSalt, request, "font/woff2", "/assets/font/gf-nunito-sans-latin.woff2")
44
+ | ["assets", ...rest] => serveAssets(system, cacheSalt, request, rest)
45
+ | _ {serveGuide(system, request)} =>
46
+ | _ =>
47
+ request.writeText("404 Not Found")
48
+ request.writeStatus("404 Not Found")
49
+ }
50
+ }
51
+ }
52
+
53
+ guides = [
54
+ Guide("/front/", [FrontPage.new()])
55
+ Guide("/getting-started/", [GettingStarted.new()])
56
+ Guide("/examples/", ExamplesOverview.mock())
57
+ Guide("/reference/", ReferenceAll.mock())
58
+ Guide("/packages/", [PackagesOverview.new()])
59
+ Guide("/community/", [CommunityOverview.new()])
60
+ ]
61
+
62
+ serveGuide(system: NodeSystem, request: WebRequest[WebResponse]): Bool {
63
+ let demos = ExamplesOverview.demos()
64
+ mutable served = False
65
+ guides.each {guide =>
66
+ if(isGuidePath(request.readPath(), guide.prefix)):
67
+ let kebab = request.readPath().dropFirst(guide.prefix.size())
68
+ let htmlAndCss = Lux.renderToString(system) {lux =>
69
+ Guide.render(lux, system.httpClient(), guide.prefix, kebab, guides, demos)
70
+ }
71
+ let title = guide.documents.find {d =>
72
+ Guide.kebabCase(d.heading()) == kebab
73
+ }.map {_.title(guide)}.else {guide.title()}
74
+ serveGuideHtml(title, htmlAndCss.first, htmlAndCss.second, request)
75
+ served = True
76
+ }
77
+ served
78
+ }
79
+
80
+ browserMain(system: BrowserSystem): Unit {
81
+ let demos = ExamplesOverview.demos()
82
+ guides.collect {guide =>
83
+ if(isGuidePath(system.urlPath(), guide.prefix)):
84
+ Lux.renderById(system, "main") {lux =>
85
+ let kebab = system.urlPath().dropFirst(guide.prefix.size())
86
+ Guide.render(lux, system.httpClient(), guide.prefix, kebab, guides, demos)
87
+ }
88
+ }
89
+ }
90
+
91
+ isGuidePath(path: String, prefix: String): Bool {
92
+ //Log.debug(path + " vs. " + prefix)
93
+ path.startsWith(prefix) || (prefix == "/front/" && (path == "/" || path == "/index.html")) // Why is index.html requested
94
+ }
95
+
96
+ buildMain(system: BuildSystem) {
97
+ let browser = system.compileForBrowser("Main.ff") // .bundle()
98
+ let assets = AssetSystem.create()
99
+ .addAssets("/js/", browser.assets())
100
+ .addAssets("/assets/", system.packageAssets().assets("/assets/"))
101
+ system.setAssets(assets)
102
+ }
103
+
104
+ serveAssets(system: NodeSystem, salt: Buffer, request: WebRequest[WebResponse], segments: List[String]): Unit {
105
+ let asset = "/assets/" + segments.join("/")
106
+ if(!system.assets().exists(asset)) {
107
+ request.writeStatus("404 Not Found")
108
+ } else:
109
+
110
+ guessContentType(asset).or {
111
+ request.writeStatus("404 Not Found")
112
+ }: contentType =>
113
+
114
+ serveAsset(system, salt, request, contentType, asset)
115
+ }
116
+
117
+ guessContentType(path: String): Option[String] {
118
+ path.split('.').last().flatMap {
119
+ | "md" => Some("text/markdown")
120
+ | "png" => Some("image/png")
121
+ | "webp" => Some("image/webp")
122
+ | "ttf" => Some("font/ttf")
123
+ | _ => None
124
+ }
125
+ }
126
+
127
+ serveAsset(system: NodeSystem, salt: Buffer, request: WebRequest[WebResponse], contentType: String, asset: String): Unit {
128
+ let etag = "\"" + system.crypto().hmacSha256(salt, asset.toBuffer()).toHex() + "\""
129
+ request.writeHeader("Content-Type", contentType)
130
+ if(request.readHeader("host") == Some("localhost:8080")) {
131
+ request.writeHeader("Cache-Control", "no-cache")
132
+ } else {
133
+ request.writeHeader("Cache-Control", "public, max-age=60, must-revalidate")
134
+ request.writeHeader("ETag", etag)
135
+ }
136
+ if(request.readHeader("if-none-match").any {_.trim() == etag}) {
137
+ request.writeStatus("304 Not Modified")
138
+ } else {
139
+ request.writeStream(system.assets().readStream(asset))
140
+ }
141
+ }
142
+
143
+ serveGuideHtml(title: String, contentHtml: String, styleTags: String, request: WebRequest[WebResponse]): Unit {
144
+ request.writeHeader("Content-Type", "text/html; charset=UTF-8")
145
+ request.writeText("<!doctype html>")
146
+ request.writeText("<html lang='en' style='background-color: #ffffff; color: #333333; width: 100%; height: 100%; color-scheme: light;'>")
147
+ request.writeText("<head>")
148
+ request.writeText("<title>" + title + "</title>")
149
+ request.writeText("<meta name='viewport' content='width=device-width, initial-scale=1.0'>")
150
+ request.writeText("<meta name='theme-color' content='#ecc45e'>")
151
+ request.writeText("<script type='module' src='/js/ff/fireflysite/Main.mjs'></script>")
152
+ request.writeText("<link rel='preload' href='/assets/font/firefly-mono.woff2' as='font' crossorigin='anonymous'>")
153
+ request.writeText("<link rel='preload' href='/assets/font/firefly-sans.woff2' as='font' crossorigin='anonymous'>")
154
+ request.writeText("<link rel='preload' href='/assets/image/firefly-logo-yellow.webp' as='image'>")
155
+ request.writeText("<style>@font-face { font-family: 'Firefly Mono'; font-display: fallback; src: url('/assets/font/firefly-mono.woff2'); }</style>")
156
+ request.writeText("<style>@font-face { font-family: 'Firefly Sans'; font-display: fallback; src: url('/assets/font/firefly-sans.woff2'); }</style>")
157
+ request.writeText(styleTags)
158
+ request.writeText("</head>")
159
+ request.writeText("<body style='margin: 0; padding: 0; width: 100%; height: 100%; touch-action: manipulation;'>")
160
+ request.writeText("<div id='main'>" + contentHtml + "</div>")
161
+ request.writeText("</body>")
162
+ request.writeText("</html>")
163
+ }
@@ -1,82 +1,82 @@
1
- import Guide
2
- import Lux from ff:lux
3
- import LuxEvent from ff:lux
4
-
5
- name = "Matching passwords"
6
-
7
- new(): Demo {
8
- Demo(
9
- name
10
- {render(_)}
11
- )
12
- }
13
-
14
- render(lux: Lux): Unit {
15
- function passwordInput(password: String, setPassword: String => Unit) {
16
- lux.div {
17
- lux.input {
18
- lux.set("type", "password")
19
- lux.set("autocomplete", "new-password")
20
- lux.setValue(password)
21
- lux.onInput {event =>
22
- setPassword(event.text())
23
- }
24
- }
25
- }
26
- }
27
- lux.form {
28
- lux.useState(""): password1, setPassword1 =>
29
- lux.useState(""): password2, setPassword2 =>
30
- passwordInput(password1, setPassword1)
31
- passwordInput(password2, setPassword2)
32
- lux.div {
33
- if(password1 != password2) {
34
- lux.text("Passwords don't match!")
35
- } elseIf {password2.size() != 0} {
36
- lux.text("Passwords match!")
37
- }
38
- }
39
- }
40
- }
41
-
42
- newDocument(): Document {
43
- ReadyDocument([
44
- Section(name, [
45
- Paragraph([Text("Two passwords fields and a check that they match.")])
46
- ])
47
- Section("Demo", [
48
- LuxDemo(name)
49
- ])
50
- Section("Code", [
51
- CodeBlock("""
52
- render(lux: Lux): Unit {
53
- function passwordInput(password: String, setPassword: String => Unit) {
54
- lux.div {
55
- lux.input {
56
- lux.set("type", "password")
57
- lux.set("autocomplete", "new-password")
58
- lux.setValue(password)
59
- lux.onInput {event =>
60
- setPassword(event.text())
61
- }
62
- }
63
- }
64
- }
65
- lux.form {
66
- lux.useState(""): password1, setPassword1 =>
67
- lux.useState(""): password2, setPassword2 =>
68
- passwordInput(password1, setPassword1)
69
- passwordInput(password2, setPassword2)
70
- lux.div {
71
- if(password1 != password2) {
72
- lux.text("Passwords don't match!")
73
- } elseIf {password2.size() != 0} {
74
- lux.text("Passwords match!")
75
- }
76
- }
77
- }
78
- }
79
- """, firefly = True)
80
- ])
81
- ])
82
- }
1
+ import Guide
2
+ import Lux from ff:lux
3
+ import LuxEvent from ff:lux
4
+
5
+ name = "Matching passwords"
6
+
7
+ new(): Demo {
8
+ Demo(
9
+ name
10
+ {render(_)}
11
+ )
12
+ }
13
+
14
+ render(lux: Lux): Unit {
15
+ function passwordInput(password: String, setPassword: String => Unit) {
16
+ lux.div {
17
+ lux.input {
18
+ lux.set("type", "password")
19
+ lux.set("autocomplete", "new-password")
20
+ lux.setValue(password)
21
+ lux.onInput {event =>
22
+ setPassword(event.text())
23
+ }
24
+ }
25
+ }
26
+ }
27
+ lux.form {
28
+ lux.useState(""): password1, setPassword1 =>
29
+ lux.useState(""): password2, setPassword2 =>
30
+ passwordInput(password1, setPassword1)
31
+ passwordInput(password2, setPassword2)
32
+ lux.div {
33
+ if(password1 != password2) {
34
+ lux.text("Passwords don't match!")
35
+ } elseIf {password2.size() != 0} {
36
+ lux.text("Passwords match!")
37
+ }
38
+ }
39
+ }
40
+ }
41
+
42
+ newDocument(): Document {
43
+ ReadyDocument([
44
+ Section(name, [
45
+ Paragraph([Text("Two passwords fields and a check that they match.")])
46
+ ])
47
+ Section("Demo", [
48
+ LuxDemo(name)
49
+ ])
50
+ Section("Code", [
51
+ CodeBlock("""
52
+ render(lux: Lux): Unit {
53
+ function passwordInput(password: String, setPassword: String => Unit) {
54
+ lux.div {
55
+ lux.input {
56
+ lux.set("type", "password")
57
+ lux.set("autocomplete", "new-password")
58
+ lux.setValue(password)
59
+ lux.onInput {event =>
60
+ setPassword(event.text())
61
+ }
62
+ }
63
+ }
64
+ }
65
+ lux.form {
66
+ lux.useState(""): password1, setPassword1 =>
67
+ lux.useState(""): password2, setPassword2 =>
68
+ passwordInput(password1, setPassword1)
69
+ passwordInput(password2, setPassword2)
70
+ lux.div {
71
+ if(password1 != password2) {
72
+ lux.text("Passwords don't match!")
73
+ } elseIf {password2.size() != 0} {
74
+ lux.text("Passwords match!")
75
+ }
76
+ }
77
+ }
78
+ }
79
+ """, firefly = True)
80
+ ])
81
+ ])
82
+ }
@@ -1,49 +1,49 @@
1
- import Guide
2
- import CountingButtonDemo
3
- import MatchingPasswordsDemo
4
-
5
- new(): Document {
6
- ReadyDocument([
7
- Section("Packages", [
8
- Paragraph([
9
- Text("Here you'll eventually find the package index.")
10
- Text("For now, here's a short list of essential packages.")
11
- ])
12
- ])
13
- Section("ff:core", [
14
- Paragraph([
15
- Text("The standard library of Firefly.")
16
- ])
17
- ])
18
- Section("ff:webserver", [
19
- Paragraph([
20
- Text("A webserver package. Has HTTPS and WebSocket support.")
21
- ])
22
- ])
23
- Section("ff:websocket", [
24
- Paragraph([
25
- Text("A package for making websocket connections.")
26
- ])
27
- ])
28
- Section("ff:postgresql", [
29
- Paragraph([
30
- Text("A package for connecting to PostgreSQL. Has connection pool support.")
31
- ])
32
- ])
33
- Section("ff:lux", [
34
- Paragraph([
35
- Text("A package for building interactive web interfaces.")
36
- ])
37
- ])
38
- Section("ff:rpc", [
39
- Paragraph([
40
- Text("A package for type safe remote procedure calls.")
41
- ])
42
- ])
43
- Section("ff:unsafejs", [
44
- Paragraph([
45
- Text("A FFI package for JavaScript.")
46
- ])
47
- ])
48
- ])
49
- }
1
+ import Guide
2
+ import CountingButtonDemo
3
+ import MatchingPasswordsDemo
4
+
5
+ new(): Document {
6
+ ReadyDocument([
7
+ Section("Packages", [
8
+ Paragraph([
9
+ Text("Here you'll eventually find the package index.")
10
+ Text("For now, here's a short list of essential packages.")
11
+ ])
12
+ ])
13
+ Section("ff:core", [
14
+ Paragraph([
15
+ Text("The standard library of Firefly.")
16
+ ])
17
+ ])
18
+ Section("ff:webserver", [
19
+ Paragraph([
20
+ Text("A webserver package. Has HTTPS and WebSocket support.")
21
+ ])
22
+ ])
23
+ Section("ff:websocket", [
24
+ Paragraph([
25
+ Text("A package for making websocket connections.")
26
+ ])
27
+ ])
28
+ Section("ff:postgresql", [
29
+ Paragraph([
30
+ Text("A package for connecting to PostgreSQL. Has connection pool support.")
31
+ ])
32
+ ])
33
+ Section("ff:lux", [
34
+ Paragraph([
35
+ Text("A package for building interactive web interfaces.")
36
+ ])
37
+ ])
38
+ Section("ff:rpc", [
39
+ Paragraph([
40
+ Text("A package for type safe remote procedure calls.")
41
+ ])
42
+ ])
43
+ Section("ff:unsafejs", [
44
+ Paragraph([
45
+ Text("A FFI package for JavaScript.")
46
+ ])
47
+ ])
48
+ ])
49
+ }