-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathcompact.cirru
406 lines (405 loc) · 19.2 KB
/
compact.cirru
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
{} (:package |app)
:configs $ {} (:init-fn |app.main/main!) (:reload-fn |app.main/reload!) (:version nil)
:modules $ [] |respo.calcit/compact.cirru |lilac/compact.cirru |memof/compact.cirru |respo-ui.calcit/compact.cirru |respo-markdown.calcit/compact.cirru |reel.calcit/compact.cirru |respo-feather.calcit/
:entries $ {}
:files $ {}
|app.comp.container $ %{} :FileEntry
:defs $ {}
|card-width $ %{} :CodeEntry (:doc |)
:code $ quote
def card-width $ js/Math.min 480 (- js/window.innerWidth 24)
|comp-avatar $ %{} :CodeEntry (:doc |)
:code $ quote
defcomp comp-avatar (on-home)
[] (effect-bump)
div
{} (:class-name css-avatar)
:on-click $ fn (e d!) (on-home nil :home d!)
div $ {} (:class-name css-avatar-outline)
|comp-card $ %{} :CodeEntry (:doc |)
:code $ quote
defcomp comp-card (idx key on-open on-close)
[] (effect-fading)
let
info $ get site-map key
div
{} (:class-name css-card)
:style $ {}
:left $ * (+ card-width 12) idx
:height 640
:top -320
:width card-width
div
{} $ :style
merge ui/row-parted $ {} (:position :absolute) (:top 0) (:left 0) (:width "\"100%") (:padding "\"4px") (:backdrop-filter "\"blur(2px)")
span
{} (:class-name "\"show-on-hover")
:style $ {} (:line-height "\"20px") (:height "\"20px")
:title "\"Command + Click to close all"
comp-icon :x
{}
:color $ hsl 0 0 0
:font-size 20
:cursor :pointer
fn (e d!)
let
all? $ or (-> e :event .-metaKey) (-> e :event .-ctrlKey)
on-close idx all? d!
<> (:title info)
{} (:font-family ui/font-fancy) (:font-size 14) (:font-weight 300)
:color $ hsl 0 0 60
span $ {} (:class-name "\"right-corner") (:inner-text "\" ")
=< nil 8
if (nil? info)
<> $ str "\"Unknown data: " key
div
{} $ :style
merge ui/expand $ {} (:padding "\"0 12px") (:overflow :auto)
=< nil 48
, &
-> info (get :content)
map $ fn (directive)
if (tuple? directive)
render-content directive $ fn (key d!) (on-open idx key d!)
<> $ str "\"Unknown " directive
=< nil 120
|comp-cards $ %{} :CodeEntry (:doc |)
:code $ quote
defcomp comp-cards (router on-open on-close)
list->
{} $ :style
merge ui/row $ {} (:margin "\"auto 0")
:width $ * (+ card-width 4) (count router)
:position :relative
:transition-duration "\"300ms"
:transition-property :width
:pointer-events :auto
-> router $ map-indexed
fn (idx key)
[] key $ comp-card idx key on-open on-close
|comp-container $ %{} :CodeEntry (:doc |)
:code $ quote
defcomp comp-container (reel)
let
store $ :store reel
states $ :states store
router $ :router store
push-tab $ fn (idx x d!)
d! $ :: :push-page idx x
close-tab $ fn (idx all? d!)
d! $ :: :close-page all? idx
div ({})
div
{} $ :class-name css-bg
create-element :iframe $ {}
:src $ if (some? js/navigator.gpu) "\"https://webgpu.art/soluble/?tab=rhombic-mirror" "\"https://r.tiye.me/Triadica/sapium/"
:class-name css-iframe
div
{} $ :class-name style-container
comp-empty (empty? router) push-tab
comp-cards router push-tab close-tab
when dev? $ comp-reel (>> states :reel) reel ({})
|comp-empty $ %{} :CodeEntry (:doc |)
:code $ quote
defcomp comp-empty (visible? on-home)
div
{} (:class-name css-profile)
:style $ if visible?
{} (:opacity 1) (:transform "\"translate(0,0px)")
div
{} $ :style
merge ui/center $ {} (:pointer-events :auto)
comp-avatar on-home
=< nil 32
div
{} $ :style
merge ui/center $ {} (:color :white) (:font-size 20) (:line-height "\"28px") (:text-shadow "\"1px 1px 4px black")
<> "\"题叶"
|css-avatar $ %{} :CodeEntry (:doc |)
:code $ quote
defstyle css-avatar $ {}
"\"$0" $ {} (:background-image "\"url(https://cdn.tiye.me/logo/tiye.jpg)") (:background-size "\"120px 120px") (:width 120) (:height 120) (:border-radius "\"50%")
:box-shadow $ str "\"1px 1px 4px " (hsl 120 80 50) "\",1px 1px 4px black"
:transition-duration "\"300ms"
:transition-timing-function "\"cubic-bezier(0.54, 0.17, 0.53, 1.88)"
:cursor :pointer
:position :relative
:background-position :center
"\"$0:hover" $ {}
:box-shadow $ str "\"1px 1px 6px " (hsl 120 90 80)
:transform "\"scale(1.04)"
"\"$0:active" $ {}
:box-shadow $ str "\"1px 1px 20px " (hsl 120 90 90)
:transform "\"scale(1.08)"
:transition-duration "\"100ms"
:animation-name "\"breathing"
:animation-duration "\"4s"
:animation-iteration-count :infinite
(str "\"$0:hover ." css-avatar-outline)
{} $ :border-color (hsl 0 0 100 0.8)
(str "\"$0:active ." css-avatar-outline)
{} $ :border-width "\"5px"
|css-avatar-outline $ %{} :CodeEntry (:doc |)
:code $ quote
defstyle css-avatar-outline $ {}
"\"$0" $ {} (:position :absolute) (:width "\"100%") (:height "\"100%") (:border-radius "\"50%")
:border $ str-spaced "\"5px solid" (hsl 0 0 100 0.9)
|css-bg $ %{} :CodeEntry (:doc |)
:code $ quote
defstyle css-bg $ {}
"\"$0" $ merge ui/global ui/fullscreen
{} (:background-image "\"url(https://r.tiye.me/tiye/logo/leaf.jpg)") (:background-size "\"cover") (:background-position :center) (:position :fixed) (:top 0) (:left 0) (:width "\"100%") (:height "\"100%") (:z-index -1) (:opacity 0.7) (:filter "\"grayscale(0.5)")
|css-card $ %{} :CodeEntry (:doc |)
:code $ quote
defstyle css-card $ {}
"\"$0" $ merge ui/column
{} (:padding "\"4px") (:box-shadow "\"1px 1px 4px black") (:border-radius "\"2px") (:border "\"2px solid white") (:margin-right 4) (:transition-duration "\"300ms") (:position :absolute) (:border-radius "\"6px") (:transform-origin "\"-10% 33.3%")
:box-shadow $ str "\"0 0 10px 4px " (hsl 200 60 90 0.3)
:background-color $ hsl 0 0 100 0.96
|css-iframe $ %{} :CodeEntry (:doc |)
:code $ quote
defstyle css-iframe $ {}
"\"$0" $ {} (:width "\"100%") (:height "\"100%")
|css-profile $ %{} :CodeEntry (:doc |)
:code $ quote
defstyle css-profile $ {}
"\"$0" $ merge ui/center
{} (:margin :auto) (:position :fixed) (:top 0) (:left 0) (:width "\"100%") (:height "\"100%") (:opacity 0.01) (:transition-duration "\"300ms") (:transform "\"translate(12vw,0px)") (:-webkit-tap-highlight-color :transparent) (:transition-delay "\"120ms") (:animation-name "\"entering") (:animation-duration "\"800ms")
|effect-bump $ %{} :CodeEntry (:doc |)
:code $ quote
defeffect effect-bump () (action el at?)
if (= action :mount)
do
-> el .-style .-scale $ set! 0.6
js/setTimeout $ fn ()
-> el .-style .-scale $ set! 1
, 0
|effect-fading $ %{} :CodeEntry (:doc |)
:code $ quote
defeffect effect-fading () (action el at?)
case-default action nil
:mount $ do
-> el .-style .-opacity $ set! 0.1
-> el .-style .-transform $ set! "\"scale(0.6) translate(-80px,0px)"
flipped js/setTimeout 0 $ fn ()
-> el .-style .-opacity $ set! 1
-> el .-style .-transform $ set! "\"scale(1) translate(0,0)"
flipped js/setTimeout 240 $ fn ()
-> el (.!querySelector "\".right-corner") (wo-js-log)
.!scrollIntoView $ js-object (:block "\"end") (:behavoir "\"smooth")
:unmount $ let
el' $ .!cloneNode el
-> el .-parentElement $ .!appendChild el'
js/setTimeout
fn ()
-> el' .-style .-opacity $ set! 0.01
-> el' .-style .-transform $ set! "\"scale(0.8) translate(-24px,0)"
-> el' .-style .-width $ set! 0
, 10
js/setTimeout
fn () $ .!remove el'
, 300
|render-content $ %{} :CodeEntry (:doc |)
:code $ quote
defn render-content (directive on-open)
tag-match directive
:title t
div
{} $ :style
{} (:font-weight 700) (:font-size 32) (:margin-top 8) (:font-weight 300)
<> t
(:text t)
div
{} $ :style
{} (:line-height "\"26px") (:margin-bottom 12)
<> t
(:links links)
div
{} $ :style
{} $ :margin "\"8px 0px"
, & $ -> links
map $ fn (xs)
if (tuple? xs) (render-content xs on-open)
<> $ str "\"Unknown " xs
(:route target title)
div
{} (:class-name "\"hover-scale")
:style $ {} (:display :inline-block) (:min-width 40) (:border "\"1px solid #ddf") (:padding "\"0 8px") (:margin-right 8) (:margin-bottom 8) (:cursor :pointer)
:on-click $ fn (e d!) (on-open target d!)
<> title
(:url target title)
div ({})
<> "\"🌐" $ {}
:color $ hsl 200 80 70
:font-size 14
=< 6 nil
a $ {}
:style $ {} (:font-size 16) (:line-height "\"20px")
:href target
:inner-text title
:target "\"_blank"
(:html h)
div $ {}
:style $ {} (:position :relative)
:innerHTML h
(:xigua url)
div $ {}
:style $ {} (:position :relative)
:innerHTML $ str "\"<iframe width=\"100%\" height=\"260px\" frameborder=\"0\" src=\"" url "\"\" referrerpolicy=\"unsafe-url\" allowfullscreen></iframe>"
(:image src alt)
img $ {} (:src src) (:alt alt) (:class-name style-embed-image)
_ $ div ({})
<> $ str "\"Unknown kind: " (nth 0 directive)
|style-container $ %{} :CodeEntry (:doc |)
:code $ quote
defstyle style-container $ {}
"\"$0" $ merge ui/global ui/fullscreen
{} (:display :flex) (:font-size 20) (; :backdrop-filter "\"blur(2px)") (:padding "\"0 12px 0 12vw") (:scroll-behavior :smooth)
:box-shadow $ str "\"inset 0 -40px 1200px " (hsl 0 0 0)
; :background-color $ hsl 180 60 20 0.01
:font-family $ str "\"Buda," (:font-family ui/global)
:pointer-events :none
|style-embed-image $ %{} :CodeEntry (:doc |)
:code $ quote
defstyle style-embed-image $ {}
"\"&" $ {} (:max-width "\"100%")
:ns $ %{} :CodeEntry (:doc |)
:code $ quote
ns app.comp.container $ :require
respo.util.format :refer $ hsl
respo-ui.core :as ui
respo.core :refer $ defcomp >> <> div button textarea span a list-> defeffect create-element img
respo.comp.space :refer $ =<
reel.comp.reel :refer $ comp-reel
respo-md.comp.md :refer $ comp-md-block comp-md
app.config :refer $ dev?
feather.core :refer $ comp-icon
respo.css :refer $ defstyle
app.schema :refer $ site-map
|app.config $ %{} :FileEntry
:defs $ {}
|dev? $ %{} :CodeEntry (:doc |)
:code $ quote (def dev? true)
|site $ %{} :CodeEntry (:doc |)
:code $ quote
def site $ {} (:storage "\"tiye-site") (:dev-ui "\"http://localhost:8100/main.css") (:release-ui "\"http://cdn.tiye.me/favored-fonts/main.css") (:cdn-url "\"http://cdn.tiye.me/tiye-site/") (:cdn-folder "\"tiye.me:cdn/tiye-site") (:title "\"题叶@jiyinyiyong") (:icon "\"http://cdn.tiye.me/logo/tiye.jpg") (:storage-key "\"tiye-site") (:upload-folder "\"tiye.me:repo/tiye/tiye.me/")
:ns $ %{} :CodeEntry (:doc |)
:code $ quote (ns app.config)
|app.main $ %{} :FileEntry
:defs $ {}
|*reel $ %{} :CodeEntry (:doc |)
:code $ quote
defatom *reel $ -> reel-schema/reel (assoc :base schema/store) (assoc :store schema/store)
|dispatch! $ %{} :CodeEntry (:doc |)
:code $ quote
defn dispatch! (op)
when config/dev? $ println "\"Dispatch:" op
reset! *reel $ reel-updater updater @*reel op
|main! $ %{} :CodeEntry (:doc |)
:code $ quote
defn main! ()
println "\"Running mode:" $ if config/dev? "\"dev" "\"release"
if config/dev? $ load-console-formatter!
render-app!
add-watch *reel :changes $ fn (r p) (render-app!)
listen-devtools! |k dispatch!
js/window.addEventListener "\"keydown" $ fn (event)
if
= "\"Escape" $ .-key event
dispatch! $ :: :reduce-page
println "|App started."
|mount-target $ %{} :CodeEntry (:doc |)
:code $ quote
def mount-target $ js/document.querySelector |.app
|reload! $ %{} :CodeEntry (:doc |)
:code $ quote
defn reload! () $ if (some? build-errors) (tip! "\"error" build-errors)
do (clear-cache!) (remove-watch *reel :changes)
add-watch *reel :changes $ fn (r p) (render-app!)
reset! *reel $ refresh-reel @*reel schema/store updater
println "|Code updated."
tip! "\"ok~" "\"Ok"
|render-app! $ %{} :CodeEntry (:doc |)
:code $ quote
defn render-app! () $ render! mount-target (comp-container @*reel) dispatch!
:ns $ %{} :CodeEntry (:doc |)
:code $ quote
ns app.main $ :require
respo.core :refer $ render! clear-cache! realize-ssr!
app.comp.container :refer $ comp-container
app.updater :refer $ updater
app.schema :as schema
reel.util :refer $ listen-devtools!
reel.core :refer $ reel-updater refresh-reel
reel.schema :as reel-schema
app.config :as config
app.page :refer $ ssr-processing!
"\"bottom-tip" :default tip!
"\"./calcit.build-errors" :default build-errors
|app.page $ %{} :FileEntry
:defs $ {}
|base-info $ %{} :CodeEntry (:doc |)
:code $ quote
def base-info $ {}
:title $ :title config/site
:icon $ :icon config/site
:ssr nil
:inline-html nil
|ssr-processing! $ %{} :CodeEntry (:doc |)
:code $ quote
defn ssr-processing! () $ let
reel $ -> reel-schema/reel (assoc :base schema/store) (assoc :store schema/store)
file "\"dist/index.html"
html-content $ make-string (comp-container reel)
content $ fs/readFileSync file "\"utf8"
fs/writeFileSync file $ replace content "\"<div class=\"app\" ></div>" (str "\"<div class=\"app\" data-ssr=\"true\" >" html-content "\"</div>")
echo "\"replaced" file
:ns $ %{} :CodeEntry (:doc |)
:code $ quote
ns app.page $ :require
respo.render.html :refer $ make-string
app.comp.container :refer $ comp-container
app.schema :as schema
reel.schema :as reel-schema
app.config :as config
"\"fs" :as fs
|app.schema $ %{} :FileEntry
:defs $ {}
|load-as-code $ %{} :CodeEntry (:doc |)
:code $ quote
defmacro load-as-code (file)
&data-to-code $ parse-cirru-edn (read-file file)
|site-map $ %{} :CodeEntry (:doc |)
:code $ quote
def site-map $ load-as-code "\"data/meta.cirru"
|store $ %{} :CodeEntry (:doc |)
:code $ quote
def store $ {}
:states $ {}
:router $ []
:ns $ %{} :CodeEntry (:doc |)
:code $ quote (ns app.schema)
|app.updater $ %{} :FileEntry
:defs $ {}
|updater $ %{} :CodeEntry (:doc |)
:code $ quote
defn updater (store op op-id op-time)
tag-match op
:states cursor s
update-states store cursor s
(:push-page idx x)
update store :router $ fn (rs)
if (nil? idx) (conj rs x) (.assoc-after rs idx x)
(:close-page all? idx)
if all?
update store :router $ fn (rs) (.slice rs 0 idx)
update store :router $ fn (rs) (dissoc rs idx)
(:reduce-page)
update store :router $ fn (r)
if (empty? r) r $ rest r
(:hydrate-storage d) d
_ $ do (eprintln "\"unknown op" op) store
:ns $ %{} :CodeEntry (:doc |)
:code $ quote
ns app.updater $ :require
respo.cursor :refer $ update-states