mirror of
https://code.naskya.net/repos/ndqEd
synced 2025-01-10 16:56:47 +09:00
UI: HTML version for getLoomClothsR, copied from getDeckTicketsR
This commit is contained in:
parent
1db56ced39
commit
1e2b3d2006
7 changed files with 276 additions and 10 deletions
|
@ -34,6 +34,7 @@ import Control.Monad.Trans.Except
|
||||||
import Control.Monad.Trans.Maybe
|
import Control.Monad.Trans.Maybe
|
||||||
import Data.Aeson
|
import Data.Aeson
|
||||||
import Data.ByteString (ByteString)
|
import Data.ByteString (ByteString)
|
||||||
|
import Data.Default.Class
|
||||||
import Data.Foldable
|
import Data.Foldable
|
||||||
import Data.Maybe
|
import Data.Maybe
|
||||||
import Data.Text (Text)
|
import Data.Text (Text)
|
||||||
|
@ -44,7 +45,7 @@ import Text.Blaze.Html (Html)
|
||||||
import Yesod.Auth (requireAuth)
|
import Yesod.Auth (requireAuth)
|
||||||
import Yesod.Core
|
import Yesod.Core
|
||||||
import Yesod.Core.Handler (redirect, setMessage, lookupPostParam, notFound)
|
import Yesod.Core.Handler (redirect, setMessage, lookupPostParam, notFound)
|
||||||
import Yesod.Form.Functions (runFormPost)
|
import Yesod.Form.Functions (runFormPost, runFormGet)
|
||||||
import Yesod.Form.Types (FormResult (..))
|
import Yesod.Form.Types (FormResult (..))
|
||||||
import Yesod.Persist.Core (runDB, get404, getBy404)
|
import Yesod.Persist.Core (runDB, get404, getBy404)
|
||||||
|
|
||||||
|
@ -74,12 +75,16 @@ import Vervis.Federation.Auth
|
||||||
import Vervis.Federation.Collab
|
import Vervis.Federation.Collab
|
||||||
import Vervis.FedURI
|
import Vervis.FedURI
|
||||||
import Vervis.Form.Project
|
import Vervis.Form.Project
|
||||||
|
import Vervis.Form.Ticket
|
||||||
import Vervis.Foundation
|
import Vervis.Foundation
|
||||||
import Vervis.Model
|
import Vervis.Model
|
||||||
import Vervis.Paginate
|
import Vervis.Paginate
|
||||||
import Vervis.Recipient
|
import Vervis.Recipient
|
||||||
import Vervis.Settings
|
import Vervis.Settings
|
||||||
|
import Vervis.Ticket
|
||||||
|
import Vervis.TicketFilter
|
||||||
import Vervis.Web.Actor
|
import Vervis.Web.Actor
|
||||||
|
import Vervis.Widget.Ticket
|
||||||
|
|
||||||
import qualified Vervis.Client as C
|
import qualified Vervis.Client as C
|
||||||
|
|
||||||
|
@ -152,7 +157,6 @@ getLoomFollowersR = getActorFollowersCollection LoomFollowersR loomActor
|
||||||
|
|
||||||
getLoomClothsR :: KeyHashid Loom -> Handler TypedContent
|
getLoomClothsR :: KeyHashid Loom -> Handler TypedContent
|
||||||
getLoomClothsR loomHash = selectRep $ do
|
getLoomClothsR loomHash = selectRep $ do
|
||||||
{-
|
|
||||||
provideRep $ do
|
provideRep $ do
|
||||||
((filtResult, filtWidget), filtEnctype) <- runFormGet ticketFilterForm
|
((filtResult, filtWidget), filtEnctype) <- runFormGet ticketFilterForm
|
||||||
let tf =
|
let tf =
|
||||||
|
@ -161,23 +165,22 @@ getLoomClothsR loomHash = selectRep $ do
|
||||||
FormMissing -> def
|
FormMissing -> def
|
||||||
FormFailure l ->
|
FormFailure l ->
|
||||||
error $ "Ticket filter form failed: " ++ show l
|
error $ "Ticket filter form failed: " ++ show l
|
||||||
|
loomID <- decodeKeyHashid404 loomHash
|
||||||
(total, pages, mpage) <- runDB $ do
|
(total, pages, mpage) <- runDB $ do
|
||||||
Entity sid _ <- getBy404 $ UniqueSharer shr
|
_ <- get404 loomID
|
||||||
Entity jid _ <- getBy404 $ UniqueProject prj sid
|
let countAllTickets = count [TicketLoomLoom ==. loomID]
|
||||||
let countAllTickets = count [TicketProjectLocalProject ==. jid]
|
|
||||||
selectTickets off lim =
|
selectTickets off lim =
|
||||||
getTicketSummaries
|
getClothSummaries
|
||||||
(filterTickets tf)
|
(filterTickets tf)
|
||||||
(Just $ \ t -> [E.asc $ t E.^. TicketId])
|
(Just $ \ t -> [E.desc $ t E.^. TicketId])
|
||||||
(Just (off, lim))
|
(Just (off, lim))
|
||||||
jid
|
loomID
|
||||||
getPageAndNavCount countAllTickets selectTickets
|
getPageAndNavCount countAllTickets selectTickets
|
||||||
case mpage of
|
case mpage of
|
||||||
Nothing -> redirectFirstPage here
|
Nothing -> redirectFirstPage here
|
||||||
Just (rows, navModel) ->
|
Just (rows, navModel) ->
|
||||||
let pageNav = navWidget navModel
|
let pageNav = navWidget navModel
|
||||||
in defaultLayout $(widgetFile "ticket/list")
|
in defaultLayout $(widgetFile "cloth/list")
|
||||||
-}
|
|
||||||
AP.provideAP' $ do
|
AP.provideAP' $ do
|
||||||
loomID <- decodeKeyHashid404 loomHash
|
loomID <- decodeKeyHashid404 loomHash
|
||||||
(total, pages, mpage) <- runDB $ do
|
(total, pages, mpage) <- runDB $ do
|
||||||
|
|
|
@ -17,6 +17,7 @@
|
||||||
module Vervis.Ticket
|
module Vervis.Ticket
|
||||||
(
|
(
|
||||||
getTicketSummaries
|
getTicketSummaries
|
||||||
|
, getClothSummaries
|
||||||
--, getTicketDepEdges
|
--, getTicketDepEdges
|
||||||
|
|
||||||
, WorkflowFieldFilter (..)
|
, WorkflowFieldFilter (..)
|
||||||
|
@ -165,6 +166,80 @@ getTicketSummaries mfilt morder offlim deckID = do
|
||||||
, tsComments = r
|
, tsComments = r
|
||||||
}
|
}
|
||||||
|
|
||||||
|
getClothSummaries
|
||||||
|
:: Maybe (E.SqlExpr (Entity Ticket) -> E.SqlExpr (E.Value Bool))
|
||||||
|
-> Maybe (E.SqlExpr (Entity Ticket) -> [E.SqlExpr E.OrderBy])
|
||||||
|
-> Maybe (Int, Int)
|
||||||
|
-> LoomId
|
||||||
|
-> AppDB [ClothSummary]
|
||||||
|
getClothSummaries mfilt morder offlim loomID = do
|
||||||
|
tickets <- E.select $ E.from $
|
||||||
|
\ ( t
|
||||||
|
`E.InnerJoin` tl
|
||||||
|
`E.LeftOuterJoin` (tal `E.InnerJoin` p `E.InnerJoin` a)
|
||||||
|
`E.LeftOuterJoin` (tar `E.InnerJoin` ra `E.InnerJoin` ro `E.InnerJoin` i)
|
||||||
|
`E.InnerJoin` d
|
||||||
|
`E.LeftOuterJoin` m
|
||||||
|
) -> do
|
||||||
|
E.on $ E.just (d E.^. DiscussionId) E.==. m E.?. MessageRoot
|
||||||
|
E.on $ t E.^. TicketDiscuss E.==. d E.^. DiscussionId
|
||||||
|
|
||||||
|
E.on $ ro E.?. RemoteObjectInstance E.==. i E.?. InstanceId
|
||||||
|
E.on $ ra E.?. RemoteActorIdent E.==. ro E.?. RemoteObjectId
|
||||||
|
E.on $ tar E.?. TicketAuthorRemoteAuthor E.==. ra E.?. RemoteActorId
|
||||||
|
E.on $ E.just (t E.^. TicketId) E.==. tar E.?. TicketAuthorRemoteTicket
|
||||||
|
|
||||||
|
E.on $ p E.?. PersonActor E.==. a E.?. ActorId
|
||||||
|
E.on $ tal E.?. TicketAuthorLocalAuthor E.==. p E.?. PersonId
|
||||||
|
E.on $ E.just (t E.^. TicketId) E.==. tal E.?. TicketAuthorLocalTicket
|
||||||
|
|
||||||
|
E.on $ t E.^. TicketId E.==. tl E.^. TicketLoomTicket
|
||||||
|
|
||||||
|
E.where_ $ tl E.^. TicketLoomLoom E.==. E.val loomID
|
||||||
|
E.groupBy
|
||||||
|
( t E.^. TicketId
|
||||||
|
, tal E.?. TicketAuthorLocalId, p E.?. PersonId, a E.?. ActorId
|
||||||
|
, ra E.?. RemoteActorId, ro E.?. RemoteObjectId, i E.?. InstanceId
|
||||||
|
)
|
||||||
|
for_ mfilt $ \ filt -> E.where_ $ filt t
|
||||||
|
for_ morder $ \ order -> E.orderBy $ order t
|
||||||
|
for_ offlim $ \ (off, lim) -> do
|
||||||
|
E.offset $ fromIntegral off
|
||||||
|
E.limit $ fromIntegral lim
|
||||||
|
|
||||||
|
return
|
||||||
|
( t E.^. TicketId
|
||||||
|
, tl E.^. TicketLoomId
|
||||||
|
, p, a
|
||||||
|
, i, ro, ra
|
||||||
|
, t E.^. TicketCreated
|
||||||
|
, t E.^. TicketTitle
|
||||||
|
, t E.^. TicketStatus
|
||||||
|
, E.count $ m E.?. MessageId
|
||||||
|
)
|
||||||
|
|
||||||
|
for tickets $
|
||||||
|
\ (E.Value tid, E.Value tlid, mp, ma, mi, mro, mra, E.Value c, E.Value t, E.Value d, E.Value r) -> do
|
||||||
|
labels <- E.select $ E.from $ \ (tpc `E.InnerJoin` wf) -> do
|
||||||
|
E.on $ tpc E.^. TicketParamClassField E.==. wf E.^. WorkflowFieldId
|
||||||
|
E.where_ $ tpc E.^. TicketParamClassTicket E.==. E.val tid
|
||||||
|
return wf
|
||||||
|
return ClothSummary
|
||||||
|
{ csId = tlid
|
||||||
|
, csCreatedBy =
|
||||||
|
case (mp, ma, mi, mro, mra) of
|
||||||
|
(Just p, Just a, Nothing, Nothing, Nothing) ->
|
||||||
|
Left (p, entityVal a)
|
||||||
|
(Nothing, Nothing, Just i, Just ro, Just ra) ->
|
||||||
|
Right (entityVal i, entityVal ro, entityVal ra)
|
||||||
|
_ -> error "Ticket author DB invalid state"
|
||||||
|
, csCreatedAt = c
|
||||||
|
, csTitle = t
|
||||||
|
, csLabels = map entityVal labels
|
||||||
|
, csStatus = d
|
||||||
|
, csComments = r
|
||||||
|
}
|
||||||
|
|
||||||
-- | Get the child-parent ticket number pairs of all the ticket dependencies
|
-- | Get the child-parent ticket number pairs of all the ticket dependencies
|
||||||
-- in the given project, in ascending order by child, and then ascending order
|
-- in the given project, in ascending order by child, and then ascending order
|
||||||
-- by parent.
|
-- by parent.
|
||||||
|
|
|
@ -15,8 +15,10 @@
|
||||||
|
|
||||||
module Vervis.Widget.Ticket
|
module Vervis.Widget.Ticket
|
||||||
( TicketSummary (..)
|
( TicketSummary (..)
|
||||||
|
, ClothSummary (..)
|
||||||
--, ticketDepW
|
--, ticketDepW
|
||||||
, ticketSummaryW
|
, ticketSummaryW
|
||||||
|
, clothSummaryW
|
||||||
--, ticketTreeVW
|
--, ticketTreeVW
|
||||||
--, ticketTreeDW
|
--, ticketTreeDW
|
||||||
)
|
)
|
||||||
|
@ -63,6 +65,18 @@ data TicketSummary = TicketSummary
|
||||||
, tsComments :: Int
|
, tsComments :: Int
|
||||||
}
|
}
|
||||||
|
|
||||||
|
data ClothSummary = ClothSummary
|
||||||
|
{ csId :: TicketLoomId
|
||||||
|
, csCreatedBy :: Either
|
||||||
|
(Entity Person, Actor)
|
||||||
|
(Instance, RemoteObject, RemoteActor)
|
||||||
|
, csCreatedAt :: UTCTime
|
||||||
|
, csTitle :: Text
|
||||||
|
, csLabels :: [WorkflowField]
|
||||||
|
, csStatus :: TicketStatus
|
||||||
|
, csComments :: Int
|
||||||
|
}
|
||||||
|
|
||||||
{-
|
{-
|
||||||
ticketDepW :: ShrIdent -> PrjIdent -> LocalTicketId -> Ticket -> Widget
|
ticketDepW :: ShrIdent -> PrjIdent -> LocalTicketId -> Ticket -> Widget
|
||||||
ticketDepW shr prj ltid ticket = do
|
ticketDepW shr prj ltid ticket = do
|
||||||
|
@ -92,6 +106,25 @@ ticketSummaryW deckHash ts mcs = do
|
||||||
ticketRoute' hashTicket summary =
|
ticketRoute' hashTicket summary =
|
||||||
TicketR deckHash (hashTicket $ tsId summary)
|
TicketR deckHash (hashTicket $ tsId summary)
|
||||||
|
|
||||||
|
clothSummaryW
|
||||||
|
:: KeyHashid Loom
|
||||||
|
-> ClothSummary
|
||||||
|
-> Maybe (HashMap Int64 Int64)
|
||||||
|
-> Widget
|
||||||
|
clothSummaryW loomHash cs mcs = do
|
||||||
|
hashTicket <- getEncodeKeyHashid
|
||||||
|
cNew <- newIdent
|
||||||
|
cTodo <- newIdent
|
||||||
|
cClosed <- newIdent
|
||||||
|
let tshow = T.pack . show
|
||||||
|
mparams = map (tshow *** tshow) . M.toList <$> mcs
|
||||||
|
ticketRoute = ticketRoute' hashTicket
|
||||||
|
mroute <- getCurrentRoute
|
||||||
|
$(widgetFile "cloth/widget/summary")
|
||||||
|
where
|
||||||
|
ticketRoute' hashTicket summary =
|
||||||
|
ClothR loomHash (hashTicket $ csId summary)
|
||||||
|
|
||||||
{-
|
{-
|
||||||
-- I'm noticing a pattern. A problem. Some of my widget functions take data and
|
-- I'm noticing a pattern. A problem. Some of my widget functions take data and
|
||||||
-- directly represent it in HTML. Others take some other more general
|
-- directly represent it in HTML. Others take some other more general
|
||||||
|
|
4
templates/cloth/list.cassius
Normal file
4
templates/cloth/list.cassius
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
.container
|
||||||
|
display: grid
|
||||||
|
grid-template-columns: 1rem 1rem 2fr 2fr 8fr 1rem 1rem
|
||||||
|
grid-column-gap: 1rem
|
32
templates/cloth/list.hamlet
Normal file
32
templates/cloth/list.hamlet
Normal file
|
@ -0,0 +1,32 @@
|
||||||
|
$# This file is part of Vervis.
|
||||||
|
$#
|
||||||
|
$# Written in 2016, 2018, 2022 by fr33domlover <fr33domlover@riseup.net>.
|
||||||
|
$#
|
||||||
|
$# ♡ Copying is an act of love. Please copy, reuse and share.
|
||||||
|
$#
|
||||||
|
$# The author(s) have dedicated all copyright and related and neighboring
|
||||||
|
$# rights to this software to the public domain worldwide. This software is
|
||||||
|
$# distributed without any warranty.
|
||||||
|
$#
|
||||||
|
$# You should have received a copy of the CC0 Public Domain Dedication along
|
||||||
|
$# with this software. If not, see
|
||||||
|
$# <http://creativecommons.org/publicdomain/zero/1.0/>.
|
||||||
|
|
||||||
|
$# <p>
|
||||||
|
$# <a href=@{ProjectTicketNewR shr prj}>Create new…
|
||||||
|
|
||||||
|
$# <p>
|
||||||
|
$# <a href=@{ProjectTicketTreeR shr prj}>View as tree…
|
||||||
|
|
||||||
|
<form method=GET action=@{LoomClothsR loomHash} enctype=#{filtEnctype}>
|
||||||
|
^{filtWidget}
|
||||||
|
<div class="submit">
|
||||||
|
<input type="submit" value="Filter">
|
||||||
|
|
||||||
|
^{pageNav}
|
||||||
|
|
||||||
|
<div .container>
|
||||||
|
$forall cs <- rows
|
||||||
|
^{clothSummaryW loomHash cs Nothing}
|
||||||
|
|
||||||
|
^{pageNav}
|
57
templates/cloth/widget/summary.cassius
Normal file
57
templates/cloth/widget/summary.cassius
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
/* This file is part of Vervis.
|
||||||
|
*
|
||||||
|
* Written in 2016, 2020 by fr33domlover <fr33domlover@riseup.net>,
|
||||||
|
* 2019 by Jason Harrer <jazzyeagle79@gmail.com>.
|
||||||
|
*
|
||||||
|
* ♡ Copying is an act of love. Please copy, reuse and share.
|
||||||
|
*
|
||||||
|
* The author(s) have dedicated all copyright and related and neighboring
|
||||||
|
* rights to this software to the public domain worldwide. This software is
|
||||||
|
* distributed without any warranty.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the CC0 Public Domain Dedication along
|
||||||
|
* with this software. If not, see
|
||||||
|
* <http://creativecommons.org/publicdomain/zero/1.0/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
.#{cNew}
|
||||||
|
color: #{dark yellow}
|
||||||
|
|
||||||
|
.#{cTodo}
|
||||||
|
color: #{dark red}
|
||||||
|
|
||||||
|
.#{cClosed}
|
||||||
|
color: #{dark green}
|
||||||
|
|
||||||
|
.ticket-status-column
|
||||||
|
grid-column: 1 / 1
|
||||||
|
|
||||||
|
.ticket-number-column
|
||||||
|
grid-column: 2 / 2
|
||||||
|
|
||||||
|
.ticket-date-column
|
||||||
|
grid-column: 3 / 3
|
||||||
|
|
||||||
|
.ticket-sharer-column
|
||||||
|
grid-column: 4 / 4
|
||||||
|
|
||||||
|
.ticket-title-column
|
||||||
|
grid-column: 5 / 5
|
||||||
|
|
||||||
|
.ticket-tree-column
|
||||||
|
grid-column: 6 / 6
|
||||||
|
|
||||||
|
.ticket-node-column
|
||||||
|
grid-column: 7 / 7
|
||||||
|
|
||||||
|
.label1
|
||||||
|
color: #{light red}
|
||||||
|
|
||||||
|
.label2
|
||||||
|
color: #{light green}
|
||||||
|
|
||||||
|
.label3
|
||||||
|
color: #{light yellow}
|
||||||
|
|
||||||
|
.label4
|
||||||
|
color: #{light blue}
|
62
templates/cloth/widget/summary.hamlet
Normal file
62
templates/cloth/widget/summary.hamlet
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
$# This file is part of Vervis.
|
||||||
|
$#
|
||||||
|
$# Written in 2016, 2019, 2020, 2022 by fr33domlover <fr33domlover@riseup.net>.
|
||||||
|
$#
|
||||||
|
$# ♡ Copying is an act of love. Please copy, reuse and share.
|
||||||
|
$#
|
||||||
|
$# The author(s) have dedicated all copyright and related and neighboring
|
||||||
|
$# rights to this software to the public domain worldwide. This software is
|
||||||
|
$# distributed without any warranty.
|
||||||
|
$#
|
||||||
|
$# You should have received a copy of the CC0 Public Domain Dedication along
|
||||||
|
$# with this software. If not, see
|
||||||
|
$# <http://creativecommons.org/publicdomain/zero/1.0/>.
|
||||||
|
|
||||||
|
<span .ticket-status-column>
|
||||||
|
$case csStatus cs
|
||||||
|
$of TSNew
|
||||||
|
<span .#{cNew}>
|
||||||
|
⬚
|
||||||
|
$of TSTodo
|
||||||
|
<span .#{cTodo}>
|
||||||
|
☐
|
||||||
|
$of TSClosed
|
||||||
|
<span .#{cClosed}>
|
||||||
|
☒
|
||||||
|
|
||||||
|
<span .ticket-number-column>
|
||||||
|
<a href=@{ticketRoute cs}>
|
||||||
|
###
|
||||||
|
|
||||||
|
<span .ticket-date-column>
|
||||||
|
#{showDate $ csCreatedAt cs}
|
||||||
|
|
||||||
|
<span .ticket-sharer-column>
|
||||||
|
^{personLinkFedW $ csCreatedBy cs}
|
||||||
|
|
||||||
|
<span .ticket-title-column>
|
||||||
|
<a href=@{ticketRoute cs}>
|
||||||
|
#{preEscapedToHtml $ csTitle cs}
|
||||||
|
$forall wf <- csLabels cs
|
||||||
|
$maybe wfcol <- workflowFieldColor wf
|
||||||
|
<span .label#{wfcol}>
|
||||||
|
[#{workflowFieldName wf}]
|
||||||
|
$nothing
|
||||||
|
<span .label-nocolor>
|
||||||
|
[#{workflowFieldName wf}]
|
||||||
|
|
||||||
|
<span .ticket-tree-column>
|
||||||
|
$if csComments cs > 0
|
||||||
|
💬
|
||||||
|
#{csComments cs}
|
||||||
|
|
||||||
|
$maybe params <- mparams
|
||||||
|
<span .ticket-node-column>
|
||||||
|
<a href="#node-#{keyHashidText $ hashTicket $ csId cs}" title="Jump to subtree">
|
||||||
|
☝
|
||||||
|
$maybe route <- mroute
|
||||||
|
<a href=@?{(route, params)} title="Move subtree here">
|
||||||
|
☚
|
||||||
|
$nothing
|
||||||
|
<span .ticket-node-column>
|
||||||
|
<a id="node-#{keyHashidText $ hashTicket $ csId cs}">
|
Loading…
Reference in a new issue