query_table.cljs 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. (ns frontend.components.query-table
  2. (:require [frontend.components.svg :as svg]
  3. [frontend.config :as config]
  4. [frontend.date :as date]
  5. [frontend.db :as db]
  6. [frontend.db.query-dsl :as query-dsl]
  7. [frontend.db.utils :as db-utils]
  8. [frontend.format.block :as block]
  9. [frontend.handler.common :as common-handler]
  10. [frontend.handler.property :as property-handler]
  11. [frontend.state :as state]
  12. [frontend.util :as util]
  13. [frontend.util.clock :as clock]
  14. [frontend.handler.file-based.property :as file-property-handler]
  15. [medley.core :as medley]
  16. [rum.core :as rum]
  17. [promesa.core :as p]
  18. [logseq.graph-parser.text :as text]
  19. [logseq.db.frontend.property :as db-property]
  20. [frontend.handler.property.util :as pu]
  21. [logseq.db.frontend.content :as db-content]
  22. [frontend.handler.db-based.property.util :as db-pu]))
  23. ;; Util fns
  24. ;; ========
  25. (defn- attach-clock-property
  26. [result]
  27. ;; FIXME: Look up by property id if still supporting clock-time
  28. (let [ks [:block/properties :clock-time]
  29. result (map (fn [b]
  30. (let [b (block/parse-title-and-body b)]
  31. (assoc-in b ks (or (clock/clock-summary (:block/body b) false) 0))))
  32. result)]
  33. (if (every? #(zero? (get-in % ks)) result)
  34. (map #(medley/dissoc-in % ks) result)
  35. result)))
  36. (defn- sort-by-fn [sort-by-column item {:keys [page? db-graph?]}]
  37. (case sort-by-column
  38. :created-at
  39. (:block/created-at item)
  40. :updated-at
  41. (:block/updated-at item)
  42. :block
  43. (:block/content item)
  44. :page
  45. (if page? (:block/name item) (get-in item [:block/page :block/name]))
  46. (if db-graph?
  47. (get-in item [:block/properties-by-name sort-by-column])
  48. (get-in item [:block/properties sort-by-column]))))
  49. (defn- locale-compare
  50. "Use locale specific comparison for strings and general comparison for others."
  51. [x y]
  52. (if (and (number? x) (number? y))
  53. (< x y)
  54. (.localeCompare (str x) (str y) (state/sub :preferred-language) #js {:numeric true})))
  55. (defn- sort-result [result {:keys [sort-by-column sort-desc? sort-nlp-date? page? db-graph?]}]
  56. (if (some? sort-by-column)
  57. (let [comp-fn (if sort-desc? #(locale-compare %2 %1) locale-compare)
  58. repo (state/get-current-repo)
  59. result' (if db-graph?
  60. (mapv (fn [block]
  61. (assoc block :block/properties-by-name (db-pu/properties-by-name repo block)))
  62. result)
  63. result)
  64. sort-by-column' (if (and db-graph? (qualified-keyword? sort-by-column))
  65. (:block/original-name (db-utils/entity repo sort-by-column))
  66. sort-by-column)]
  67. (sort-by (fn [item]
  68. (block/normalize-block (sort-by-fn sort-by-column' item {:page? page? :db-graph? db-graph?})
  69. sort-nlp-date?))
  70. comp-fn
  71. result'))
  72. result))
  73. (defn- get-sort-state
  74. "Return current sort direction and column being sorted, respectively
  75. :sort-desc? and :sort-by-column. :sort-by-column is nil if no sorting is to be
  76. done"
  77. [current-block {:keys [db-graph?]}]
  78. (let [properties (:block/properties current-block)
  79. p-desc? (pu/lookup properties :logseq.property/query-sort-desc)
  80. desc? (if (some? p-desc?) p-desc? true)
  81. properties (:block/properties current-block)
  82. query-sort-by (pu/lookup properties :logseq.property/query-sort-by)
  83. ;; Starting with #6105, we started putting properties under namespaces.
  84. nlp-date? (and (not db-graph?) (:logseq.query/nlp-date properties))
  85. sort-by-column (or (keyword query-sort-by)
  86. (if (query-dsl/query-contains-filter? (:block/content current-block) "sort-by")
  87. nil
  88. :updated-at))]
  89. {:sort-desc? desc?
  90. :sort-by-column sort-by-column
  91. :sort-nlp-date? nlp-date?}))
  92. ;; Components
  93. ;; ==========
  94. (rum/defc sortable-title
  95. [title column {:keys [sort-by-column sort-desc?]} block-id {:keys [db-graph?]}]
  96. (let [repo (state/get-current-repo)]
  97. [:th.whitespace-nowrap
  98. [:a {:on-click (fn []
  99. (p/do!
  100. (property-handler/set-block-property! repo block-id
  101. (pu/get-pid :logseq.property/query-sort-by)
  102. (if db-graph? column (name column)))
  103. (property-handler/set-block-property! repo block-id
  104. (pu/get-pid :logseq.property/query-sort-desc)
  105. (not sort-desc?))))}
  106. [:div.flex.items-center
  107. [:span.mr-1 title]
  108. (when (= sort-by-column column)
  109. [:span
  110. (if sort-desc? (svg/caret-down) (svg/caret-up))])]]]))
  111. (defn get-all-columns-for-result
  112. "Gets all possible columns for a given result. For a db graph, this is a mix
  113. of property db-idents and special keywords like :page. For a file graph, these are
  114. all property names as keywords"
  115. [result page?]
  116. (let [repo (state/get-current-repo)
  117. db-graph? (config/db-based-graph? repo)
  118. hidden-properties (if db-graph?
  119. ;; TODO: Support additional hidden properties e.g. from user config
  120. ;; or gp-property/built-in-extended properties
  121. (set (keys db-property/built-in-properties))
  122. (conj (file-property-handler/built-in-properties) :template))
  123. properties-fn (if db-graph? db-property/properties :block/properties)
  124. prop-keys* (->> (distinct (mapcat keys (map properties-fn result)))
  125. (remove hidden-properties))
  126. prop-keys (cond-> (if page? (cons :page prop-keys*) (concat '(:block :page) prop-keys*))
  127. (or db-graph? page?)
  128. (concat [:created-at :updated-at]))]
  129. prop-keys))
  130. (defn get-columns [current-block result {:keys [page?]}]
  131. (let [properties (:block/properties current-block)
  132. query-properties (pu/lookup properties :logseq.property/query-properties)
  133. query-properties (if (config/db-based-graph? (state/get-current-repo))
  134. query-properties
  135. (some-> query-properties
  136. (common-handler/safe-read-string "Parsing query properties failed")))
  137. query-properties (if page? (remove #{:block} query-properties) query-properties)
  138. columns (if (seq query-properties)
  139. query-properties
  140. (get-all-columns-for-result result page?))]
  141. (distinct columns)))
  142. (defn- build-column-value
  143. "Builds a column's tuple value for a query table given a row, column and
  144. options"
  145. [db row column {:keys [db-graph? page? ->elem map-inline comma-separated-property?]}]
  146. (case column
  147. :page
  148. [:string (if page?
  149. (or (:block/original-name row)
  150. (:block/name row))
  151. (or (get-in row [:block/page :block/original-name])
  152. (get-in row [:block/page :block/name])))]
  153. :block ; block title
  154. (let [content (:block/content row)
  155. uuid (:block/uuid row)
  156. {:block/keys [title]} (block/parse-title-and-body
  157. (:block/uuid row)
  158. (:block/format row)
  159. (:block/pre-block? row)
  160. content)]
  161. (if (seq title)
  162. [:element (->elem :div (map-inline {:block/uuid uuid} title))]
  163. [:string content]))
  164. :created-at
  165. [:string (when-let [created-at (:block/created-at row)]
  166. (date/int->local-time-2 created-at))]
  167. :updated-at
  168. [:string (when-let [updated-at (:block/updated-at row)]
  169. (date/int->local-time-2 updated-at))]
  170. [:string
  171. (if db-graph?
  172. (db-property/get-property-value-names-from-ref db (get row column))
  173. (if comma-separated-property?
  174. ;; Return original properties since comma properties need to
  175. ;; return collections for display purposes
  176. (get-in row [:block/properties column])
  177. (or (get-in row [:block/properties-text-values column])
  178. ;; Fallback to original properties for page blocks
  179. (get-in row [:block/properties column]))))]))
  180. (defn- render-column-value
  181. [{:keys [row-block row-format cell-format value]} page-cp inline-text {:keys [db-graph?]}]
  182. (cond
  183. ;; elements should be rendered as they are provided
  184. (= :element cell-format) value
  185. (coll? value) (if db-graph?
  186. (->> value
  187. (map #(if-let [page (db/get-page %)]
  188. (page-cp {} page)
  189. (inline-text row-block row-format %)))
  190. (interpose [:span ", "]))
  191. (->> (map #(page-cp {} {:block/name %}) value)
  192. (interpose [:span ", "])))
  193. ;; boolean values need to first be stringified
  194. (boolean? value) (str value)
  195. ;; string values will attempt to be rendered as pages, falling back to
  196. ;; inline-text when no page entity is found
  197. (string? value) (if-let [page (db/get-page value)]
  198. (page-cp {} page)
  199. (inline-text row-block row-format value))
  200. ;; anything else should just be rendered as provided
  201. :else value))
  202. (rum/defcs result-table-v1 < rum/reactive
  203. (rum/local false ::select?)
  204. (rum/local false ::mouse-down?)
  205. [state config current-block sort-result sort-state columns {:keys [page?]} map-inline page-cp ->elem inline-text]
  206. (let [select? (get state ::select?)
  207. *mouse-down? (::mouse-down? state)
  208. clock-time-total (when-not page?
  209. (->> (map #(get-in % [:block/properties :clock-time] 0) sort-result)
  210. (apply +)))
  211. property-separated-by-commas? (partial text/separated-by-commas? (state/get-config))
  212. db-graph? (config/db-based-graph? (state/get-current-repo))
  213. db (db/get-db (state/get-current-repo))]
  214. [:div.overflow-x-auto {:on-pointer-down (fn [e] (.stopPropagation e))
  215. :style {:width "100%"}
  216. :class (when-not page? "query-table")}
  217. [:table.table-auto
  218. [:thead
  219. [:tr.cursor
  220. (for [column columns]
  221. (let [title (if (and (= column :clock-time) (integer? clock-time-total))
  222. (util/format "clock-time(total: %s)" (clock/seconds->days:hours:minutes:seconds
  223. clock-time-total))
  224. (name column))]
  225. (sortable-title title column sort-state (:block/uuid current-block) {:db-graph? db-graph?})))]]
  226. [:tbody
  227. (for [row sort-result]
  228. (let [format (:block/format row)]
  229. [:tr.cursor
  230. (for [column columns]
  231. (let [[cell-format value] (build-column-value db
  232. row
  233. column
  234. {:page? page?
  235. :->elem ->elem
  236. :map-inline map-inline
  237. :config config
  238. :db-graph? db-graph?
  239. :comma-separated-property? (property-separated-by-commas? column)})]
  240. [:td.whitespace-nowrap {:on-pointer-down (fn []
  241. (reset! *mouse-down? true)
  242. (reset! select? false))
  243. :on-mouse-move (fn [] (reset! select? true))
  244. :on-pointer-up (fn []
  245. (when (and @*mouse-down? (not @select?))
  246. (state/sidebar-add-block!
  247. (state/get-current-repo)
  248. (:db/id row)
  249. :block-ref)
  250. (reset! *mouse-down? false)))}
  251. (when value
  252. (render-column-value {:row-block row
  253. :row-format format
  254. :cell-format cell-format
  255. :value value}
  256. page-cp
  257. inline-text
  258. {:db-graph? db-graph?}))]))]))]]]))
  259. (rum/defc result-table < rum/reactive
  260. [config current-block result {:keys [page?] :as options} map-inline page-cp ->elem inline-text]
  261. (when current-block
  262. (let [db-graph? (config/db-based-graph? (state/get-current-repo))
  263. result' (cond-> (if page? result (attach-clock-property result))
  264. db-graph?
  265. ((fn [res]
  266. (map #(if (:block/content %)
  267. (update % :block/content
  268. db-content/special-id-ref->page-ref
  269. ;; Lookup here instead of initial query as advanced queries
  270. ;; won't usually have a ref's name
  271. (map (fn [m] (db/entity (:db/id m))) (:block/refs %)))
  272. %)
  273. res))))
  274. columns (get-columns current-block result' {:page? page?})
  275. ;; Sort state needs to be in sync between final result and sortable title
  276. ;; as user needs to know if there result is sorted
  277. sort-state (get-sort-state current-block {:db-graph? db-graph?})
  278. sort-result (sort-result result' (assoc sort-state :page? page? :db-graph? db-graph?))]
  279. (result-table-v1 config current-block sort-result sort-state columns options map-inline page-cp ->elem inline-text))))