views.cljs 104 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164
  1. (ns frontend.components.views
  2. "Different views of blocks"
  3. (:require [cljs-bean.core :as bean]
  4. [cljs-time.coerce :as tc]
  5. [cljs-time.core :as t]
  6. [cljs-time.format :as tf]
  7. [clojure.set :as set]
  8. [clojure.string :as string]
  9. [datascript.impl.entity :as de]
  10. [dommy.core :as dom]
  11. [frontend.common.missionary :as c.m]
  12. [frontend.components.dnd :as dnd]
  13. [frontend.components.icon :as icon-component]
  14. [frontend.components.property.config :as property-config]
  15. [frontend.components.property.value :as pv]
  16. [frontend.components.select :as select]
  17. [frontend.components.selection :as selection]
  18. [frontend.config :as config]
  19. [frontend.context.i18n :refer [t]]
  20. [frontend.date :as date]
  21. [frontend.db :as db]
  22. [frontend.db-mixins :as db-mixins]
  23. [frontend.db.async :as db-async]
  24. [frontend.db.react :as react]
  25. [frontend.handler.db-based.export :as db-export-handler]
  26. [frontend.handler.db-based.property :as db-property-handler]
  27. [frontend.handler.editor :as editor-handler]
  28. [frontend.handler.property :as property-handler]
  29. [frontend.handler.property.util :as pu]
  30. [frontend.handler.route :as route-handler]
  31. [frontend.handler.ui :as ui-handler]
  32. [frontend.mixins :as mixins]
  33. [frontend.modules.outliner.op :as outliner-op]
  34. [frontend.modules.outliner.ui :as ui-outliner-tx]
  35. [frontend.state :as state]
  36. [frontend.ui :as ui]
  37. [frontend.util :as util]
  38. [goog.dom :as gdom]
  39. [logseq.common.config :as common-config]
  40. [logseq.db :as ldb]
  41. [logseq.db.common.view :as db-view]
  42. [logseq.db.frontend.property :as db-property]
  43. [logseq.shui.hooks :as hooks]
  44. [logseq.shui.ui :as shui]
  45. [medley.core :as medley]
  46. [missionary.core :as m]
  47. [promesa.core :as p]
  48. [rum.core :as rum]))
  49. (defn- get-scroll-parent
  50. [config]
  51. (if (:sidebar? config)
  52. (dom/sel1 ".sidebar-item-list")
  53. (gdom/getElement "main-content-container")))
  54. (rum/defc header-checkbox < rum/static
  55. [{:keys [selected-all? selected-some? toggle-selected-all!] :as table}]
  56. (let [[show? set-show!] (rum/use-state false)]
  57. [:label.h-8.w-8.flex.items-center.justify-center.cursor-pointer
  58. {:html-for "header-checkbox"
  59. :on-mouse-over #(set-show! true)
  60. :on-mouse-out #(set-show! false)}
  61. (shui/checkbox
  62. {:id "header-checkbox"
  63. :checked (or selected-all? (and selected-some? "indeterminate"))
  64. :on-checked-change (fn [value]
  65. (p/do
  66. (when value
  67. (db-async/<get-blocks (state/get-current-repo) (:rows table) {}))
  68. (toggle-selected-all! table value)))
  69. :aria-label "Select all"
  70. :class (str "flex transition-opacity "
  71. (if (or show? selected-all? selected-some?) "opacity-100" "opacity-0"))})]))
  72. (rum/defc header-index < rum/static
  73. []
  74. [:label.h-8.w-6.flex.items-center.justify-center
  75. {:html-for "header-index"
  76. :title "Row number"}
  77. "ID"])
  78. (rum/defc row-checkbox < rum/static
  79. [{:keys [row-selected? row-toggle-selected! data state data-fns]} row _column]
  80. (let [idx (.indexOf data (:db/id row))
  81. id (str (:db/id row) "-" "checkbox")
  82. [show? set-show!] (rum/use-state false)
  83. checked? (row-selected? row)
  84. {:keys [last-selected-idx row-selection]} state
  85. {:keys [set-last-selected-idx! set-row-selection!]} data-fns]
  86. [:label.h-8.w-8.flex.items-center.justify-center.cursor-pointer
  87. {:html-for (str (:db/id row) "-" "checkbox")
  88. :on-mouse-over #(set-show! true)
  89. :on-mouse-out #(set-show! false)}
  90. (shui/checkbox
  91. {:id id
  92. :checked checked?
  93. :on-click (fn [e]
  94. (when (and (.-shiftKey e) last-selected-idx)
  95. ;; add selection
  96. (util/stop e)
  97. (when (not= last-selected-idx idx)
  98. (let [new-ids (keep (fn [idx] (util/nth-safe data idx)) (range (min last-selected-idx idx) (inc (max last-selected-idx idx))))]
  99. (when (seq new-ids)
  100. (let [row-selection' (update row-selection :selected-ids set/union (set new-ids))]
  101. (set-row-selection! row-selection')))))))
  102. :on-checked-change (fn [v]
  103. (p/do!
  104. (when v (db-async/<get-block (state/get-current-repo) (:db/id row) {:skip-refresh? true
  105. :children? false}))
  106. (if v
  107. (set-last-selected-idx! idx)
  108. (when (= (:db/id row) last-selected-idx)
  109. (set-last-selected-idx! nil)))
  110. (row-toggle-selected! row-selection row v)))
  111. :aria-label "Select row"
  112. :class (str "flex transition-opacity "
  113. (if (or show? checked?) "opacity-100" "opacity-0"))})]))
  114. (defonce *last-header-action-target (atom nil))
  115. (defn header-cp
  116. [{:keys [view-entity column-set-sorting! state]} column]
  117. (let [sorting (:sorting state)
  118. db-based? (config/db-based-graph?)
  119. [asc?] (some (fn [item] (when (= (:id item) (:id column))
  120. (when-some [asc? (:asc? item)]
  121. [asc?]))) sorting)
  122. property (db/entity (:id column))
  123. pinned? (when property
  124. (contains? (set (map :db/id (:logseq.property.table/pinned-columns view-entity)))
  125. (:db/id property)))
  126. sub-content (fn [{:keys [id]}]
  127. (let [table-options [(shui/dropdown-menu-item
  128. {:key "asc"
  129. :on-click #(column-set-sorting! sorting column true)}
  130. [:div.flex.flex-row.items-center.gap-1
  131. (ui/icon "arrow-up" {:size 15})
  132. [:div "Sort ascending"]])
  133. (shui/dropdown-menu-item
  134. {:key "desc"
  135. :on-click #(column-set-sorting! sorting column false)}
  136. [:div.flex.flex-row.items-center.gap-1
  137. (ui/icon "arrow-down" {:size 15})
  138. [:div "Sort descending"]])
  139. (when (and db-based? property)
  140. (shui/dropdown-menu-item
  141. {:on-click (fn [_e]
  142. (if pinned?
  143. (db-property-handler/delete-property-value! (:db/id view-entity)
  144. :logseq.property.table/pinned-columns
  145. (:db/id property))
  146. (property-handler/set-block-property! (state/get-current-repo)
  147. (:db/id view-entity)
  148. :logseq.property.table/pinned-columns
  149. (:db/id property)))
  150. (shui/popup-hide! id))}
  151. [:div.flex.flex-row.items-center.gap-1
  152. (ui/icon "pin" {:size 15})
  153. [:div (if pinned? "Unpin" "Pin")]]))]
  154. tag (when-let [entity (:logseq.property/view-for view-entity)]
  155. (when (ldb/class? entity)
  156. entity))
  157. option (cond->
  158. {:with-title? false
  159. :more-options table-options}
  160. (some? tag)
  161. (assoc :class-schema? true))]
  162. [:div.ls-property-dropdown
  163. (property-config/property-dropdown property tag option)]))]
  164. (shui/button
  165. {:variant "text"
  166. :class "h-8 !pl-4 !px-2 !py-0 hover:text-foreground w-full justify-start"
  167. :on-click (fn [^js e]
  168. (let [popup-id (str "table-column-" (:id column))]
  169. (when-let [^js el (some-> (.-target e) (.closest "[aria-roledescription=sortable]"))]
  170. (when (and (or (nil? @*last-header-action-target)
  171. (not= el @*last-header-action-target))
  172. (string/blank? (some-> el (.-style) (.-transform))))
  173. (shui/popup-show! el sub-content
  174. {:id popup-id
  175. :align "start"
  176. :as-dropdown? true
  177. :on-before-hide (fn []
  178. (reset! *last-header-action-target el)
  179. (js/setTimeout #(reset! *last-header-action-target nil) 128))})))))}
  180. (let [title (str (:name column))]
  181. [:span {:title title
  182. :class "max-w-full overflow-hidden text-ellipsis"}
  183. title])
  184. (case asc?
  185. true
  186. (ui/icon "arrow-up")
  187. false
  188. (ui/icon "arrow-down")
  189. nil))))
  190. (defn- timestamp-cell-cp
  191. [_table row column]
  192. (some-> (get row (:id column))
  193. date/int->local-time-2))
  194. (defn- get-property-value-content
  195. [entity]
  196. (db-view/get-property-value-content (db/get-db) entity))
  197. (rum/defc block-container
  198. [config row]
  199. (let [container (state/get-component :block/container)
  200. config' (cond-> config
  201. (not (:popup? config))
  202. (assoc :view? true))]
  203. [:div.relative.w-full {:style {:min-height 24}}
  204. (if row
  205. (container config' row)
  206. [:div])]))
  207. (defn- save-block-and-focus
  208. [*ref set-focus-timeout! hide-popup?]
  209. (let [node (rum/deref *ref)
  210. cell (util/rec-get-node node "ls-table-cell")]
  211. (p/do!
  212. (editor-handler/save-current-block!)
  213. (when hide-popup?
  214. (shui/popup-hide!))
  215. (state/exit-editing-and-set-selected-blocks! [cell])
  216. (set-focus-timeout! (js/setTimeout #(.focus cell) 100)))))
  217. (rum/defc block-title
  218. "Used on table view"
  219. [block* {:keys [create-new-block width row property]}]
  220. (let [*ref (hooks/use-ref nil)
  221. [opacity set-opacity!] (hooks/use-state 0)
  222. [focus-timeout set-focus-timeout!] (hooks/use-state nil)
  223. inline-title (state/get-component :block/inline-title)
  224. many? (db-property/many? property)
  225. block (if many? (first block*) block*)
  226. add-to-sidebar! #(state/sidebar-add-block! (state/get-current-repo)
  227. (or (and many? (:db/id row)) (:db/id block))
  228. :block)
  229. redirect! #(some-> (:block/uuid block) route-handler/redirect-to-page!)]
  230. (hooks/use-effect!
  231. (fn []
  232. #(some-> focus-timeout js/clearTimeout))
  233. [])
  234. [:div.table-block-title.relative.flex.items-center.w-full.h-full.cursor-pointer.items-center
  235. {:ref *ref
  236. :on-mouse-over #(set-opacity! 100)
  237. :on-mouse-out #(set-opacity! 0)
  238. :on-click (fn [e]
  239. (p/let [block (or block (and (fn? create-new-block) (create-new-block)))]
  240. (when block
  241. (cond
  242. (util/meta-key? e)
  243. (redirect!)
  244. (.-shiftKey e)
  245. (add-to-sidebar!)
  246. :else
  247. (let [popup (fn []
  248. (let [width (-> (max 160 width) (- 18))]
  249. (if many?
  250. [:div.ls-table-block
  251. {:style {:width width :max-width width}
  252. :on-click util/stop-propagation}
  253. (pv/property-value row property {})]
  254. [:div.ls-table-block
  255. {:style {:width width :max-width width}
  256. :on-click util/stop-propagation}
  257. (block-container
  258. {:popup? true
  259. :view? true
  260. :table-block-title? true
  261. :on-key-down
  262. (fn [e]
  263. (when (= (util/ekey e) "Enter")
  264. (util/stop e)
  265. (save-block-and-focus *ref set-focus-timeout! true)))}
  266. block)])))]
  267. (p/do!
  268. (shui/popup-show!
  269. (.closest (.-target e) ".ls-table-cell")
  270. popup
  271. {:id :ls-table-block-editor
  272. :as-mask? true
  273. :on-after-hide (fn []
  274. (save-block-and-focus *ref set-focus-timeout! false))})
  275. (editor-handler/edit-block! block :max {:container-id :unknown-container})))))))}
  276. (if block
  277. [:div.flex.flex-row
  278. (let [render (fn [block]
  279. [:div
  280. (inline-title
  281. (some->> (:block/title block)
  282. string/trim
  283. string/split-lines
  284. first))])]
  285. (if many?
  286. (->> (map render block*)
  287. (interpose [:div.mr-1 ","]))
  288. (render block*)))]
  289. [:div])
  290. (let [class (str "h-6 w-6 !p-1 text-muted-foreground transition-opacity duration-100 ease-in bg-gray-01 "
  291. "opacity-" opacity)]
  292. [:div.absolute.-right-1
  293. [:div.flex.flex-row.items-center
  294. (shui/button
  295. {:variant :ghost
  296. :title "Open"
  297. :on-click (fn [e]
  298. (util/stop-propagation e)
  299. (redirect!))
  300. :class class}
  301. (ui/icon "arrow-right"))
  302. (shui/button
  303. {:variant :ghost
  304. :title "Open in sidebar"
  305. :class class
  306. :on-click (fn [e]
  307. (util/stop-propagation e)
  308. (add-to-sidebar!))}
  309. (ui/icon "layout-sidebar-right"))]])]))
  310. (defn build-columns
  311. [config properties & {:keys [with-object-name? with-id? add-tags-column?]
  312. :or {with-object-name? true
  313. with-id? true
  314. add-tags-column? true}}]
  315. (let [;; FIXME: Shouldn't file graphs have :block/tags?
  316. add-tags-column?' (and (config/db-based-graph? (state/get-current-repo)) add-tags-column?)
  317. properties' (->>
  318. (if (or (some #(= (:db/ident %) :block/tags) properties) (not add-tags-column?'))
  319. properties
  320. (conj properties (db/entity :block/tags)))
  321. (remove nil?))]
  322. (->> (concat
  323. [{:id :select
  324. :name "Select"
  325. :header (fn [table _column] (header-checkbox table))
  326. :cell (fn [table row column]
  327. (row-checkbox table row column))
  328. :column-list? false
  329. :resizable? false}
  330. (when with-id?
  331. {:id :id
  332. :name "ID"
  333. :header (fn [_table _column] (header-index))
  334. :cell (fn [table row _column]
  335. (inc (.indexOf (:rows table) (:db/id row))))
  336. :resizable? false})
  337. (when with-object-name?
  338. {:id :block/title
  339. :name "Name"
  340. :type :string
  341. :header header-cp
  342. :cell (fn [_table row _column style]
  343. (block-title row {:property-ident :block/title
  344. :sidebar? (:sidebar? config)
  345. :width (:width style)}))
  346. :disable-hide? true})]
  347. (keep
  348. (fn [property]
  349. (when-let [ident (or (:db/ident property) (:id property))]
  350. ;; Hide properties that shouldn't ever be editable or that do not display well in a table
  351. (when-not (or (contains? #{:logseq.property/built-in? :logseq.property.asset/checksum :logseq.property.class/properties
  352. :block/created-at :block/updated-at :block/order :block/collapsed?
  353. :logseq.property/created-from-property}
  354. ident)
  355. (and with-object-name? (= :block/title ident))
  356. (contains? #{:map :entity} (:logseq.property/type property)))
  357. (let [property (if (de/entity? property)
  358. property
  359. (or (merge (db/entity ident) property) property)) ; otherwise, :cell/:header/etc. will be removed
  360. get-value (when (de/entity? property)
  361. (fn [row] (db-view/get-property-value-for-search row property)))]
  362. {:id ident
  363. :name (or (:name property)
  364. (:block/title property))
  365. :header (or (:header property)
  366. header-cp)
  367. :cell (or (:cell property)
  368. (when (de/entity? property)
  369. (fn [_table row _column style]
  370. (pv/property-value row property {:view? true
  371. :table-view? true
  372. :table-text-property-render
  373. (fn [block opts]
  374. (block-title block (assoc opts
  375. :row row
  376. :property property
  377. :width (:width style)
  378. :sidebar? (:sidebar? config))))}))))
  379. :get-value get-value
  380. :type (:type property)}))))
  381. properties')
  382. [{:id :block/created-at
  383. :name (t :page/created-at)
  384. :type :datetime
  385. :header header-cp
  386. :cell timestamp-cell-cp}
  387. {:id :block/updated-at
  388. :name (t :page/updated-at)
  389. :type :datetime
  390. :header header-cp
  391. :cell timestamp-cell-cp}])
  392. (remove nil?))))
  393. (defn- sort-columns
  394. [columns ordered-column-ids]
  395. (if (seq ordered-column-ids)
  396. (let [id->columns (zipmap (map :id columns) columns)
  397. ordered-id-set (set ordered-column-ids)]
  398. (concat
  399. (keep (fn [id]
  400. (get id->columns id))
  401. ordered-column-ids)
  402. (remove
  403. (fn [column] (ordered-id-set (:id column)))
  404. columns)))
  405. columns))
  406. (rum/defc more-actions
  407. [view-entity columns {:keys [column-visible? rows column-toggle-visibility]} {:keys [group-by-property-ident]}]
  408. (let [display-type (:db/ident (:logseq.property.view/type view-entity))
  409. table? (= display-type :logseq.property.view/type.table)
  410. group-by-columns (concat (when (or
  411. (contains? #{:linked-references :unlinked-references}
  412. (:logseq.property.view/feature-type view-entity))
  413. (:logseq.property/query view-entity))
  414. [{:id :block/page
  415. :name "Block Page"}])
  416. (filter (fn [column]
  417. (when (:id column)
  418. (when-let [p (db/entity (:id column))]
  419. (and (not (db-property/many? p))
  420. (contains? #{:default :number :checkbox :url :node :date}
  421. (:logseq.property/type p)))))) columns))]
  422. (shui/dropdown-menu
  423. (shui/dropdown-menu-trigger
  424. {:asChild true}
  425. (shui/button
  426. {:variant "ghost"
  427. :class "text-muted-foreground !px-1"
  428. :size :sm}
  429. (ui/icon "dots" {:size 15})))
  430. (shui/dropdown-menu-content
  431. {:align "end"}
  432. (shui/dropdown-menu-group
  433. (when table?
  434. (shui/dropdown-menu-sub
  435. (shui/dropdown-menu-sub-trigger
  436. "Columns visibility")
  437. (shui/dropdown-menu-sub-content
  438. (for [column (remove #(or (false? (:column-list? %))
  439. (:disable-hide? %)) columns)]
  440. (shui/dropdown-menu-checkbox-item
  441. {:key (str (:id column))
  442. :className "capitalize"
  443. :checked (column-visible? column)
  444. :onCheckedChange #(column-toggle-visibility column %)
  445. :onSelect (fn [e] (.preventDefault e))}
  446. (:name column))))))
  447. (when (seq group-by-columns)
  448. (shui/dropdown-menu-sub
  449. (shui/dropdown-menu-sub-trigger
  450. "Group by")
  451. (shui/dropdown-menu-sub-content
  452. (for [column group-by-columns]
  453. (shui/dropdown-menu-checkbox-item
  454. {:key (str (:id column))
  455. :className "capitalize"
  456. :checked (= (:id column) group-by-property-ident)
  457. :onCheckedChange (fn [result]
  458. (if result
  459. (db-property-handler/set-block-property! (:db/id view-entity) :logseq.property.view/group-by-property
  460. (:db/id (db/entity (:id column))))
  461. (db-property-handler/remove-block-property! (:db/id view-entity) :logseq.property.view/group-by-property)))
  462. :onSelect (fn [e] (.preventDefault e))}
  463. (:name column))))))
  464. (shui/dropdown-menu-item
  465. {:key "export-edn"
  466. :on-click #(db-export-handler/export-view-nodes-data rows)}
  467. "Export EDN"))))))
  468. (defn- get-column-size
  469. [column sized-columns]
  470. (let [id (:id column)
  471. size (get sized-columns id)]
  472. (cond
  473. (= id :id)
  474. 48
  475. (number? size)
  476. size
  477. (= id :logseq.property/query)
  478. 400
  479. :else
  480. (case id
  481. :select 32
  482. :add-property 160
  483. (:block/title :block/name) 360
  484. (:block/created-at :block/updated-at) 160
  485. 180))))
  486. (rum/defc add-property-button < rum/static
  487. []
  488. [:div.ls-table-header-cell.!border-0
  489. (shui/button
  490. {:variant "text"
  491. :class "h-8 !pl-4 !px-2 !py-0 hover:text-foreground w-full justify-start"}
  492. (ui/icon "plus")
  493. "New property")])
  494. (rum/defc action-bar < rum/static
  495. [table selected-rows {:keys [on-delete-rows]}]
  496. (shui/table-actions
  497. {}
  498. [:div (str (count selected-rows) " selected")]
  499. (selection/action-bar
  500. {:on-cut #(on-delete-rows table selected-rows)
  501. :selected-blocks selected-rows
  502. :hide-dots? true
  503. :button-border? true})))
  504. (rum/defc column-resizer
  505. [_column on-sized!]
  506. (let [*el (rum/use-ref nil)
  507. [dx set-dx!] (rum/use-state nil)
  508. [width set-width!] (rum/use-state nil)
  509. add-resizing-class #(dom/add-class! js/document.documentElement "is-resizing-buf")
  510. remove-resizing-class #(dom/remove-class! js/document.documentElement "is-resizing-buf")]
  511. (hooks/use-effect!
  512. (fn []
  513. (when (number? dx)
  514. (some-> (rum/deref *el)
  515. (dom/set-style! :transform (str "translate3D(" dx "px , 0, 0)")))))
  516. [dx])
  517. (hooks/use-effect!
  518. (fn []
  519. (when-let [el (and (fn? js/window.interact) (rum/deref *el))]
  520. (let [*field-rect (atom nil)
  521. min-width 40
  522. max-width 500]
  523. (-> (js/interact el)
  524. (.draggable
  525. (bean/->js
  526. {:listeners
  527. {:start (fn []
  528. (let [{:keys [width right] :as rect} (bean/->clj (.toJSON (.getBoundingClientRect (.closest el ".ls-table-header-cell"))))
  529. left-dx (if (>= width min-width) (- min-width width) 0)
  530. right-dx (if (<= width max-width) (- max-width width) 0)]
  531. (reset! *field-rect rect)
  532. (swap! *field-rect assoc
  533. ;; calculate left/right boundary
  534. :left-dx left-dx
  535. :right-dx right-dx
  536. :left-b (inc (+ left-dx right))
  537. :right-b (inc (+ right-dx right)))
  538. (dom/add-class! el "is-active")))
  539. :move (fn [^js e]
  540. (let [dx (.-dx e)
  541. pointer-x (js/Math.floor (.-clientX e))
  542. {:keys [left-b right-b]} @*field-rect
  543. left-b (js/Math.floor left-b)
  544. right-b (js/Math.floor right-b)]
  545. (when (and (> pointer-x left-b)
  546. (< pointer-x right-b))
  547. (set-dx! (fn [dx']
  548. (if (contains? #{min-width max-width} (abs dx'))
  549. dx'
  550. (let [to-dx (+ (or dx' 0) dx)
  551. {:keys [left-dx right-dx]} @*field-rect]
  552. (cond
  553. ;; left
  554. (neg? to-dx) (if (> (abs left-dx) (abs to-dx)) to-dx left-dx)
  555. ;; right
  556. (pos? to-dx) (if (> right-dx to-dx) to-dx right-dx)))))))))
  557. :end (fn []
  558. (set-dx!
  559. (fn [dx]
  560. (let [w (js/Math.round (+ dx (:width @*field-rect)))]
  561. (set-width! (cond
  562. (< w min-width) min-width
  563. (> w max-width) max-width
  564. :else w)))
  565. (reset! *field-rect nil)
  566. (dom/remove-class! el "is-active")
  567. 0)))}}))
  568. (.styleCursor false)
  569. (.on "dragstart" add-resizing-class)
  570. (.on "dragend" remove-resizing-class)
  571. (.on "mousedown" util/stop-propagation)))))
  572. [])
  573. (hooks/use-effect!
  574. (fn []
  575. (when (number? width)
  576. (on-sized! width)))
  577. [width])
  578. [:a.ls-table-resize-handle
  579. {:data-no-dnd true
  580. :ref *el}]))
  581. (defn- table-header-cell
  582. [table column]
  583. (let [header-fn (:header column)
  584. sized-columns (get-in table [:state :sized-columns])
  585. set-sized-columns! (get-in table [:data-fns :set-sized-columns!])
  586. width (get-column-size column sized-columns)
  587. select? (= :select (:id column))]
  588. [:div.ls-table-header-cell
  589. {:style {:width width
  590. :min-width width}
  591. :class (when select? "!border-0")}
  592. (if (fn? header-fn)
  593. (header-fn table column)
  594. header-fn)
  595. ;; resize handle
  596. (when-not (false? (:resizable? column))
  597. (column-resizer column
  598. (fn [size]
  599. (set-sized-columns! (assoc sized-columns (:id column) size)))))]))
  600. (defn- on-delete-rows
  601. [view-parent view-feature-type table selected-ids]
  602. (let [selected-rows (->> (map db/entity selected-ids)
  603. (remove :logseq.property/built-in?))
  604. pages (filter ldb/page? selected-rows)
  605. blocks (remove ldb/page? selected-rows)
  606. page-ids (map :db/id pages)
  607. {:keys [set-data! set-row-selection!]} (:data-fns table)
  608. update-table-state! (fn []
  609. (let [data (:full-data table)
  610. selected-ids (set (map :db/id selected-rows))
  611. new-data (if (every? number? data)
  612. (remove selected-ids data)
  613. ;; group
  614. (map (fn [[by-value col]]
  615. [by-value (remove selected-ids col)]) data))]
  616. (set-data! new-data)
  617. (set-row-selection! {})))]
  618. (p/do!
  619. (ui-outliner-tx/transact!
  620. {:outliner-op :delete-blocks}
  621. (when (seq blocks)
  622. (outliner-op/delete-blocks! blocks nil))
  623. (case view-feature-type
  624. :class-objects
  625. (when (seq page-ids)
  626. (let [tx-data (map (fn [pid] [:db/retract pid :block/tags (:db/id view-parent)]) page-ids)]
  627. (when (seq tx-data)
  628. (outliner-op/transact! tx-data {:outliner-op :save-block}))))
  629. :property-objects
  630. ;; Relationships with built-in properties must not be deleted e.g. built-in? or parent
  631. (when-not (:logseq.property/built-in? view-parent)
  632. (let [tx-data (map (fn [pid] [:db/retract pid (:db/ident view-parent)]) page-ids)]
  633. (when (seq tx-data)
  634. (outliner-op/transact! tx-data {:outliner-op :save-block}))))
  635. :query-result
  636. (doseq [page pages]
  637. (when-let [id (:block/uuid page)]
  638. (outliner-op/delete-page! id)))
  639. :all-pages
  640. (state/pub-event! [:page/show-delete-dialog selected-rows update-table-state!])
  641. nil))
  642. (when-not (or (= view-feature-type :all-pages)
  643. (and (= view-feature-type :property-objects) (:logseq.property/built-in? view-parent)))
  644. (update-table-state!)))))
  645. (defn- table-header
  646. [table {:keys [show-add-property? add-property! view-parent view-feature-type] :as option} selected-rows]
  647. (let [set-ordered-columns! (get-in table [:data-fns :set-ordered-columns!])
  648. pinned (get-in table [:state :pinned-columns])
  649. unpinned (get-in table [:state :unpinned-columns])
  650. build-item (fn [column]
  651. {:id (:name column)
  652. :value (:id column)
  653. :content (table-header-cell table column)
  654. :disabled? (= (:id column) :select)})
  655. pinned-items (mapv build-item pinned)
  656. unpinned-items (if show-add-property?
  657. (conj (mapv build-item unpinned)
  658. {:id "add property"
  659. :prop {:style {:width "-webkit-fill-available"
  660. :min-width 160}
  661. :on-click (fn [e] (when (fn? add-property!) (add-property! e)))}
  662. :value :add-new-property
  663. :content (add-property-button)
  664. :disabled? true})
  665. (mapv build-item unpinned))
  666. selection-rows-count (count selected-rows)]
  667. (shui/table-header
  668. (when (seq pinned-items)
  669. [:div.sticky-columns.flex.flex-row
  670. (dnd/items pinned-items {:vertical? false
  671. :on-drag-end (fn [ordered-columns _m]
  672. (set-ordered-columns! ordered-columns))})])
  673. (when (seq unpinned-items)
  674. [:div.flex.flex-row
  675. (dnd/items unpinned-items
  676. {:vertical? false
  677. :on-drag-end (fn [ordered-columns _m]
  678. (set-ordered-columns! ordered-columns))})])
  679. (when (pos? selection-rows-count)
  680. [:div.table-action-bar.absolute.top-0.left-8
  681. (action-bar table selected-rows
  682. (assoc option
  683. :on-delete-rows (fn [table selected-ids]
  684. (on-delete-rows view-parent view-feature-type table selected-ids))))]))))
  685. (rum/defc lazy-table-cell
  686. [cell-render-f cell-placeholder]
  687. (let [^js state (ui/useInView #js {:rootMargin "0px"})
  688. in-view? (.-inView state)]
  689. [:div.h-full
  690. {:ref (.-ref state)}
  691. (if in-view?
  692. (cell-render-f)
  693. cell-placeholder)]))
  694. (defn- click-cell
  695. [node]
  696. (when-let [trigger (or (dom/sel1 node ".jtrigger")
  697. (dom/sel1 node ".table-block-title"))]
  698. (.click trigger)))
  699. (defn navigate-to-cell
  700. [e cell direction]
  701. (util/stop e)
  702. (let [row (util/rec-get-node cell "ls-table-row")
  703. cells (dom/sel row ".ls-table-cell")
  704. idx (.indexOf cells cell)
  705. rows-container (util/rec-get-node row "ls-table-rows")
  706. rows (dom/sel rows-container ".ls-table-row")
  707. row-idx (.indexOf rows row)
  708. container-left (.-left (.getBoundingClientRect rows-container))
  709. next-cell (case direction
  710. :left (if (> idx 1) ; don't focus on checkbox
  711. (nth cells (dec idx))
  712. ;; last cell in the prev row
  713. (let [prev-row (when (> row-idx 0)
  714. (nth rows (dec row-idx)))]
  715. (when prev-row
  716. (let [cells (dom/sel prev-row ".ls-table-cell")]
  717. (last cells)))))
  718. :right (if (< idx (dec (count cells)))
  719. (nth cells (inc idx))
  720. ;; first cell in the next row
  721. (let [next-row (when (< row-idx (dec (count rows)))
  722. (nth rows (inc row-idx)))]
  723. (when next-row
  724. (let [cells (dom/sel next-row ".ls-table-cell")]
  725. (second cells)))))
  726. :up (let [prev-row (when (> row-idx 0)
  727. (nth rows (dec row-idx)))]
  728. (when prev-row
  729. (let [cells (dom/sel prev-row ".ls-table-cell")]
  730. (nth cells idx))))
  731. :down (let [next-row (when (< row-idx (dec (count rows)))
  732. (nth rows (inc row-idx)))]
  733. (when next-row
  734. (let [cells (dom/sel next-row ".ls-table-cell")]
  735. (nth cells idx)))))]
  736. (when next-cell
  737. (let [next-cell-left (.-left (.getBoundingClientRect next-cell))]
  738. (state/clear-selection!)
  739. (dom/add-class! next-cell "selected")
  740. (.focus next-cell)
  741. (when (< next-cell-left container-left)
  742. (.scrollIntoView next-cell #js {:inline "center"
  743. :block "nearest"}))))))
  744. (rum/defc table-cell-container
  745. [cell-opts body]
  746. (let [*ref (hooks/use-ref nil)]
  747. (shui/table-cell
  748. (assoc cell-opts
  749. :tabIndex 0
  750. :ref *ref
  751. :on-click (fn [e]
  752. (when-not (dom/has-class? (.-target e) "jtrigger")
  753. (click-cell (rum/deref *ref))))
  754. :on-key-down (fn [e]
  755. (let [container (rum/deref *ref)]
  756. (case (util/ekey e)
  757. "Escape"
  758. (do
  759. (if (util/input? (.-target e))
  760. (do
  761. (state/exit-editing-and-set-selected-blocks! [container])
  762. (.focus container))
  763. (do
  764. (dom/remove-class! container "selected")
  765. (let [row (util/rec-get-node container "ls-table-row")]
  766. (state/exit-editing-and-set-selected-blocks! [row]))))
  767. (util/stop e))
  768. "Enter"
  769. (do
  770. (if (util/input? (.-target e)) ; number
  771. (do
  772. (state/exit-editing-and-set-selected-blocks! [container])
  773. (.focus container))
  774. (click-cell container))
  775. (util/stop e))
  776. "ArrowUp"
  777. (navigate-to-cell e container :up)
  778. "ArrowDown"
  779. (navigate-to-cell e container :down)
  780. "ArrowLeft"
  781. (navigate-to-cell e container :left)
  782. "ArrowRight"
  783. (navigate-to-cell e container :right)
  784. nil))))
  785. body)))
  786. (rum/defc table-row-inner < rum/static
  787. [{:keys [row-selected?] :as table} row props {:keys [show-add-property? scrolling?]}]
  788. (let [*ref (hooks/use-ref nil)
  789. pinned-columns (get-in table [:state :pinned-columns])
  790. unpinned (get-in table [:state :unpinned-columns])
  791. unpinned-columns (if show-add-property?
  792. (conj (vec unpinned)
  793. {:id :add-property
  794. :cell (fn [_table _row _column])})
  795. unpinned)
  796. sized-columns (get-in table [:state :sized-columns])
  797. row-cell-f (fn [column {:keys [_lazy?]}]
  798. (let [id (str (:id row) "-" (:id column))
  799. width (get-column-size column sized-columns)
  800. select? (= (:id column) :select)
  801. add-property? (= (:id column) :add-property)
  802. style {:width width :min-width width}
  803. cell-opts {:key id
  804. :select? select?
  805. :add-property? add-property?
  806. :style style}
  807. cell-placeholder (table-cell-container cell-opts nil)]
  808. (if (and scrolling? (not (:block/title row)))
  809. cell-placeholder
  810. (when-let [render (get column :cell)]
  811. (lazy-table-cell
  812. (fn []
  813. (table-cell-container
  814. cell-opts (render table row column style)))
  815. cell-placeholder)))))]
  816. (shui/table-row
  817. (merge
  818. props
  819. {:key (str (:db/id row))
  820. :tabIndex 0
  821. :ref *ref
  822. :data-state (when (row-selected? row) "selected")
  823. :data-id (:db/id row)
  824. :blockid (str (:block/uuid row))
  825. :on-pointer-down (fn [_e] (db-async/<get-block (state/get-current-repo) (:db/id row) {:children? false}))
  826. :on-key-down (fn [e]
  827. (let [container (rum/deref *ref)]
  828. (when (dom/has-class? container "selected")
  829. (case (util/ekey e)
  830. "Enter"
  831. (do
  832. (state/sidebar-add-block! (state/get-current-repo) (:db/id row) :block)
  833. (state/clear-selection!)
  834. (util/stop e))
  835. "ArrowLeft"
  836. (do
  837. (when-let [cell (->> (dom/sel container ".ls-table-cell")
  838. (remove (fn [node]
  839. (some? (dom/sel1 node ".ui__checkbox"))))
  840. first)]
  841. (state/clear-selection!)
  842. (dom/add-class! cell "selected")
  843. (.focus cell))
  844. (util/stop e))
  845. "ArrowRight"
  846. (do
  847. (when-let [cell (->> (dom/sel container ".ls-table-cell")
  848. (remove (fn [node]
  849. (some? (dom/sel1 node ".ui__checkbox"))))
  850. last)]
  851. (state/clear-selection!)
  852. (dom/remove-class! container "selected")
  853. (dom/add-class! cell "selected")
  854. (.focus cell))
  855. (util/stop e))
  856. "Escape"
  857. (do
  858. (state/clear-selection!)
  859. (util/stop e))
  860. nil))))})
  861. (when (seq pinned-columns)
  862. [:div.sticky-columns.flex.flex-row
  863. (map #(row-cell-f % {}) pinned-columns)])
  864. (when (seq unpinned-columns)
  865. [:div.flex.flex-row
  866. (map #(row-cell-f % {:lazy? true}) unpinned-columns)]))))
  867. (rum/defc table-row < rum/reactive db-mixins/query
  868. [table row props option]
  869. (let [block (db/sub-block (:db/id row))
  870. row' (if (:block.temp/fully-loaded? block)
  871. (assoc block :block.temp/refs-count (:block.temp/refs-count row))
  872. row)]
  873. (table-row-inner table row' props option)))
  874. (rum/defc search
  875. [input {:keys [on-change set-input!]}]
  876. (let [[show-input? set-show-input!] (rum/use-state false)]
  877. (if show-input?
  878. [:div.flex.flex-row.items-center
  879. (shui/input
  880. {:placeholder "Type to search"
  881. :auto-focus true
  882. :value input
  883. :on-change (fn [e]
  884. (let [value (util/evalue e)]
  885. (on-change value)))
  886. :on-key-down (fn [e]
  887. (when (= "Escape" (util/ekey e))
  888. (set-show-input! false)
  889. (set-input! "")))
  890. :class "max-w-sm !h-7 !py-0 border-none focus-visible:ring-0 focus-visible:ring-offset-0"})
  891. (shui/button
  892. {:variant "ghost"
  893. :class "text-muted-foreground !px-1"
  894. :size :sm
  895. :on-click #(do
  896. (set-show-input! false)
  897. (set-input! ""))}
  898. (ui/icon "x"))]
  899. (shui/button
  900. {:variant "ghost"
  901. ;; FIXME: remove ring when focused
  902. :class "text-muted-foreground !px-1"
  903. :size :sm
  904. :on-click #(set-show-input! true)}
  905. (ui/icon "search" {:size 15})))))
  906. (defn datetime-property?
  907. [property]
  908. (or
  909. (= :datetime (:logseq.property/type property))
  910. (contains? #{:block/created-at :block/updated-at} (:db/ident property))))
  911. (def timestamp-options
  912. [{:value "1 day ago"
  913. :label "1 day ago"}
  914. {:value "3 days ago"
  915. :label "3 days ago"}
  916. {:value "1 week ago"
  917. :label "1 week ago"}
  918. {:value "1 month ago"
  919. :label "1 month ago"}
  920. {:value "3 months ago"
  921. :label "3 months ago"}
  922. {:value "1 year ago"
  923. :label "1 year ago"}
  924. {:value "Custom date"
  925. :label "Custom date"}])
  926. (rum/defc filter-property < rum/static
  927. [view-entity columns {:keys [data-fns] :as table} opts]
  928. (let [[property set-property!] (rum/use-state nil)
  929. [values set-values!] (rum/use-state nil)
  930. schema (:schema (db/get-db))
  931. timestamp? (datetime-property? property)
  932. set-filters! (:set-filters! data-fns)
  933. filters (get-in table [:state :filters])
  934. columns (remove #(or (false? (:column-list? %))
  935. (= :id (:id %))) columns)
  936. items (map (fn [column]
  937. {:label (:name column)
  938. :value column}) columns)
  939. option {:input-default-placeholder "Filter"
  940. :input-opts {:class "!px-2 !py-1"}
  941. :items items
  942. :extract-fn :label
  943. :extract-chosen-fn :value
  944. :on-chosen (fn [column]
  945. (let [id (:id column)
  946. property (db/entity id)
  947. internal-property {:db/ident (:id column)
  948. :block/title (:name column)
  949. :logseq.property/type (:type column)}]
  950. (if (or property
  951. (= :db.cardinality/many (:db/cardinality (get schema id)))
  952. (not= (:type column) :string))
  953. (set-property! (or property internal-property))
  954. (do
  955. (shui/popup-hide!)
  956. (let [property internal-property
  957. new-filter [(:db/ident property) :text-contains]
  958. filters' (if (seq (:filters filters))
  959. (conj (:filters filters) new-filter)
  960. [new-filter])]
  961. (set-filters! {:or? (:or? filters)
  962. :filters filters'}))))))}
  963. checkbox? (= :checkbox (:logseq.property/type property))
  964. property-ident (:db/ident property)]
  965. (hooks/use-effect!
  966. (fn []
  967. (when (and view-entity property-ident (not (or timestamp? checkbox?)))
  968. (p/let [data (db-async/<get-property-values property-ident {:view-id (:db/id view-entity)
  969. :query-entity-ids (:query-entity-ids opts)})]
  970. (set-values! data))))
  971. [property-ident])
  972. (let [option (cond
  973. timestamp?
  974. (merge option
  975. {:items timestamp-options
  976. :input-default-placeholder (if property (:block/title property) "Select")
  977. :on-chosen (fn [value _ _ e]
  978. (shui/popup-hide!)
  979. (let [set-filter-fn (fn [value]
  980. (let [filters' (conj (:filters filters) [(:db/ident property) :after value])]
  981. (set-filters! {:or? (:or? filters)
  982. :filters filters'})))]
  983. (if (= value "Custom date")
  984. (shui/popup-show!
  985. (.-target e)
  986. (ui/nlp-calendar
  987. {:initial-focus true
  988. :datetime? false
  989. :on-day-click (fn [value]
  990. (set-filter-fn value)
  991. (shui/popup-hide!))})
  992. {})
  993. (set-filter-fn value))))})
  994. property
  995. (if checkbox?
  996. (let [items [{:value true :label "true"}
  997. {:value false :label "false"}]]
  998. (merge option
  999. {:items items
  1000. :input-default-placeholder (if property (:block/title property) "Select")
  1001. :on-chosen (fn [value]
  1002. (let [filters' (conj (:filters filters) [(:db/ident property) :is value])]
  1003. (set-filters! {:or? (:or? filters)
  1004. :filters filters'})))}))
  1005. (let [items values]
  1006. (merge option
  1007. {:items items
  1008. :input-default-placeholder (if property (:block/title property) "Select")
  1009. :multiple-choices? true
  1010. :on-chosen (fn [_value _selected? selected]
  1011. (let [selected-value (if (and (map? (first selected))
  1012. (:block/uuid (first selected)))
  1013. (set (map :block/uuid selected))
  1014. selected)
  1015. filters' (if (seq selected)
  1016. (conj (:filters filters) [(:db/ident property) :is selected-value])
  1017. (:filters filters))]
  1018. (set-filters! {:or? (:or? filters)
  1019. :filters filters'})))})))
  1020. :else
  1021. option)]
  1022. (select/select option))))
  1023. (rum/defc filter-properties < rum/static
  1024. [view-entity columns table opts]
  1025. (shui/button
  1026. {:variant "ghost"
  1027. :class "text-muted-foreground !px-1"
  1028. :size :sm
  1029. :on-click (fn [e]
  1030. (shui/popup-show! (.-target e)
  1031. (fn []
  1032. (filter-property view-entity columns table opts))
  1033. {:align :end
  1034. :auto-focus? true}))}
  1035. (ui/icon "filter")))
  1036. (defn operator->text
  1037. [operator]
  1038. (case operator
  1039. :is "is"
  1040. :is-not "is not"
  1041. :text-contains "text contains"
  1042. :text-not-contains "text not contains"
  1043. :date-before "date before"
  1044. :date-after "date after"
  1045. :before "before"
  1046. :after "after"
  1047. :number-gt ">"
  1048. :number-lt "<"
  1049. :number-gte ">="
  1050. :number-lte "<="
  1051. :between "between"))
  1052. (defn get-property-operators
  1053. [property]
  1054. (if (datetime-property? property)
  1055. [:before :after]
  1056. (concat
  1057. [:is :is-not]
  1058. (case (:logseq.property/type property)
  1059. (:default :url :node)
  1060. [:text-contains :text-not-contains]
  1061. (:date)
  1062. [:date-before :date-after]
  1063. :number
  1064. [:number-gt :number-lt :number-gte :number-lte :between]
  1065. nil))))
  1066. (defn- get-filter-with-changed-operator
  1067. [_property operator value]
  1068. (case operator
  1069. (:is :is-not)
  1070. (when (set? value) value)
  1071. (:text-contains :text-not-contains)
  1072. (when (string? value) value)
  1073. (:number-gt :number-lt :number-gte :number-lte)
  1074. (when (number? value) value)
  1075. :between
  1076. (when (and (vector? value) (every? number? value))
  1077. value)
  1078. (:date-before :date-after :before :after)
  1079. ;; FIXME: should be a valid date number
  1080. (when (number? value) value)))
  1081. (rum/defc filter-operator < rum/static
  1082. [property operator filters set-filters! idx]
  1083. (shui/dropdown-menu
  1084. (shui/dropdown-menu-trigger
  1085. {:asChild true}
  1086. (shui/button
  1087. {:class "!px-2 rounded-none border-r"
  1088. :variant "ghost"
  1089. :size :sm}
  1090. [:span.text-xs (operator->text operator)]))
  1091. (shui/dropdown-menu-content
  1092. {:align "start"}
  1093. (let [operators (get-property-operators property)]
  1094. (for [operator operators]
  1095. (shui/dropdown-menu-item
  1096. {:on-click (fn []
  1097. (let [new-filters (update filters :filters
  1098. (fn [col]
  1099. (update col idx
  1100. (fn [[property _old-operator value]]
  1101. (let [value' (get-filter-with-changed-operator property operator value)]
  1102. (if value'
  1103. [property operator value']
  1104. [property operator]))))))]
  1105. (set-filters! new-filters)))}
  1106. (operator->text operator)))))))
  1107. (rum/defc between < rum/static
  1108. [_property [start end] filters set-filters! idx]
  1109. [:<>
  1110. (shui/input
  1111. {:auto-focus true
  1112. :placeholder "from"
  1113. :value (str start)
  1114. :onChange (fn [e]
  1115. (let [input-value (util/evalue e)
  1116. number-value (when-not (string/blank? input-value)
  1117. (util/safe-parse-float input-value))
  1118. value [number-value end]
  1119. value (if (every? nil? value) nil value)]
  1120. (let [new-filters (update filters :filters
  1121. (fn [col]
  1122. (update col idx
  1123. (fn [[property operator _old_value]]
  1124. (if (nil? value)
  1125. [property operator]
  1126. [property operator value])))))]
  1127. (set-filters! new-filters))))
  1128. :class "w-24 !h-6 !py-0 border-none focus-visible:ring-0 focus-visible:ring-offset-0"})
  1129. (shui/input
  1130. {:value (str end)
  1131. :placeholder "to"
  1132. :onChange (fn [e]
  1133. (let [input-value (util/evalue e)
  1134. number-value (when-not (string/blank? input-value)
  1135. (util/safe-parse-float input-value))
  1136. value [start number-value]
  1137. value (if (every? nil? value) nil value)]
  1138. (let [new-filters (update filters :filters
  1139. (fn [col]
  1140. (update col idx
  1141. (fn [[property operator _old_value]]
  1142. (if (nil? value)
  1143. [property operator]
  1144. [property operator value])))))]
  1145. (set-filters! new-filters))))
  1146. :class "w-24 !h-6 !py-0 border-none focus-visible:ring-0 focus-visible:ring-offset-0"})])
  1147. (rum/defc ^:large-vars/cleanup-todo filter-value-select < rum/static
  1148. [view-entity {:keys [data-fns] :as table} property value operator idx opts]
  1149. (let [type (:logseq.property/type property)
  1150. property-ident (:db/ident property)]
  1151. (hooks/use-effect!
  1152. (fn []
  1153. (let [values (if (coll? value) value [value])
  1154. ids (filter #(and (uuid? %) (nil? (db/entity [:block/uuid %]))) values)]
  1155. (when (seq ids) (db-async/<get-blocks (state/get-current-repo) ids))))
  1156. [])
  1157. (let [filters (get-in table [:state :filters])
  1158. set-filters! (:set-filters! data-fns)
  1159. many? (if (or (contains? #{:date-before :date-after :before :after} operator)
  1160. (contains? #{:checkbox} type))
  1161. false
  1162. true)]
  1163. (shui/button
  1164. {:class "!px-2 rounded-none border-r"
  1165. :variant "ghost"
  1166. :size :sm
  1167. :on-click (fn [e]
  1168. (p/let [values (when (and property-ident
  1169. (not (contains? #{:data :datetime :checkbox} type)))
  1170. (p/let [data (db-async/<get-property-values property-ident {:view-id (:db/id view-entity)
  1171. :query-entity-ids (:query-entity-ids opts)})]
  1172. (map (fn [v] (if (map? (:value v))
  1173. (assoc v :value (:block/uuid (:value v)))
  1174. v)) data)))
  1175. items (cond
  1176. (contains? #{:before :after} operator)
  1177. timestamp-options
  1178. (= type :checkbox)
  1179. [{:value true :label "true"} {:value false :label "false"}]
  1180. :else
  1181. values)]
  1182. (shui/popup-show!
  1183. (.-target e)
  1184. (fn []
  1185. (let [option (cond->
  1186. {:input-default-placeholder (:block/title property)
  1187. :input-opts {:class "!px-3 !py-1"}
  1188. :items items
  1189. :extract-fn :label
  1190. :extract-chosen-fn :value
  1191. :on-chosen (fn [value _selected? selected e]
  1192. (when-not many?
  1193. (shui/popup-hide!))
  1194. (let [value' (if many? selected value)
  1195. set-filters-fn (fn [value']
  1196. (let [new-filters (update filters :filters
  1197. (fn [col]
  1198. (update col idx
  1199. (fn [[property operator _value]]
  1200. [property operator value']))))]
  1201. (set-filters! new-filters)))]
  1202. (if (= value "Custom date")
  1203. (shui/popup-show!
  1204. (.-target e)
  1205. (ui/nlp-calendar
  1206. {:initial-focus true
  1207. :datetime? false
  1208. :on-day-click (fn [value]
  1209. (set-filters-fn value)
  1210. (shui/popup-hide!))})
  1211. {})
  1212. (set-filters-fn value'))))}
  1213. many?
  1214. (assoc
  1215. :multiple-choices? true
  1216. :selected-choices value))]
  1217. (select/select option)))
  1218. {:align :start})))}
  1219. (let [value (cond
  1220. (uuid? value)
  1221. (db/entity [:block/uuid value])
  1222. (instance? js/Date value)
  1223. (some->> (tc/to-date value)
  1224. (t/to-default-time-zone)
  1225. (tf/unparse (tf/formatter "yyyy-MM-dd")))
  1226. (and (coll? value) (every? uuid? value))
  1227. (keep #(db/entity [:block/uuid %]) value)
  1228. :else
  1229. value)]
  1230. [:div.flex.flex-row.items-center.gap-1.text-xs
  1231. (cond
  1232. (de/entity? value)
  1233. [:div (get-property-value-content value)]
  1234. (string? value)
  1235. [:div value]
  1236. (boolean? value)
  1237. [:div (str value)]
  1238. (seq value)
  1239. (->> (map (fn [v] [:div (get-property-value-content v)]) value)
  1240. (interpose [:div "or"]))
  1241. :else
  1242. "All")])))))
  1243. (rum/defc filter-value < rum/static
  1244. [view-entity table property operator value filters set-filters! idx opts]
  1245. (let [number-operator? (string/starts-with? (name operator) "number-")]
  1246. (case operator
  1247. :between
  1248. (between property value filters set-filters! idx)
  1249. (:text-contains :text-not-contains :number-gt :number-lt :number-gte :number-lte)
  1250. (shui/input
  1251. {:auto-focus false
  1252. :value (or value "")
  1253. :onChange (fn [e]
  1254. (let [value (util/evalue e)
  1255. number-value (and number-operator? (when-not (string/blank? value)
  1256. (util/safe-parse-float value)))]
  1257. (let [new-filters (update filters :filters
  1258. (fn [col]
  1259. (update col idx
  1260. (fn [[property operator _value]]
  1261. (if (and number-operator? (nil? number-value))
  1262. [property operator]
  1263. [property operator (or number-value value)])))))]
  1264. (set-filters! new-filters))))
  1265. :class "w-24 !h-6 !py-0 border-none focus-visible:ring-0 focus-visible:ring-offset-0"})
  1266. (filter-value-select view-entity table property value operator idx opts))))
  1267. (rum/defc filters-row < rum/static ;
  1268. [view-entity {:keys [data-fns columns] :as table} opts]
  1269. (let [filters (get-in table [:state :filters])
  1270. {:keys [set-filters!]} data-fns]
  1271. (when (seq (:filters filters))
  1272. [:div.filters-row.flex.flex-row.items-center.gap-4.justify-between.flex-wrap.py-2
  1273. [:div.flex.flex-row.items-center.gap-2
  1274. (map-indexed
  1275. (fn [idx filter']
  1276. (let [[property-ident operator value] filter'
  1277. property (if (= property-ident :block/title)
  1278. {:db/ident property-ident
  1279. :block/title "Name"}
  1280. (or (db/entity property-ident)
  1281. (some (fn [column] (when (= (:id column) property-ident)
  1282. {:db/ident (:id column)
  1283. :block/title (:name column)})) columns)))]
  1284. [:div.flex.flex-row.items-center.border.rounded
  1285. (shui/button
  1286. {:class "!px-2 rounded-none border-r"
  1287. :variant "ghost"
  1288. :size :sm
  1289. :disabled true}
  1290. [:span.text-xs (:block/title property)])
  1291. (filter-operator property operator filters set-filters! idx)
  1292. (filter-value view-entity table property operator value filters set-filters! idx opts)
  1293. (shui/button
  1294. {:class "!px-1 rounded-none text-muted-foreground"
  1295. :variant "ghost"
  1296. :size :sm
  1297. :on-click (fn [_e]
  1298. (let [new-filters (update filters :filters (fn [col] (vec (remove #{filter'} col))))]
  1299. (set-filters! new-filters)))}
  1300. (ui/icon "x"))]))
  1301. (:filters filters))]
  1302. (when (> (count (:filters filters)) 1)
  1303. [:div
  1304. (shui/select
  1305. {:default-value (if (:or? filters) "or" "and")
  1306. :on-value-change (fn [v]
  1307. (set-filters! (assoc filters :or? (= v "or"))))}
  1308. (shui/select-trigger
  1309. {:class "opacity-75 hover:opacity-100 !px-2 !py-0 !h-6"}
  1310. (shui/select-value
  1311. {:placeholder "Match"}))
  1312. (shui/select-content
  1313. (shui/select-group
  1314. (shui/select-item {:value "and"} "Match all filters")
  1315. (shui/select-item {:value "or"} "Match any filter"))))])])))
  1316. (rum/defc new-record-button < rum/static
  1317. [table view-entity]
  1318. (let [asset? (and (:logseq.property/built-in? view-entity)
  1319. (= (:block/name view-entity) "asset"))]
  1320. (ui/tooltip
  1321. (shui/button
  1322. {:variant "ghost"
  1323. :class "!px-1 text-muted-foreground"
  1324. :size :sm
  1325. :on-click (fn [_]
  1326. (let [f (get-in table [:data-fns :add-new-object!])]
  1327. (f view-entity table)))}
  1328. (ui/icon (if asset? "upload" "plus")))
  1329. [:div "New node"])))
  1330. (rum/defc add-new-row < rum/static
  1331. [view-entity table]
  1332. [:div.py-1.px-2.cursor-pointer.flex.flex-row.items-center.gap-1.text-muted-foreground.hover:text-foreground.w-full.text-sm.border-b
  1333. {:on-click (fn [_]
  1334. (let [f (get-in table [:data-fns :add-new-object!])]
  1335. (f view-entity table)))}
  1336. (ui/icon "plus" {:size 14})
  1337. [:div "New"]])
  1338. (defn- table-filters->persist-state
  1339. [filters]
  1340. (mapv
  1341. (fn [[property operator matches]]
  1342. (let [matches' (cond
  1343. (de/entity? matches)
  1344. (:block/uuid matches)
  1345. (and (coll? matches) (every? de/entity? matches))
  1346. (set (map :block/uuid matches))
  1347. :else
  1348. matches)]
  1349. (if (some? matches')
  1350. [property operator matches']
  1351. [property operator])))
  1352. filters))
  1353. (defn- db-set-table-state!
  1354. [entity {:keys [set-sorting! set-filters! set-visible-columns!
  1355. set-ordered-columns! set-sized-columns!]}]
  1356. (let [repo (state/get-current-repo)
  1357. db-based? (config/db-based-graph?)]
  1358. {:set-sorting!
  1359. (fn [sorting]
  1360. (p/do!
  1361. (when db-based? (property-handler/set-block-property! repo (:db/id entity) :logseq.property.table/sorting sorting))
  1362. (set-sorting! sorting)))
  1363. :set-filters!
  1364. (fn [filters]
  1365. (let [filters (-> (update filters :filters table-filters->persist-state)
  1366. (update :or? boolean))]
  1367. (p/do!
  1368. (when db-based? (property-handler/set-block-property! repo (:db/id entity) :logseq.property.table/filters filters))
  1369. (set-filters! filters))))
  1370. :set-visible-columns!
  1371. (fn [columns]
  1372. (let [hidden-columns (vec (keep (fn [[column visible?]]
  1373. (when (false? visible?)
  1374. column)) columns))]
  1375. (p/do!
  1376. (when db-based? (property-handler/set-block-property! repo (:db/id entity) :logseq.property.table/hidden-columns hidden-columns))
  1377. (set-visible-columns! columns))))
  1378. :set-ordered-columns!
  1379. (fn [ordered-columns]
  1380. (let [ids (vec (remove #{:select} ordered-columns))]
  1381. (p/do!
  1382. (when db-based? (property-handler/set-block-property! repo (:db/id entity) :logseq.property.table/ordered-columns ids))
  1383. (set-ordered-columns! ordered-columns))))
  1384. :set-sized-columns!
  1385. (fn [sized-columns]
  1386. (p/do!
  1387. (when db-based? (property-handler/set-block-property! repo (:db/id entity) :logseq.property.table/sized-columns sized-columns))
  1388. (set-sized-columns! sized-columns)))}))
  1389. (rum/defc lazy-item
  1390. [data idx {:keys [properties list-view? scrolling?]} item-render]
  1391. (let [item (util/nth-safe data idx)
  1392. db-id (cond (map? item) (:db/id item)
  1393. (number? item) item
  1394. :else nil)
  1395. [item set-item!] (hooks/use-state nil)
  1396. opts (if list-view?
  1397. {:skip-refresh? true
  1398. :children? false}
  1399. {:children? false
  1400. :properties properties
  1401. :skip-transact? true
  1402. :skip-refresh? true})]
  1403. (hooks/use-effect!
  1404. #(c.m/run-task*
  1405. (m/sp
  1406. (when (and db-id (not item) (not scrolling?))
  1407. (let [block (c.m/<? (db-async/<get-block (state/get-current-repo) db-id opts))
  1408. block' (if list-view? (db/entity db-id) block)]
  1409. (set-item! block')))))
  1410. [db-id scrolling?])
  1411. (let [item' (cond (map? item) item (number? item) {:db/id item})]
  1412. (item-render item'))))
  1413. (rum/defc table-body < rum/static
  1414. [table option rows *scroller-ref set-items-rendered!]
  1415. (let [[scrolling? set-scrolling!] (hooks/use-state false)]
  1416. (when (seq rows)
  1417. (ui/virtualized-list
  1418. {:ref #(reset! *scroller-ref %)
  1419. :increase-viewport-by {:top 300 :bottom 300}
  1420. :custom-scroll-parent (get-scroll-parent (:config option))
  1421. :compute-item-key (fn [idx]
  1422. (let [block-id (util/nth-safe rows idx)]
  1423. (str "table-row-" block-id)))
  1424. :skipAnimationFrameInResizeObserver true
  1425. :total-count (count rows)
  1426. :context {:scrolling scrolling?}
  1427. :is-scrolling set-scrolling!
  1428. :item-content (fn [idx _user ^js context]
  1429. (let [option (assoc option
  1430. :scrolling? (.-scrolling context)
  1431. :table-view? true)]
  1432. (lazy-item (:data table) idx option
  1433. (fn [row]
  1434. (table-row table row {} option)))))
  1435. :items-rendered (fn [props]
  1436. (when (seq props)
  1437. (set-items-rendered! true)))}))))
  1438. (rum/defc table-view < rum/static
  1439. [table option row-selection *scroller-ref]
  1440. (let [selected-rows (shui/table-get-selection-rows row-selection (:rows table))
  1441. [items-rendered? set-items-rendered!] (hooks/use-state false)]
  1442. (shui/table
  1443. (let [rows (:rows table)]
  1444. [:div.ls-table-rows.content.overflow-x-auto.force-visible-scrollbar
  1445. [:div.relative
  1446. (table-header table option selected-rows)
  1447. (table-body table option rows *scroller-ref set-items-rendered!)
  1448. (when (and (get-in table [:data-fns :add-new-object!]) (or (empty? rows) items-rendered?))
  1449. (shui/table-footer (add-new-row (:view-entity option) table)))]]))))
  1450. (rum/defc list-view < rum/static
  1451. [{:keys [config] :as option} _view-entity {:keys [rows]} *scroller-ref]
  1452. (let [lazy-item-render (fn [rows idx]
  1453. (lazy-item rows idx (assoc option :list-view? true)
  1454. (fn [block]
  1455. (block-container (assoc config :list-view? true) block))))
  1456. list-cp (fn [rows]
  1457. (when (seq rows)
  1458. (ui/virtualized-list
  1459. {:ref #(reset! *scroller-ref %)
  1460. :class "content"
  1461. :custom-scroll-parent (get-scroll-parent config)
  1462. :increase-viewport-by {:top 64 :bottom 64}
  1463. :compute-item-key (fn [idx]
  1464. (let [block-id (util/nth-safe rows idx)]
  1465. (str "list-row-" block-id)))
  1466. :total-count (count rows)
  1467. :skipAnimationFrameInResizeObserver true
  1468. :item-content (fn [idx] (lazy-item-render rows idx))})))
  1469. breadcrumb (state/get-component :block/breadcrumb)
  1470. all-numbers? (every? number? rows)]
  1471. (if all-numbers?
  1472. (list-cp rows)
  1473. (for [[idx row] (medley/indexed rows)]
  1474. (if (and (vector? row) (uuid? (first row)))
  1475. (let [[first-block-id blocks] row]
  1476. [:div
  1477. {:key (str "partition-" first-block-id)}
  1478. [:div.ml-6.text-sm.opacity-70.hover:opacity-100.mt-1
  1479. (breadcrumb (assoc config :list-view? true)
  1480. (state/get-current-repo) first-block-id
  1481. {:show-page? false})]
  1482. (list-cp blocks)])
  1483. (rum/with-key
  1484. (lazy-item-render rows idx)
  1485. (str "partition-" idx)))))))
  1486. (rum/defc gallery-card-item
  1487. [view-entity block config]
  1488. [:div.ls-card-item.content
  1489. {:key (str "view-card-" (:db/id view-entity) "-" (:db/id block))}
  1490. [:div.-ml-4
  1491. (block-container (assoc config
  1492. :id (str (:block/uuid block))
  1493. :gallery-view? true
  1494. :view? true)
  1495. block)]])
  1496. (rum/defcs gallery-view < rum/static mixins/container-id
  1497. [state {:keys [config]} table view-entity blocks *scroller-ref]
  1498. (let [config' (assoc config :container-id (:container-id state))]
  1499. [:div.ls-cards
  1500. (when (seq blocks)
  1501. (ui/virtualized-grid
  1502. {:ref #(reset! *scroller-ref %)
  1503. :total-count (count blocks)
  1504. :custom-scroll-parent (get-scroll-parent config)
  1505. :skipAnimationFrameInResizeObserver true
  1506. :compute-item-key (fn [idx]
  1507. (str (:db/id view-entity) "-card-" idx))
  1508. :item-content (fn [idx]
  1509. (lazy-item (:data table) idx {}
  1510. (fn [block]
  1511. (gallery-card-item view-entity block config'))))}))]))
  1512. (defn- run-effects!
  1513. [option {:keys [data]} *scroller-ref gallery?]
  1514. (hooks/use-effect!
  1515. (fn []
  1516. (when (and (:current-page? (:config option)) (seq data) (map? (first data)) (:block/uuid (first data)))
  1517. (ui-handler/scroll-to-anchor-block @*scroller-ref data gallery?)
  1518. (state/set-state! :editor/virtualized-scroll-fn #(ui-handler/scroll-to-anchor-block @*scroller-ref data gallery?))))
  1519. []))
  1520. (rum/defc view-sorting-item
  1521. [table sorting id name asc? set-sorting!]
  1522. [:div.flex.flex-row.gap-2.items-center.justify-between.px-2
  1523. [:div:div.flex.flex-row.gap-1.items-center
  1524. (shui/button
  1525. {:size :sm
  1526. :class "!px-1"
  1527. :variant :ghost
  1528. :title "Drag && Drop to reorder"}
  1529. (shui/tabler-icon "grip-vertical" {:size 14}))
  1530. [:div.text-muted-foreground.whitespace-nowrap (str name ":")]]
  1531. [:div.flex.flex-row.gap-2.items-center
  1532. (shui/select
  1533. {:default-value (if asc? "asc" "desc")
  1534. :on-value-change (fn [v]
  1535. (let [asc? (= v "asc")
  1536. f (:column-set-sorting! table)]
  1537. (when f
  1538. (f sorting {:id id} asc?))))}
  1539. (shui/select-trigger
  1540. {:class "order-button !px-2 !py-0 !h-8"}
  1541. (shui/select-value
  1542. {:placeholder "Select order"}))
  1543. (shui/select-content
  1544. (shui/select-group
  1545. (shui/select-item {:value "asc"} "Ascending")
  1546. (shui/select-item {:value "desc"} "Descending"))))
  1547. (shui/button
  1548. {:variant "ghost"
  1549. :class "text-muted-foreground !px-1"
  1550. :size :sm
  1551. :on-click (fn []
  1552. (let [f (:column-set-sorting! table)
  1553. new-sorting (f sorting {:id id} nil)
  1554. f (get-in table [:data-fns :set-sorting!])]
  1555. (set-sorting! new-sorting)
  1556. (f new-sorting)
  1557. (when (empty? new-sorting)
  1558. (shui/popup-hide!))))}
  1559. (ui/icon "x"))]])
  1560. (rum/defc view-sorting-config
  1561. [table sorting columns]
  1562. (let [[sorting set-sorting!] (rum/use-state sorting)]
  1563. [:div.ls-view-order-setting.flex.flex-col.gap-2.py-2.text-sm
  1564. (let [items (for [{:keys [id asc?]} sorting]
  1565. (when-let [name (some (fn [column] (when (= id (:id column))
  1566. (:name column))) columns)]
  1567. {:id (str id)
  1568. :value id
  1569. :content (view-sorting-item table sorting id name asc? set-sorting!)}))]
  1570. (dnd/items items
  1571. {:on-drag-end (fn [ordered-columns]
  1572. (let [f (get-in table [:data-fns :set-sorting!])
  1573. new-sorting (mapv (fn [column] (some #(when (= column (:id %)) %) sorting)) ordered-columns)]
  1574. (set-sorting! new-sorting)
  1575. (f new-sorting)))}))
  1576. (shui/dropdown-menu-item
  1577. {:class "text-muted-foreground pl-3"
  1578. :on-click (fn []
  1579. (let [f (get-in table [:data-fns :set-sorting!])]
  1580. (set-sorting! nil)
  1581. (f nil)
  1582. (shui/popup-hide!)))}
  1583. (ui/icon "trash" {:size 15})
  1584. [:span.ml-1 "Delete sort"])]))
  1585. (rum/defc view-sorting
  1586. [table columns sorting]
  1587. (shui/button
  1588. {:variant "ghost"
  1589. :class "text-muted-foreground !px-1"
  1590. :size :sm
  1591. :on-click (fn [e]
  1592. (shui/popup-show! (.-target e)
  1593. (fn [] (view-sorting-config table sorting columns))
  1594. {:align :end}))}
  1595. (ui/icon "arrows-up-down")))
  1596. (defn- view-cp
  1597. [view-entity table option* {:keys [*scroller-ref display-type row-selection]}]
  1598. (let [option (assoc option* :view-entity view-entity)]
  1599. (case display-type
  1600. :logseq.property.view/type.list
  1601. (list-view option view-entity table *scroller-ref)
  1602. :logseq.property.view/type.gallery
  1603. (gallery-view option table view-entity (:rows table) *scroller-ref)
  1604. (table-view table option row-selection *scroller-ref))))
  1605. (defn- get-views
  1606. [ent view-feature-type]
  1607. (let [entity (db/entity (:db/id ent))
  1608. views (->> (:logseq.property/_view-for entity)
  1609. (filter (fn [view]
  1610. (= view-feature-type (:logseq.property.view/feature-type view)))))]
  1611. (ldb/sort-by-order views)))
  1612. (defn- create-view!
  1613. [view-parent view-feature-type]
  1614. (when-let [page (db/get-case-page common-config/views-page-name)]
  1615. (p/let [properties (cond->
  1616. {:logseq.property/view-for (:db/id view-parent)
  1617. :logseq.property.view/feature-type view-feature-type}
  1618. (contains? #{:linked-references :unlinked-references} view-feature-type)
  1619. (assoc :logseq.property.view/type (:db/id (db/entity :logseq.property.view/type.list))
  1620. :logseq.property.view/group-by-property (:db/id (db/entity :block/page))))
  1621. view-exists? (seq (get-views view-parent view-feature-type))
  1622. view-title (if view-exists?
  1623. ""
  1624. (case view-feature-type
  1625. :linked-references
  1626. "Linked references"
  1627. :unlinked-references
  1628. "Unlinked references"
  1629. :class-objects
  1630. "All"
  1631. :property-objects
  1632. "All"
  1633. :all-pages
  1634. "All pages"
  1635. ""))
  1636. result (editor-handler/api-insert-new-block! view-title
  1637. {:page (:block/uuid page)
  1638. :properties properties
  1639. :edit-block? false})]
  1640. (db/entity [:block/uuid (:block/uuid result)]))))
  1641. (rum/defc views-tab < rum/reactive db-mixins/query
  1642. [view-parent current-view {:keys [views data items-count set-view-entity! set-data! set-views! view-feature-type show-items-count? references? opacity]}]
  1643. [:div.views
  1644. (for [view* views]
  1645. (let [view (db/sub-block (:db/id view*))
  1646. current-view? (= (:db/id current-view) (:db/id view))]
  1647. (shui/button
  1648. {:variant :text
  1649. :size :sm
  1650. :class (str "text-sm px-0 py-0 h-6 " (when-not current-view? "text-muted-foreground"))
  1651. :on-click (fn [e]
  1652. (if (and current-view? (not= (:db/id view) (:db/id view-parent)))
  1653. (shui/popup-show!
  1654. (.-target e)
  1655. (fn []
  1656. [:<>
  1657. (shui/dropdown-menu-sub
  1658. (shui/dropdown-menu-sub-trigger
  1659. "Rename")
  1660. (shui/dropdown-menu-sub-content
  1661. (when-let [block-container-cp (state/get-component :block/container)]
  1662. (block-container-cp {} view))))
  1663. (shui/dropdown-menu-item
  1664. {:key "Delete"
  1665. :on-click (fn []
  1666. (p/do!
  1667. (editor-handler/delete-block-aux! view)
  1668. (let [views' (remove (fn [v] (= (:db/id v) (:db/id view))) views)]
  1669. (set-views! views')
  1670. (set-view-entity! (first views'))
  1671. (shui/popup-hide!))))}
  1672. "Delete")])
  1673. {:as-dropdown? true
  1674. :align "start"
  1675. :content-props {:onClick shui/popup-hide!}})
  1676. (do
  1677. (set-view-entity! view)
  1678. (set-data! nil))))}
  1679. (when-not references?
  1680. (let [display-type (or (:db/ident (get view :logseq.property.view/type))
  1681. :logseq.property.view/type.table)]
  1682. (when-let [icon (:logseq.property/icon (db/entity display-type))]
  1683. (icon-component/icon icon {:color? true
  1684. :size 15}))))
  1685. (let [title (:block/title view)]
  1686. (if (= title "")
  1687. "New view"
  1688. title))
  1689. (when (and current-view? show-items-count? (> items-count 0) (seq data))
  1690. [:span.text-muted-foreground.text-xs
  1691. items-count]))))
  1692. (shui/button
  1693. {:variant :text
  1694. :size :sm
  1695. :title "Add new view"
  1696. :class (str "!px-1 -ml-1 text-muted-foreground hover:text-foreground transition-opacity ease-in duration-300 " opacity)
  1697. :on-click (fn []
  1698. (p/let [view (create-view! view-parent view-feature-type)]
  1699. (set-views! (concat views [view]))))}
  1700. (ui/icon "plus" {:size 15}))])
  1701. (rum/defc view-head < rum/static
  1702. [view-parent view-entity table columns input sorting
  1703. set-input! add-new-object!
  1704. {:keys [view-feature-type title-key additional-actions]
  1705. :as option}]
  1706. (let [[hover? set-hover?] (hooks/use-state nil)
  1707. db-based? (config/db-based-graph? (state/get-current-repo))
  1708. references? (contains? #{:linked-references :unlinked-references} view-feature-type)
  1709. opacity (cond
  1710. (and references? (not hover?)) "opacity-0"
  1711. hover? "opacity-100"
  1712. :else "opacity-75")]
  1713. [:div.flex.flex-1.flex-nowrap.items-center.justify-between.gap-1.overflow-hidden
  1714. {:on-mouse-over #(set-hover? true)
  1715. :on-mouse-out #(set-hover? false)}
  1716. [:div.flex.flex-row.items-center.gap-2
  1717. (if db-based?
  1718. (if (= view-feature-type :query-result)
  1719. [:div.font-medium.opacity-50.text-sm
  1720. (t (or title-key :views.table/default-title)
  1721. (count (:rows table)))]
  1722. (views-tab view-parent view-entity (assoc option
  1723. :hover? hover?
  1724. :opacity opacity
  1725. :references? references?)))
  1726. [:div.font-medium.text-sm
  1727. [:span
  1728. (case view-feature-type
  1729. :all-pages "All pages"
  1730. :linked-references "Linked references"
  1731. :unlinked-references "Unlinked references"
  1732. "Nodes")]
  1733. [:span.ml-1 (count (:rows table))]])]
  1734. [:div.view-actions.flex.items-center.gap-1.transition-opacity.ease-in.duration-300
  1735. {:class opacity}
  1736. (when (seq additional-actions)
  1737. [:<> (for [action additional-actions]
  1738. (if (fn? action)
  1739. (action option)
  1740. action))])
  1741. (when (and db-based? (seq sorting))
  1742. (view-sorting table columns sorting))
  1743. (when db-based? (filter-properties view-entity columns table option))
  1744. (search input {:on-change set-input!
  1745. :set-input! set-input!})
  1746. (when db-based?
  1747. [:div.text-muted-foreground.text-sm
  1748. (pv/property-value view-entity (db/entity :logseq.property.view/type) {})])
  1749. (when db-based? (more-actions view-entity columns table option))
  1750. (when (and db-based? add-new-object!) (new-record-button table view-entity))]]))
  1751. (rum/defc ^:large-vars/cleanup-todo view-inner < rum/static
  1752. [view-entity {:keys [view-parent data full-data set-data! columns add-new-object! foldable-options input set-input! sorting set-sorting! filters set-filters! display-type group-by-property-ident] :as option*}
  1753. *scroller-ref]
  1754. (let [db-based? (config/db-based-graph?)
  1755. option (assoc option* :properties
  1756. (-> (remove #{:id :select} (map :id columns))
  1757. (conj :block/uuid :block/name)
  1758. vec))
  1759. default-visible-columns (if-let [hidden-columns (conj (:logseq.property.table/hidden-columns view-entity) :id)]
  1760. (zipmap hidden-columns (repeat false))
  1761. ;; This case can happen for imported tables
  1762. (if (seq (:logseq.property.table/ordered-columns view-entity))
  1763. (zipmap (set/difference (set (map :id columns))
  1764. (set (:logseq.property.table/ordered-columns view-entity))
  1765. #{:select :block/created-at :block/updated-at})
  1766. (repeat false))
  1767. {}))
  1768. [visible-columns set-visible-columns!] (rum/use-state default-visible-columns)
  1769. ordered-columns (vec (concat [:select] (:logseq.property.table/ordered-columns view-entity)))
  1770. sized-columns (:logseq.property.table/sized-columns view-entity)
  1771. [ordered-columns set-ordered-columns!] (rum/use-state ordered-columns)
  1772. [sized-columns set-sized-columns!] (rum/use-state sized-columns)
  1773. {:keys [set-sorting! set-filters! set-visible-columns! set-ordered-columns! set-sized-columns!]}
  1774. (db-set-table-state! view-entity {:set-sorting! set-sorting!
  1775. :set-filters! set-filters!
  1776. :set-visible-columns! set-visible-columns!
  1777. :set-sized-columns! set-sized-columns!
  1778. :set-ordered-columns! set-ordered-columns!})
  1779. [row-selection set-row-selection!] (rum/use-state {})
  1780. [last-selected-idx set-last-selected-idx!] (rum/use-state nil)
  1781. columns (sort-columns columns ordered-columns)
  1782. select? (first (filter (fn [item] (= (:id item) :select)) columns))
  1783. id? (first (filter (fn [item] (= (:id item) :id)) columns))
  1784. pinned-properties (set (cond->> (map :db/ident (:logseq.property.table/pinned-columns view-entity))
  1785. id?
  1786. (cons :id)
  1787. select?
  1788. (cons :select)))
  1789. {pinned true unpinned false} (group-by (fn [item]
  1790. (contains? pinned-properties (:id item)))
  1791. (remove (fn [column]
  1792. (false? (get visible-columns (:id column))))
  1793. columns))
  1794. group-by-property (or (:logseq.property.view/group-by-property view-entity)
  1795. (db/entity group-by-property-ident))
  1796. table-map {:view-entity view-entity
  1797. :data data
  1798. :full-data full-data
  1799. :columns columns
  1800. :state {:sorting sorting
  1801. :filters filters
  1802. :row-selection row-selection
  1803. :visible-columns visible-columns
  1804. :sized-columns sized-columns
  1805. :ordered-columns ordered-columns
  1806. :pinned-columns pinned
  1807. :unpinned-columns unpinned
  1808. :group-by-property group-by-property
  1809. :last-selected-idx last-selected-idx}
  1810. :data-fns {:set-data! set-data!
  1811. :set-filters! set-filters!
  1812. :set-sorting! set-sorting!
  1813. :set-visible-columns! set-visible-columns!
  1814. :set-ordered-columns! set-ordered-columns!
  1815. :set-sized-columns! set-sized-columns!
  1816. :set-row-selection! set-row-selection!
  1817. :add-new-object! add-new-object!
  1818. :set-last-selected-idx! set-last-selected-idx!}}
  1819. table (shui/table-option table-map)
  1820. *view-ref (rum/use-ref nil)
  1821. gallery? (= display-type :logseq.property.view/type.gallery)
  1822. list-view? (= display-type :logseq.property.view/type.list)]
  1823. (run-effects! option table-map *scroller-ref gallery?)
  1824. [:div.flex.flex-col.gap-2.grid
  1825. {:ref *view-ref}
  1826. (ui/foldable
  1827. (view-head view-parent view-entity table columns input sorting set-input! add-new-object! option)
  1828. [:div.ls-view-body.flex.flex-col.gap-2.grid.mt-1
  1829. (filters-row view-entity table option)
  1830. (let [view-opts {:*scroller-ref *scroller-ref
  1831. :display-type display-type
  1832. :row-selection row-selection
  1833. :add-new-object! add-new-object!}]
  1834. (if (and group-by-property-ident (not (number? (first (:rows table)))))
  1835. (when (seq (:rows table))
  1836. [:div.flex.flex-col.border-t.pt-2.gap-2
  1837. (map-indexed
  1838. (fn [idx [value group]]
  1839. (let [add-new-object! (when (fn? add-new-object!)
  1840. (fn [_]
  1841. (add-new-object! view-entity table
  1842. {:properties {(:db/ident group-by-property) (or (and (map? value) (:db/id value)) value)}})))
  1843. table' (shui/table-option (-> table-map
  1844. (assoc-in [:data-fns :add-new-object!] add-new-object!)
  1845. (assoc :data group ; data for this group
  1846. )))
  1847. readable-property-value #(cond (and (map? %) (or (:block/title %) (:logseq.property/value %)))
  1848. (db-property/property-value-content %)
  1849. (= (:db/ident %) :logseq.property/empty-placeholder)
  1850. "Empty"
  1851. :else
  1852. (str %))
  1853. group-by-page? (or (= :block/page group-by-property-ident)
  1854. (and (not db-based?) (contains? #{:linked-references :unlinked-references} display-type)))]
  1855. (rum/with-key
  1856. (ui/foldable
  1857. [:div
  1858. {:class (when-not list-view? "my-4")}
  1859. (cond
  1860. group-by-page?
  1861. (if value
  1862. (let [c (state/get-component :block/page-cp)]
  1863. (c {:disable-preview? true} value))
  1864. [:div.text-muted-foreground.text-sm
  1865. "Pages"])
  1866. (some? value)
  1867. (let [icon (pu/get-block-property-value value :logseq.property/icon)]
  1868. [:div.flex.flex-row.gap-1.items-center
  1869. (when icon (icon-component/icon icon {:color? true}))
  1870. (readable-property-value value)])
  1871. :else
  1872. (str "No " (:block/title group-by-property)))]
  1873. (let [render (view-cp view-entity (assoc table' :rows group) option view-opts)]
  1874. (if list-view? [:div.-ml-2 render] render))
  1875. {:title-trigger? false})
  1876. (str (:db/id view-entity) "-group-idx-" idx))))
  1877. (:rows table))])
  1878. (view-cp view-entity table
  1879. (assoc option :group-by-property-ident group-by-property-ident)
  1880. view-opts)))]
  1881. (merge {:title-trigger? false} foldable-options))]))
  1882. (rum/defcs view-container
  1883. "Provides a view for data like query results and tagged objects, multiple
  1884. layouts such as table and list are supported. Args:
  1885. * view-entity: a db Entity
  1886. * option:
  1887. * title-key: dict key defaults to `:views.table/default-title`
  1888. * data: a collections of entities
  1889. * set-data!: `fn` to update `data`
  1890. * columns: view columns including properties and db attributes, which could be built by `build-columns`
  1891. * add-new-object!: `fn` to create a new object (or row)
  1892. * show-add-property?: whether to show `Add property`
  1893. * add-property!: `fn` to add a new property (or column)"
  1894. < (rum/local nil ::scroller-ref)
  1895. [state view-entity option]
  1896. (rum/with-key (view-inner view-entity
  1897. (cond-> option
  1898. (or config/publishing? (:logseq.property.view/group-by-property view-entity))
  1899. (dissoc :add-new-object!))
  1900. (::scroller-ref state))
  1901. (str "view-" (:db/id view-entity))))
  1902. (defn <load-view-data
  1903. [view opts]
  1904. (state/<invoke-db-worker :thread-api/get-view-data (state/get-current-repo) (:db/id view) opts))
  1905. (rum/defc view-aux
  1906. [view-entity {:keys [view-parent view-feature-type data query-entity-ids set-view-entity!] :as option}]
  1907. (let [[input set-input!] (hooks/use-state "")
  1908. db-based? (config/db-based-graph?)
  1909. group-by-property (:logseq.property.view/group-by-property view-entity)
  1910. display-type (if (config/db-based-graph?)
  1911. (or (:db/ident (get view-entity :logseq.property.view/type))
  1912. (when (= (:view-type option) :linked-references)
  1913. :logseq.property.view/type.list)
  1914. :logseq.property.view/type.table)
  1915. (if (= view-feature-type :all-pages)
  1916. :logseq.property.view/type.table
  1917. :logseq.property.view/type.list))
  1918. list-view? (= display-type :logseq.property.view/type.list)
  1919. group-by-property-ident (or (:db/ident group-by-property)
  1920. (when (and list-view? (nil? group-by-property))
  1921. :block/page)
  1922. (when (and (not db-based?) (contains? #{:linked-references :unlinked-references} view-feature-type))
  1923. :block/page))
  1924. sorting* (:logseq.property.table/sorting view-entity)
  1925. sorting (if (or (= sorting* :logseq.property/empty-placeholder) (empty? sorting*))
  1926. [{:id :block/updated-at, :asc? false}]
  1927. sorting*)
  1928. [sorting set-sorting!] (rum/use-state sorting)
  1929. view-filters (:logseq.property.table/filters view-entity)
  1930. [filters set-filters!] (rum/use-state (or view-filters {}))
  1931. query? (= view-feature-type :query-result)
  1932. [loading? set-loading!] (hooks/use-state (not query?))
  1933. [data set-data!] (hooks/use-state data)
  1934. [ref-pages-count set-ref-pages-count!] (hooks/use-state nil)
  1935. load-view-data (fn load-view-data []
  1936. (c.m/run-task*
  1937. (m/sp
  1938. (let [need-query? (and query? (seq query-entity-ids) (or sorting filters (not (string/blank? input))))]
  1939. (cond
  1940. (and query? (empty? query-entity-ids))
  1941. (set-data! nil)
  1942. (and query? (not (or sorting filters)) (string/blank? input))
  1943. (set-data! query-entity-ids)
  1944. :else
  1945. (when (or (not query?) need-query?)
  1946. (try
  1947. (let [{:keys [data ref-pages-count]}
  1948. (c.m/<?
  1949. (<load-view-data view-entity
  1950. (cond->
  1951. {:view-for-id (or (:db/id (:logseq.property/view-for view-entity))
  1952. (:db/id view-parent))
  1953. :view-feature-type view-feature-type
  1954. :group-by-property-ident group-by-property-ident
  1955. :input input
  1956. :filters filters
  1957. :sorting sorting}
  1958. query?
  1959. (assoc :query-entity-ids query-entity-ids))))]
  1960. (set-data! data)
  1961. (when ref-pages-count
  1962. (set-ref-pages-count! ref-pages-count)))
  1963. (finally
  1964. (set-loading! false)))))))))]
  1965. (let [sorting-filters {:sorting sorting
  1966. :filters filters}]
  1967. (hooks/use-effect!
  1968. load-view-data
  1969. [(:db/id view-entity)
  1970. (hooks/use-debounced-value input 300)
  1971. sorting-filters
  1972. group-by-property-ident
  1973. (:db/id (:logseq.property.view/type view-entity))
  1974. ;; page filters
  1975. (:logseq.property.linked-references/includes view-parent)
  1976. (:logseq.property.linked-references/excludes view-parent)
  1977. (:filters view-parent)
  1978. query-entity-ids
  1979. (:data-changes-version option)]))
  1980. (if loading?
  1981. [:div.flex.flex-col.space-2.gap-2.my-2
  1982. (repeat 3 (shui/skeleton {:class "h-6 w-full"}))]
  1983. [:div.flex.flex-col.gap-2
  1984. (view-container view-entity (assoc option
  1985. :data data
  1986. :full-data data
  1987. :filters filters
  1988. :sorting sorting
  1989. :set-filters! set-filters!
  1990. :set-sorting! set-sorting!
  1991. :set-data! set-data!
  1992. :set-input! set-input!
  1993. :input input
  1994. :items-count (if (every? number? data)
  1995. (count data)
  1996. ;; grouped
  1997. (reduce (fn [total [_ col]]
  1998. (+ total (count col))) 0 data))
  1999. :group-by-property-ident group-by-property-ident
  2000. :ref-pages-count ref-pages-count
  2001. :display-type display-type
  2002. :load-view-data load-view-data
  2003. :set-view-entity! set-view-entity!))])))
  2004. (defn sub-view-data-changes
  2005. [view-parent view-feature-type]
  2006. (when view-parent
  2007. (when-let [repo (state/get-current-repo)]
  2008. (when-let [k (case view-feature-type
  2009. :class-objects :frontend.worker.react/objects
  2010. :linked-references :frontend.worker.react/refs
  2011. nil)]
  2012. (let [*version (atom 0)]
  2013. (react/q repo [k (:db/id view-parent)]
  2014. {:query-fn (fn [_] (swap! *version inc))}
  2015. nil))))))
  2016. (rum/defc sub-view < rum/reactive db-mixins/query
  2017. [view-entity option]
  2018. (let [view (or (some-> (:db/id view-entity) db/sub-block) view-entity)
  2019. data-changes-version (some-> (sub-view-data-changes (:view-parent option) (:view-feature-type option)) rum/react)]
  2020. (view-aux view (assoc option :data-changes-version data-changes-version))))
  2021. (rum/defc view < rum/static
  2022. [{:keys [view-parent view-feature-type view-entity] :as option}]
  2023. (let [[views set-views!] (hooks/use-state nil)
  2024. [view-entity set-view-entity!] (hooks/use-state view-entity)
  2025. query? (= view-feature-type :query-result)
  2026. db-based? (config/db-based-graph?)]
  2027. (hooks/use-effect!
  2028. #(c.m/run-task*
  2029. (m/sp
  2030. (when-not query?
  2031. (let [repo (state/get-current-repo)]
  2032. (when (and db-based? (not view-entity))
  2033. (c.m/<? (db-async/<get-views repo (:db/id view-parent) view-feature-type))
  2034. (let [views (get-views view-parent view-feature-type)]
  2035. (if-let [v (first views)]
  2036. (do
  2037. (set-views! views)
  2038. (when-not view-entity (set-view-entity! v)))
  2039. (when (and view-parent view-feature-type (not view-entity))
  2040. (let [new-view (c.m/<? (create-view! view-parent view-feature-type))]
  2041. (set-views! (concat views [new-view]))
  2042. (set-view-entity! new-view))))))))))
  2043. [])
  2044. (when (if db-based? view-entity (or view-entity view-parent
  2045. (= view-feature-type :all-pages)))
  2046. (let [option' (assoc option
  2047. :view-feature-type (or view-feature-type
  2048. (:logseq.property.view/feature-type view-entity))
  2049. :views views
  2050. :set-views! set-views!
  2051. :set-view-entity! set-view-entity!)]
  2052. (rum/with-key
  2053. (sub-view view-entity option')
  2054. (str "view-" (:db/id view-entity)))))))