Fix and improve tag search menu

Show also "empty tags", where the count is 0. Before only tags with a
count > 0 were displayed. When searching this is fine, but when using
drag&drop to attach tags to items, it is good to see all. They can be
hidden via a button.

The tags are now ordered by their count descending, but regarding to
the overall count – not the current view. Otherwise the tags are
reordered when clicking on them, which is confusing. Also it then
shows the "more important" (most used) tags first, even when the
result is a subset.

A fix was made related to updating the menu. When coming back from
the detail view where a tag with prior count=0 was associated, the
menu didn't show it, because it relied on a previous state, where this
tag were not included.
This commit is contained in:
Eike Kettner 2021-01-11 12:07:37 +01:00
parent 3fccc3df39
commit 7beda302b2
3 changed files with 316 additions and 187 deletions

View File

@ -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 )

View File

@ -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 =

View File

@ -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 {