Compare commits
	
		
			9 Commits
		
	
	
		
			b3737b312b
			...
			7983f9b187
		
	
	| Author | SHA1 | Date | 
|---|---|---|
|  | 7983f9b187 | |
|  | 6b93991cb3 | |
|  | 8c100e167b | |
|  | 1612bc6ff6 | |
|  | c8610caa54 | |
|  | bd1bcf51e9 | |
|  | 1276ffe020 | |
|  | 56c15cb7f5 | |
|  | 60d8502741 | 
|  | @ -162,7 +162,7 @@ resolveMutation :: (MonadRequest m, MonadEmail m, MonadRandom m, MonadTime m, | ||||||
|     MonadDB m, MonadError GQLError m, MonadPermissions m) => Mutation m |     MonadDB m, MonadError GQLError m, MonadPermissions m) => Mutation m | ||||||
| resolveMutation = Mutation | resolveMutation = Mutation | ||||||
|     { apply = newUser |     { apply = newUser | ||||||
|     , verifyEmail = \(Arg secret) -> voidU $ dbVerifyEmail secret |     , verifyEmail = \(Arg secret) -> either (const False) (const True) <$> dbVerifyEmail secret | ||||||
|     , resendVerificationEmail = \(Arg id) -> targetUser id >>= dbGetUserPendingEmail >>= liftDBEither >>= |     , resendVerificationEmail = \(Arg id) -> targetUser id >>= dbGetUserPendingEmail >>= liftDBEither >>= | ||||||
|         maybe (pure Unit) (dbGetUserByEmail >=> liftDBEither >=> voidU . sendVerificationSecret) |         maybe (pure Unit) (dbGetUserByEmail >=> liftDBEither >=> voidU . sendVerificationSecret) | ||||||
|     , update = \updateArgs (Arg id) -> targetUser id >>= \user -> |     , update = \updateArgs (Arg id) -> targetUser id >>= \user -> | ||||||
|  | @ -258,7 +258,7 @@ data Query m = Query | ||||||
| 
 | 
 | ||||||
| data Mutation m = Mutation | data Mutation m = Mutation | ||||||
|     { apply :: ApplicationArgs -> m (User m) |     { apply :: ApplicationArgs -> m (User m) | ||||||
|     , verifyEmail :: Arg "secret" Text -> m Unit |     , verifyEmail :: Arg "secret" Text -> m Bool | ||||||
|     , resendVerificationEmail :: Arg "user" (Maybe UserID) -> m Unit |     , resendVerificationEmail :: Arg "user" (Maybe UserID) -> m Unit | ||||||
|     , update :: UpdateArgs -> Arg "user" (Maybe UserID) -> m (User m) |     , update :: UpdateArgs -> Arg "user" (Maybe UserID) -> m (User m) | ||||||
|     , newToken :: NewTokenArgs -> m (Token m) |     , newToken :: NewTokenArgs -> m (Token m) | ||||||
|  |  | ||||||
|  | @ -33,6 +33,7 @@ main = do | ||||||
|       GCEmails -> gcEmailsMain |       GCEmails -> gcEmailsMain | ||||||
|       GCApplications -> gcApplicationsMain |       GCApplications -> gcApplicationsMain | ||||||
|       GCAll -> gcAllMain |       GCAll -> gcAllMain | ||||||
|  |       UpdateUser opts -> updateUserMain opts | ||||||
| 
 | 
 | ||||||
| addUserMain :: AddUserOpts -> CLIM () | addUserMain :: AddUserOpts -> CLIM () | ||||||
| addUserMain AddUserOpts {..} = do | addUserMain AddUserOpts {..} = do | ||||||
|  | @ -74,6 +75,15 @@ gcAllMain = do | ||||||
|     gcEmailsMain |     gcEmailsMain | ||||||
|     gcApplicationsMain |     gcApplicationsMain | ||||||
| 
 | 
 | ||||||
|  | updateUserMain :: UpdateUserOpts -> CLIM () | ||||||
|  | updateUserMain UpdateUserOpts {..} = runQuery $ do | ||||||
|  |     case updateUserApplication of | ||||||
|  |       Nothing -> pure () | ||||||
|  |       Just application -> Sql.updateUserData updateUserId [] [Sql.SetUserApplication application] | ||||||
|  |     case updateUserPermissions of | ||||||
|  |       Nothing -> pure () | ||||||
|  |       Just permissions -> Sql.setPermissions updateUserId $ show permissions | ||||||
|  | 
 | ||||||
| cliOptions :: Parser CLIOptions | cliOptions :: Parser CLIOptions | ||||||
| cliOptions = CLIOptions | cliOptions = CLIOptions | ||||||
|     <$> strOption (short 'u' <> long "db-url" <> metavar "URL" <> value "postgres:///datarekisteri-backend") |     <$> strOption (short 'u' <> long "db-url" <> metavar "URL" <> value "postgres:///datarekisteri-backend") | ||||||
|  | @ -85,6 +95,7 @@ cliCommandParser = hsubparser | ||||||
|     <> command "gc-expired-emails" (info (pure GCEmails) (progDesc "Delete emails that have not been verified in time.")) |     <> command "gc-expired-emails" (info (pure GCEmails) (progDesc "Delete emails that have not been verified in time.")) | ||||||
|     <> command "gc-applications" (info (pure GCApplications) (progDesc "Delete users without an email address")) |     <> command "gc-applications" (info (pure GCApplications) (progDesc "Delete users without an email address")) | ||||||
|     <> command "gc" (info (pure GCAll) (progDesc "Run all garbage collection jobs.")) |     <> command "gc" (info (pure GCAll) (progDesc "Run all garbage collection jobs.")) | ||||||
|  |     <> command "update-user" (info updateUserCommand (progDesc "Update a datarekisteri user")) | ||||||
| 
 | 
 | ||||||
| addUserCommand :: Parser CLISubCommand | addUserCommand :: Parser CLISubCommand | ||||||
| addUserCommand = fmap AddUser $ AddUserOpts | addUserCommand = fmap AddUser $ AddUserOpts | ||||||
|  | @ -94,9 +105,18 @@ addUserCommand = fmap AddUser $ AddUserOpts | ||||||
|     <*> strOption (long "homeplace" <> metavar "NAME" <> help "The user's homeplace, usually a city") |     <*> strOption (long "homeplace" <> metavar "NAME" <> help "The user's homeplace, usually a city") | ||||||
|     <*> option (maybeReader $ toPhoneNumber . toText) (long "phone-number" <> metavar "PHONE" <> help "The user's phone number. Only numbers, spaces and the plus-sign are allowed") |     <*> option (maybeReader $ toPhoneNumber . toText) (long "phone-number" <> metavar "PHONE" <> help "The user's phone number. Only numbers, spaces and the plus-sign are allowed") | ||||||
|     <*> option (maybeReader $ toEmail . toText) (long "email" <> metavar "EMAIL" <> help "The user's email address.") |     <*> option (maybeReader $ toEmail . toText) (long "email" <> metavar "EMAIL" <> help "The user's email address.") | ||||||
|     <*> (fromList <$> many permissionParser) |     <*> permissionsParser | ||||||
|     <*> strOption (long "application" <> metavar "TEXT" <> value "Added by the admin.") |     <*> strOption (long "application" <> metavar "TEXT" <> value "Added by the admin.") | ||||||
| 
 | 
 | ||||||
|  | updateUserCommand :: Parser CLISubCommand | ||||||
|  | updateUserCommand = fmap UpdateUser $ UpdateUserOpts | ||||||
|  |     <$> (UserID <$> argument auto (metavar "USER")) | ||||||
|  |     <*> optional (strOption (long "application" <> metavar "TEXT")) | ||||||
|  |     <*> optional permissionsParser | ||||||
|  | 
 | ||||||
|  | permissionsParser :: Parser (Map Scope Permission) | ||||||
|  | permissionsParser = fromList <$> many permissionParser | ||||||
|  | 
 | ||||||
| permissionParser :: Parser (Scope, Permission) | permissionParser :: Parser (Scope, Permission) | ||||||
| permissionParser = (,) | permissionParser = (,) | ||||||
|     <$> option auto (long "scope") |     <$> option auto (long "scope") | ||||||
|  | @ -111,6 +131,13 @@ data CLISubCommand = AddUser AddUserOpts | ||||||
|                    | GCEmails |                    | GCEmails | ||||||
|                    | GCApplications |                    | GCApplications | ||||||
|                    | GCAll |                    | GCAll | ||||||
|  |                    | UpdateUser UpdateUserOpts | ||||||
|  | 
 | ||||||
|  | data UpdateUserOpts = UpdateUserOpts | ||||||
|  |     { updateUserId :: UserID | ||||||
|  |     , updateUserApplication :: Maybe Text | ||||||
|  |     , updateUserPermissions :: Maybe (Map Scope Permission) | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
| data AddUserOpts = AddUserOpts | data AddUserOpts = AddUserOpts | ||||||
|         { addUserNickname :: Maybe Text |         { addUserNickname :: Maybe Text | ||||||
|  |  | ||||||
|  | @ -6,7 +6,31 @@ | ||||||
| 
 | 
 | ||||||
| {-# LANGUAGE NoImplicitPrelude #-} | {-# LANGUAGE NoImplicitPrelude #-} | ||||||
| 
 | 
 | ||||||
| module Datarekisteri.Core.Types where | module Datarekisteri.Core.Types | ||||||
|  |     ( Base64 | ||||||
|  |     , Date(..) | ||||||
|  |     , Email | ||||||
|  |     , KeyID(..) | ||||||
|  |     , Permission(..) | ||||||
|  |     , PhoneNumber | ||||||
|  |     , Scope(..) | ||||||
|  |     , Time(..) | ||||||
|  |     , TokenID(..) | ||||||
|  |     , Unit(Unit) | ||||||
|  |     , UserID(..) | ||||||
|  |     , addTime | ||||||
|  |     , base64Decode | ||||||
|  |     , base64Encode | ||||||
|  |     , readPermission | ||||||
|  |     , renderDate | ||||||
|  |     , renderEmail | ||||||
|  |     , renderPhoneNumber | ||||||
|  |     , renderTime | ||||||
|  |     , toDate | ||||||
|  |     , toEmail | ||||||
|  |     , toPhoneNumber | ||||||
|  |     , toTime | ||||||
|  |     ) where | ||||||
| 
 | 
 | ||||||
| import Relude | import Relude | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
|  | @ -65,7 +65,7 @@ type Query { | ||||||
| 
 | 
 | ||||||
| type Mutation { | type Mutation { | ||||||
|   apply(email: Email!, phoneNumber: PhoneNumber!, password: String!, name: String!, nickname: String, birthdate: Date!, homeplace: String!, application: String!): User! |   apply(email: Email!, phoneNumber: PhoneNumber!, password: String!, name: String!, nickname: String, birthdate: Date!, homeplace: String!, application: String!): User! | ||||||
|   verifyEmail(secret: String!): Unit! |   verifyEmail(secret: String!): Boolean! | ||||||
|   resendVerificationEmail(user: UserID): Unit! |   resendVerificationEmail(user: UserID): Unit! | ||||||
|   update(email: Email, phoneNumber: PhoneNumber, password: String, name: String, nickname: String, homeplace: String, application: String, user: UserID): User! |   update(email: Email, phoneNumber: PhoneNumber, password: String, name: String, nickname: String, homeplace: String, application: String, user: UserID): User! | ||||||
|   newToken(comment: String, name: String, permissions: String): Token! |   newToken(comment: String, name: String, permissions: String): Token! | ||||||
|  |  | ||||||
|  | @ -33,12 +33,13 @@ telephoneField = Field | ||||||
|     { fieldParse = \rawValues _ -> |     { fieldParse = \rawValues _ -> | ||||||
|         case rawValues of |         case rawValues of | ||||||
|           [] -> pure $ Right Nothing |           [] -> pure $ Right Nothing | ||||||
|  |           [""] -> pure $ Right Nothing | ||||||
|           [x] -> pure $ maybe (Left "could not parse as a phone number") (Right . Just) $ toPhoneNumber x |           [x] -> pure $ maybe (Left "could not parse as a phone number") (Right . Just) $ toPhoneNumber x | ||||||
|           _ -> pure $ Left $ "Expected one value" |           _ -> pure $ Left $ "Expected one value" | ||||||
|     , fieldView = \id name otherAttributes result isRequired -> |     , fieldView = \id name otherAttributes result isRequired -> | ||||||
|         let result' = either (\x -> x) renderPhoneNumber result |         let result' = either (\x -> x) renderPhoneNumber result | ||||||
|         in [whamlet| |         in [whamlet| | ||||||
|                <input type="tel" id="#{id}" name="#{name}" value="#{result'}" *{otherAttributes} :isRequired:required="true"> |                <input type="tel" id="#{id}" name="#{name}" value="#{result'}" pattern="[+ 0123456789]*" title="Only '+', spaces and numbers are allowed" "*{otherAttributes} :isRequired:required="true"> | ||||||
|            |] |            |] | ||||||
|     , fieldEnctype = UrlEncoded |     , fieldEnctype = UrlEncoded | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  | @ -14,7 +14,6 @@ module Datarekisteri.Frontend.Handlers.Profile where | ||||||
| 
 | 
 | ||||||
| import Relude hiding (id) | import Relude hiding (id) | ||||||
| 
 | 
 | ||||||
| import Data.Maybe (fromJust) |  | ||||||
| import Data.Morpheus.Client (raw, declareLocalTypesInline) | import Data.Morpheus.Client (raw, declareLocalTypesInline) | ||||||
| 
 | 
 | ||||||
| import Yesod hiding (emailField) | import Yesod hiding (emailField) | ||||||
|  | @ -62,28 +61,22 @@ mutation UpdatePassword($user: UserID, $password: String!) { | ||||||
| passwordForm :: Form Text | passwordForm :: Form Text | ||||||
| passwordForm = renderDivs $ areq verifiedPasswordField "Uusi salasana" Nothing | passwordForm = renderDivs $ areq verifiedPasswordField "Uusi salasana" Nothing | ||||||
| 
 | 
 | ||||||
| profileForm :: (Maybe UserID) -> (Maybe ProfilePageUser) -> Form UpdateProfileArgs | profileForm :: ProfilePageUser -> Form UpdateProfileArgs | ||||||
| profileForm userID user extraHtml = do | profileForm ProfilePageUser {..} extraHtml = do | ||||||
|     (nameRes, nameView) <- mopt textField "Nimi" |     (nameRes, nameView) <- mopt textField "Nimi" (Just $ Just name) | ||||||
|         (Just $ maybe Nothing (\x -> Just $ let ProfilePageUser {..} = x in name) user) |     (homeRes, homeView) <- mopt textField "Kotipaikka" (Just $ Just homeplace) | ||||||
|     (homeRes, homeView) <- mopt textField "Kotipaikka" |     (nicknameRes, nicknameView) <- mopt textField "Kutsumanimi" (Just $ Just nickname) | ||||||
|         (Just $ maybe Nothing (\x -> Just $ let ProfilePageUser {..} = x in homeplace) user) |     (emailRes, emailView) <- mopt emailField "Sähköposti" (Just email) | ||||||
|     (nicknameRes, nicknameView) <- mopt textField "Kutsumanimi" |     (phoneNumberRes, phoneNumberView) <- mopt telephoneField "Puhelinnumero" (Just $ Just phoneNumber) | ||||||
|         (Just $ maybe Nothing (\x -> Just $ let ProfilePageUser {..} = x in nickname) user) |     (applicationRes, applicationView) <- mopt textareaField' "Jäsenhakemus" (Just $ Just application) | ||||||
|     (emailRes, emailView) <- mopt emailField "Sähköposti" |     let profileUpdateRes = UpdateProfileArgs (Just id) <$> | ||||||
|         (maybe Nothing (\x -> Just $ let ProfilePageUser {..} = x in email) user) |  | ||||||
|     (phoneNumberRes, phoneNumberView) <- mopt telephoneField "Puhelinnumero" |  | ||||||
|         (Just $ maybe Nothing (\x -> Just $ let ProfilePageUser {..} = x in phoneNumber) user) |  | ||||||
|     (applicationRes, applicationView) <- mopt textareaField' "Jäsenhakemus" |  | ||||||
|         (Just $ maybe Nothing (\x -> Just $ let ProfilePageUser {..} = x in application) user) |  | ||||||
|     let profileUpdateRes = UpdateProfileArgs userID <$> |  | ||||||
|             nameRes <*> homeRes <*> nicknameRes <*> emailRes <*> phoneNumberRes <*> applicationRes |             nameRes <*> homeRes <*> nicknameRes <*> emailRes <*> phoneNumberRes <*> applicationRes | ||||||
|         maybePendingEmail = user >>= \x -> let ProfilePageUser {..} = x in pendingEmail |  | ||||||
|         canUpdateApplication = maybe False (\x -> let ProfilePageUser {..} = x in not isMember) user |  | ||||||
|         inputField FieldView {..} = [whamlet| |         inputField FieldView {..} = [whamlet| | ||||||
|             <label for="#{fvId}"> |             <label for="#{fvId}"> | ||||||
|                 ^{fvLabel} |                 ^{fvLabel} | ||||||
|                 ^{fvInput} |                 ^{fvInput} | ||||||
|  |                 $maybe err <- fvErrors | ||||||
|  |                     <div class=".errors">#{err} | ||||||
|             |] |             |] | ||||||
|         widget = [whamlet| |         widget = [whamlet| | ||||||
|             #{extraHtml} |             #{extraHtml} | ||||||
|  | @ -92,12 +85,12 @@ profileForm userID user extraHtml = do | ||||||
|             ^{inputField nicknameView} |             ^{inputField nicknameView} | ||||||
|             ^{inputField phoneNumberView} |             ^{inputField phoneNumberView} | ||||||
|             ^{inputField emailView} |             ^{inputField emailView} | ||||||
|             $maybe pendingEmail <- maybePendingEmail |             $maybe pending <- pendingEmail | ||||||
|                 <p>Päivitys osoitteeseen # |                 <p>Päivitys osoitteeseen # | ||||||
|                     <a href="mailto:#{renderEmail pendingEmail}">#{renderEmail pendingEmail} |                     <a href="mailto:#{renderEmail pending}">#{renderEmail pending} | ||||||
|                     odottaa vahvistusta. # |                     odottaa vahvistusta. # | ||||||
|                     <a href="@{VerifyEmailR}">Siirry vahvistamaan |                     <a href="@{VerifyEmailR}">Siirry vahvistamaan | ||||||
|             $if canUpdateApplication |             $if not isMember | ||||||
|                 ^{inputField applicationView} |                 ^{inputField applicationView} | ||||||
|             |] |             |] | ||||||
|     return (profileUpdateRes, widget) |     return (profileUpdateRes, widget) | ||||||
|  | @ -127,10 +120,11 @@ getProfile userID = do | ||||||
|     ProfilePage {..} <- apiRequest True (ProfilePageArgs {id = userID}) |     ProfilePage {..} <- apiRequest True (ProfilePageArgs {id = userID}) | ||||||
|     passwordForm <- liftHandler $ generateFormPost passwordForm |     passwordForm <- liftHandler $ generateFormPost passwordForm | ||||||
|     profileForm <- liftHandler $ generateFormPost $ |     profileForm <- liftHandler $ generateFormPost $ | ||||||
|         profileForm ((\x -> let ProfilePageUser {..} = x in id) <$> user) user |         profileForm (fromJust user) | ||||||
|     defaultLayout $ profile (fromJust user) |     defaultLayout $ profile (fromJust user) | ||||||
|         profileForm passwordForm |         profileForm passwordForm | ||||||
|         where fromJust = fromMaybe $ error "Tried to access the profile of an inexistent user" | 
 | ||||||
|  | fromJust = fromMaybe $ error "Tried to access the profile of an inexistent user" | ||||||
| 
 | 
 | ||||||
| getOwnProfileR :: (YesodAuth DataIdClient, AuthId DataIdClient ~ Text) => Handler Html | getOwnProfileR :: (YesodAuth DataIdClient, AuthId DataIdClient ~ Text) => Handler Html | ||||||
| getOwnProfileR = getProfile Nothing | getOwnProfileR = getProfile Nothing | ||||||
|  | @ -140,12 +134,12 @@ getProfileR = getProfile . Just | ||||||
| 
 | 
 | ||||||
| postProfileR :: (YesodAuth DataIdClient, AuthId DataIdClient ~ Text) => UserID -> Handler Html | postProfileR :: (YesodAuth DataIdClient, AuthId DataIdClient ~ Text) => UserID -> Handler Html | ||||||
| postProfileR userID = do | postProfileR userID = do | ||||||
|     ((result, widget), enctype) <- runFormPost $ profileForm (Just userID) Nothing |     ProfilePage {..} <- apiRequest True (ProfilePageArgs {id = Just userID}) | ||||||
|  |     ((result, widget), enctype) <- runFormPost $ profileForm (fromJust user) | ||||||
|     case result of |     case result of | ||||||
|       FormSuccess update -> apiRequest @UpdateProfile True update >> redirect (ProfileR userID) |       FormSuccess update -> apiRequest @UpdateProfile True update >> redirect (ProfileR userID) | ||||||
|       _ -> do |       _ -> do | ||||||
|           passwordForm <- liftHandler $ generateFormPost passwordForm |           passwordForm <- liftHandler $ generateFormPost passwordForm | ||||||
|           ProfilePage {..} <- apiRequest True (ProfilePageArgs {id = Just userID}) |  | ||||||
|           defaultLayout $ profile (fromJust user) (widget, enctype) passwordForm |           defaultLayout $ profile (fromJust user) (widget, enctype) passwordForm | ||||||
|           --- XXX fromJust explodes if the user no longer exists |           --- XXX fromJust explodes if the user no longer exists | ||||||
| 
 | 
 | ||||||
|  | @ -156,7 +150,7 @@ postUpdatePasswordR userID = do | ||||||
|       FormSuccess new -> |       FormSuccess new -> | ||||||
|           apiRequest @UpdatePassword True (UpdatePasswordArgs {password = new, user = Just userID}) >> redirect (ProfileR userID) |           apiRequest @UpdatePassword True (UpdatePasswordArgs {password = new, user = Just userID}) >> redirect (ProfileR userID) | ||||||
|       _ -> do |       _ -> do | ||||||
|           profileForm <- liftHandler $ generateFormPost $ profileForm (Just userID) Nothing |  | ||||||
|           ProfilePage {..} <- apiRequest True (ProfilePageArgs {id = Just userID}) |           ProfilePage {..} <- apiRequest True (ProfilePageArgs {id = Just userID}) | ||||||
|  |           profileForm <- liftHandler $ generateFormPost $ profileForm (fromJust user) | ||||||
|           defaultLayout $ profile (fromJust user) profileForm (widget, enctype) |           defaultLayout $ profile (fromJust user) profileForm (widget, enctype) | ||||||
|           --- XXX fromJust explodes if the user no longer exists |           --- XXX fromJust explodes if the user no longer exists | ||||||
|  |  | ||||||
|  | @ -36,7 +36,11 @@ postVerifyEmailR :: (YesodAuth DataIdClient, AuthId DataIdClient ~ Text) => Hand | ||||||
| postVerifyEmailR = do | postVerifyEmailR = do | ||||||
|     ((result, widget), enctype) <- runFormPost verifyForm |     ((result, widget), enctype) <- runFormPost verifyForm | ||||||
|     case result of |     case result of | ||||||
|       FormSuccess verify -> apiRequest @VerifyEmail False verify >> setMessage "Sähköpostiosoite vahvistettu" >> redirect OwnProfileR |       FormSuccess verify -> do | ||||||
|  |           success <- apiRequest @VerifyEmail False verify | ||||||
|  |           case success of | ||||||
|  |             VerifyEmail True -> setMessage "Sähköpostiosoite vahvistettu" >> redirect OwnProfileR | ||||||
|  |             VerifyEmail False -> setMessage "Virheellinen vahvistuskoodi" >> redirect VerifyEmailR | ||||||
|       _ -> defaultLayout $ verifyEmailW (widget, enctype) |       _ -> defaultLayout $ verifyEmailW (widget, enctype) | ||||||
| 
 | 
 | ||||||
| verifyEmailW (codeWidget, codeEnctype) = do | verifyEmailW (codeWidget, codeEnctype) = do | ||||||
|  |  | ||||||
|  | @ -287,6 +287,9 @@ instance Yesod DataIdClient where | ||||||
|                 input[type="submit"].reject-button:hover { |                 input[type="submit"].reject-button:hover { | ||||||
|                   background-color: #aa3968; |                   background-color: #aa3968; | ||||||
|                 } |                 } | ||||||
|  |                 input:invalid { | ||||||
|  |                   border-color: #8a003a; | ||||||
|  |                 } | ||||||
|                 nav { |                 nav { | ||||||
|                   display: block; |                   display: block; | ||||||
|                   position: fixed; |                   position: fixed; | ||||||
|  | @ -322,6 +325,18 @@ instance Yesod DataIdClient where | ||||||
|                 .left-nav { |                 .left-nav { | ||||||
|                   float: right; |                   float: right; | ||||||
|                 } |                 } | ||||||
|  |                 aside.messages > ul { | ||||||
|  |                   list-style: none; | ||||||
|  |                   padding: 0; | ||||||
|  |                   margin: 0; | ||||||
|  |                 } | ||||||
|  |                 aside.messages li.message { | ||||||
|  |                   display: block; | ||||||
|  |                   padding: 0.7em; | ||||||
|  |                   background-color: #3b4553; | ||||||
|  |                   color: white; | ||||||
|  |                   border-radius: 0.3em; | ||||||
|  |                 } | ||||||
|                 main { |                 main { | ||||||
|                   margin: 0 auto; |                   margin: 0 auto; | ||||||
|                   max-width: 50em; |                   max-width: 50em; | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue