Merge pull request #146 from eikek/search-improvements

Search improvements
This commit is contained in:
mergify[bot] 2020-06-13 20:44:40 +00:00 committed by GitHub
commit e05d435d1a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
14 changed files with 323 additions and 173 deletions

View File

@ -22,7 +22,8 @@
"justinmimbs/date": "3.1.2", "justinmimbs/date": "3.1.2",
"norpan/elm-html5-drag-drop": "3.1.4", "norpan/elm-html5-drag-drop": "3.1.4",
"ryannhg/date-format": "2.3.0", "ryannhg/date-format": "2.3.0",
"truqu/elm-base64": "2.0.4" "truqu/elm-base64": "2.0.4",
"ursi/elm-throttle": "1.0.1"
}, },
"indirect": { "indirect": {
"elm/bytes": "1.0.8", "elm/bytes": "1.0.8",

View File

@ -3354,6 +3354,13 @@ components:
- outgoing - outgoing
name: name:
type: string type: string
description: |
Search in item names.
allNames:
type: string
description: |
Search in item names, correspondents, concerned entities
and notes.
corrOrg: corrOrg:
type: string type: string
format: ident format: ident

View File

@ -121,6 +121,7 @@ trait Conversions {
m.dateUntil, m.dateUntil,
m.dueDateFrom, m.dueDateFrom,
m.dueDateUntil, m.dueDateUntil,
m.allNames,
None None
) )

View File

@ -164,6 +164,7 @@ object QItem {
dateTo: Option[Timestamp], dateTo: Option[Timestamp],
dueDateFrom: Option[Timestamp], dueDateFrom: Option[Timestamp],
dueDateTo: Option[Timestamp], dueDateTo: Option[Timestamp],
allNames: Option[String],
orderAsc: Option[RItem.Columns.type => Column] orderAsc: Option[RItem.Columns.type => Column]
) )
@ -184,6 +185,7 @@ object QItem {
None, None,
None, None,
None, None,
None,
None None
) )
} }
@ -282,13 +284,26 @@ object QItem {
RTagItem.Columns.tagId.isOneOf(q.tagsExclude) RTagItem.Columns.tagId.isOneOf(q.tagsExclude)
) )
val name = q.name.map(queryWildcard) val name = q.name.map(_.toLowerCase).map(queryWildcard)
val allNames = q.allNames.map(_.toLowerCase).map(queryWildcard)
val cond = and( val cond = and(
IC.cid.prefix("i").is(q.collective), IC.cid.prefix("i").is(q.collective),
IC.state.prefix("i").isOneOf(q.states), IC.state.prefix("i").isOneOf(q.states),
IC.incoming.prefix("i").isOrDiscard(q.direction), IC.incoming.prefix("i").isOrDiscard(q.direction),
name name
.map(n => or(IC.name.prefix("i").lowerLike(n), IC.notes.prefix("i").lowerLike(n))) .map(n => IC.name.prefix("i").lowerLike(n))
.getOrElse(Fragment.empty),
allNames
.map(n =>
or(
OC.name.prefix("o0").lowerLike(n),
PC.name.prefix("p0").lowerLike(n),
PC.name.prefix("p1").lowerLike(n),
EC.name.prefix("e1").lowerLike(n),
IC.name.prefix("i").lowerLike(n),
IC.notes.prefix("i").lowerLike(n)
)
)
.getOrElse(Fragment.empty), .getOrElse(Fragment.empty),
RPerson.Columns.pid.prefix("p0").isOrDiscard(q.corrPerson), RPerson.Columns.pid.prefix("p0").isOrDiscard(q.corrPerson),
ROrganization.Columns.oid.prefix("o0").isOrDiscard(q.corrOrg), ROrganization.Columns.oid.prefix("o0").isOrDiscard(q.corrOrg),

View File

@ -40,44 +40,44 @@ update msg model =
( m, c, s ) = ( m, c, s ) =
updateWithSub msg model updateWithSub msg model
in in
( { m | subs = Sub.batch [ m.subs, s ] }, c ) ( { m | subs = s }, c )
updateWithSub : Msg -> Model -> ( Model, Cmd Msg, Sub Msg ) updateWithSub : Msg -> Model -> ( Model, Cmd Msg, Sub Msg )
updateWithSub msg model = updateWithSub msg model =
case msg of case msg of
HomeMsg lm -> HomeMsg lm ->
updateHome lm model |> noSub updateHome lm model
LoginMsg lm -> LoginMsg lm ->
updateLogin lm model |> noSub updateLogin lm model
ManageDataMsg lm -> ManageDataMsg lm ->
updateManageData lm model |> noSub updateManageData lm model
CollSettingsMsg m -> CollSettingsMsg m ->
updateCollSettings m model |> noSub updateCollSettings m model
UserSettingsMsg m -> UserSettingsMsg m ->
updateUserSettings m model |> noSub updateUserSettings m model
QueueMsg m -> QueueMsg m ->
updateQueue m model |> noSub updateQueue m model
RegisterMsg m -> RegisterMsg m ->
updateRegister m model |> noSub updateRegister m model
UploadMsg m -> UploadMsg m ->
updateUpload m model updateUpload m model
NewInviteMsg m -> NewInviteMsg m ->
updateNewInvite m model |> noSub updateNewInvite m model
ItemDetailMsg m -> ItemDetailMsg m ->
updateItemDetail m model updateItemDetail m model
VersionResp (Ok info) -> VersionResp (Ok info) ->
( { model | version = info }, Cmd.none ) |> noSub ( { model | version = info }, Cmd.none, Sub.none )
VersionResp (Err _) -> VersionResp (Err _) ->
( model, Cmd.none, Sub.none ) ( model, Cmd.none, Sub.none )
@ -162,25 +162,27 @@ updateWithSub msg model =
check = check =
checkPage model.flags page checkPage model.flags page
( m, c ) = ( m, c, s ) =
initPage model page initPage model page
in in
if check == page then if check == page then
( { m | page = page }, c, Sub.none ) ( { m | page = page }, c, s )
else else
( model, Page.goto check, Sub.none ) ( model, Page.goto check, Sub.none )
ToggleNavMenu -> ToggleNavMenu ->
( { model | navMenuOpen = not model.navMenuOpen }, Cmd.none, Sub.none ) ( { model | navMenuOpen = not model.navMenuOpen }
, Cmd.none
, Sub.none
)
GetUiSettings settings -> GetUiSettings settings ->
Util.Update.andThen1 Util.Update.andThen2
[ updateUserSettings Page.UserSettings.Data.UpdateSettings [ updateUserSettings Page.UserSettings.Data.UpdateSettings
, updateHome Page.Home.Data.DoSearch , updateHome Page.Home.Data.DoSearch
] ]
{ model | uiSettings = settings } { model | uiSettings = settings }
|> noSub
updateItemDetail : Page.ItemDetail.Data.Msg -> Model -> ( Model, Cmd Msg, Sub Msg ) updateItemDetail : Page.ItemDetail.Data.Msg -> Model -> ( Model, Cmd Msg, Sub Msg )
@ -200,7 +202,7 @@ updateItemDetail lmsg model =
) )
updateNewInvite : Page.NewInvite.Data.Msg -> Model -> ( Model, Cmd Msg ) updateNewInvite : Page.NewInvite.Data.Msg -> Model -> ( Model, Cmd Msg, Sub Msg )
updateNewInvite lmsg model = updateNewInvite lmsg model =
let let
( lm, lc ) = ( lm, lc ) =
@ -208,6 +210,7 @@ updateNewInvite lmsg model =
in in
( { model | newInviteModel = lm } ( { model | newInviteModel = lm }
, Cmd.map NewInviteMsg lc , Cmd.map NewInviteMsg lc
, Sub.none
) )
@ -227,7 +230,7 @@ updateUpload lmsg model =
) )
updateRegister : Page.Register.Data.Msg -> Model -> ( Model, Cmd Msg ) updateRegister : Page.Register.Data.Msg -> Model -> ( Model, Cmd Msg, Sub Msg )
updateRegister lmsg model = updateRegister lmsg model =
let let
( lm, lc ) = ( lm, lc ) =
@ -235,10 +238,11 @@ updateRegister lmsg model =
in in
( { model | registerModel = lm } ( { model | registerModel = lm }
, Cmd.map RegisterMsg lc , Cmd.map RegisterMsg lc
, Sub.none
) )
updateQueue : Page.Queue.Data.Msg -> Model -> ( Model, Cmd Msg ) updateQueue : Page.Queue.Data.Msg -> Model -> ( Model, Cmd Msg, Sub Msg )
updateQueue lmsg model = updateQueue lmsg model =
let let
( lm, lc ) = ( lm, lc ) =
@ -246,10 +250,11 @@ updateQueue lmsg model =
in in
( { model | queueModel = lm } ( { model | queueModel = lm }
, Cmd.map QueueMsg lc , Cmd.map QueueMsg lc
, Sub.none
) )
updateUserSettings : Page.UserSettings.Data.Msg -> Model -> ( Model, Cmd Msg ) updateUserSettings : Page.UserSettings.Data.Msg -> Model -> ( Model, Cmd Msg, Sub Msg )
updateUserSettings lmsg model = updateUserSettings lmsg model =
let let
( lm, lc, ls ) = ( lm, lc, ls ) =
@ -257,17 +262,13 @@ updateUserSettings lmsg model =
in in
( { model ( { model
| userSettingsModel = lm | userSettingsModel = lm
, subs =
Sub.batch
[ model.subs
, Sub.map UserSettingsMsg ls
]
} }
, Cmd.map UserSettingsMsg lc , Cmd.map UserSettingsMsg lc
, Sub.map UserSettingsMsg ls
) )
updateCollSettings : Page.CollectiveSettings.Data.Msg -> Model -> ( Model, Cmd Msg ) updateCollSettings : Page.CollectiveSettings.Data.Msg -> Model -> ( Model, Cmd Msg, Sub Msg )
updateCollSettings lmsg model = updateCollSettings lmsg model =
let let
( lm, lc ) = ( lm, lc ) =
@ -277,10 +278,11 @@ updateCollSettings lmsg model =
in in
( { model | collSettingsModel = lm } ( { model | collSettingsModel = lm }
, Cmd.map CollSettingsMsg lc , Cmd.map CollSettingsMsg lc
, Sub.none
) )
updateLogin : Page.Login.Data.Msg -> Model -> ( Model, Cmd Msg ) updateLogin : Page.Login.Data.Msg -> Model -> ( Model, Cmd Msg, Sub Msg )
updateLogin lmsg model = updateLogin lmsg model =
let let
( lm, lc, ar ) = ( lm, lc, ar ) =
@ -295,21 +297,25 @@ updateLogin lmsg model =
in in
( { model | loginModel = lm, flags = newFlags } ( { model | loginModel = lm, flags = newFlags }
, Cmd.map LoginMsg lc , Cmd.map LoginMsg lc
, Sub.none
) )
updateHome : Page.Home.Data.Msg -> Model -> ( Model, Cmd Msg ) updateHome : Page.Home.Data.Msg -> Model -> ( Model, Cmd Msg, Sub Msg )
updateHome lmsg model = updateHome lmsg model =
let let
( lm, lc ) = ( lm, lc, ls ) =
Page.Home.Update.update model.key model.flags model.uiSettings lmsg model.homeModel Page.Home.Update.update model.key model.flags model.uiSettings lmsg model.homeModel
in in
( { model | homeModel = lm } ( { model
| homeModel = lm
}
, Cmd.map HomeMsg lc , Cmd.map HomeMsg lc
, Sub.map HomeMsg ls
) )
updateManageData : Page.ManageData.Data.Msg -> Model -> ( Model, Cmd Msg ) updateManageData : Page.ManageData.Data.Msg -> Model -> ( Model, Cmd Msg, Sub Msg )
updateManageData lmsg model = updateManageData lmsg model =
let let
( lm, lc ) = ( lm, lc ) =
@ -317,14 +323,15 @@ updateManageData lmsg model =
in in
( { model | manageDataModel = lm } ( { model | manageDataModel = lm }
, Cmd.map ManageDataMsg lc , Cmd.map ManageDataMsg lc
, Sub.none
) )
initPage : Model -> Page -> ( Model, Cmd Msg ) initPage : Model -> Page -> ( Model, Cmd Msg, Sub Msg )
initPage model page = initPage model page =
case page of case page of
HomePage -> HomePage ->
Util.Update.andThen1 Util.Update.andThen2
[ updateHome Page.Home.Data.Init [ updateHome Page.Home.Data.Init
, updateQueue Page.Queue.Data.StopRefresh , updateQueue Page.Queue.Data.StopRefresh
] ]
@ -337,14 +344,14 @@ initPage model page =
updateQueue Page.Queue.Data.StopRefresh model updateQueue Page.Queue.Data.StopRefresh model
CollectiveSettingPage -> CollectiveSettingPage ->
Util.Update.andThen1 Util.Update.andThen2
[ updateQueue Page.Queue.Data.StopRefresh [ updateQueue Page.Queue.Data.StopRefresh
, updateCollSettings Page.CollectiveSettings.Data.Init , updateCollSettings Page.CollectiveSettings.Data.Init
] ]
model model
UserSettingPage -> UserSettingPage ->
Util.Update.andThen1 Util.Update.andThen2
[ updateQueue Page.Queue.Data.StopRefresh [ updateQueue Page.Queue.Data.StopRefresh
] ]
model model
@ -362,21 +369,8 @@ initPage model page =
updateQueue Page.Queue.Data.StopRefresh model updateQueue Page.Queue.Data.StopRefresh model
ItemDetailPage id -> ItemDetailPage id ->
let Util.Update.andThen2
updateDetail m__ = [ updateItemDetail (Page.ItemDetail.Data.Init id)
let
( m, c, s ) =
updateItemDetail (Page.ItemDetail.Data.Init id) m__
in
( { m | subs = Sub.batch [ m.subs, s ] }, c )
in
Util.Update.andThen1
[ updateDetail
, updateQueue Page.Queue.Data.StopRefresh , updateQueue Page.Queue.Data.StopRefresh
] ]
model model
noSub : ( Model, Cmd Msg ) -> ( Model, Cmd Msg, Sub Msg )
noSub ( m, c ) =
( m, c, Sub.none )

View File

@ -209,7 +209,7 @@ viewItem settings item =
[ class "item" [ class "item"
, title "Correspondent" , title "Correspondent"
] ]
[ Icons.correspondentIcon [ Icons.correspondentIcon ""
, text " " , text " "
, Util.String.withDefault "-" corr |> text , Util.String.withDefault "-" corr |> text
] ]

View File

@ -1787,7 +1787,7 @@ renderItemInfo settings model =
[ class "item" [ class "item"
, title "Correspondent" , title "Correspondent"
] ]
[ Icons.correspondentIcon [ Icons.correspondentIcon ""
, List.filterMap identity [ model.item.corrOrg, model.item.corrPerson ] , List.filterMap identity [ model.item.corrOrg, model.item.corrPerson ]
|> List.map .name |> List.map .name
|> String.join ", " |> String.join ", "
@ -1994,7 +1994,7 @@ renderEditForm settings model =
, renderDueDateSuggestions model , renderDueDateSuggestions model
] ]
, h4 [ class "ui dividing header" ] , h4 [ class "ui dividing header" ]
[ Icons.correspondentIcon [ Icons.correspondentIcon ""
, text "Correspondent" , text "Correspondent"
] ]
, div [ class "field" ] , div [ class "field" ]

View File

@ -20,12 +20,14 @@ import Comp.DatePicker
import Comp.Dropdown exposing (isDropdownChangeMsg) import Comp.Dropdown exposing (isDropdownChangeMsg)
import Data.Direction exposing (Direction) import Data.Direction exposing (Direction)
import Data.Flags exposing (Flags) import Data.Flags exposing (Flags)
import Data.Icons as Icons
import Data.UiSettings exposing (UiSettings) import Data.UiSettings exposing (UiSettings)
import DatePicker exposing (DatePicker) import DatePicker exposing (DatePicker)
import Html exposing (..) import Html exposing (..)
import Html.Attributes exposing (..) import Html.Attributes exposing (..)
import Html.Events exposing (onCheck, onInput) import Html.Events exposing (onCheck, onInput)
import Http import Http
import Util.Maybe
import Util.Tag import Util.Tag
import Util.Update import Util.Update
@ -52,6 +54,7 @@ type alias Model =
, untilDueDateModel : DatePicker , untilDueDateModel : DatePicker
, untilDueDate : Maybe Int , untilDueDate : Maybe Int
, nameModel : Maybe String , nameModel : Maybe String
, allNameModel : Maybe String
, datePickerInitialized : Bool , datePickerInitialized : Bool
} }
@ -107,6 +110,7 @@ init =
, untilDueDateModel = Comp.DatePicker.emptyModel , untilDueDateModel = Comp.DatePicker.emptyModel
, untilDueDate = Nothing , untilDueDate = Nothing
, nameModel = Nothing , nameModel = Nothing
, allNameModel = Nothing
, datePickerInitialized = False , datePickerInitialized = False
} }
@ -130,6 +134,7 @@ type Msg
| GetEquipResp (Result Http.Error EquipmentList) | GetEquipResp (Result Http.Error EquipmentList)
| GetPersonResp (Result Http.Error ReferenceList) | GetPersonResp (Result Http.Error ReferenceList)
| SetName String | SetName String
| SetAllName String
| ResetForm | ResetForm
@ -152,6 +157,17 @@ getItemSearch model =
let let
e = e =
Api.Model.ItemSearch.empty Api.Model.ItemSearch.empty
amendWildcards s =
if String.startsWith "\"" s && String.endsWith "\"" s then
String.dropLeft 1 s
|> String.dropRight 1
else if String.contains "*" s then
s
else
"*" ++ s ++ "*"
in in
{ e { e
| tagsInclude = Comp.Dropdown.getSelected model.tagInclModel |> List.map .id | tagsInclude = Comp.Dropdown.getSelected model.tagInclModel |> List.map .id
@ -166,7 +182,12 @@ getItemSearch model =
, dateUntil = model.untilDate , dateUntil = model.untilDate
, dueDateFrom = model.fromDueDate , dueDateFrom = model.fromDueDate
, dueDateUntil = model.untilDueDate , dueDateUntil = model.untilDueDate
, name = model.nameModel , name =
model.nameModel
|> Maybe.map amendWildcards
, allNames =
model.allNameModel
|> Maybe.map amendWildcards
} }
@ -444,11 +465,7 @@ update flags settings msg model =
SetName str -> SetName str ->
let let
next = next =
if str == "" then Util.Maybe.fromString str
Nothing
else
Just str
in in
NextState NextState
( { model | nameModel = next } ( { model | nameModel = next }
@ -456,6 +473,17 @@ update flags settings msg model =
) )
(model.nameModel /= next) (model.nameModel /= next)
SetAllName str ->
let
next =
Util.Maybe.fromString str
in
NextState
( { model | allNameModel = next }
, Cmd.none
)
(model.allNameModel /= next)
-- View -- View
@ -463,6 +491,18 @@ update flags settings msg model =
view : UiSettings -> Model -> Html Msg view : UiSettings -> Model -> Html Msg
view settings model = view settings model =
let
formHeader icon headline =
div [ class "ui small dividing header" ]
[ icon
, div [ class "content" ]
[ text headline
]
]
nameIcon =
i [ class "left align icon" ] []
in
div [ class "ui form" ] div [ class "ui form" ]
[ div [ class "inline field" ] [ div [ class "inline field" ]
[ div [ class "ui checkbox" ] [ div [ class "ui checkbox" ]
@ -477,8 +517,21 @@ view settings model =
] ]
] ]
] ]
, formHeader nameIcon "Names"
, div [ class "field" ] , div [ class "field" ]
[ label [] [ text "Name or Notes" ] [ label [] [ text "All Names" ]
, input
[ type_ "text"
, onInput SetAllName
, model.allNameModel |> Maybe.withDefault "" |> value
]
[]
, span [ class "small-info" ]
[ text "Looks in correspondents, concerned, item name and notes."
]
]
, div [ class "field" ]
[ label [] [ text "Name" ]
, input , input
[ type_ "text" [ type_ "text"
, onInput SetName , onInput SetName
@ -486,18 +539,16 @@ view settings model =
] ]
[] []
, span [ class "small-info" ] , span [ class "small-info" ]
[ text "May contain wildcard " [ text "Looks in item name."
]
]
, span [ class "small-info" ]
[ text "Use wildcards "
, code [] [ text "*" ] , code [] [ text "*" ]
, text " at beginning or end" , text " at beginning or end. Added automatically if not "
] , text "present and not quoted."
]
, div [ class "field" ]
[ label [] [ text "Direction" ]
, Html.map DirectionMsg (Comp.Dropdown.view settings model.directionModel)
]
, h3 [ class "ui header" ]
[ text "Tags"
] ]
, formHeader (Icons.tagsIcon "") "Tags"
, div [ class "field" ] , div [ class "field" ]
[ label [] [ text "Include (and)" ] [ label [] [ text "Include (and)" ]
, Html.map TagIncMsg (Comp.Dropdown.view settings model.tagInclModel) , Html.map TagIncMsg (Comp.Dropdown.view settings model.tagInclModel)
@ -506,17 +557,17 @@ view settings model =
[ label [] [ text "Exclude (or)" ] [ label [] [ text "Exclude (or)" ]
, Html.map TagExcMsg (Comp.Dropdown.view settings model.tagExclModel) , Html.map TagExcMsg (Comp.Dropdown.view settings model.tagExclModel)
] ]
, h3 [ class "ui header" ] , formHeader (Icons.correspondentIcon "")
[ case getDirection model of (case getDirection model of
Just Data.Direction.Incoming -> Just Data.Direction.Incoming ->
text "Sender" "Sender"
Just Data.Direction.Outgoing -> Just Data.Direction.Outgoing ->
text "Recipient" "Recipient"
Nothing -> Nothing ->
text "Correspondent" "Correspondent"
] )
, div [ class "field" ] , div [ class "field" ]
[ label [] [ text "Organization" ] [ label [] [ text "Organization" ]
, Html.map OrgMsg (Comp.Dropdown.view settings model.orgModel) , Html.map OrgMsg (Comp.Dropdown.view settings model.orgModel)
@ -525,9 +576,7 @@ view settings model =
[ label [] [ text "Person" ] [ label [] [ text "Person" ]
, Html.map CorrPersonMsg (Comp.Dropdown.view settings model.corrPersonModel) , Html.map CorrPersonMsg (Comp.Dropdown.view settings model.corrPersonModel)
] ]
, h3 [ class "ui header" ] , formHeader Icons.concernedIcon "Concerned"
[ text "Concerned"
]
, div [ class "field" ] , div [ class "field" ]
[ label [] [ text "Person" ] [ label [] [ text "Person" ]
, Html.map ConcPersonMsg (Comp.Dropdown.view settings model.concPersonModel) , Html.map ConcPersonMsg (Comp.Dropdown.view settings model.concPersonModel)
@ -536,9 +585,7 @@ view settings model =
[ label [] [ text "Equipment" ] [ label [] [ text "Equipment" ]
, Html.map ConcEquipmentMsg (Comp.Dropdown.view settings model.concEquipmentModel) , Html.map ConcEquipmentMsg (Comp.Dropdown.view settings model.concEquipmentModel)
] ]
, h3 [ class "ui header" ] , formHeader (Icons.dateIcon "") "Date"
[ text "Date"
]
, div [ class "fields" ] , div [ class "fields" ]
[ div [ class "field" ] [ div [ class "field" ]
[ label [] [ label []
@ -561,9 +608,7 @@ view settings model =
) )
] ]
] ]
, h3 [ class "ui header" ] , formHeader (Icons.dueDateIcon "") "Due Date"
[ text "Due Date"
]
, div [ class "fields" ] , div [ class "fields" ]
[ div [ class "field" ] [ div [ class "field" ]
[ label [] [ label []
@ -586,4 +631,8 @@ view settings model =
) )
] ]
] ]
, formHeader (Icons.directionIcon "") "Direction"
, div [ class "field" ]
[ Html.map DirectionMsg (Comp.Dropdown.view settings model.directionModel)
]
] ]

View File

@ -44,9 +44,9 @@ correspondent =
"address card outline icon" "address card outline icon"
correspondentIcon : Html msg correspondentIcon : String -> Html msg
correspondentIcon = correspondentIcon classes =
i [ class correspondent ] [] i [ class (correspondent ++ " " ++ classes) ] []
date : String date : String

View File

@ -45,12 +45,12 @@ init flags url key =
page = page =
checkPage flags im.page checkPage flags im.page
( m, cmd ) = ( m, cmd, s ) =
if im.page == page then if im.page == page then
App.Update.initPage im page App.Update.initPage im page
else else
( im, Page.goto page ) ( im, Page.goto page, Sub.none )
sessionCheck = sessionCheck =
case m.flags.account of case m.flags.account of
@ -60,7 +60,7 @@ init flags url key =
Nothing -> Nothing ->
Cmd.none Cmd.none
in in
( m ( { m | subs = s }
, Cmd.batch , Cmd.batch
[ cmd [ cmd
, ic , ic

View File

@ -16,6 +16,7 @@ import Data.Flags exposing (Flags)
import Data.Items import Data.Items
import Data.UiSettings exposing (UiSettings) import Data.UiSettings exposing (UiSettings)
import Http import Http
import Throttle exposing (Throttle)
type alias Model = type alias Model =
@ -27,6 +28,7 @@ type alias Model =
, searchOffset : Int , searchOffset : Int
, moreAvailable : Bool , moreAvailable : Bool
, moreInProgress : Bool , moreInProgress : Bool
, throttle : Throttle Msg
} }
@ -36,10 +38,11 @@ init _ =
, itemListModel = Comp.ItemCardList.init , itemListModel = Comp.ItemCardList.init
, searchInProgress = False , searchInProgress = False
, viewMode = Listing , viewMode = Listing
, menuCollapsed = False , menuCollapsed = True
, searchOffset = 0 , searchOffset = 0
, moreAvailable = True , moreAvailable = True
, moreInProgress = False , moreInProgress = False
, throttle = Throttle.create 1
} }
@ -53,6 +56,8 @@ type Msg
| DoSearch | DoSearch
| ToggleSearchMenu | ToggleSearchMenu
| LoadMore | LoadMore
| UpdateThrottle
| SetBasicSearch String
type ViewMode type ViewMode

View File

@ -7,17 +7,15 @@ import Data.Flags exposing (Flags)
import Data.UiSettings exposing (UiSettings) import Data.UiSettings exposing (UiSettings)
import Page exposing (Page(..)) import Page exposing (Page(..))
import Page.Home.Data exposing (..) import Page.Home.Data exposing (..)
import Util.Update import Throttle
import Time
update : Nav.Key -> Flags -> UiSettings -> Msg -> Model -> ( Model, Cmd Msg ) update : Nav.Key -> Flags -> UiSettings -> Msg -> Model -> ( Model, Cmd Msg, Sub Msg )
update key flags settings msg model = update key flags settings msg model =
case msg of case msg of
Init -> Init ->
Util.Update.andThen1 update key flags settings (SearchMenuMsg Comp.SearchMenu.Init) model
[ update key flags settings (SearchMenuMsg Comp.SearchMenu.Init)
]
model
ResetSearch -> ResetSearch ->
let let
@ -34,14 +32,20 @@ update key flags settings msg model =
newModel = newModel =
{ model | searchMenuModel = Tuple.first nextState.modelCmd } { model | searchMenuModel = Tuple.first nextState.modelCmd }
( m2, c2 ) = ( m2, c2, s2 ) =
if nextState.stateChange then if nextState.stateChange && not model.searchInProgress then
doSearch flags settings newModel doSearch flags settings newModel
else else
( newModel, Cmd.none ) withSub ( newModel, Cmd.none )
in in
( m2, Cmd.batch [ c2, Cmd.map SearchMenuMsg (Tuple.second nextState.modelCmd) ] ) ( m2
, Cmd.batch
[ c2
, Cmd.map SearchMenuMsg (Tuple.second nextState.modelCmd)
]
, s2
)
ItemCardListMsg m -> ItemCardListMsg m ->
let let
@ -56,6 +60,7 @@ update key flags settings msg model =
Nothing -> Nothing ->
Cmd.none Cmd.none
in in
withSub
( { model | itemListModel = m2 } ( { model | itemListModel = m2 }
, Cmd.batch [ Cmd.map ItemCardListMsg c2, cmd ] , Cmd.batch [ Cmd.map ItemCardListMsg c2, cmd ]
) )
@ -92,6 +97,7 @@ update key flags settings msg model =
update key flags settings (ItemCardListMsg (Comp.ItemCardList.AddResults list)) m update key flags settings (ItemCardListMsg (Comp.ItemCardList.AddResults list)) m
ItemSearchAddResp (Err _) -> ItemSearchAddResp (Err _) ->
withSub
( { model ( { model
| moreInProgress = False | moreInProgress = False
} }
@ -99,6 +105,7 @@ update key flags settings msg model =
) )
ItemSearchResp (Err _) -> ItemSearchResp (Err _) ->
withSub
( { model ( { model
| searchInProgress = False | searchInProgress = False
} }
@ -110,31 +117,59 @@ update key flags settings msg model =
nm = nm =
{ model | searchOffset = 0 } { model | searchOffset = 0 }
in in
if model.searchInProgress then
withSub ( model, Cmd.none )
else
doSearch flags settings nm doSearch flags settings nm
ToggleSearchMenu -> ToggleSearchMenu ->
withSub
( { model | menuCollapsed = not model.menuCollapsed } ( { model | menuCollapsed = not model.menuCollapsed }
, Cmd.none , Cmd.none
) )
LoadMore -> LoadMore ->
if model.moreAvailable then if model.moreAvailable then
doSearchMore flags settings model doSearchMore flags settings model |> withSub
else else
( model, Cmd.none ) withSub ( model, Cmd.none )
UpdateThrottle ->
let
( newThrottle, cmd ) =
Throttle.update model.throttle
in
withSub ( { model | throttle = newThrottle }, cmd )
SetBasicSearch str ->
let
m =
SearchMenuMsg (Comp.SearchMenu.SetAllName str)
in
update key flags settings m model
doSearch : Flags -> UiSettings -> Model -> ( Model, Cmd Msg )
--- Helpers
doSearch : Flags -> UiSettings -> Model -> ( Model, Cmd Msg, Sub Msg )
doSearch flags settings model = doSearch flags settings model =
let let
cmd = searchCmd =
doSearchCmd flags settings 0 model doSearchCmd flags settings 0 model
( newThrottle, cmd ) =
Throttle.try searchCmd model.throttle
in in
withSub
( { model ( { model
| searchInProgress = True | searchInProgress = cmd /= Cmd.none
, viewMode = Listing , viewMode = Listing
, searchOffset = 0 , searchOffset = 0
, throttle = newThrottle
} }
, cmd , cmd
) )
@ -149,3 +184,13 @@ doSearchMore flags settings model =
( { model | moreInProgress = True, viewMode = Listing } ( { model | moreInProgress = True, viewMode = Listing }
, cmd , cmd
) )
withSub : ( Model, Cmd Msg ) -> ( Model, Cmd Msg, Sub Msg )
withSub ( m, c ) =
( m
, c
, Throttle.ifNeeded
(Time.every 150 (\_ -> UpdateThrottle))
m.throttle
)

View File

@ -1,11 +1,12 @@
module Page.Home.View exposing (view) module Page.Home.View exposing (view)
import Api.Model.ItemSearch
import Comp.ItemCardList import Comp.ItemCardList
import Comp.SearchMenu import Comp.SearchMenu
import Data.UiSettings exposing (UiSettings) import Data.UiSettings exposing (UiSettings)
import Html exposing (..) import Html exposing (..)
import Html.Attributes exposing (..) import Html.Attributes exposing (..)
import Html.Events exposing (onClick) import Html.Events exposing (onClick, onInput)
import Page exposing (Page(..)) import Page exposing (Page(..))
import Page.Home.Data exposing (..) import Page.Home.Data exposing (..)
@ -47,8 +48,15 @@ view settings model =
, onClick DoSearch , onClick DoSearch
, title "Run search query" , title "Run search query"
, href "" , href ""
, disabled model.searchInProgress
] ]
[ i [ class "ui search icon" ] [] [ i
[ classList
[ ( "search icon", not model.searchInProgress )
, ( "loading spinner icon", model.searchInProgress )
]
]
[]
] ]
] ]
] ]
@ -68,24 +76,48 @@ view settings model =
[ div [ div
[ classList [ classList
[ ( "invisible hidden", not model.menuCollapsed ) [ ( "invisible hidden", not model.menuCollapsed )
, ( "ui segment container", True ) , ( "ui menu container", True )
] ]
] ]
[ a [ a
[ class "ui basic large circular label" [ class "item"
, onClick ToggleSearchMenu , onClick ToggleSearchMenu
, href "#" , href "#"
, title "Open search menu"
]
[ i [ class "angle left icon" ] []
, i [ class "icons" ]
[ i [ class "grey bars icon" ] []
, i [ class "bottom left corner search icon" ] []
, if hasMoreSearch model then
i [ class "top right blue corner circle icon" ] []
else
span [ class "hidden invisible" ] []
]
]
, div [ class "ui category search item" ]
[ div [ class "ui transparent icon input" ]
[ input
[ type_ "text"
, placeholder "Basic search"
, onInput SetBasicSearch
, Maybe.map value model.searchMenuModel.allNameModel
|> Maybe.withDefault (value "")
]
[]
, i
[ classList
[ ( "search link icon", not model.searchInProgress )
, ( "loading spinner icon", model.searchInProgress )
]
]
[]
] ]
[ i [ class "search icon" ] []
, text "Search Menu"
] ]
] ]
, case model.viewMode of , case model.viewMode of
Listing -> Listing ->
if model.searchInProgress then
resultPlaceholder
else
Html.map ItemCardListMsg Html.map ItemCardListMsg
(Comp.ItemCardList.view settings model.itemListModel) (Comp.ItemCardList.view settings model.itemListModel)
@ -125,32 +157,13 @@ view settings model =
] ]
resultPlaceholder : Html Msg hasMoreSearch : Model -> Bool
resultPlaceholder = hasMoreSearch model =
div [ class "ui basic segment" ] let
[ div [ class "ui active inverted dimmer" ] is =
[ div [ class "ui medium text loader" ] Comp.SearchMenu.getItemSearch model.searchMenuModel
[ text "Searching "
] is_ =
] { is | allNames = Nothing }
, div [ class "ui middle aligned very relaxed divided basic list segment" ] in
[ div [ class "item" ] is_ /= Api.Model.ItemSearch.empty
[ div [ class "ui fluid placeholder" ]
[ div [ class "full line" ] []
, div [ class "full line" ] []
]
]
, div [ class "item" ]
[ div [ class "ui fluid placeholder" ]
[ div [ class "full line" ] []
, div [ class "full line" ] []
]
]
, div [ class "item" ]
[ div [ class "ui fluid placeholder" ]
[ div [ class "full line" ] []
, div [ class "full line" ] []
]
]
]
]

View File

@ -1,4 +1,4 @@
module Util.Update exposing (andThen1) module Util.Update exposing (andThen1, andThen2)
andThen1 : List (a -> ( a, Cmd b )) -> a -> ( a, Cmd b ) andThen1 : List (a -> ( a, Cmd b )) -> a -> ( a, Cmd b )
@ -16,3 +16,23 @@ andThen1 fs a =
in in
List.foldl update init fs List.foldl update init fs
|> Tuple.mapSecond Cmd.batch |> Tuple.mapSecond Cmd.batch
andThen2 : List (model -> ( model, Cmd msg, Sub msg )) -> model -> ( model, Cmd msg, Sub msg )
andThen2 fs m =
let
init =
( m, [], [] )
update el ( m1, c1, s1 ) =
let
( m2, c2, s2 ) =
el m1
in
( m2, c2 :: c1, s2 :: s1 )
combine ( m1, cl, sl ) =
( m1, Cmd.batch cl, Sub.batch sl )
in
List.foldl update init fs
|> combine