mirror of
https://code.naskya.net/repos/ndqEd
synced 2025-01-10 20:56:45 +09:00
Ticket update form
This commit is contained in:
parent
88569a08ad
commit
9d3b7b686f
6 changed files with 120 additions and 15 deletions
|
@ -72,5 +72,7 @@ Ticket
|
||||||
title Text
|
title Text
|
||||||
desc Text -- Assume this is Pandoc Markdown
|
desc Text -- Assume this is Pandoc Markdown
|
||||||
done Bool
|
done Bool
|
||||||
|
closed UTCTime
|
||||||
|
closer PersonId
|
||||||
|
|
||||||
UniqueTicket project number
|
UniqueTicket project number
|
||||||
|
|
|
@ -15,6 +15,7 @@
|
||||||
|
|
||||||
module Vervis.Form.Ticket
|
module Vervis.Form.Ticket
|
||||||
( newTicketForm
|
( newTicketForm
|
||||||
|
, editTicketForm
|
||||||
)
|
)
|
||||||
where
|
where
|
||||||
|
|
||||||
|
@ -22,7 +23,8 @@ import Prelude
|
||||||
|
|
||||||
import Control.Monad.IO.Class (liftIO)
|
import Control.Monad.IO.Class (liftIO)
|
||||||
import Control.Monad.Trans.Class (lift)
|
import Control.Monad.Trans.Class (lift)
|
||||||
import Data.Time.Clock (getCurrentTime)
|
import Data.Time.Calendar (Day (..))
|
||||||
|
import Data.Time.Clock (getCurrentTime, UTCTime (..))
|
||||||
import Yesod.Form
|
import Yesod.Form
|
||||||
|
|
||||||
import Vervis.Foundation (Form, Handler)
|
import Vervis.Foundation (Form, Handler)
|
||||||
|
@ -32,15 +34,53 @@ import Vervis.Model
|
||||||
--TODO stuff like number and created - do I generate them here using monadic
|
--TODO stuff like number and created - do I generate them here using monadic
|
||||||
-- form or do I rely on handler to provide? which approach is better?
|
-- form or do I rely on handler to provide? which approach is better?
|
||||||
|
|
||||||
|
defTime :: UTCTime
|
||||||
|
defTime = UTCTime (ModifiedJulianDay 0) 0
|
||||||
|
|
||||||
|
now :: AForm Handler UTCTime
|
||||||
|
now = lift $ liftIO getCurrentTime
|
||||||
|
|
||||||
newTicketAForm :: ProjectId -> Int -> PersonId -> AForm Handler Ticket
|
newTicketAForm :: ProjectId -> Int -> PersonId -> AForm Handler Ticket
|
||||||
newTicketAForm pid number author = Ticket
|
newTicketAForm pid number author = Ticket
|
||||||
<$> pure pid
|
<$> pure pid
|
||||||
<*> pure number
|
<*> pure number
|
||||||
<*> lift (liftIO getCurrentTime)
|
<*> now
|
||||||
<*> pure author
|
<*> pure author
|
||||||
<*> areq textField "Title*" Nothing
|
<*> areq textField "Title*" Nothing
|
||||||
<*> (maybe "" unTextarea <$> aopt textareaField "Description (Markdown)" Nothing)
|
<*> ( maybe "" unTextarea <$>
|
||||||
|
aopt textareaField "Description (Markdown)" Nothing
|
||||||
|
)
|
||||||
<*> pure False
|
<*> pure False
|
||||||
|
<*> pure defTime
|
||||||
|
<*> pure author
|
||||||
|
|
||||||
newTicketForm :: ProjectId -> Int -> PersonId -> Form Ticket
|
newTicketForm :: ProjectId -> Int -> PersonId -> Form Ticket
|
||||||
newTicketForm pid number author = renderDivs $ newTicketAForm pid number author
|
newTicketForm pid number author = renderDivs $ newTicketAForm pid number author
|
||||||
|
|
||||||
|
editTicketAForm :: Ticket -> PersonId -> AForm Handler Ticket
|
||||||
|
editTicketAForm ticket pid = fmap fixDone $ Ticket
|
||||||
|
<$> pure (ticketProject ticket)
|
||||||
|
<*> pure (ticketNumber ticket)
|
||||||
|
<*> pure (ticketCreated ticket)
|
||||||
|
<*> pure (ticketCreator ticket)
|
||||||
|
<*> areq textField "Title*" (Just $ ticketTitle ticket)
|
||||||
|
<*> ( maybe "" unTextarea <$>
|
||||||
|
aopt
|
||||||
|
textareaField
|
||||||
|
"Description (Markdown)"
|
||||||
|
(Just $ Just $ Textarea $ ticketDesc ticket)
|
||||||
|
)
|
||||||
|
<*> areq checkBoxField "Done*" (Just $ ticketDone ticket)
|
||||||
|
<*> now
|
||||||
|
<*> pure (ticketCloser ticket)
|
||||||
|
where
|
||||||
|
fixDone result = case (ticketDone ticket, ticketDone result) of
|
||||||
|
(True, True) -> result { ticketClosed = ticketClosed ticket }
|
||||||
|
(False, False) -> result { ticketClosed = ticketClosed ticket }
|
||||||
|
(False, True) -> result { ticketCloser = pid }
|
||||||
|
(True, False) -> result { ticketClosed = defTime
|
||||||
|
, ticketCloser = ticketCreator ticket
|
||||||
|
}
|
||||||
|
|
||||||
|
editTicketForm :: Ticket -> PersonId -> Form Ticket
|
||||||
|
editTicketForm t p = renderDivs $ editTicketAForm t p
|
||||||
|
|
|
@ -116,6 +116,10 @@ instance Yesod App where
|
||||||
isAuthorized (KeyNewR user) _ =
|
isAuthorized (KeyNewR user) _ =
|
||||||
loggedInAs user "You can’t add keys for other users"
|
loggedInAs user "You can’t add keys for other users"
|
||||||
isAuthorized (TicketNewR _ _) _ = loggedIn
|
isAuthorized (TicketNewR _ _) _ = loggedIn
|
||||||
|
isAuthorized (TicketR user _ _) True =
|
||||||
|
loggedInAs user "Only project members can modify this ticket"
|
||||||
|
isAuthorized (TicketEditR user _ _) _ =
|
||||||
|
loggedInAs user "Only project members can modify this ticket"
|
||||||
isAuthorized _ _ = return Authorized
|
isAuthorized _ _ = return Authorized
|
||||||
|
|
||||||
-- This function creates static content files in the static folder
|
-- This function creates static content files in the static folder
|
||||||
|
|
|
@ -34,7 +34,7 @@ import Database.Persist
|
||||||
import Text.Blaze.Html (Html, toHtml)
|
import Text.Blaze.Html (Html, toHtml)
|
||||||
import Yesod.Auth (requireAuthId)
|
import Yesod.Auth (requireAuthId)
|
||||||
import Yesod.Core (defaultLayout)
|
import Yesod.Core (defaultLayout)
|
||||||
import Yesod.Core.Handler (redirectUltDest, setMessage)
|
import Yesod.Core.Handler (setMessage, redirect)
|
||||||
import Yesod.Core.Widget (setTitle)
|
import Yesod.Core.Widget (setTitle)
|
||||||
import Yesod.Form.Functions (runFormPost)
|
import Yesod.Form.Functions (runFormPost)
|
||||||
import Yesod.Form.Types (FormResult (..))
|
import Yesod.Form.Types (FormResult (..))
|
||||||
|
@ -50,10 +50,6 @@ import Vervis.Settings (widgetFile)
|
||||||
|
|
||||||
getTicketsR :: Text -> Text -> Handler Html
|
getTicketsR :: Text -> Text -> Handler Html
|
||||||
getTicketsR shar proj = do
|
getTicketsR shar proj = do
|
||||||
--tickets <- runDB $ do
|
|
||||||
-- Entity sid _sharer <- getBy404 $ UniqueSharerIdent shar
|
|
||||||
-- Entity pid _project <- getBy404 $ UniqueProject proj sid
|
|
||||||
-- selectList [TicketProject ==. pid] [Asc TicketNumber]
|
|
||||||
rows <- runDB $ select $ from $ \ (ticket, person, sharer) -> do
|
rows <- runDB $ select $ from $ \ (ticket, person, sharer) -> do
|
||||||
where_ $
|
where_ $
|
||||||
ticket ^. TicketCreator E.==. person ^. PersonId &&.
|
ticket ^. TicketCreator E.==. person ^. PersonId &&.
|
||||||
|
@ -84,7 +80,7 @@ postTicketsR shar proj = do
|
||||||
update pid [ProjectNextTicket +=. 1]
|
update pid [ProjectNextTicket +=. 1]
|
||||||
insert_ ticket
|
insert_ ticket
|
||||||
setMessage "Ticket created."
|
setMessage "Ticket created."
|
||||||
redirectUltDest HomeR
|
redirect $ TicketR shar proj (ticketNumber ticket)
|
||||||
FormMissing -> do
|
FormMissing -> do
|
||||||
setMessage "Field(s) missing."
|
setMessage "Field(s) missing."
|
||||||
defaultLayout $(widgetFile "ticket/new")
|
defaultLayout $(widgetFile "ticket/new")
|
||||||
|
@ -106,23 +102,59 @@ getTicketNewR shar proj = do
|
||||||
|
|
||||||
getTicketR :: Text -> Text -> Int -> Handler Html
|
getTicketR :: Text -> Text -> Int -> Handler Html
|
||||||
getTicketR shar proj num = do
|
getTicketR shar proj num = do
|
||||||
(author, ticket) <- runDB $ do
|
(author, closer, ticket) <- runDB $ do
|
||||||
Entity sid _sharer <- getBy404 $ UniqueSharerIdent shar
|
Entity sid _sharer <- getBy404 $ UniqueSharerIdent shar
|
||||||
Entity pid _project <- getBy404 $ UniqueProject proj sid
|
Entity pid _project <- getBy404 $ UniqueProject proj sid
|
||||||
Entity _tid ticket <- getBy404 $ UniqueTicket pid num
|
Entity _tid ticket <- getBy404 $ UniqueTicket pid num
|
||||||
person <- get404 $ ticketCreator ticket
|
person <- get404 $ ticketCreator ticket
|
||||||
author <- get404 $ personIdent person
|
author <- get404 $ personIdent person
|
||||||
return (author, ticket)
|
closer <-
|
||||||
|
if ticketDone ticket
|
||||||
|
then do
|
||||||
|
person' <- get404 $ ticketCloser ticket
|
||||||
|
get404 $ personIdent person'
|
||||||
|
else return author
|
||||||
|
return (author, closer, ticket)
|
||||||
defaultLayout $ do
|
defaultLayout $ do
|
||||||
setTitle $ toHtml $ T.intercalate " :: "
|
setTitle $ toHtml $ T.intercalate " :: "
|
||||||
[shar, proj, "Tickets", T.pack ('#' : show num)]
|
[shar, proj, "Tickets", T.pack ('#' : show num)]
|
||||||
$(widgetFile "ticket/one")
|
$(widgetFile "ticket/one")
|
||||||
|
|
||||||
putTicketR :: Text -> Text -> Int -> Handler Html
|
putTicketR :: Text -> Text -> Int -> Handler Html
|
||||||
putTicketR shar proj num = error "Not implemented"
|
putTicketR shar proj num = do
|
||||||
|
Entity tid ticket <- runDB $ do
|
||||||
|
Entity sid _sharer <- getBy404 $ UniqueSharerIdent shar
|
||||||
|
Entity pid _project <- getBy404 $ UniqueProject proj sid
|
||||||
|
getBy404 $ UniqueTicket pid num
|
||||||
|
user <- requireAuthId
|
||||||
|
((result, widget), enctype) <- runFormPost $ editTicketForm ticket user
|
||||||
|
case result of
|
||||||
|
FormSuccess ticket' -> do
|
||||||
|
runDB $ replace tid ticket'
|
||||||
|
setMessage "Ticket created."
|
||||||
|
redirect $ TicketR shar proj num
|
||||||
|
FormMissing -> do
|
||||||
|
setMessage "Field(s) missing."
|
||||||
|
defaultLayout $(widgetFile "ticket/edit")
|
||||||
|
FormFailure _l -> do
|
||||||
|
setMessage "Ticket update failed, see errors below."
|
||||||
|
defaultLayout $(widgetFile "ticket/edit")
|
||||||
|
|
||||||
deleteTicketR :: Text -> Text -> Int -> Handler Html
|
deleteTicketR :: Text -> Text -> Int -> Handler Html
|
||||||
deleteTicketR shar proj num = error "Not implemented"
|
deleteTicketR shar proj num =
|
||||||
|
--TODO: I can easily implement this, but should it even be possible to
|
||||||
|
--delete tickets?
|
||||||
|
error "Not implemented"
|
||||||
|
|
||||||
getTicketEditR :: Text -> Text -> Int -> Handler Html
|
getTicketEditR :: Text -> Text -> Int -> Handler Html
|
||||||
getTicketEditR shar proj num = error "Not implemented"
|
getTicketEditR shar proj num = do
|
||||||
|
Entity _tid ticket <- runDB $ do
|
||||||
|
Entity sid _sharer <- getBy404 $ UniqueSharerIdent shar
|
||||||
|
Entity pid _project <- getBy404 $ UniqueProject proj sid
|
||||||
|
getBy404 $ UniqueTicket pid num
|
||||||
|
user <- requireAuthId
|
||||||
|
((_result, widget), enctype) <- runFormPost $ editTicketForm ticket user
|
||||||
|
defaultLayout $ do
|
||||||
|
setTitle $ toHtml $ T.intercalate " :: "
|
||||||
|
[shar, proj, "Tickets", T.pack ('#' : show num), "Edit"]
|
||||||
|
$(widgetFile "ticket/edit")
|
||||||
|
|
21
templates/ticket/edit.hamlet
Normal file
21
templates/ticket/edit.hamlet
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
$# This file is part of Vervis.
|
||||||
|
$#
|
||||||
|
$# Written in 2016 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/>.
|
||||||
|
|
||||||
|
<h1>#{shar} :: #{proj} :: Tickets :: ##{num} :: Edit
|
||||||
|
|
||||||
|
Enter the details and click "Submit" to update the ticket.
|
||||||
|
|
||||||
|
<form method=PUT action=@{TicketR shar proj num} enctype=#{enctype}>
|
||||||
|
^{widget}
|
||||||
|
<input type=submit>
|
|
@ -24,7 +24,13 @@ $# <http://creativecommons.org/publicdomain/zero/1.0/>.
|
||||||
Created on #{formatTime defaultTimeLocale "%F" $ ticketCreated ticket} by
|
Created on #{formatTime defaultTimeLocale "%F" $ ticketCreated ticket} by
|
||||||
#{fromMaybe (sharerIdent author) $ sharerName author}
|
#{fromMaybe (sharerIdent author) $ sharerName author}
|
||||||
|
|
||||||
<p>Done: #{ticketDone ticket}
|
<p>
|
||||||
|
Status:
|
||||||
|
$if ticketDone ticket
|
||||||
|
Closed on #{formatTime defaultTimeLocale "%F" $ ticketClosed ticket} by
|
||||||
|
#{fromMaybe (sharerIdent closer) $ sharerName closer}
|
||||||
|
$else
|
||||||
|
Open
|
||||||
|
|
||||||
<h2>#{ticketTitle ticket}
|
<h2>#{ticketTitle ticket}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue