mirror of
https://github.com/TheAnachronism/docspell.git
synced 2025-04-05 19:09:32 +00:00
Merge pull request #755 from eikek/fix-category-count
Fix category count
This commit is contained in:
commit
cdbd65388d
@ -1,4 +1,12 @@
|
|||||||
pull_request_rules:
|
pull_request_rules:
|
||||||
|
- name: assign and label scala-steward's PRs
|
||||||
|
conditions:
|
||||||
|
- author=scala-steward
|
||||||
|
actions:
|
||||||
|
assign:
|
||||||
|
users: [eikek]
|
||||||
|
label:
|
||||||
|
add: ["type: dependencies"]
|
||||||
- name: automatically merge Scala Steward PRs on CI success
|
- name: automatically merge Scala Steward PRs on CI success
|
||||||
conditions:
|
conditions:
|
||||||
- author=scala-steward
|
- author=scala-steward
|
||||||
|
@ -71,6 +71,9 @@ object OCollective {
|
|||||||
type TagCount = docspell.store.queries.TagCount
|
type TagCount = docspell.store.queries.TagCount
|
||||||
val TagCount = docspell.store.queries.TagCount
|
val TagCount = docspell.store.queries.TagCount
|
||||||
|
|
||||||
|
type CategoryCount = docspell.store.queries.CategoryCount
|
||||||
|
val CategoryCount = docspell.store.queries.CategoryCount
|
||||||
|
|
||||||
type InsightData = QCollective.InsightData
|
type InsightData = QCollective.InsightData
|
||||||
val insightData = QCollective.InsightData
|
val insightData = QCollective.InsightData
|
||||||
|
|
||||||
|
@ -4242,6 +4242,7 @@ components:
|
|||||||
required:
|
required:
|
||||||
- count
|
- count
|
||||||
- tagCloud
|
- tagCloud
|
||||||
|
- tagCategoryCloud
|
||||||
- fieldStats
|
- fieldStats
|
||||||
- folderStats
|
- folderStats
|
||||||
properties:
|
properties:
|
||||||
@ -4250,6 +4251,8 @@ components:
|
|||||||
format: int32
|
format: int32
|
||||||
tagCloud:
|
tagCloud:
|
||||||
$ref: "#/components/schemas/TagCloud"
|
$ref: "#/components/schemas/TagCloud"
|
||||||
|
tagCategoryCloud:
|
||||||
|
$ref: "#/components/schemas/NameCloud"
|
||||||
fieldStats:
|
fieldStats:
|
||||||
type: array
|
type: array
|
||||||
items:
|
items:
|
||||||
@ -4354,7 +4357,7 @@ components:
|
|||||||
$ref: "#/components/schemas/TagCount"
|
$ref: "#/components/schemas/TagCount"
|
||||||
TagCount:
|
TagCount:
|
||||||
description: |
|
description: |
|
||||||
Generic structure for counting something.
|
Structure for counting tags.
|
||||||
required:
|
required:
|
||||||
- tag
|
- tag
|
||||||
- count
|
- count
|
||||||
@ -4364,6 +4367,30 @@ components:
|
|||||||
count:
|
count:
|
||||||
type: integer
|
type: integer
|
||||||
format: int32
|
format: int32
|
||||||
|
|
||||||
|
NameCloud:
|
||||||
|
description: |
|
||||||
|
A set of counters.
|
||||||
|
required:
|
||||||
|
- items
|
||||||
|
properties:
|
||||||
|
items:
|
||||||
|
type: array
|
||||||
|
items:
|
||||||
|
$ref: "#/components/schemas/NameCount"
|
||||||
|
NameCount:
|
||||||
|
description: |
|
||||||
|
Generic structure for counting something.
|
||||||
|
required:
|
||||||
|
- name
|
||||||
|
- count
|
||||||
|
properties:
|
||||||
|
name:
|
||||||
|
type: string
|
||||||
|
count:
|
||||||
|
type: integer
|
||||||
|
format: int32
|
||||||
|
|
||||||
AttachmentMeta:
|
AttachmentMeta:
|
||||||
description: |
|
description: |
|
||||||
Extracted meta data of an attachment.
|
Extracted meta data of an attachment.
|
||||||
|
@ -31,6 +31,7 @@ trait Conversions {
|
|||||||
SearchStats(
|
SearchStats(
|
||||||
sum.count,
|
sum.count,
|
||||||
mkTagCloud(sum.tags),
|
mkTagCloud(sum.tags),
|
||||||
|
mkTagCategoryCloud(sum.cats),
|
||||||
sum.fields.map(mkFieldStats),
|
sum.fields.map(mkFieldStats),
|
||||||
sum.folders.map(mkFolderStats)
|
sum.folders.map(mkFolderStats)
|
||||||
)
|
)
|
||||||
@ -63,6 +64,9 @@ trait Conversions {
|
|||||||
def mkTagCloud(tags: List[OCollective.TagCount]) =
|
def mkTagCloud(tags: List[OCollective.TagCount]) =
|
||||||
TagCloud(tags.map(tc => TagCount(mkTag(tc.tag), tc.count)))
|
TagCloud(tags.map(tc => TagCount(mkTag(tc.tag), tc.count)))
|
||||||
|
|
||||||
|
def mkTagCategoryCloud(tags: List[OCollective.CategoryCount]) =
|
||||||
|
NameCloud(tags.map(tc => NameCount(tc.category, tc.count)))
|
||||||
|
|
||||||
// attachment meta
|
// attachment meta
|
||||||
def mkAttachmentMeta(rm: RAttachmentMeta): AttachmentMeta =
|
def mkAttachmentMeta(rm: RAttachmentMeta): AttachmentMeta =
|
||||||
AttachmentMeta(
|
AttachmentMeta(
|
||||||
|
@ -9,11 +9,11 @@ object DBFunction {
|
|||||||
val countAll: DBFunction = CountAll
|
val countAll: DBFunction = CountAll
|
||||||
|
|
||||||
def countAs[A](column: Column[A]): DBFunction =
|
def countAs[A](column: Column[A]): DBFunction =
|
||||||
Count(column)
|
Count(column, false)
|
||||||
|
|
||||||
case object CountAll extends DBFunction
|
case object CountAll extends DBFunction
|
||||||
|
|
||||||
case class Count(column: Column[_]) extends DBFunction
|
case class Count(column: Column[_], distinct: Boolean) extends DBFunction
|
||||||
|
|
||||||
case class Max(expr: SelectExpr) extends DBFunction
|
case class Max(expr: SelectExpr) extends DBFunction
|
||||||
|
|
||||||
|
@ -63,7 +63,10 @@ trait DSL extends DoobieMeta {
|
|||||||
FromExpr.From(sel, alias)
|
FromExpr.From(sel, alias)
|
||||||
|
|
||||||
def count(c: Column[_]): DBFunction =
|
def count(c: Column[_]): DBFunction =
|
||||||
DBFunction.Count(c)
|
DBFunction.Count(c, false)
|
||||||
|
|
||||||
|
def countDistinct(c: Column[_]): DBFunction =
|
||||||
|
DBFunction.Count(c, true)
|
||||||
|
|
||||||
def countAll: DBFunction =
|
def countAll: DBFunction =
|
||||||
DBFunction.CountAll
|
DBFunction.CountAll
|
||||||
|
@ -13,8 +13,9 @@ object DBFunctionBuilder extends CommonBuilder {
|
|||||||
case DBFunction.CountAll =>
|
case DBFunction.CountAll =>
|
||||||
sql"COUNT(*)"
|
sql"COUNT(*)"
|
||||||
|
|
||||||
case DBFunction.Count(col) =>
|
case DBFunction.Count(col, distinct) =>
|
||||||
sql"COUNT(" ++ column(col) ++ fr")"
|
if (distinct) sql"COUNT(DISTINCT " ++ column(col) ++ fr")"
|
||||||
|
else sql"COUNT(" ++ column(col) ++ fr")"
|
||||||
|
|
||||||
case DBFunction.Max(expr) =>
|
case DBFunction.Max(expr) =>
|
||||||
sql"MAX(" ++ SelectExprBuilder.build(expr) ++ fr")"
|
sql"MAX(" ++ SelectExprBuilder.build(expr) ++ fr")"
|
||||||
|
@ -0,0 +1,3 @@
|
|||||||
|
package docspell.store.queries
|
||||||
|
|
||||||
|
final case class CategoryCount(category: String, count: Int)
|
@ -190,9 +190,38 @@ object QItem {
|
|||||||
for {
|
for {
|
||||||
count <- searchCountSummary(today)(q)
|
count <- searchCountSummary(today)(q)
|
||||||
tags <- searchTagSummary(today)(q)
|
tags <- searchTagSummary(today)(q)
|
||||||
|
cats <- searchTagCategorySummary(today)(q)
|
||||||
fields <- searchFieldSummary(today)(q)
|
fields <- searchFieldSummary(today)(q)
|
||||||
folders <- searchFolderSummary(today)(q)
|
folders <- searchFolderSummary(today)(q)
|
||||||
} yield SearchSummary(count, tags, fields, folders)
|
} yield SearchSummary(count, tags, cats, fields, folders)
|
||||||
|
|
||||||
|
def searchTagCategorySummary(
|
||||||
|
today: LocalDate
|
||||||
|
)(q: Query): ConnectionIO[List[CategoryCount]] = {
|
||||||
|
val tagFrom =
|
||||||
|
from(ti)
|
||||||
|
.innerJoin(tag, tag.tid === ti.tagId)
|
||||||
|
.innerJoin(i, i.id === ti.itemId)
|
||||||
|
|
||||||
|
val tagCloud =
|
||||||
|
findItemsBase(q.fix, today, 0).unwrap
|
||||||
|
.withSelect(select(tag.category).append(countDistinct(i.id).as("num")))
|
||||||
|
.changeFrom(_.prepend(tagFrom))
|
||||||
|
.changeWhere(c => c && queryCondition(today, q.fix.account.collective, q.cond))
|
||||||
|
.groupBy(tag.category)
|
||||||
|
.build
|
||||||
|
.query[CategoryCount]
|
||||||
|
.to[List]
|
||||||
|
|
||||||
|
// the previous query starts from tags, so items with tag-count=0
|
||||||
|
// are not included they are fetched separately
|
||||||
|
for {
|
||||||
|
existing <- tagCloud
|
||||||
|
allCats <- RTag.listCategories(q.fix.account.collective)
|
||||||
|
other = allCats.diff(existing.map(_.category))
|
||||||
|
} yield existing ++ other.map(CategoryCount(_, 0))
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
def searchTagSummary(today: LocalDate)(q: Query): ConnectionIO[List[TagCount]] = {
|
def searchTagSummary(today: LocalDate)(q: Query): ConnectionIO[List[TagCount]] = {
|
||||||
val tagFrom =
|
val tagFrom =
|
||||||
|
@ -3,6 +3,7 @@ package docspell.store.queries
|
|||||||
case class SearchSummary(
|
case class SearchSummary(
|
||||||
count: Int,
|
count: Int,
|
||||||
tags: List[TagCount],
|
tags: List[TagCount],
|
||||||
|
cats: List[CategoryCount],
|
||||||
fields: List[FieldStats],
|
fields: List[FieldStats],
|
||||||
folders: List[FolderCount]
|
folders: List[FolderCount]
|
||||||
)
|
)
|
||||||
|
@ -2,4 +2,4 @@ package docspell.store.queries
|
|||||||
|
|
||||||
import docspell.store.records.RTag
|
import docspell.store.records.RTag
|
||||||
|
|
||||||
case class TagCount(tag: RTag, count: Int)
|
final case class TagCount(tag: RTag, count: Int)
|
||||||
|
@ -92,7 +92,7 @@ type TextSearchModel
|
|||||||
|
|
||||||
init : Flags -> Model
|
init : Flags -> Model
|
||||||
init flags =
|
init flags =
|
||||||
{ tagSelectModel = Comp.TagSelect.init [] []
|
{ tagSelectModel = Comp.TagSelect.init [] [] [] []
|
||||||
, tagSelection = Comp.TagSelect.emptySelection
|
, tagSelection = Comp.TagSelect.emptySelection
|
||||||
, directionModel =
|
, directionModel =
|
||||||
Comp.Dropdown.makeSingleList
|
Comp.Dropdown.makeSingleList
|
||||||
@ -483,7 +483,9 @@ updateDrop ddm flags settings msg model =
|
|||||||
GetAllTagsResp (Ok stats) ->
|
GetAllTagsResp (Ok stats) ->
|
||||||
let
|
let
|
||||||
tagSel =
|
tagSel =
|
||||||
Comp.TagSelect.modifyAll stats.tagCloud.items model.tagSelectModel
|
Comp.TagSelect.modifyAll stats.tagCloud.items
|
||||||
|
stats.tagCategoryCloud.items
|
||||||
|
model.tagSelectModel
|
||||||
in
|
in
|
||||||
{ model = { model | tagSelectModel = tagSel }
|
{ model = { model | tagSelectModel = tagSel }
|
||||||
, cmd = Cmd.none
|
, cmd = Cmd.none
|
||||||
@ -500,9 +502,14 @@ updateDrop ddm flags settings msg model =
|
|||||||
|
|
||||||
GetStatsResp (Ok stats) ->
|
GetStatsResp (Ok stats) ->
|
||||||
let
|
let
|
||||||
selectModel =
|
tagCount =
|
||||||
List.sortBy .count stats.tagCloud.items
|
List.sortBy .count stats.tagCloud.items
|
||||||
|> Comp.TagSelect.modifyCount model.tagSelectModel
|
|
||||||
|
catCount =
|
||||||
|
List.sortBy .count stats.tagCategoryCloud.items
|
||||||
|
|
||||||
|
selectModel =
|
||||||
|
Comp.TagSelect.modifyCount model.tagSelectModel tagCount catCount
|
||||||
|
|
||||||
model_ =
|
model_ =
|
||||||
{ model
|
{ model
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
module Comp.TagSelect exposing
|
module Comp.TagSelect exposing
|
||||||
( Category
|
( Model
|
||||||
, Model
|
|
||||||
, Msg
|
, Msg
|
||||||
, Selection
|
, Selection
|
||||||
, WorkModel
|
, WorkModel
|
||||||
@ -18,6 +17,7 @@ module Comp.TagSelect exposing
|
|||||||
, viewTagsDrop2
|
, viewTagsDrop2
|
||||||
)
|
)
|
||||||
|
|
||||||
|
import Api.Model.NameCount exposing (NameCount)
|
||||||
import Api.Model.Tag exposing (Tag)
|
import Api.Model.Tag exposing (Tag)
|
||||||
import Api.Model.TagCount exposing (TagCount)
|
import Api.Model.TagCount exposing (TagCount)
|
||||||
import Data.Icons as I
|
import Data.Icons as I
|
||||||
@ -38,9 +38,9 @@ import Util.Maybe
|
|||||||
|
|
||||||
type alias Model =
|
type alias Model =
|
||||||
{ availableTags : Dict String TagCount
|
{ availableTags : Dict String TagCount
|
||||||
, availableCats : Dict String Category
|
, availableCats : Dict String NameCount
|
||||||
, tagCounts : List TagCount
|
, tagCounts : List TagCount
|
||||||
, categoryCounts : List Category
|
, categoryCounts : List NameCount
|
||||||
, filterTerm : Maybe String
|
, filterTerm : Maybe String
|
||||||
, expandedTags : Bool
|
, expandedTags : Bool
|
||||||
, expandedCats : Bool
|
, expandedCats : Bool
|
||||||
@ -48,23 +48,16 @@ type alias Model =
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
type alias Category =
|
init : List TagCount -> List NameCount -> List TagCount -> List NameCount -> Model
|
||||||
{ name : String
|
init allTags allCats tags cats =
|
||||||
, count : Int
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
init : List TagCount -> List TagCount -> Model
|
|
||||||
init allTags tags =
|
|
||||||
{ availableTags =
|
{ availableTags =
|
||||||
List.map (\e -> ( e.tag.id, e )) allTags
|
List.map (\e -> ( e.tag.id, e )) allTags
|
||||||
|> Dict.fromList
|
|> Dict.fromList
|
||||||
, availableCats = sumCategories allTags
|
, availableCats =
|
||||||
|
List.map (\e -> ( e.name, e )) allCats
|
||||||
|
|> Dict.fromList
|
||||||
, tagCounts = tags
|
, tagCounts = tags
|
||||||
, categoryCounts =
|
, categoryCounts = cats
|
||||||
sumCategories tags
|
|
||||||
|> Dict.toList
|
|
||||||
|> List.map Tuple.second
|
|
||||||
, filterTerm = Nothing
|
, filterTerm = Nothing
|
||||||
, expandedTags = False
|
, expandedTags = False
|
||||||
, expandedCats = False
|
, expandedCats = False
|
||||||
@ -72,24 +65,23 @@ init allTags tags =
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
modifyAll : List TagCount -> Model -> Model
|
modifyAll : List TagCount -> List NameCount -> Model -> Model
|
||||||
modifyAll allTags model =
|
modifyAll allTags allCats model =
|
||||||
{ model
|
{ model
|
||||||
| availableTags =
|
| availableTags =
|
||||||
List.map (\e -> ( e.tag.id, e )) allTags
|
List.map (\e -> ( e.tag.id, e )) allTags
|
||||||
|> Dict.fromList
|
|> Dict.fromList
|
||||||
, availableCats = sumCategories allTags
|
, availableCats =
|
||||||
|
List.map (\e -> ( e.name, e )) allCats
|
||||||
|
|> Dict.fromList
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
modifyCount : Model -> List TagCount -> Model
|
modifyCount : Model -> List TagCount -> List NameCount -> Model
|
||||||
modifyCount model tags =
|
modifyCount model tags cats =
|
||||||
{ model
|
{ model
|
||||||
| tagCounts = tags
|
| tagCounts = tags
|
||||||
, categoryCounts =
|
, categoryCounts = cats
|
||||||
sumCategories tags
|
|
||||||
|> Dict.toList
|
|
||||||
|> List.map Tuple.second
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -108,34 +100,11 @@ toggleTag id =
|
|||||||
ToggleTag id
|
ToggleTag id
|
||||||
|
|
||||||
|
|
||||||
sumCategories : List TagCount -> Dict String Category
|
|
||||||
sumCategories tags =
|
|
||||||
let
|
|
||||||
filterCat tc =
|
|
||||||
Maybe.map (\cat -> Category cat tc.count) tc.tag.category
|
|
||||||
|
|
||||||
withCats =
|
|
||||||
List.filterMap filterCat tags
|
|
||||||
|
|
||||||
sum cat mc =
|
|
||||||
Maybe.map ((+) cat.count) mc
|
|
||||||
|> Maybe.withDefault cat.count
|
|
||||||
|> Just
|
|
||||||
|
|
||||||
sumCounts cat dict =
|
|
||||||
Dict.update cat.name (sum cat) dict
|
|
||||||
|
|
||||||
cats =
|
|
||||||
List.foldl sumCounts Dict.empty withCats
|
|
||||||
in
|
|
||||||
Dict.map (\name -> \count -> Category name count) cats
|
|
||||||
|
|
||||||
|
|
||||||
type alias Selection =
|
type alias Selection =
|
||||||
{ includeTags : List TagCount
|
{ includeTags : List TagCount
|
||||||
, excludeTags : List TagCount
|
, excludeTags : List TagCount
|
||||||
, includeCats : List Category
|
, includeCats : List NameCount
|
||||||
, excludeCats : List Category
|
, excludeCats : List NameCount
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -145,7 +114,7 @@ emptySelection =
|
|||||||
|
|
||||||
|
|
||||||
type alias WorkModel =
|
type alias WorkModel =
|
||||||
{ filteredCats : List Category
|
{ filteredCats : List NameCount
|
||||||
, filteredTags : List TagCount
|
, filteredTags : List TagCount
|
||||||
, selectedTags : Dict String Bool
|
, selectedTags : Dict String Bool
|
||||||
, selectedCats : Dict String Bool
|
, selectedCats : Dict String Bool
|
||||||
@ -166,7 +135,7 @@ orderTagCountStable model tagCounts =
|
|||||||
List.sortBy order tagCounts
|
List.sortBy order tagCounts
|
||||||
|
|
||||||
|
|
||||||
orderCatCountStable : Model -> List Category -> List Category
|
orderCatCountStable : Model -> List NameCount -> List NameCount
|
||||||
orderCatCountStable model catCounts =
|
orderCatCountStable model catCounts =
|
||||||
let
|
let
|
||||||
order cat =
|
order cat =
|
||||||
@ -193,7 +162,7 @@ removeEmptyTagCounts sel tagCounts =
|
|||||||
List.filter (\tc -> isSelected tc || tc.count > 0) tagCounts
|
List.filter (\tc -> isSelected tc || tc.count > 0) tagCounts
|
||||||
|
|
||||||
|
|
||||||
removeEmptyCatCounts : Selection -> List Category -> List Category
|
removeEmptyCatCounts : Selection -> List NameCount -> List NameCount
|
||||||
removeEmptyCatCounts sel catCounts =
|
removeEmptyCatCounts sel catCounts =
|
||||||
let
|
let
|
||||||
selected =
|
selected =
|
||||||
@ -548,7 +517,7 @@ viewTagItem2 ddm settings model tag =
|
|||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
viewCategoryItem2 : UiSettings -> WorkModel -> Category -> Html Msg
|
viewCategoryItem2 : UiSettings -> WorkModel -> NameCount -> Html Msg
|
||||||
viewCategoryItem2 settings model cat =
|
viewCategoryItem2 settings model cat =
|
||||||
let
|
let
|
||||||
state =
|
state =
|
||||||
|
Loading…
x
Reference in New Issue
Block a user