Some predefined boxes for a dashboard

This commit is contained in:
eikek 2022-01-26 21:22:20 +01:00
parent 0337be98f9
commit 370679daed
25 changed files with 1004 additions and 80 deletions

View File

@ -1961,7 +1961,7 @@ paths:
- $ref: "#/components/parameters/bookmarkId" - $ref: "#/components/parameters/bookmarkId"
delete: delete:
operationId: "sec-querybookmark-delete" operationId: "sec-querybookmark-delete"
tags: [Query Bookmark] tags: [Query Bookmarks]
summary: Delete a bookmark. summary: Delete a bookmark.
description: | description: |
Deletes a bookmarks by its id. Deletes a bookmarks by its id.

View File

@ -101,7 +101,9 @@ module Api exposing
, itemDetailShare , itemDetailShare
, itemIndexSearch , itemIndexSearch
, itemSearch , itemSearch
, itemSearchBookmark
, itemSearchStats , itemSearchStats
, itemSearchStatsBookmark
, login , login
, loginSession , loginSession
, logout , logout
@ -2028,24 +2030,70 @@ itemIndexSearch flags query receive =
} }
itemSearch : Flags -> ItemQuery -> (Result Http.Error ItemLightList -> msg) -> Cmd msg itemSearchTask : Flags -> ItemQuery -> Task.Task Http.Error ItemLightList
itemSearch flags search receive = itemSearchTask flags search =
Http2.authPost Http2.authTask
{ url = flags.config.baseUrl ++ "/api/v1/sec/item/search" { url = flags.config.baseUrl ++ "/api/v1/sec/item/search"
, method = "POST"
, headers = []
, account = getAccount flags , account = getAccount flags
, body = Http.jsonBody (Api.Model.ItemQuery.encode search) , body = Http.jsonBody (Api.Model.ItemQuery.encode search)
, expect = Http.expectJson receive Api.Model.ItemLightList.decoder , resolver = Http2.jsonResolver Api.Model.ItemLightList.decoder
, timeout = Nothing
}
itemSearch : Flags -> ItemQuery -> (Result Http.Error ItemLightList -> msg) -> Cmd msg
itemSearch flags search receive =
itemSearchTask flags search |> Task.attempt receive
{-| Same as `itemSearch` but interprets the `query` field as a bookmark id.
-}
itemSearchBookmark : Flags -> ItemQuery -> (Result Http.Error ItemLightList -> msg) -> Cmd msg
itemSearchBookmark flags bmSearch receive =
let
getBookmark =
getBookmarkByIdTask flags bmSearch.query
|> Task.map (\bm -> { bmSearch | query = bm.query })
search q =
itemSearchTask flags q
in
Task.andThen search getBookmark
|> Task.attempt receive
itemSearchStatsTask : Flags -> ItemQuery -> Task.Task Http.Error SearchStats
itemSearchStatsTask flags search =
Http2.authTask
{ url = flags.config.baseUrl ++ "/api/v1/sec/item/searchStats"
, method = "POST"
, headers = []
, account = getAccount flags
, body = Http.jsonBody (Api.Model.ItemQuery.encode search)
, resolver = Http2.jsonResolver Api.Model.SearchStats.decoder
, timeout = Nothing
} }
itemSearchStats : Flags -> ItemQuery -> (Result Http.Error SearchStats -> msg) -> Cmd msg itemSearchStats : Flags -> ItemQuery -> (Result Http.Error SearchStats -> msg) -> Cmd msg
itemSearchStats flags search receive = itemSearchStats flags search receive =
Http2.authPost itemSearchStatsTask flags search |> Task.attempt receive
{ url = flags.config.baseUrl ++ "/api/v1/sec/item/searchStats"
, account = getAccount flags
, body = Http.jsonBody (Api.Model.ItemQuery.encode search) itemSearchStatsBookmark : Flags -> ItemQuery -> (Result Http.Error SearchStats -> msg) -> Cmd msg
, expect = Http.expectJson receive Api.Model.SearchStats.decoder itemSearchStatsBookmark flags search receive =
} let
getBookmark =
getBookmarkByIdTask flags search.query
|> Task.map (\bm -> { search | query = bm.query })
getStats q =
itemSearchStatsTask flags q
in
Task.andThen getStats getBookmark
|> Task.attempt receive
itemDetail : Flags -> String -> (Result Http.Error ItemDetail -> msg) -> Cmd msg itemDetail : Flags -> String -> (Result Http.Error ItemDetail -> msg) -> Cmd msg
@ -2335,6 +2383,21 @@ getBookmarksTask flags =
} }
getBookmarkByIdTask : Flags -> String -> Task.Task Http.Error BookmarkedQuery
getBookmarkByIdTask flags id =
let
findBm all =
Data.Bookmarks.findById id all
mapNotFound maybeBookmark =
Maybe.map Task.succeed maybeBookmark
|> Maybe.withDefault (Task.fail (Http.BadStatus 404))
in
getBookmarksTask flags
|> Task.map findBm
|> Task.andThen mapNotFound
getBookmarks : Flags -> (Result Http.Error AllBookmarks -> msg) -> Cmd msg getBookmarks : Flags -> (Result Http.Error AllBookmarks -> msg) -> Cmd msg
getBookmarks flags receive = getBookmarks flags receive =
let let

View File

@ -18,15 +18,18 @@ import Api.Model.BasicResult exposing (BasicResult)
import Api.Model.VersionInfo exposing (VersionInfo) import Api.Model.VersionInfo exposing (VersionInfo)
import Browser exposing (UrlRequest) import Browser exposing (UrlRequest)
import Browser.Navigation exposing (Key) import Browser.Navigation exposing (Key)
import Data.Dashboard exposing (Dashboard)
import Data.Flags exposing (Flags) import Data.Flags exposing (Flags)
import Data.ServerEvent exposing (ServerEvent) import Data.ServerEvent exposing (ServerEvent)
import Data.UiSettings exposing (StoredUiSettings, UiSettings) import Data.UiSettings exposing (StoredUiSettings, UiSettings)
import Data.UiTheme exposing (UiTheme) import Data.UiTheme exposing (UiTheme)
import Http import Http
import Messages
import Messages.UiLanguage exposing (UiLanguage) import Messages.UiLanguage exposing (UiLanguage)
import Page exposing (Page(..)) import Page exposing (Page(..))
import Page.CollectiveSettings.Data import Page.CollectiveSettings.Data
import Page.Dashboard.Data import Page.Dashboard.Data
import Page.Dashboard.DefaultDashboard
import Page.ItemDetail.Data import Page.ItemDetail.Data
import Page.Login.Data import Page.Login.Data
import Page.ManageData.Data import Page.ManageData.Data
@ -102,6 +105,7 @@ init key url flags_ settings =
( dbm, dbc ) = ( dbm, dbc ) =
Page.Dashboard.Data.init flags Page.Dashboard.Data.init flags
(Page.Dashboard.DefaultDashboard.getDefaultDashboard flags settings)
searchViewMode = searchViewMode =
if settings.searchMenuVisible then if settings.searchMenuVisible then
@ -214,9 +218,4 @@ defaultPage _ =
getUiLanguage : Model -> UiLanguage getUiLanguage : Model -> UiLanguage
getUiLanguage model = getUiLanguage model =
case model.flags.account of Data.UiSettings.getUiLanguage model.flags model.uiSettings model.anonymousUiLang
Just _ ->
model.uiSettings.uiLang
Nothing ->
model.anonymousUiLang

View File

@ -364,6 +364,7 @@ applyClientSettings texts model settings =
, setTheme , setTheme
, Sub.none , Sub.none
) )
, updateDashboard texts Page.Dashboard.Data.reloadUiSettings
, updateUserSettings texts Page.UserSettings.Data.UpdateSettings , updateUserSettings texts Page.UserSettings.Data.UpdateSettings
, updateSearch texts Page.Search.Data.UiSettingsUpdated , updateSearch texts Page.Search.Data.UiSettingsUpdated
, updateItemDetail texts Page.ItemDetail.Data.UiSettingsUpdated , updateItemDetail texts Page.ItemDetail.Data.UiSettingsUpdated
@ -375,7 +376,12 @@ updateDashboard : Messages -> Page.Dashboard.Data.Msg -> Model -> ( Model, Cmd M
updateDashboard texts lmsg model = updateDashboard texts lmsg model =
let let
( dbm, dbc, dbs ) = ( dbm, dbc, dbs ) =
Page.Dashboard.Update.update texts.dashboard model.key model.flags lmsg model.dashboardModel Page.Dashboard.Update.update texts.dashboard
model.uiSettings
model.key
model.flags
lmsg
model.dashboardModel
in in
( { model | dashboardModel = dbm } ( { model | dashboardModel = dbm }
, Cmd.map DashboardMsg dbc , Cmd.map DashboardMsg dbc

View File

@ -207,7 +207,7 @@ loadingDimmer : { label : String, active : Bool } -> Html msg
loadingDimmer cfg = loadingDimmer cfg =
let let
content = content =
div [ class "text-gray-200" ] div [ class "text-gray-200 " ]
[ i [ class "fa fa-circle-notch animate-spin" ] [] [ i [ class "fa fa-circle-notch animate-spin" ] []
, span [ class "ml-2" ] , span [ class "ml-2" ]
[ text cfg.label [ text cfg.label

View File

@ -0,0 +1,184 @@
module Comp.BoxQueryView exposing (..)
import Api
import Api.Model.ItemLight exposing (ItemLight)
import Api.Model.ItemLightList exposing (ItemLightList)
import Api.Model.ItemQuery exposing (ItemQuery)
import Comp.Basic
import Data.BoxContent exposing (QueryData, SearchQuery(..))
import Data.Flags exposing (Flags)
import Data.ItemTemplate as IT
import Data.Items
import Data.SearchMode
import Html exposing (Html, a, div, i, table, tbody, td, text, th, thead, tr)
import Html.Attributes exposing (class, classList)
import Http
import Messages.Comp.BoxQueryView exposing (Texts)
import Page exposing (Page(..))
import Styles
type alias Model =
{ results : ViewResult
, meta : QueryData
}
type ViewResult
= Loading
| Loaded ItemLightList
| Failed Http.Error
type Msg
= ItemsResp (Result Http.Error ItemLightList)
init : Flags -> QueryData -> ( Model, Cmd Msg )
init flags data =
( { results = Loading
, meta = data
}
, case data.query of
SearchQueryString q ->
Api.itemSearch flags (mkQuery q data) ItemsResp
SearchQueryBookmark bmId ->
Api.itemSearchBookmark flags (mkQuery bmId data) ItemsResp
)
--- Update
update : Msg -> Model -> ( Model, Cmd Msg )
update msg model =
case msg of
ItemsResp (Ok list) ->
( { model | results = Loaded list }, Cmd.none )
ItemsResp (Err err) ->
( { model | results = Failed err }, Cmd.none )
--- View
view : Texts -> Model -> Html Msg
view texts model =
case model.results of
Loading ->
div [ class "h-24 " ]
[ Comp.Basic.loadingDimmer
{ label = ""
, active = True
}
]
Failed err ->
div
[ class "py-4"
, class Styles.errorMessage
]
[ text texts.errorOccurred
, text ": "
, text (texts.httpError err)
]
Loaded list ->
if list.groups == [] then
viewEmpty texts
else
viewItems texts model.meta list
viewItems : Texts -> QueryData -> ItemLightList -> Html Msg
viewItems texts meta list =
let
items =
Data.Items.flatten list
in
table [ class "w-full divide-y dark:divide-slate-500" ]
(viewItemHead meta ++ [ tbody [] <| List.map (viewItemRow texts meta) items ])
viewItemHead : QueryData -> List (Html Msg)
viewItemHead meta =
case meta.header of
[] ->
[]
labels ->
[ thead []
[ tr []
(List.map (\n -> th [ class "text-left text-sm" ] [ text n ]) labels)
]
]
viewItemRow : Texts -> QueryData -> ItemLight -> Html Msg
viewItemRow texts meta item =
let
( col1, cols ) =
getColumns meta
render tpl =
IT.render tpl texts.templateCtx item
td1 =
td [ class "py-2 px-1" ]
[ a
[ class Styles.link
, Page.href (ItemDetailPage item.id)
]
[ text (render col1)
]
]
tdRem index col =
td
[ class "py-2 px-1"
, classList [ ( "hidden md:table-cell", index > 1 ) ]
]
[ text (render col)
]
in
tr []
(td1 :: List.indexedMap tdRem cols)
viewEmpty : Texts -> Html Msg
viewEmpty texts =
div [ class "flex justify-center items-center h-full" ]
[ div [ class "px-4 py-4 text-center align-middle text-lg" ]
[ i [ class "fa fa-eraser mr-2" ] []
, text texts.noResults
]
]
--- Helpers
getColumns : QueryData -> ( IT.ItemTemplate, List IT.ItemTemplate )
getColumns meta =
case meta.columns of
x :: xs ->
( x, xs )
[] ->
( IT.name, [ IT.correspondent, IT.dateShort ] )
mkQuery : String -> QueryData -> ItemQuery
mkQuery q meta =
{ query = q
, limit = Just meta.limit
, offset = Nothing
, searchMode = Just <| Data.SearchMode.asString Data.SearchMode.Normal
, withDetails = Just meta.details
}

View File

@ -0,0 +1,160 @@
module Comp.BoxSummaryView exposing (..)
import Api
import Api.Model.ItemQuery exposing (ItemQuery)
import Api.Model.SearchStats exposing (SearchStats)
import Comp.Basic
import Comp.SearchStatsView
import Data.BoxContent exposing (SearchQuery(..), SummaryData, SummaryShow(..))
import Data.Flags exposing (Flags)
import Html exposing (Html, div, text)
import Html.Attributes exposing (class)
import Http
import Messages.Comp.BoxSummaryView exposing (Texts)
import Styles
import Util.List
type alias Model =
{ results : ViewResult
, show : SummaryShow
}
type ViewResult
= Loading
| Loaded SearchStats
| Failed Http.Error
type Msg
= StatsResp (Result Http.Error SearchStats)
init : Flags -> SummaryData -> ( Model, Cmd Msg )
init flags data =
( { results = Loading
, show = data.show
}
, case data.query of
SearchQueryString q ->
Api.itemSearchStats flags (mkQuery q) StatsResp
SearchQueryBookmark bmId ->
Api.itemSearchStatsBookmark flags (mkQuery bmId) StatsResp
)
--- Update
update : Msg -> Model -> ( Model, Cmd Msg )
update msg model =
case msg of
StatsResp (Ok stats) ->
( { model | results = Loaded stats }, Cmd.none )
StatsResp (Err err) ->
( { model | results = Failed err }, Cmd.none )
--- View
view : Texts -> Model -> Html Msg
view texts model =
case model.results of
Loading ->
div [ class "h-24 " ]
[ Comp.Basic.loadingDimmer
{ label = ""
, active = True
}
]
Failed err ->
div
[ class "py-4"
, class Styles.errorMessage
]
[ text texts.errorOccurred
, text ": "
, text (texts.httpError err)
]
Loaded stats ->
case model.show of
Data.BoxContent.SummaryShowFields flag ->
Comp.SearchStatsView.view2
texts.statsView
flag
""
stats
SummaryShowGeneral ->
viewGeneral texts stats
viewGeneral : Texts -> SearchStats -> Html Msg
viewGeneral texts stats =
let
tagCount =
List.length stats.tagCloud.items
fieldCount =
List.length stats.fieldStats
orgCount =
List.length stats.corrOrgStats
persCount =
(stats.corrPersStats ++ stats.concPersStats)
|> List.map (.ref >> .id)
|> Util.List.distinct
|> List.length
equipCount =
List.length stats.concEquipStats
mklabel name =
div [ class "py-1 text-lg" ] [ text name ]
value num =
div [ class "py-1 font-mono text-lg" ] [ text <| String.fromInt num ]
in
div [ class "opacity-90" ]
[ div [ class "flex flex-row" ]
[ div [ class "flex flex-col mr-4" ]
[ mklabel texts.basics.items
, mklabel texts.basics.tags
, mklabel texts.basics.customFields
, mklabel texts.basics.organization
, mklabel texts.basics.person
, mklabel texts.basics.equipment
]
, div [ class "flex flex-col" ]
[ value stats.count
, value tagCount
, value fieldCount
, value orgCount
, value persCount
, value equipCount
]
]
]
--- Helpers
mkQuery : String -> ItemQuery
mkQuery query =
{ query = query
, limit = Nothing
, offset = Nothing
, searchMode = Nothing
, withDetails = Nothing
}

View File

@ -1,28 +1,180 @@
module Comp.BoxView exposing (..) module Comp.BoxView exposing (..)
import Comp.BoxQueryView
import Comp.BoxSummaryView
import Data.Box exposing (Box) import Data.Box exposing (Box)
import Data.BoxContent exposing (BoxContent(..), MessageData)
import Data.Flags exposing (Flags) import Data.Flags exposing (Flags)
import Html exposing (Html, div) import Html exposing (Html, div, text)
import Html.Attributes exposing (class, classList)
import Markdown
import Messages.Comp.BoxView exposing (Texts)
import Styles as S
type alias Model = type alias Model =
{} { box : Box
, content : ContentModel
}
type ContentModel
= ContentMessage Data.BoxContent.MessageData
| ContentUpload (Maybe String)
| ContentQuery Comp.BoxQueryView.Model
| ContentSummary Comp.BoxSummaryView.Model
type Msg type Msg
= Dummy = QueryMsg Comp.BoxQueryView.Msg
| SummaryMsg Comp.BoxSummaryView.Msg
init : Flags -> Box -> ( Model, Cmd Msg ) init : Flags -> Box -> ( Model, Cmd Msg )
init flags box = init flags box =
( {}, Cmd.none ) let
( cm, cc ) =
contentInit flags box.content
in
( { box = box
, content = cm
}
, cc
)
contentInit : Flags -> BoxContent -> ( ContentModel, Cmd Msg )
contentInit flags content =
case content of
BoxMessage data ->
( ContentMessage data, Cmd.none )
BoxUpload source ->
( ContentUpload source, Cmd.none )
BoxQuery data ->
let
( qm, qc ) =
Comp.BoxQueryView.init flags data
in
( ContentQuery qm, Cmd.map QueryMsg qc )
BoxSummary data ->
let
( sm, sc ) =
Comp.BoxSummaryView.init flags data
in
( ContentSummary sm, Cmd.map SummaryMsg sc )
--- Update --- Update
update : Msg -> Model -> ( Model, Cmd Msg )
update msg model =
case msg of
QueryMsg lm ->
case model.content of
ContentQuery qm ->
let
( cm, cc ) =
Comp.BoxQueryView.update lm qm
in
( { model | content = ContentQuery cm }, Cmd.map QueryMsg cc )
_ ->
unit model
SummaryMsg lm ->
case model.content of
ContentSummary qm ->
let
( cm, cc ) =
Comp.BoxSummaryView.update lm qm
in
( { model | content = ContentSummary cm }, Cmd.map SummaryMsg cc )
_ ->
unit model
unit : Model -> ( Model, Cmd Msg )
unit model =
( model, Cmd.none )
--- View --- View
view : Model -> Html Msg view : Texts -> Model -> Html Msg
view model = view texts model =
div [] [] div
[ classList [ ( S.box ++ "rounded", model.box.decoration ) ]
, class (spanStyle model.box)
, class "relative h-full"
, classList [ ( "hidden", not model.box.visible ) ]
]
[ boxHeader model
, div [ class "px-2 py-1 h-5/6" ]
[ boxContent texts model
]
]
boxHeader : Model -> Html Msg
boxHeader model =
div
[ class "border-b dark:border-slate-500 flex flex-row py-1 bg-blue-50 dark:bg-slate-700 rounded-t"
, classList [ ( "hidden", not model.box.decoration || model.box.name == "" ) ]
]
[ div [ class "flex text-lg tracking-medium italic px-2" ]
[ text model.box.name
]
]
boxContent : Texts -> Model -> Html Msg
boxContent texts model =
case model.content of
ContentMessage m ->
messageContent m
ContentUpload sourceId ->
Debug.todo "not implemented"
ContentQuery qm ->
Html.map QueryMsg
(Comp.BoxQueryView.view texts.queryView qm)
ContentSummary qm ->
Html.map SummaryMsg
(Comp.BoxSummaryView.view texts.summaryView qm)
spanStyle : Box -> String
spanStyle box =
case box.colspan of
1 ->
""
2 ->
"col-span-1 md:col-span-2"
3 ->
"col-span-1 md:col-span-3"
4 ->
"col-span-1 md:col-span-4"
_ ->
"col-span-1 md:col-span-5"
messageContent : MessageData -> Html msg
messageContent data =
div [ class "markdown-preview" ]
[ Markdown.toHtml [] data.title
, Markdown.toHtml [] data.body
]

View File

@ -1,4 +1,4 @@
module Comp.DashboardView exposing (Model, Msg, init, view, viewBox) module Comp.DashboardView exposing (Model, Msg, init, update, view, viewBox)
import Comp.BoxView import Comp.BoxView
import Data.Box exposing (Box) import Data.Box exposing (Box)
@ -7,16 +7,17 @@ import Data.Flags exposing (Flags)
import Dict exposing (Dict) import Dict exposing (Dict)
import Html exposing (Html, div) import Html exposing (Html, div)
import Html.Attributes exposing (class) import Html.Attributes exposing (class)
import Messages.Comp.DashboardView exposing (Texts)
type alias Model = type alias Model =
{ dashboard : Dashboard { dashboard : Dashboard
, boxModels : List Comp.BoxView.Model , boxModels : Dict Int Comp.BoxView.Model
} }
type Msg type Msg
= BoxMsg Comp.BoxView.Msg = BoxMsg Int Comp.BoxView.Msg
init : Flags -> Dashboard -> ( Model, Cmd Msg ) init : Flags -> Dashboard -> ( Model, Cmd Msg )
@ -24,32 +25,61 @@ init flags db =
let let
( boxModels, cmds ) = ( boxModels, cmds ) =
List.map (Comp.BoxView.init flags) db.boxes List.map (Comp.BoxView.init flags) db.boxes
|> List.map (Tuple.mapSecond <| Cmd.map BoxMsg) |> List.indexedMap (\a -> \( bm, bc ) -> ( bm, Cmd.map (BoxMsg a) bc ))
|> List.unzip |> List.unzip
in in
( { dashboard = db ( { dashboard = db
, boxModels = boxModels , boxModels =
List.indexedMap Tuple.pair boxModels
|> Dict.fromList
} }
, Cmd.batch cmds , Cmd.batch cmds
) )
--- Update
update : Msg -> Model -> ( Model, Cmd Msg )
update msg model =
case msg of
BoxMsg index lm ->
case Dict.get index model.boxModels of
Just bm ->
let
( cm, cc ) =
Comp.BoxView.update lm bm
in
( { model | boxModels = Dict.insert index cm model.boxModels }
, Cmd.map (BoxMsg index) cc
)
Nothing ->
unit model
unit : Model -> ( Model, Cmd Msg )
unit model =
( model, Cmd.none )
--- View --- View
view : Model -> Html Msg view : Texts -> Model -> Html Msg
view model = view texts model =
div div
[ class (gridStyle model.dashboard) [ class (gridStyle model.dashboard)
] ]
(List.map viewBox model.boxModels) (List.indexedMap (viewBox texts) <| Dict.values model.boxModels)
viewBox : Comp.BoxView.Model -> Html Msg viewBox : Texts -> Int -> Comp.BoxView.Model -> Html Msg
viewBox box = viewBox texts index box =
Html.map BoxMsg Html.map (BoxMsg index)
(Comp.BoxView.view box) (Comp.BoxView.view texts.boxView box)

View File

@ -8,6 +8,7 @@
module Comp.SearchStatsView exposing module Comp.SearchStatsView exposing
( nameOrLabel ( nameOrLabel
, sortFields , sortFields
, view
, view2 , view2
) )
@ -36,8 +37,13 @@ sortFields fields =
--- View2 --- View2
view2 : Texts -> String -> SearchStats -> Html msg view : Texts -> String -> SearchStats -> Html msg
view2 texts classes stats = view texts classes stats =
view2 texts True classes stats
view2 : Texts -> Bool -> String -> SearchStats -> Html msg
view2 texts showCount classes stats =
let let
isNumField f = isNumField f =
f.sum > 0 f.sum > 0
@ -78,7 +84,10 @@ view2 texts classes stats =
in in
div [ class classes ] div [ class classes ]
[ div [ class "flex flex-col md:flex-row" ] [ div [ class "flex flex-col md:flex-row" ]
[ div [ class "px-8 py-4" ] [ div
[ class "px-8 py-4"
, classList [ ( "hidden", not showCount ) ]
]
[ B.stats [ B.stats
{ rootClass = "" { rootClass = ""
, valueClass = "text-4xl" , valueClass = "text-4xl"

View File

@ -11,6 +11,7 @@ module Data.Bookmarks exposing
, bookmarksDecoder , bookmarksDecoder
, empty , empty
, exists , exists
, findById
, sort , sort
) )
@ -34,6 +35,12 @@ type alias Bookmarks =
List BookmarkedQuery List BookmarkedQuery
findById : String -> Bookmarks -> Maybe BookmarkedQuery
findById id all =
List.filter (\e -> e.id == id) all
|> List.head
{-| Checks wether a bookmark of this name already exists. {-| Checks wether a bookmark of this name already exists.
-} -}
exists : String -> Bookmarks -> Bool exists : String -> Bookmarks -> Bool

View File

@ -1,10 +1,17 @@
module Data.BoxContent exposing (BoxContent(..), MessageData, QueryData, SummaryData) module Data.BoxContent exposing
( BoxContent(..)
, MessageData
, QueryData
, SearchQuery(..)
, SummaryData
, SummaryShow(..)
)
import Data.ItemArrange exposing (ItemArrange) import Data.ItemTemplate exposing (ItemTemplate)
type BoxContent type BoxContent
= BoxUpload = BoxUpload (Maybe String)
| BoxMessage MessageData | BoxMessage MessageData
| BoxQuery QueryData | BoxQuery QueryData
| BoxSummary SummaryData | BoxSummary SummaryData
@ -17,11 +24,25 @@ type alias MessageData =
type alias QueryData = type alias QueryData =
{ query : String { query : SearchQuery
, view : ItemArrange , limit : Int
, details : Bool
, header : List String
, columns : List ItemTemplate
} }
type alias SummaryData = type alias SummaryData =
{ query : String { query : SearchQuery
, show : SummaryShow
} }
type SummaryShow
= SummaryShowFields Bool
| SummaryShowGeneral
type SearchQuery
= SearchQueryString String
| SearchQueryBookmark String

View File

@ -18,6 +18,7 @@ module Data.UiSettings exposing
, defaults , defaults
, fieldHidden , fieldHidden
, fieldVisible , fieldVisible
, getUiLanguage
, merge , merge
, mergeDefaults , mergeDefaults
, pdfUrl , pdfUrl
@ -444,6 +445,16 @@ pdfUrl settings flags originalUrl =
Data.Pdf.serverUrl originalUrl Data.Pdf.serverUrl originalUrl
getUiLanguage : Flags -> UiSettings -> UiLanguage -> UiLanguage
getUiLanguage flags settings default =
case flags.account of
Just _ ->
settings.uiLang
Nothing ->
default
--- Helpers --- Helpers

View File

@ -0,0 +1,46 @@
module Messages.Comp.BoxQueryView exposing (Texts, de, gb)
import Data.ItemTemplate as IT
import Http
import Messages.Basics
import Messages.Comp.HttpError
import Messages.Data.Direction
import Messages.DateFormat as DF
import Messages.UiLanguage
type alias Texts =
{ httpError : Http.Error -> String
, errorOccurred : String
, basics : Messages.Basics.Texts
, noResults : String
, templateCtx : IT.TemplateContext
}
gb : Texts
gb =
{ httpError = Messages.Comp.HttpError.gb
, errorOccurred = "Error retrieving data."
, basics = Messages.Basics.gb
, noResults = "No items found."
, templateCtx =
{ dateFormatLong = DF.formatDateLong Messages.UiLanguage.English
, dateFormatShort = DF.formatDateShort Messages.UiLanguage.English
, directionLabel = Messages.Data.Direction.gb
}
}
de : Texts
de =
{ httpError = Messages.Comp.HttpError.de
, errorOccurred = "Fehler beim Laden der Daten."
, basics = Messages.Basics.de
, noResults = "Keine Dokumente gefunden."
, templateCtx =
{ dateFormatLong = DF.formatDateLong Messages.UiLanguage.German
, dateFormatShort = DF.formatDateShort Messages.UiLanguage.German
, directionLabel = Messages.Data.Direction.de
}
}

View File

@ -0,0 +1,32 @@
module Messages.Comp.BoxSummaryView exposing (Texts, de, gb)
import Http
import Messages.Basics
import Messages.Comp.HttpError
import Messages.Comp.SearchStatsView
type alias Texts =
{ httpError : Http.Error -> String
, errorOccurred : String
, statsView : Messages.Comp.SearchStatsView.Texts
, basics : Messages.Basics.Texts
}
gb : Texts
gb =
{ httpError = Messages.Comp.HttpError.gb
, errorOccurred = "Error retrieving data."
, statsView = Messages.Comp.SearchStatsView.gb
, basics = Messages.Basics.gb
}
de : Texts
de =
{ httpError = Messages.Comp.HttpError.de
, errorOccurred = "Fehler beim Laden der Daten."
, statsView = Messages.Comp.SearchStatsView.de
, basics = Messages.Basics.de
}

View File

@ -0,0 +1,24 @@
module Messages.Comp.BoxView exposing (Texts, de, gb)
import Messages.Comp.BoxQueryView
import Messages.Comp.BoxSummaryView
type alias Texts =
{ queryView : Messages.Comp.BoxQueryView.Texts
, summaryView : Messages.Comp.BoxSummaryView.Texts
}
gb : Texts
gb =
{ queryView = Messages.Comp.BoxQueryView.gb
, summaryView = Messages.Comp.BoxSummaryView.gb
}
de : Texts
de =
{ queryView = Messages.Comp.BoxQueryView.de
, summaryView = Messages.Comp.BoxSummaryView.de
}

View File

@ -0,0 +1,20 @@
module Messages.Comp.DashboardView exposing (Texts, de, gb)
import Messages.Comp.BoxView
type alias Texts =
{ boxView : Messages.Comp.BoxView.Texts
}
gb : Texts
gb =
{ boxView = Messages.Comp.BoxView.gb
}
de : Texts
de =
{ boxView = Messages.Comp.BoxView.de
}

View File

@ -10,6 +10,7 @@ module Messages.DateFormat exposing
, formatDateLong , formatDateLong
, formatDateShort , formatDateShort
, formatDateTimeLong , formatDateTimeLong
, formatDateTimeShort
) )
import DateFormat exposing (Token) import DateFormat exposing (Token)
@ -68,6 +69,11 @@ formatDateShort lang millis =
format lang .dateShort millis format lang .dateShort millis
formatDateTimeShort : UiLanguage -> Int -> String
formatDateTimeShort lang millis =
format lang .dateTimeShort millis
--- Language Definitions --- Language Definitions

View File

@ -1,6 +1,7 @@
module Messages.Page.Dashboard exposing (Texts, de, gb) module Messages.Page.Dashboard exposing (Texts, de, gb)
import Messages.Comp.BookmarkChooser import Messages.Comp.BookmarkChooser
import Messages.Comp.DashboardView
import Messages.Comp.EquipmentManage import Messages.Comp.EquipmentManage
import Messages.Comp.FolderManage import Messages.Comp.FolderManage
import Messages.Comp.NotificationHookManage import Messages.Comp.NotificationHookManage
@ -10,6 +11,7 @@ import Messages.Comp.PersonManage
import Messages.Comp.ShareManage import Messages.Comp.ShareManage
import Messages.Comp.SourceManage import Messages.Comp.SourceManage
import Messages.Comp.TagManage import Messages.Comp.TagManage
import Messages.Page.DefaultDashboard
type alias Texts = type alias Texts =
@ -23,6 +25,8 @@ type alias Texts =
, equipManage : Messages.Comp.EquipmentManage.Texts , equipManage : Messages.Comp.EquipmentManage.Texts
, tagManage : Messages.Comp.TagManage.Texts , tagManage : Messages.Comp.TagManage.Texts
, folderManage : Messages.Comp.FolderManage.Texts , folderManage : Messages.Comp.FolderManage.Texts
, dashboard : Messages.Comp.DashboardView.Texts
, defaultDashboard : Messages.Page.DefaultDashboard.Texts
} }
@ -38,6 +42,8 @@ gb =
, equipManage = Messages.Comp.EquipmentManage.gb , equipManage = Messages.Comp.EquipmentManage.gb
, tagManage = Messages.Comp.TagManage.gb , tagManage = Messages.Comp.TagManage.gb
, folderManage = Messages.Comp.FolderManage.gb , folderManage = Messages.Comp.FolderManage.gb
, dashboard = Messages.Comp.DashboardView.gb
, defaultDashboard = Messages.Page.DefaultDashboard.gb
} }
@ -53,4 +59,6 @@ de =
, equipManage = Messages.Comp.EquipmentManage.de , equipManage = Messages.Comp.EquipmentManage.de
, tagManage = Messages.Comp.TagManage.de , tagManage = Messages.Comp.TagManage.de
, folderManage = Messages.Comp.FolderManage.de , folderManage = Messages.Comp.FolderManage.de
, dashboard = Messages.Comp.DashboardView.de
, defaultDashboard = Messages.Page.DefaultDashboard.de
} }

View File

@ -0,0 +1,57 @@
module Messages.Page.DefaultDashboard exposing (Texts, de, gb)
import Messages.Basics
type alias Texts =
{ basics : Messages.Basics.Texts
, default : String
, welcomeName : String
, welcomeTitle : String
, welcomeBody : String
, summaryName : String
, dueInDays : Int -> String
, dueHeaderColumns : List String
, newDocsName : String
}
gb : Texts
gb =
let
b =
Messages.Basics.gb
in
{ basics = b
, default = "Default"
, welcomeName = "Welcome Message"
, welcomeTitle = "# Welcome to Docspell"
, welcomeBody = "Docspell keeps your documents organized."
, summaryName = "Summary"
, dueInDays = \n -> "Due in " ++ String.fromInt n ++ " days"
, dueHeaderColumns = dueHeaderCols b
, newDocsName = "New Documents"
}
de : Texts
de =
let
b =
Messages.Basics.de
in
{ basics = b
, default = "Standard"
, welcomeName = "Willkommens-Nachricht"
, welcomeTitle = "# Willkommen zu Docspell"
, welcomeBody = "Docspell behält die Übersicht über deine Dokumene."
, summaryName = "Zahlen"
, dueInDays = \n -> "Fällig in " ++ String.fromInt n ++ " Tagen"
, newDocsName = "Neue Dokumente"
, dueHeaderColumns = dueHeaderCols b
}
dueHeaderCols : Messages.Basics.Texts -> List String
dueHeaderCols b =
[ b.name, b.correspondent, b.date ]

View File

@ -11,6 +11,8 @@ module Page.Dashboard.Data exposing
, Msg(..) , Msg(..)
, SideMenuModel , SideMenuModel
, init , init
, reloadDashboard
, reloadUiSettings
) )
import Api import Api
@ -26,8 +28,8 @@ import Comp.ShareManage
import Comp.SourceManage import Comp.SourceManage
import Comp.TagManage import Comp.TagManage
import Data.Bookmarks exposing (AllBookmarks) import Data.Bookmarks exposing (AllBookmarks)
import Data.Dashboard exposing (Dashboard)
import Data.Flags exposing (Flags) import Data.Flags exposing (Flags)
import Page.Dashboard.DefaultDashboard as DefaultDashboard
type alias SideMenuModel = type alias SideMenuModel =
@ -41,11 +43,11 @@ type alias Model =
} }
init : Flags -> ( Model, Cmd Msg ) init : Flags -> Dashboard -> ( Model, Cmd Msg )
init flags = init flags db =
let let
( dm, dc ) = ( dm, dc ) =
Comp.DashboardView.init flags DefaultDashboard.value Comp.DashboardView.init flags db
in in
( { sideMenu = ( { sideMenu =
{ bookmarkChooser = Comp.BookmarkChooser.init Data.Bookmarks.empty { bookmarkChooser = Comp.BookmarkChooser.init Data.Bookmarks.empty
@ -69,6 +71,16 @@ initCmd flags =
Api.getBookmarks flags ignoreBookmarkError Api.getBookmarks flags ignoreBookmarkError
reloadDashboard : Msg
reloadDashboard =
InitDashboard
reloadUiSettings : Msg
reloadUiSettings =
InitDashboard
type Msg type Msg
= GetBookmarksResp AllBookmarks = GetBookmarksResp AllBookmarks
| BookmarkMsg Comp.BookmarkChooser.Msg | BookmarkMsg Comp.BookmarkChooser.Msg

View File

@ -1,57 +1,121 @@
module Page.Dashboard.DefaultDashboard exposing (..) module Page.Dashboard.DefaultDashboard exposing (getDefaultDashboard, value)
import Data.Box exposing (Box) import Data.Box exposing (Box)
import Data.BoxContent exposing (BoxContent(..)) import Data.BoxContent exposing (BoxContent(..), SearchQuery(..), SummaryShow(..))
import Data.Dashboard exposing (Dashboard) import Data.Dashboard exposing (Dashboard)
import Data.ItemArrange import Data.Flags exposing (Flags)
import Data.ItemTemplate as IT
import Data.UiSettings exposing (UiSettings)
import Messages
import Messages.Page.DefaultDashboard exposing (Texts)
import Messages.UiLanguage
value : Dashboard value : Texts -> Dashboard
value = value texts =
{ name = "Default" { name = texts.default
, columns = 2 , columns = 2
, boxes = , boxes =
[ messageBox [ messageBox texts
, newDocuments , summary2
, summary , newDocuments texts
, dueDocuments texts
, summary texts
] ]
} }
messageBox : Box getDefaultDashboard : Flags -> UiSettings -> Dashboard
messageBox = getDefaultDashboard flags settings =
{ name = "Welcome Message" let
lang =
Data.UiSettings.getUiLanguage flags settings Messages.UiLanguage.English
texts =
Messages.get lang
in
value texts.dashboard.defaultDashboard
--- Boxes
messageBox : Texts -> Box
messageBox texts =
{ name = texts.welcomeName
, visible = True , visible = True
, decoration = False , decoration = False
, colspan = 2 , colspan = 2
, content = , content =
BoxMessage BoxMessage
{ title = "Welcome to Docspell" { title = texts.welcomeTitle
, body = "" , body = texts.welcomeBody
} }
} }
newDocuments : Box newDocuments : Texts -> Box
newDocuments = newDocuments texts =
{ name = "New Documents" { name = texts.newDocsName
, visible = True , visible = True
, decoration = True , decoration = True
, colspan = 1 , colspan = 1
, content = , content =
BoxQuery BoxQuery
{ query = "inbox:yes" { query = SearchQueryString "inbox:yes"
, view = Data.ItemArrange.List , limit = 5
, details = True
, header = []
, columns = []
} }
} }
summary : Box dueDocuments : Texts -> Box
summary = dueDocuments texts =
{ name = "Summary" { name = texts.dueInDays 10
, visible = True , visible = True
, decoration = True , decoration = True
, colspan = 1 , colspan = 1
, content = , content =
BoxSummary { query = "" } BoxQuery
{ query = SearchQueryString "due>today;-10d due<today;+10d"
, limit = 5
, details = True
, header = texts.dueHeaderColumns
, columns =
[ IT.name
, IT.correspondent
, IT.dueDateShort
]
}
}
summary : Texts -> Box
summary texts =
{ name = texts.summaryName
, visible = True
, decoration = True
, colspan = 1
, content =
BoxSummary
{ query = SearchQueryString ""
, show = SummaryShowGeneral
}
}
summary2 : Box
summary2 =
{ name = ""
, visible = True
, decoration = True
, colspan = 2
, content =
BoxSummary
{ query = SearchQueryString ""
, show = SummaryShowFields False
}
} }

View File

@ -20,6 +20,7 @@ import Comp.ShareManage
import Comp.SourceManage import Comp.SourceManage
import Comp.TagManage import Comp.TagManage
import Data.Flags exposing (Flags) import Data.Flags exposing (Flags)
import Data.UiSettings exposing (UiSettings)
import Messages.Page.Dashboard exposing (Texts) import Messages.Page.Dashboard exposing (Texts)
import Page exposing (Page(..)) import Page exposing (Page(..))
import Page.Dashboard.Data exposing (..) import Page.Dashboard.Data exposing (..)
@ -27,8 +28,8 @@ import Page.Dashboard.DefaultDashboard
import Set import Set
update : Texts -> Nav.Key -> Flags -> Msg -> Model -> ( Model, Cmd Msg, Sub Msg ) update : Texts -> UiSettings -> Nav.Key -> Flags -> Msg -> Model -> ( Model, Cmd Msg, Sub Msg )
update texts navKey flags msg model = update texts settings navKey flags msg model =
case msg of case msg of
GetBookmarksResp list -> GetBookmarksResp list ->
let let
@ -59,8 +60,11 @@ update texts navKey flags msg model =
InitDashboard -> InitDashboard ->
let let
board =
Page.Dashboard.DefaultDashboard.getDefaultDashboard flags settings
( dm, dc ) = ( dm, dc ) =
Comp.DashboardView.init flags Page.Dashboard.DefaultDashboard.value Comp.DashboardView.init flags board
in in
( { model | content = Home dm }, Cmd.map DashboardMsg dc, Sub.none ) ( { model | content = Home dm }, Cmd.map DashboardMsg dc, Sub.none )
@ -242,7 +246,16 @@ update texts navKey flags msg model =
unit model unit model
DashboardMsg lm -> DashboardMsg lm ->
unit model case model.content of
Home m ->
let
( dm, dc ) =
Comp.DashboardView.update lm m
in
( { model | content = Home dm }, Cmd.map DashboardMsg dc, Sub.none )
_ ->
unit model
unit : Model -> ( Model, Cmd Msg, Sub Msg ) unit : Model -> ( Model, Cmd Msg, Sub Msg )

View File

@ -48,7 +48,7 @@ viewContent texts flags settings model =
[ case model.content of [ case model.content of
Home m -> Home m ->
Html.map DashboardMsg Html.map DashboardMsg
(Comp.DashboardView.view m) (Comp.DashboardView.view texts.dashboard m)
Webhook m -> Webhook m ->
viewHookManage texts settings m viewHookManage texts settings m

View File

@ -563,7 +563,7 @@ editMenuBar texts model svm =
searchStats : Texts -> Flags -> UiSettings -> Model -> List (Html Msg) searchStats : Texts -> Flags -> UiSettings -> Model -> List (Html Msg)
searchStats texts _ settings model = searchStats texts _ settings model =
if settings.searchStatsVisible then if settings.searchStatsVisible then
[ Comp.SearchStatsView.view2 texts.searchStatsView "my-2" model.searchStats [ Comp.SearchStatsView.view texts.searchStatsView "my-2" model.searchStats
] ]
else else