sidebar.cljs 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543
  1. (ns frontend.components.sidebar
  2. (:require [cljs-drag-n-drop.core :as dnd]
  3. [clojure.string :as string]
  4. [frontend.components.command-palette :as command-palette]
  5. [frontend.components.header :as header]
  6. [frontend.components.journal :as journal]
  7. [frontend.components.repo :as repo]
  8. [frontend.components.right-sidebar :as right-sidebar]
  9. [frontend.components.settings :as settings]
  10. [frontend.components.theme :as theme]
  11. [frontend.components.widgets :as widgets]
  12. [frontend.components.plugins :as plugins]
  13. [frontend.components.select :as select]
  14. [frontend.config :as config]
  15. [frontend.context.i18n :refer [t]]
  16. [frontend.db :as db]
  17. [frontend.db.model :as db-model]
  18. [frontend.components.svg :as svg]
  19. [frontend.db-mixins :as db-mixins]
  20. [frontend.handler.editor :as editor-handler]
  21. [frontend.handler.route :as route-handler]
  22. [frontend.handler.page :as page-handler]
  23. [frontend.mixins :as mixins]
  24. [frontend.modules.shortcut.data-helper :as shortcut-dh]
  25. [frontend.state :as state]
  26. [frontend.ui :as ui]
  27. [frontend.util :as util]
  28. [reitit.frontend.easy :as rfe]
  29. [goog.dom :as gdom]
  30. [goog.object :as gobj]
  31. [rum.core :as rum]
  32. [frontend.extensions.srs :as srs]
  33. [frontend.extensions.pdf.assets :as pdf-assets]
  34. [frontend.mobile.util :as mobile-util]
  35. [frontend.handler.mobile.swipe :as swipe]))
  36. (rum/defc nav-content-item
  37. [name {:keys [class]} child]
  38. [:div.nav-content-item.is-expand
  39. {:class class}
  40. [:div.header.items-center.mb-1
  41. {:on-click (fn [^js/MouseEvent e]
  42. (let [^js target (.-target e)
  43. ^js parent (.closest target ".nav-content-item")]
  44. (.toggle (.-classList parent) "is-expand")))}
  45. [:div.font-medium.fade-link name]
  46. [:span
  47. [:a.more svg/arrow-down-v2]]]
  48. [:div.bd child]])
  49. (defn- delta-y
  50. [e]
  51. (when-let [target (.. e -target)]
  52. (let [rect (.. target getBoundingClientRect)]
  53. (- (.. e -pageY) (.. rect -top)))))
  54. (defn- move-up?
  55. [e]
  56. (let [delta (delta-y e)]
  57. (< delta 14)))
  58. (rum/defc page-name
  59. [name icon]
  60. (let [original-name (db-model/get-page-original-name name)]
  61. [:a {:on-click (fn [e]
  62. (let [name (util/safe-page-name-sanity-lc name)]
  63. (if (gobj/get e "shiftKey")
  64. (when-let [page-entity (db/entity [:block/name name])]
  65. (state/sidebar-add-block!
  66. (state/get-current-repo)
  67. (:db/id page-entity)
  68. :page
  69. {:page page-entity}))
  70. (route-handler/redirect-to-page! name))))}
  71. [:span.page-icon icon]
  72. (pdf-assets/fix-local-asset-filename original-name)]))
  73. (defn get-page-icon [page-entity]
  74. (let [default-icon "◦"
  75. from-properties (get-in (into {} page-entity) [:block/properties :icon])]
  76. (or
  77. (when (not= from-properties "") from-properties)
  78. default-icon))) ;; Fall back to default if icon is undefined or empty
  79. (rum/defcs favorite-item <
  80. (rum/local nil ::up?)
  81. (rum/local nil ::dragging-over)
  82. [state _t name icon]
  83. (let [up? (get state ::up?)
  84. dragging-over (get state ::dragging-over)
  85. target (state/sub :favorites/dragging)]
  86. [:li.favorite-item
  87. {:key name
  88. :data-ref name
  89. :class (if (and target @dragging-over (not= target @dragging-over))
  90. "dragging-target"
  91. "")
  92. :draggable true
  93. :on-drag-start (fn [_event]
  94. (state/set-state! :favorites/dragging name))
  95. :on-drag-over (fn [e]
  96. (util/stop e)
  97. (reset! dragging-over name)
  98. (when-not (= name (get @state/state :favorites/dragging))
  99. (reset! up? (move-up? e))))
  100. :on-drag-leave (fn [_e]
  101. (reset! dragging-over nil))
  102. :on-drop (fn [e]
  103. (page-handler/reorder-favorites! {:to name
  104. :up? (move-up? e)})
  105. (reset! up? nil)
  106. (reset! dragging-over nil))}
  107. (page-name name icon)]))
  108. (rum/defc favorites < rum/reactive
  109. [t]
  110. (nav-content-item
  111. [:a.flex.items-center.text-sm.font-medium.rounded-md
  112. (ui/icon "star mr-1" {:style {:font-size 18}})
  113. [:span.flex-1.ml-1 {:style {:padding-top 2}}
  114. (t :left-side-bar/nav-favorites)]]
  115. {:class "favorites"
  116. :edit-fn
  117. (fn [e]
  118. (rfe/push-state :page {:name "Favorites"})
  119. (util/stop e))}
  120. (let [favorites (->> (:favorites (state/sub-graph-config))
  121. (remove string/blank?)
  122. (filter string?))]
  123. (when (seq favorites)
  124. [:ul.favorites.text-sm
  125. (for [name favorites]
  126. (when-not (string/blank? name)
  127. (when-let [entity (db/entity [:block/name (util/safe-page-name-sanity-lc name)])]
  128. (let [icon (get-page-icon entity)]
  129. (favorite-item t name icon)))))]))))
  130. (rum/defc recent-pages < rum/reactive db-mixins/query
  131. [t]
  132. (nav-content-item
  133. [:a.flex.items-center.text-sm.font-medium.rounded-md
  134. (ui/icon "history mr-2" {:style {:font-size 18}})
  135. [:span.flex-1 {:style {:padding-top 2}}
  136. (t :left-side-bar/nav-recent-pages)]]
  137. {:class "recent"}
  138. (let [pages (->> (db/sub-key-value :recent/pages)
  139. (remove string/blank?)
  140. (filter string?)
  141. (map (fn [page] {:lowercase (util/safe-page-name-sanity-lc page)
  142. :page page}))
  143. (util/distinct-by :lowercase)
  144. (map :page))]
  145. [:ul.text-sm
  146. (for [name pages]
  147. (when-let [entity (db/entity [:block/name (util/safe-page-name-sanity-lc name)])]
  148. [:li.recent-item
  149. {:key name
  150. :data-ref name}
  151. (page-name name (get-page-icon entity))]))])))
  152. (rum/defcs flashcards < db-mixins/query rum/reactive
  153. {:did-mount (fn [state]
  154. (srs/update-cards-due-count!)
  155. state)}
  156. [state]
  157. (let [num (state/sub :srs/cards-due-count)]
  158. [:a.item.group.flex.items-center.px-2.py-2.text-sm.font-medium.rounded-md {:on-click #(state/pub-event! [:modal/show-cards])}
  159. (ui/icon "infinity mr-3" {:style {:font-size 20}})
  160. [:span.flex-1 (t :right-side-bar/flashcards)]
  161. (when (and num (not (zero? num)))
  162. [:span.ml-3.inline-block.py-0.5.px-3.text-xs.font-medium.rounded-full.fade-in num])]))
  163. (defn get-default-home-if-valid
  164. []
  165. (when-let [default-home (state/get-default-home)]
  166. (let [page (:page default-home)
  167. page (when (and (string? page)
  168. (not (string/blank? page)))
  169. (db/entity [:block/name (util/safe-page-name-sanity-lc page)]))]
  170. (if page
  171. default-home
  172. (dissoc default-home :page)))))
  173. (defn sidebar-item
  174. [{on-click-handler :on-click-handler
  175. class :class
  176. title :title
  177. icon :icon
  178. href :href}]
  179. [:div
  180. {:class class}
  181. [:a.item.group.flex.items-center.px-2.py-2.text-sm.font-medium.rounded-md
  182. {:on-click on-click-handler
  183. :href href}
  184. (ui/icon (str icon " mr-3") {:style {:font-size 20}})
  185. [:span.flex-1 title]]])
  186. (rum/defc sidebar-nav
  187. [_route-match close-modal-fn left-sidebar-open?]
  188. (let [default-home (get-default-home-if-valid)]
  189. [:div.left-sidebar-inner.flex-1.flex.flex-col.min-h-0
  190. {:on-click #(when-let [^js target (and (util/sm-breakpoint?) (.-target %))]
  191. (when (some (fn [sel] (boolean (.closest target sel)))
  192. [".favorites .bd" ".recent .bd" ".dropdown-wrapper" ".nav-header"])
  193. (close-modal-fn)))}
  194. [:div.flex.flex-col.pb-4.wrap
  195. [:nav.px-2.space-y-1 {:aria-label "Sidebar"}
  196. (repo/repos-dropdown)
  197. [:div.nav-header
  198. (if (:page default-home)
  199. (sidebar-item
  200. {:class "home-nav"
  201. :title (:page default-home)
  202. :on-click-handler route-handler/redirect-to-home!
  203. :icon "home"})
  204. (sidebar-item
  205. {:class "journals-nav"
  206. :title (t :right-side-bar/journals)
  207. :on-click-handler route-handler/go-to-journals!
  208. :icon "calendar"}))
  209. [:div.flashcards-nav
  210. (flashcards)]
  211. (sidebar-item
  212. {:class "graph-view-nav"
  213. :title (t :right-side-bar/graph-view)
  214. :href (rfe/href :graph)
  215. :icon "hierarchy"})
  216. (sidebar-item
  217. {:class "all-pages-nav"
  218. :title (t :right-side-bar/all-pages)
  219. :href (rfe/href :all-pages)
  220. :icon "files"})]]
  221. (favorites t)
  222. (when left-sidebar-open? (recent-pages t))
  223. [:nav.px-2 {:aria-label "Sidebar"
  224. :class "new-page"}
  225. (when-not config/publishing?
  226. [:a.item.group.flex.items-center.px-2.py-2.text-sm.font-medium.rounded-md
  227. {:on-click (fn []
  228. (and (util/sm-breakpoint?)
  229. (state/toggle-left-sidebar!))
  230. (state/pub-event! [:go/search]))}
  231. (ui/icon "circle-plus mr-3" {:style {:font-size 20}})
  232. [:span.flex-1 (t :right-side-bar/new-page)]])]]]))
  233. (rum/defc left-sidebar < rum/reactive
  234. [{:keys [left-sidebar-open? route-match]}]
  235. (let [close-fn #(state/set-left-sidebar-open! false)]
  236. [:div#left-sidebar.cp__sidebar-left-layout
  237. {:class (util/classnames [{:is-open left-sidebar-open?}])}
  238. ;; sidebar contents
  239. (sidebar-nav route-match close-fn left-sidebar-open?)
  240. [:span.shade-mask {:on-click close-fn}]]))
  241. (rum/defc main <
  242. {:did-mount (fn [state]
  243. (when-let [element (gdom/getElement "main-content-container")]
  244. (dnd/subscribe!
  245. element
  246. :upload-files
  247. {:drop (fn [_e files]
  248. (when-let [id (state/get-edit-input-id)]
  249. (let [format (:block/format (state/get-edit-block))]
  250. (editor-handler/upload-asset id files format editor-handler/*asset-uploading? true))))}))
  251. state)}
  252. [{:keys [route-match global-graph-pages? route-name indexeddb-support? db-restoring? main-content]}]
  253. (let [left-sidebar-open? (state/sub :ui/left-sidebar-open?)]
  254. [:div#main-container.cp__sidebar-main-layout.flex-1.flex
  255. {:class (util/classnames [{:is-left-sidebar-open left-sidebar-open?}])}
  256. ;; desktop left sidebar layout
  257. (left-sidebar {:left-sidebar-open? left-sidebar-open?
  258. :route-match route-match})
  259. [:div#main-content-container.scrollbar-spacing.w-full.flex.justify-center
  260. [:div.cp__sidebar-main-content
  261. {:data-is-global-graph-pages global-graph-pages?
  262. :data-is-full-width (or global-graph-pages?
  263. (contains? #{:all-files :all-pages :my-publishing} route-name))}
  264. (when-not (mobile-util/is-native-platform?)
  265. (widgets/demo-graph-alert))
  266. (widgets/github-integration-soon-deprecated-alert)
  267. (cond
  268. (not indexeddb-support?)
  269. nil
  270. db-restoring?
  271. [:div.mt-20
  272. [:div.ls-center
  273. (ui/loading (t :loading))]]
  274. :else
  275. [:div {:class (if global-graph-pages? "" (util/hiccup->class "max-w-7xl.mx-auto.pb-24"))
  276. :style {:margin-bottom (if global-graph-pages? 0 120)
  277. :padding-bottom (when (mobile-util/native-iphone?) "7rem")}}
  278. main-content])]]]))
  279. (rum/defc footer
  280. []
  281. (when-let [user-footer (and config/publishing? (get-in (state/get-config) [:publish-common-footer]))]
  282. [:div.p-6 user-footer]))
  283. (defonce sidebar-inited? (atom false))
  284. ;; TODO: simplify logic
  285. (rum/defc main-content < rum/reactive db-mixins/query
  286. {:init (fn [state]
  287. (when-not @sidebar-inited?
  288. (let [current-repo (state/sub :git/current-repo)
  289. default-home (get-default-home-if-valid)
  290. sidebar (:sidebar default-home)
  291. sidebar (if (string? sidebar) [sidebar] sidebar)]
  292. (when-let [pages (->> (seq sidebar)
  293. (remove string/blank?))]
  294. (doseq [page pages]
  295. (let [page (util/safe-page-name-sanity-lc page)
  296. [db-id block-type] (if (= page "contents")
  297. ["contents" :contents]
  298. [page :page])]
  299. (state/sidebar-add-block! current-repo db-id block-type nil)))
  300. (reset! sidebar-inited? true))))
  301. state)}
  302. []
  303. (let [cloning? (state/sub :repo/cloning?)
  304. default-home (get-default-home-if-valid)
  305. parsing? (state/sub :repo/parsing-files?)
  306. current-repo (state/sub :git/current-repo)
  307. loading-files? (when current-repo (state/sub [:repo/loading-files? current-repo]))
  308. me (state/sub :me)
  309. journals-length (state/sub :journals-length)
  310. latest-journals (db/get-latest-journals (state/get-current-repo) journals-length)
  311. preferred-format (state/sub [:me :preferred_format])
  312. logged? (:name me)]
  313. [:div
  314. (cond
  315. (and default-home
  316. (= :home (state/get-current-route))
  317. (not (state/route-has-p?))
  318. (:page default-home))
  319. (route-handler/redirect-to-page! (:page default-home))
  320. (and config/publishing?
  321. (not default-home)
  322. (empty? latest-journals))
  323. (route-handler/redirect! {:to :all-pages})
  324. parsing?
  325. (ui/loading (t :parsing-files))
  326. loading-files?
  327. (ui/loading (t :loading-files))
  328. (and (not logged?) latest-journals)
  329. (journal/journals latest-journals)
  330. (and logged? (not preferred-format))
  331. (widgets/choose-preferred-format)
  332. ;; TODO: delay this
  333. (and logged? (nil? (:email me)))
  334. (settings/set-email)
  335. cloning?
  336. (ui/loading (t :cloning))
  337. (seq latest-journals)
  338. (journal/journals latest-journals)
  339. (or
  340. (and (mobile-util/is-native-platform?)
  341. (nil? (state/get-current-repo)))
  342. (and logged? (empty? (:repos me))))
  343. (widgets/add-graph)
  344. ;; FIXME: why will this happen?
  345. :else
  346. [:div])]))
  347. (rum/defc custom-context-menu < rum/reactive
  348. []
  349. (when (state/sub :custom-context-menu/show?)
  350. (when-let [links (state/sub :custom-context-menu/links)]
  351. (ui/css-transition
  352. {:class-names "fade"
  353. :timeout {:enter 500
  354. :exit 300}}
  355. links))))
  356. (rum/defc new-block-mode < rum/reactive
  357. []
  358. (when (state/sub [:document/mode?])
  359. (ui/tippy {:html [:div.p-2
  360. [:p.mb-2 [:b "Document mode"]]
  361. [:ul
  362. [:li
  363. [:div.inline-block.mr-1 (ui/render-keyboard-shortcut (shortcut-dh/gen-shortcut-seq :editor/new-line))]
  364. [:p.inline-block "to create new block"]]
  365. [:li
  366. [:p.inline-block.mr-1 "Click `D` or type"]
  367. [:div.inline-block.mr-1 (ui/render-keyboard-shortcut (shortcut-dh/gen-shortcut-seq :ui/toggle-document-mode))]
  368. [:p.inline-block "to toggle document mode"]]]]}
  369. [:a.block.px-1.text-sm.font-medium.bg-base-2.rounded-md.mx-2
  370. {:on-click state/toggle-document-mode!}
  371. "D"])))
  372. (rum/defc help-button < rum/reactive
  373. []
  374. (when-not (state/sub :ui/sidebar-open?)
  375. [:div.cp__sidebar-help-btn
  376. {:title (t :help-shortcut-title)
  377. :on-click (fn []
  378. (state/sidebar-add-block! (state/get-current-repo) "help" :help nil))}
  379. "?"]))
  380. (defn- hide-context-menu-and-clear-selection
  381. [e]
  382. (state/hide-custom-context-menu!)
  383. (when-not (gobj/get e "shiftKey")
  384. (editor-handler/clear-selection!)))
  385. (rum/defcs ^:large-vars/cleanup-todo sidebar <
  386. (mixins/modal :modal/show?)
  387. rum/reactive
  388. (mixins/event-mixin
  389. (fn [state]
  390. (mixins/listen state js/window "click" hide-context-menu-and-clear-selection)
  391. (mixins/listen state js/window "keydown"
  392. (fn [e]
  393. (when (= 27 (.-keyCode e))
  394. (if (and (state/modal-opened?)
  395. (not
  396. (and
  397. ;; FIXME: this does not work on CI tests
  398. util/node-test?
  399. (:editor/editing? @state/state))))
  400. (state/close-modal!)
  401. (hide-context-menu-and-clear-selection e)))))))
  402. {:did-mount (fn [state]
  403. (swipe/setup-listeners!)
  404. state)}
  405. [state route-match main-content]
  406. (let [{:keys [open-fn]} state
  407. me (state/sub :me)
  408. current-repo (state/sub :git/current-repo)
  409. granted? (state/sub [:nfs/user-granted? (state/get-current-repo)])
  410. theme (state/sub :ui/theme)
  411. system-theme? (state/sub :ui/system-theme?)
  412. white? (= "white" (state/sub :ui/theme))
  413. sidebar-open? (state/sub :ui/sidebar-open?)
  414. settings-open? (state/sub :ui/settings-open?)
  415. left-sidebar-open? (state/sub :ui/left-sidebar-open?)
  416. right-sidebar-blocks (state/sub-right-sidebar-blocks)
  417. route-name (get-in route-match [:data :name])
  418. global-graph-pages? (= :graph route-name)
  419. logged? (:name me)
  420. db-restoring? (state/sub :db/restoring?)
  421. indexeddb-support? (state/sub :indexeddb/support?)
  422. page? (= :page route-name)
  423. home? (= :home route-name)
  424. edit? (:editor/editing? @state/state)
  425. default-home (get-default-home-if-valid)]
  426. (theme/container
  427. {:t t
  428. :theme theme
  429. :route route-match
  430. :current-repo current-repo
  431. :edit? edit?
  432. :nfs-granted? granted?
  433. :db-restoring? db-restoring?
  434. :sidebar-open? sidebar-open?
  435. :settings-open? settings-open?
  436. :sidebar-blocks-len (count right-sidebar-blocks)
  437. :system-theme? system-theme?
  438. :on-click (fn [e]
  439. (editor-handler/unhighlight-blocks!)
  440. (util/fix-open-external-with-shift! e))}
  441. [:div.theme-inner
  442. {:class (util/classnames [{:ls-left-sidebar-open left-sidebar-open?
  443. :ls-right-sidebar-open sidebar-open?}])}
  444. [:div.#app-container
  445. [:div#left-container
  446. {:class (if (state/sub :ui/sidebar-open?) "overflow-hidden" "w-full")}
  447. (header/header {:open-fn open-fn
  448. :white? white?
  449. :current-repo current-repo
  450. :logged? logged?
  451. :page? page?
  452. :route-match route-match
  453. :me me
  454. :default-home default-home
  455. :new-block-mode new-block-mode})
  456. (main {:route-match route-match
  457. :global-graph-pages? global-graph-pages?
  458. :logged? logged?
  459. :home? home?
  460. :route-name route-name
  461. :indexeddb-support? indexeddb-support?
  462. :white? white?
  463. :db-restoring? db-restoring?
  464. :main-content main-content})
  465. (footer)]
  466. (right-sidebar/sidebar)
  467. [:div#app-single-container]]
  468. (ui/notification)
  469. (ui/modal)
  470. (ui/sub-modal)
  471. (command-palette/command-palette-modal)
  472. (select/select-modal)
  473. (custom-context-menu)
  474. (plugins/custom-js-installer {:t t
  475. :current-repo current-repo
  476. :nfs-granted? granted?
  477. :db-restoring? db-restoring?})
  478. [:a#download.hidden]
  479. (when
  480. (and (not config/mobile?)
  481. (not config/publishing?))
  482. (help-button))])))