mirror of
https://github.com/TheAnachronism/docspell.git
synced 2025-04-05 19:09:32 +00:00
commit
1b79d7b36d
@ -250,14 +250,22 @@ object QItem {
|
|||||||
.innerJoin(tag, tag.tid === ti.tagId)
|
.innerJoin(tag, tag.tid === ti.tagId)
|
||||||
.innerJoin(i, i.id === ti.itemId)
|
.innerJoin(i, i.id === ti.itemId)
|
||||||
|
|
||||||
findItemsBase(q, 0).unwrap
|
val tagCloud =
|
||||||
.withSelect(select(tag.all).append(count(i.id).as("num")))
|
findItemsBase(q, 0).unwrap
|
||||||
.changeFrom(_.prepend(tagFrom))
|
.withSelect(select(tag.all).append(count(i.id).as("num")))
|
||||||
.changeWhere(c => c && queryCondition(q))
|
.changeFrom(_.prepend(tagFrom))
|
||||||
.groupBy(tag.tid)
|
.changeWhere(c => c && queryCondition(q))
|
||||||
.build
|
.groupBy(tag.tid)
|
||||||
.query[TagCount]
|
.build
|
||||||
.to[List]
|
.query[TagCount]
|
||||||
|
.to[List]
|
||||||
|
|
||||||
|
// the previous query starts from tags, so items with tag-count=0
|
||||||
|
// are not included they are fetched separately
|
||||||
|
for {
|
||||||
|
existing <- tagCloud
|
||||||
|
other <- RTag.findOthers(q.account.collective, existing.map(_.tag.tagId))
|
||||||
|
} yield existing ++ other.map(TagCount(_, 0))
|
||||||
}
|
}
|
||||||
|
|
||||||
def searchCountSummary(q: Query): ConnectionIO[Int] =
|
def searchCountSummary(q: Query): ConnectionIO[Int] =
|
||||||
|
@ -135,6 +135,19 @@ object RTag {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
def findOthers(coll: Ident, excludeTags: List[Ident]): ConnectionIO[List[RTag]] = {
|
||||||
|
val excl =
|
||||||
|
NonEmptyList
|
||||||
|
.fromList(excludeTags)
|
||||||
|
.map(nel => T.tid.notIn(nel))
|
||||||
|
|
||||||
|
Select(
|
||||||
|
select(T.all),
|
||||||
|
from(T),
|
||||||
|
T.cid === coll &&? excl
|
||||||
|
).orderBy(T.name.asc).build.query[RTag].to[List]
|
||||||
|
}
|
||||||
|
|
||||||
def delete(tagId: Ident, coll: Ident): ConnectionIO[Int] =
|
def delete(tagId: Ident, coll: Ident): ConnectionIO[Int] =
|
||||||
DML.delete(T, T.tid === tagId && T.cid === coll)
|
DML.delete(T, T.tid === tagId && T.cid === coll)
|
||||||
}
|
}
|
||||||
|
@ -24,6 +24,7 @@ import Api.Model.ItemSearch exposing (ItemSearch)
|
|||||||
import Api.Model.PersonList exposing (PersonList)
|
import Api.Model.PersonList exposing (PersonList)
|
||||||
import Api.Model.ReferenceList exposing (ReferenceList)
|
import Api.Model.ReferenceList exposing (ReferenceList)
|
||||||
import Api.Model.SearchStats exposing (SearchStats)
|
import Api.Model.SearchStats exposing (SearchStats)
|
||||||
|
import Api.Model.TagList exposing (TagList)
|
||||||
import Comp.CustomFieldMultiInput
|
import Comp.CustomFieldMultiInput
|
||||||
import Comp.DatePicker
|
import Comp.DatePicker
|
||||||
import Comp.Dropdown exposing (isDropdownChangeMsg)
|
import Comp.Dropdown exposing (isDropdownChangeMsg)
|
||||||
@ -84,7 +85,7 @@ type TextSearchModel
|
|||||||
|
|
||||||
init : Flags -> Model
|
init : Flags -> Model
|
||||||
init flags =
|
init flags =
|
||||||
{ tagSelectModel = Comp.TagSelect.init Comp.TagSelect.emptySelection []
|
{ tagSelectModel = Comp.TagSelect.init [] []
|
||||||
, tagSelection = Comp.TagSelect.emptySelection
|
, tagSelection = Comp.TagSelect.emptySelection
|
||||||
, directionModel =
|
, directionModel =
|
||||||
Comp.Dropdown.makeSingleList
|
Comp.Dropdown.makeSingleList
|
||||||
@ -358,6 +359,7 @@ type Msg
|
|||||||
| CustomFieldMsg Comp.CustomFieldMultiInput.Msg
|
| CustomFieldMsg Comp.CustomFieldMultiInput.Msg
|
||||||
| SetSource String
|
| SetSource String
|
||||||
| GetStatsResp (Result Http.Error SearchStats)
|
| GetStatsResp (Result Http.Error SearchStats)
|
||||||
|
| GetAllTagsResp (Result Http.Error SearchStats)
|
||||||
|
|
||||||
|
|
||||||
type alias NextState =
|
type alias NextState =
|
||||||
@ -424,7 +426,7 @@ updateDrop ddm flags settings msg model =
|
|||||||
{ model = mdp
|
{ model = mdp
|
||||||
, cmd =
|
, cmd =
|
||||||
Cmd.batch
|
Cmd.batch
|
||||||
[ Api.itemSearchStats flags (getItemSearch model) GetStatsResp
|
[ Api.itemSearchStats flags Api.Model.ItemSearch.empty GetAllTagsResp
|
||||||
, Api.getOrgLight flags GetOrgResp
|
, Api.getOrgLight flags GetOrgResp
|
||||||
, Api.getEquipments flags "" GetEquipResp
|
, Api.getEquipments flags "" GetEquipResp
|
||||||
, Api.getPersons flags "" GetPersonResp
|
, Api.getPersons flags "" GetPersonResp
|
||||||
@ -437,7 +439,7 @@ updateDrop ddm flags settings msg model =
|
|||||||
|
|
||||||
ResetForm ->
|
ResetForm ->
|
||||||
{ model = resetModel model
|
{ model = resetModel model
|
||||||
, cmd = Cmd.none
|
, cmd = Api.itemSearchStats flags Api.Model.ItemSearch.empty GetAllTagsResp
|
||||||
, stateChange = True
|
, stateChange = True
|
||||||
, dragDrop = DD.DragDropData ddm Nothing
|
, dragDrop = DD.DragDropData ddm Nothing
|
||||||
}
|
}
|
||||||
@ -473,12 +475,29 @@ updateDrop ddm flags settings msg model =
|
|||||||
SetTag id ->
|
SetTag id ->
|
||||||
resetAndSet (TagSelectMsg (Comp.TagSelect.toggleTag id))
|
resetAndSet (TagSelectMsg (Comp.TagSelect.toggleTag id))
|
||||||
|
|
||||||
|
GetAllTagsResp (Ok stats) ->
|
||||||
|
let
|
||||||
|
tagSel =
|
||||||
|
Comp.TagSelect.modifyAll stats.tagCloud.items model.tagSelectModel
|
||||||
|
in
|
||||||
|
{ model = { model | tagSelectModel = tagSel }
|
||||||
|
, cmd = Cmd.none
|
||||||
|
, stateChange = False
|
||||||
|
, dragDrop = DD.DragDropData ddm Nothing
|
||||||
|
}
|
||||||
|
|
||||||
|
GetAllTagsResp (Err _) ->
|
||||||
|
{ model = model
|
||||||
|
, cmd = Cmd.none
|
||||||
|
, stateChange = False
|
||||||
|
, dragDrop = DD.DragDropData ddm Nothing
|
||||||
|
}
|
||||||
|
|
||||||
GetStatsResp (Ok stats) ->
|
GetStatsResp (Ok stats) ->
|
||||||
let
|
let
|
||||||
selectModel =
|
selectModel =
|
||||||
List.sortBy .count stats.tagCloud.items
|
List.sortBy .count stats.tagCloud.items
|
||||||
|> List.reverse
|
|> Comp.TagSelect.modifyCount model.tagSelectModel
|
||||||
|> Comp.TagSelect.modify model.tagSelection model.tagSelectModel
|
|
||||||
|
|
||||||
model_ =
|
model_ =
|
||||||
{ model
|
{ model
|
||||||
@ -567,7 +586,7 @@ updateDrop ddm flags settings msg model =
|
|||||||
TagSelectMsg m ->
|
TagSelectMsg m ->
|
||||||
let
|
let
|
||||||
( m_, sel, ddd ) =
|
( m_, sel, ddd ) =
|
||||||
Comp.TagSelect.updateDrop ddm m model.tagSelectModel
|
Comp.TagSelect.updateDrop ddm model.tagSelection m model.tagSelectModel
|
||||||
in
|
in
|
||||||
{ model =
|
{ model =
|
||||||
{ model
|
{ model
|
||||||
@ -968,14 +987,26 @@ viewDrop ddd flags settings model =
|
|||||||
, ( "invisible hidden", fieldHidden Data.Fields.Tag && fieldHidden Data.Fields.Folder )
|
, ( "invisible hidden", fieldHidden Data.Fields.Tag && fieldHidden Data.Fields.Folder )
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
[ optional [ Data.Fields.Tag ] <|
|
((if fieldVisible Data.Fields.Tag then
|
||||||
Html.map TagSelectMsg (Comp.TagSelect.viewTagsDrop ddd.model settings model.tagSelectModel)
|
List.map (Html.map TagSelectMsg)
|
||||||
, optional [ Data.Fields.Tag ] <|
|
(Comp.TagSelect.viewAll
|
||||||
Html.map TagSelectMsg (Comp.TagSelect.viewCats settings model.tagSelectModel)
|
ddd.model
|
||||||
, optional [ Data.Fields.Folder ] <|
|
settings
|
||||||
Html.map FolderSelectMsg
|
model.tagSelection
|
||||||
(Comp.FolderSelect.viewDrop ddd.model settings.searchMenuFolderCount model.folderList)
|
model.tagSelectModel
|
||||||
]
|
)
|
||||||
|
|
||||||
|
else
|
||||||
|
[]
|
||||||
|
)
|
||||||
|
++ [ optional [ Data.Fields.Folder ] <|
|
||||||
|
Html.map FolderSelectMsg
|
||||||
|
(Comp.FolderSelect.viewDrop ddd.model
|
||||||
|
settings.searchMenuFolderCount
|
||||||
|
model.folderList
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
, div
|
, div
|
||||||
[ classList
|
[ classList
|
||||||
[ ( segmentClass, True )
|
[ ( segmentClass, True )
|
||||||
|
@ -5,16 +5,16 @@ module Comp.TagSelect exposing
|
|||||||
, Selection
|
, Selection
|
||||||
, emptySelection
|
, emptySelection
|
||||||
, init
|
, init
|
||||||
, modify
|
, modifyAll
|
||||||
|
, modifyCount
|
||||||
, reset
|
, reset
|
||||||
, toggleTag
|
, toggleTag
|
||||||
, update
|
, update
|
||||||
, updateDrop
|
, updateDrop
|
||||||
, viewCats
|
, viewAll
|
||||||
, viewTags
|
|
||||||
, viewTagsDrop
|
|
||||||
)
|
)
|
||||||
|
|
||||||
|
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
|
||||||
import Data.UiSettings exposing (UiSettings)
|
import Data.UiSettings exposing (UiSettings)
|
||||||
@ -22,21 +22,23 @@ import Dict exposing (Dict)
|
|||||||
import Html exposing (..)
|
import Html exposing (..)
|
||||||
import Html.Attributes exposing (..)
|
import Html.Attributes exposing (..)
|
||||||
import Html.Events exposing (onClick, onInput)
|
import Html.Events exposing (onClick, onInput)
|
||||||
|
import Set
|
||||||
import Simple.Fuzzy
|
import Simple.Fuzzy
|
||||||
|
import String as S
|
||||||
import Util.ExpandCollapse
|
import Util.ExpandCollapse
|
||||||
import Util.ItemDragDrop as DD
|
import Util.ItemDragDrop as DD
|
||||||
import Util.String as S
|
import Util.Maybe
|
||||||
|
|
||||||
|
|
||||||
type alias Model =
|
type alias Model =
|
||||||
{ all : List TagCount
|
{ availableTags : Dict String TagCount
|
||||||
, filteredTags : List TagCount
|
, availableCats : Dict String Category
|
||||||
, categories : List Category
|
, tagCounts : List TagCount
|
||||||
, filteredCats : List Category
|
, categoryCounts : List Category
|
||||||
, selectedTags : Dict String Bool
|
, filterTerm : Maybe String
|
||||||
, selectedCats : Dict String Bool
|
|
||||||
, expandedTags : Bool
|
, expandedTags : Bool
|
||||||
, expandedCats : Bool
|
, expandedCats : Bool
|
||||||
|
, showEmpty : Bool
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -46,82 +48,52 @@ type alias Category =
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
init : Selection -> List TagCount -> Model
|
init : List TagCount -> List TagCount -> Model
|
||||||
init sel tags =
|
init allTags tags =
|
||||||
let
|
{ availableTags =
|
||||||
tagId t =
|
List.map (\e -> ( e.tag.id, e )) allTags
|
||||||
t.tag.id
|
|> Dict.fromList
|
||||||
|
, availableCats = sumCategories allTags
|
||||||
constDict mkId flag list =
|
, tagCounts = tags
|
||||||
List.map (\e -> ( mkId e, flag )) list
|
, categoryCounts =
|
||||||
|> Dict.fromList
|
sumCategories tags
|
||||||
|
|> Dict.toList
|
||||||
selTag =
|
|> List.map Tuple.second
|
||||||
constDict tagId True sel.includeTags
|
, filterTerm = Nothing
|
||||||
|> Dict.union (constDict tagId False sel.excludeTags)
|
|
||||||
|
|
||||||
selCat =
|
|
||||||
constDict .name True sel.includeCats
|
|
||||||
|> Dict.union (constDict .name False sel.excludeCats)
|
|
||||||
|
|
||||||
cats =
|
|
||||||
sumCategories tags
|
|
||||||
in
|
|
||||||
{ all = tags
|
|
||||||
, filteredTags = tags
|
|
||||||
, categories = cats
|
|
||||||
, filteredCats = cats
|
|
||||||
, selectedTags = selTag
|
|
||||||
, selectedCats = selCat
|
|
||||||
, expandedTags = False
|
, expandedTags = False
|
||||||
, expandedCats = False
|
, expandedCats = False
|
||||||
|
, showEmpty = True
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
modify : Selection -> Model -> List TagCount -> Model
|
modifyAll : List TagCount -> Model -> Model
|
||||||
modify sel model tags =
|
modifyAll allTags model =
|
||||||
let
|
{ model
|
||||||
newModel =
|
| availableTags =
|
||||||
init sel tags
|
List.map (\e -> ( e.tag.id, e )) allTags
|
||||||
in
|
|> Dict.fromList
|
||||||
if List.isEmpty model.all then
|
, availableCats = sumCategories allTags
|
||||||
newModel
|
}
|
||||||
|
|
||||||
else
|
|
||||||
let
|
|
||||||
tagId t =
|
|
||||||
t.tag.id
|
|
||||||
|
|
||||||
catId c =
|
modifyCount : Model -> List TagCount -> Model
|
||||||
c.name
|
modifyCount model tags =
|
||||||
|
{ model
|
||||||
tagDict =
|
| tagCounts = tags
|
||||||
List.map (\e -> ( tagId e, e )) tags
|
, categoryCounts =
|
||||||
|> Dict.fromList
|
sumCategories tags
|
||||||
|
|> Dict.toList
|
||||||
catDict =
|
|> List.map Tuple.second
|
||||||
List.map (\e -> ( catId e, e )) newModel.categories
|
}
|
||||||
|> Dict.fromList
|
|
||||||
|
|
||||||
replaceTag e =
|
|
||||||
Dict.get e.tag.id tagDict |> Maybe.withDefault { e | count = 0 }
|
|
||||||
|
|
||||||
replaceCat c =
|
|
||||||
Dict.get c.name catDict |> Maybe.withDefault { c | count = 0 }
|
|
||||||
in
|
|
||||||
{ model
|
|
||||||
| all = List.map replaceTag model.all
|
|
||||||
, filteredTags = List.map replaceTag model.filteredTags
|
|
||||||
, categories = List.map replaceCat model.categories
|
|
||||||
, filteredCats = List.map replaceCat model.filteredCats
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
reset : Model -> Model
|
reset : Model -> Model
|
||||||
reset model =
|
reset model =
|
||||||
{ model
|
{ model
|
||||||
| selectedTags = Dict.empty
|
| filterTerm = Nothing
|
||||||
, selectedCats = Dict.empty
|
, expandedTags = False
|
||||||
|
, expandedCats = False
|
||||||
|
, showEmpty = True
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -130,7 +102,7 @@ toggleTag id =
|
|||||||
ToggleTag id
|
ToggleTag id
|
||||||
|
|
||||||
|
|
||||||
sumCategories : List TagCount -> List Category
|
sumCategories : List TagCount -> Dict String Category
|
||||||
sumCategories tags =
|
sumCategories tags =
|
||||||
let
|
let
|
||||||
filterCat tc =
|
filterCat tc =
|
||||||
@ -150,21 +122,7 @@ sumCategories tags =
|
|||||||
cats =
|
cats =
|
||||||
List.foldl sumCounts Dict.empty withCats
|
List.foldl sumCounts Dict.empty withCats
|
||||||
in
|
in
|
||||||
Dict.toList cats
|
Dict.map (\name -> \count -> Category name count) cats
|
||||||
|> List.map (\( n, c ) -> Category n c)
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
--- Update
|
|
||||||
|
|
||||||
|
|
||||||
type Msg
|
|
||||||
= ToggleTag String
|
|
||||||
| ToggleCat String
|
|
||||||
| ToggleExpandTags
|
|
||||||
| ToggleExpandCats
|
|
||||||
| TagDDMsg DD.Msg
|
|
||||||
| Search String
|
|
||||||
|
|
||||||
|
|
||||||
type alias Selection =
|
type alias Selection =
|
||||||
@ -180,47 +138,185 @@ emptySelection =
|
|||||||
Selection [] [] [] []
|
Selection [] [] [] []
|
||||||
|
|
||||||
|
|
||||||
update : Msg -> Model -> ( Model, Selection )
|
type alias WorkModel =
|
||||||
update msg model =
|
{ filteredCats : List Category
|
||||||
|
, filteredTags : List TagCount
|
||||||
|
, selectedTags : Dict String Bool
|
||||||
|
, selectedCats : Dict String Bool
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
{-| Orders the list of tag counts by their overall counts and not by
|
||||||
|
their direct counts.
|
||||||
|
-}
|
||||||
|
orderTagCountStable : Model -> List TagCount -> List TagCount
|
||||||
|
orderTagCountStable model tagCounts =
|
||||||
|
let
|
||||||
|
order tc =
|
||||||
|
Dict.get tc.tag.id model.availableTags
|
||||||
|
|> Maybe.map (\e -> ( e.count * -1, S.toLower e.tag.name ))
|
||||||
|
|> Maybe.withDefault ( 0, S.toLower tc.tag.name )
|
||||||
|
in
|
||||||
|
List.sortBy order tagCounts
|
||||||
|
|
||||||
|
|
||||||
|
orderCatCountStable : Model -> List Category -> List Category
|
||||||
|
orderCatCountStable model catCounts =
|
||||||
|
let
|
||||||
|
order cat =
|
||||||
|
Dict.get cat.name model.availableCats
|
||||||
|
|> Maybe.map (\e -> ( e.count * -1, S.toLower e.name ))
|
||||||
|
|> Maybe.withDefault ( 0, S.toLower cat.name )
|
||||||
|
in
|
||||||
|
List.sortBy order catCounts
|
||||||
|
|
||||||
|
|
||||||
|
removeEmptyTagCounts : Selection -> List TagCount -> List TagCount
|
||||||
|
removeEmptyTagCounts sel tagCounts =
|
||||||
|
let
|
||||||
|
selected =
|
||||||
|
List.concat
|
||||||
|
[ List.map (.tag >> .id) sel.includeTags
|
||||||
|
, List.map (.tag >> .id) sel.excludeTags
|
||||||
|
]
|
||||||
|
|> Set.fromList
|
||||||
|
|
||||||
|
isSelected tc =
|
||||||
|
Set.member tc.tag.id selected
|
||||||
|
in
|
||||||
|
List.filter (\tc -> isSelected tc || tc.count > 0) tagCounts
|
||||||
|
|
||||||
|
|
||||||
|
removeEmptyCatCounts : Selection -> List Category -> List Category
|
||||||
|
removeEmptyCatCounts sel catCounts =
|
||||||
|
let
|
||||||
|
selected =
|
||||||
|
List.concat
|
||||||
|
[ List.map .name sel.includeCats
|
||||||
|
, List.map .name sel.excludeCats
|
||||||
|
]
|
||||||
|
|> Set.fromList
|
||||||
|
|
||||||
|
isSelected cat =
|
||||||
|
Set.member cat.name selected
|
||||||
|
in
|
||||||
|
List.filter (\tc -> isSelected tc || tc.count > 0) catCounts
|
||||||
|
|
||||||
|
|
||||||
|
makeWorkModel : Selection -> Model -> WorkModel
|
||||||
|
makeWorkModel sel model =
|
||||||
|
let
|
||||||
|
tagCounts =
|
||||||
|
orderTagCountStable model
|
||||||
|
(if model.showEmpty then
|
||||||
|
model.tagCounts
|
||||||
|
|
||||||
|
else
|
||||||
|
removeEmptyTagCounts sel model.tagCounts
|
||||||
|
)
|
||||||
|
|
||||||
|
catCounts =
|
||||||
|
orderCatCountStable model
|
||||||
|
(if model.showEmpty then
|
||||||
|
model.categoryCounts
|
||||||
|
|
||||||
|
else
|
||||||
|
removeEmptyCatCounts sel model.categoryCounts
|
||||||
|
)
|
||||||
|
|
||||||
|
( tags, cats ) =
|
||||||
|
case model.filterTerm of
|
||||||
|
Just filter ->
|
||||||
|
( List.filter (\t -> Simple.Fuzzy.match filter t.tag.name) tagCounts
|
||||||
|
, List.filter (\c -> Simple.Fuzzy.match filter c.name) catCounts
|
||||||
|
)
|
||||||
|
|
||||||
|
Nothing ->
|
||||||
|
( tagCounts, catCounts )
|
||||||
|
|
||||||
|
tagId t =
|
||||||
|
t.tag.id
|
||||||
|
|
||||||
|
constDict mkId flag list =
|
||||||
|
List.map (\e -> ( mkId e, flag )) list
|
||||||
|
|> Dict.fromList
|
||||||
|
|
||||||
|
selTag =
|
||||||
|
constDict tagId True sel.includeTags
|
||||||
|
|> Dict.union (constDict tagId False sel.excludeTags)
|
||||||
|
|
||||||
|
selCat =
|
||||||
|
constDict .name True sel.includeCats
|
||||||
|
|> Dict.union (constDict .name False sel.excludeCats)
|
||||||
|
in
|
||||||
|
{ filteredCats = cats
|
||||||
|
, filteredTags = tags
|
||||||
|
, selectedTags = selTag
|
||||||
|
, selectedCats = selCat
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
type Msg
|
||||||
|
= ToggleTag String
|
||||||
|
| ToggleCat String
|
||||||
|
| ToggleExpandTags
|
||||||
|
| ToggleExpandCats
|
||||||
|
| ToggleShowEmpty
|
||||||
|
| TagDDMsg DD.Msg
|
||||||
|
| Search String
|
||||||
|
|
||||||
|
|
||||||
|
update : Selection -> Msg -> Model -> ( Model, Selection )
|
||||||
|
update sel msg model =
|
||||||
let
|
let
|
||||||
( m, s, _ ) =
|
( m, s, _ ) =
|
||||||
updateDrop DD.init msg model
|
updateDrop DD.init sel msg model
|
||||||
in
|
in
|
||||||
( m, s )
|
( m, s )
|
||||||
|
|
||||||
|
|
||||||
updateDrop : DD.Model -> Msg -> Model -> ( Model, Selection, DD.DragDropData )
|
updateDrop : DD.Model -> Selection -> Msg -> Model -> ( Model, Selection, DD.DragDropData )
|
||||||
updateDrop ddm msg model =
|
updateDrop ddm sel msg model =
|
||||||
|
let
|
||||||
|
wm =
|
||||||
|
makeWorkModel sel model
|
||||||
|
in
|
||||||
case msg of
|
case msg of
|
||||||
|
ToggleShowEmpty ->
|
||||||
|
( { model | showEmpty = not model.showEmpty }
|
||||||
|
, sel
|
||||||
|
, DD.DragDropData ddm Nothing
|
||||||
|
)
|
||||||
|
|
||||||
ToggleTag id ->
|
ToggleTag id ->
|
||||||
let
|
let
|
||||||
next =
|
next =
|
||||||
updateSelection id model.selectedTags
|
updateSelection id wm.selectedTags
|
||||||
|
|
||||||
model_ =
|
wm_ =
|
||||||
{ model | selectedTags = next }
|
{ wm | selectedTags = next }
|
||||||
in
|
in
|
||||||
( model_, getSelection model_, DD.DragDropData ddm Nothing )
|
( model, getSelection wm_, DD.DragDropData ddm Nothing )
|
||||||
|
|
||||||
ToggleCat name ->
|
ToggleCat name ->
|
||||||
let
|
let
|
||||||
next =
|
next =
|
||||||
updateSelection name model.selectedCats
|
updateSelection name wm.selectedCats
|
||||||
|
|
||||||
model_ =
|
wm_ =
|
||||||
{ model | selectedCats = next }
|
{ wm | selectedCats = next }
|
||||||
in
|
in
|
||||||
( model_, getSelection model_, DD.DragDropData ddm Nothing )
|
( model, getSelection wm_, DD.DragDropData ddm Nothing )
|
||||||
|
|
||||||
ToggleExpandTags ->
|
ToggleExpandTags ->
|
||||||
( { model | expandedTags = not model.expandedTags }
|
( { model | expandedTags = not model.expandedTags }
|
||||||
, getSelection model
|
, sel
|
||||||
, DD.DragDropData ddm Nothing
|
, DD.DragDropData ddm Nothing
|
||||||
)
|
)
|
||||||
|
|
||||||
ToggleExpandCats ->
|
ToggleExpandCats ->
|
||||||
( { model | expandedCats = not model.expandedCats }
|
( { model | expandedCats = not model.expandedCats }
|
||||||
, getSelection model
|
, sel
|
||||||
, DD.DragDropData ddm Nothing
|
, DD.DragDropData ddm Nothing
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -229,31 +325,11 @@ updateDrop ddm msg model =
|
|||||||
ddd =
|
ddd =
|
||||||
DD.update lm ddm
|
DD.update lm ddm
|
||||||
in
|
in
|
||||||
( model, getSelection model, ddd )
|
( model, sel, ddd )
|
||||||
|
|
||||||
Search str ->
|
Search str ->
|
||||||
let
|
( { model | filterTerm = Util.Maybe.fromString str }
|
||||||
forTags t =
|
, sel
|
||||||
Simple.Fuzzy.match str t.tag.name
|
|
||||||
|
|
||||||
forCats c =
|
|
||||||
Simple.Fuzzy.match str c.name
|
|
||||||
in
|
|
||||||
( { model
|
|
||||||
| filteredTags =
|
|
||||||
if S.isBlank str then
|
|
||||||
model.all
|
|
||||||
|
|
||||||
else
|
|
||||||
List.filter forTags model.all
|
|
||||||
, filteredCats =
|
|
||||||
if S.isBlank str then
|
|
||||||
model.categories
|
|
||||||
|
|
||||||
else
|
|
||||||
List.filter forCats model.categories
|
|
||||||
}
|
|
||||||
, getSelection model
|
|
||||||
, DD.DragDropData ddm Nothing
|
, DD.DragDropData ddm Nothing
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -275,14 +351,14 @@ updateSelection id selected =
|
|||||||
Dict.remove id selected
|
Dict.remove id selected
|
||||||
|
|
||||||
|
|
||||||
getSelection : Model -> Selection
|
getSelection : WorkModel -> Selection
|
||||||
getSelection model =
|
getSelection model =
|
||||||
let
|
let
|
||||||
( inclTags, exclTags ) =
|
( inclTags, exclTags ) =
|
||||||
getSelection1 (\t -> t.tag.id) model.selectedTags model.all
|
getSelection1 (\t -> t.tag.id) model.selectedTags model.filteredTags
|
||||||
|
|
||||||
( inclCats, exclCats ) =
|
( inclCats, exclCats ) =
|
||||||
getSelection1 (\c -> c.name) model.selectedCats model.categories
|
getSelection1 (\c -> c.name) model.selectedCats model.filteredCats
|
||||||
in
|
in
|
||||||
Selection inclTags exclTags inclCats exclCats
|
Selection inclTags exclTags inclCats exclCats
|
||||||
|
|
||||||
@ -311,7 +387,7 @@ type SelState
|
|||||||
| Deselect
|
| Deselect
|
||||||
|
|
||||||
|
|
||||||
tagState : Model -> String -> SelState
|
tagState : WorkModel -> String -> SelState
|
||||||
tagState model id =
|
tagState model id =
|
||||||
case Dict.get id model.selectedTags of
|
case Dict.get id model.selectedTags of
|
||||||
Just True ->
|
Just True ->
|
||||||
@ -324,7 +400,7 @@ tagState model id =
|
|||||||
Deselect
|
Deselect
|
||||||
|
|
||||||
|
|
||||||
catState : Model -> String -> SelState
|
catState : WorkModel -> String -> SelState
|
||||||
catState model name =
|
catState model name =
|
||||||
case Dict.get name model.selectedCats of
|
case Dict.get name model.selectedCats of
|
||||||
Just True ->
|
Just True ->
|
||||||
@ -337,38 +413,60 @@ catState model name =
|
|||||||
Deselect
|
Deselect
|
||||||
|
|
||||||
|
|
||||||
viewTags : UiSettings -> Model -> Html Msg
|
viewAll : DD.Model -> UiSettings -> Selection -> Model -> List (Html Msg)
|
||||||
viewTags =
|
viewAll ddm settings sel model =
|
||||||
viewTagsDrop DD.init
|
let
|
||||||
|
wm =
|
||||||
|
makeWorkModel sel model
|
||||||
|
in
|
||||||
|
viewTagsDrop ddm wm settings model ++ [ viewCats settings wm model ]
|
||||||
|
|
||||||
|
|
||||||
viewTagsDrop : DD.Model -> UiSettings -> Model -> Html Msg
|
viewTagsDrop : DD.Model -> WorkModel -> UiSettings -> Model -> List (Html Msg)
|
||||||
viewTagsDrop ddm settings model =
|
viewTagsDrop ddm wm settings model =
|
||||||
div [ class "ui list" ]
|
[ div [ class "ui tiny fluid secondary menu" ]
|
||||||
|
[ a
|
||||||
|
[ class "borderless item"
|
||||||
|
, href "#"
|
||||||
|
, onClick ToggleShowEmpty
|
||||||
|
]
|
||||||
|
[ if model.showEmpty then
|
||||||
|
text " Hide empty"
|
||||||
|
|
||||||
|
else
|
||||||
|
text " Show empty"
|
||||||
|
]
|
||||||
|
, div [ class "right menu" ]
|
||||||
|
[ div [ class "right fitted item width-80" ]
|
||||||
|
[ div [ class "ui small transparent icon input" ]
|
||||||
|
[ input
|
||||||
|
[ type_ "text"
|
||||||
|
, placeholder "Filter …"
|
||||||
|
, onInput Search
|
||||||
|
]
|
||||||
|
[]
|
||||||
|
, i [ class "search icon" ] []
|
||||||
|
]
|
||||||
|
]
|
||||||
|
]
|
||||||
|
]
|
||||||
|
, div [ class "ui list" ]
|
||||||
[ div [ class "item" ]
|
[ div [ class "item" ]
|
||||||
[ I.tagIcon ""
|
[ I.tagIcon ""
|
||||||
, div [ class "content" ]
|
, div [ class "content" ]
|
||||||
[ div [ class "header" ]
|
[ div [ class "header" ]
|
||||||
[ text "Tags"
|
[ text "Tags"
|
||||||
, div [ class "ui small transparent icon right floated input width-70" ]
|
|
||||||
[ input
|
|
||||||
[ type_ "text"
|
|
||||||
, placeholder "Filter …"
|
|
||||||
, onInput Search
|
|
||||||
]
|
|
||||||
[]
|
|
||||||
, i [ class "search icon" ] []
|
|
||||||
]
|
|
||||||
]
|
]
|
||||||
, div [ class "ui relaxed list" ]
|
, div [ class "ui relaxed list" ]
|
||||||
(renderTagItems ddm settings model)
|
(renderTagItems ddm settings model wm)
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
viewCats : UiSettings -> Model -> Html Msg
|
viewCats : UiSettings -> WorkModel -> Model -> Html Msg
|
||||||
viewCats settings model =
|
viewCats settings wm model =
|
||||||
div [ class "ui list" ]
|
div [ class "ui list" ]
|
||||||
[ div [ class "item" ]
|
[ div [ class "item" ]
|
||||||
[ I.tagsIcon ""
|
[ I.tagsIcon ""
|
||||||
@ -377,75 +475,75 @@ viewCats settings model =
|
|||||||
[ text "Categories"
|
[ text "Categories"
|
||||||
]
|
]
|
||||||
, div [ class "ui relaxed list" ]
|
, div [ class "ui relaxed list" ]
|
||||||
(renderCatItems settings model)
|
(renderCatItems settings model wm)
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
renderTagItems : DD.Model -> UiSettings -> Model -> List (Html Msg)
|
renderTagItems : DD.Model -> UiSettings -> Model -> WorkModel -> List (Html Msg)
|
||||||
renderTagItems ddm settings model =
|
renderTagItems ddm settings model wm =
|
||||||
let
|
let
|
||||||
tags =
|
tags =
|
||||||
model.filteredTags
|
wm.filteredTags
|
||||||
|
|
||||||
max =
|
max =
|
||||||
settings.searchMenuTagCount
|
settings.searchMenuTagCount
|
||||||
|
|
||||||
exp =
|
expLink =
|
||||||
Util.ExpandCollapse.expandToggle
|
Util.ExpandCollapse.expandToggle
|
||||||
max
|
max
|
||||||
(List.length tags)
|
(List.length tags)
|
||||||
ToggleExpandTags
|
ToggleExpandTags
|
||||||
|
|
||||||
cps =
|
cpsLink =
|
||||||
Util.ExpandCollapse.collapseToggle
|
Util.ExpandCollapse.collapseToggle
|
||||||
max
|
max
|
||||||
(List.length tags)
|
(List.length tags)
|
||||||
ToggleExpandTags
|
ToggleExpandTags
|
||||||
in
|
in
|
||||||
if max <= 0 then
|
if max <= 0 then
|
||||||
List.map (viewTagItem ddm settings model) tags
|
List.map (viewTagItem ddm settings wm) tags
|
||||||
|
|
||||||
else if model.expandedTags then
|
else if model.expandedTags then
|
||||||
List.map (viewTagItem ddm settings model) tags ++ cps
|
List.map (viewTagItem ddm settings wm) tags ++ cpsLink
|
||||||
|
|
||||||
else
|
else
|
||||||
List.map (viewTagItem ddm settings model) (List.take max tags) ++ exp
|
List.map (viewTagItem ddm settings wm) (List.take max tags) ++ expLink
|
||||||
|
|
||||||
|
|
||||||
renderCatItems : UiSettings -> Model -> List (Html Msg)
|
renderCatItems : UiSettings -> Model -> WorkModel -> List (Html Msg)
|
||||||
renderCatItems settings model =
|
renderCatItems settings model wm =
|
||||||
let
|
let
|
||||||
cats =
|
cats =
|
||||||
model.filteredCats
|
wm.filteredCats
|
||||||
|
|
||||||
max =
|
max =
|
||||||
settings.searchMenuTagCatCount
|
settings.searchMenuTagCatCount
|
||||||
|
|
||||||
exp =
|
expLink =
|
||||||
Util.ExpandCollapse.expandToggle
|
Util.ExpandCollapse.expandToggle
|
||||||
max
|
max
|
||||||
(List.length cats)
|
(List.length cats)
|
||||||
ToggleExpandCats
|
ToggleExpandCats
|
||||||
|
|
||||||
cps =
|
cpsLink =
|
||||||
Util.ExpandCollapse.collapseToggle
|
Util.ExpandCollapse.collapseToggle
|
||||||
max
|
max
|
||||||
(List.length cats)
|
(List.length cats)
|
||||||
ToggleExpandCats
|
ToggleExpandCats
|
||||||
in
|
in
|
||||||
if max <= 0 then
|
if max <= 0 then
|
||||||
List.map (viewCategoryItem settings model) cats
|
List.map (viewCategoryItem settings wm) cats
|
||||||
|
|
||||||
else if model.expandedCats then
|
else if model.expandedCats then
|
||||||
List.map (viewCategoryItem settings model) cats ++ cps
|
List.map (viewCategoryItem settings wm) cats ++ cpsLink
|
||||||
|
|
||||||
else
|
else
|
||||||
List.map (viewCategoryItem settings model) (List.take max cats) ++ exp
|
List.map (viewCategoryItem settings wm) (List.take max cats) ++ expLink
|
||||||
|
|
||||||
|
|
||||||
viewCategoryItem : UiSettings -> Model -> Category -> Html Msg
|
viewCategoryItem : UiSettings -> WorkModel -> Category -> Html Msg
|
||||||
viewCategoryItem settings model cat =
|
viewCategoryItem settings model cat =
|
||||||
let
|
let
|
||||||
state =
|
state =
|
||||||
@ -479,7 +577,7 @@ viewCategoryItem settings model cat =
|
|||||||
]
|
]
|
||||||
|
|
||||||
|
|
||||||
viewTagItem : DD.Model -> UiSettings -> Model -> TagCount -> Html Msg
|
viewTagItem : DD.Model -> UiSettings -> WorkModel -> TagCount -> Html Msg
|
||||||
viewTagItem ddm settings model tag =
|
viewTagItem ddm settings model tag =
|
||||||
let
|
let
|
||||||
state =
|
state =
|
||||||
|
@ -291,8 +291,8 @@ label span.muted {
|
|||||||
margin-left: 0.5em;
|
margin-left: 0.5em;
|
||||||
}
|
}
|
||||||
|
|
||||||
.width-70 {
|
.width-80 {
|
||||||
width: 70%;
|
width: 80%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.ui.search.dropdown.open, .ui.selection.dropdown.open {
|
.ui.search.dropdown.open, .ui.selection.dropdown.open {
|
||||||
|
Loading…
x
Reference in New Issue
Block a user