Compare commits
	
		
			28 Commits
		
	
	
		
			1.7.14
			...
			e7c840a4d4
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | e7c840a4d4 | ||
|  | 96dc80fef9 | ||
|  | ef76611d33 | ||
|  | febd04c374 | ||
|  | 54ee3594eb | ||
|  | d7e90614c8 | ||
|  | b5dbfe336d | ||
|  | b673a2fe6b | ||
|  | f26682e6de | ||
|  | 218be2f0b1 | ||
|  | d7515f3611 | ||
|  | 6ae1096d79 | ||
|  | 1f0d7adae3 | ||
|  | fc96ae526f | ||
|  | 8881ef0f85 | ||
|  | 5e82a687f9 | ||
|  | de4ecf8dd6 | ||
|  | 064efd4386 | ||
|  | 65b8ba3b47 | ||
| 9d452ed8cb | |||
|  | 441419b92b | ||
|  | aecf2c15be | ||
|  | 79da56c794 | ||
|  | ef10902c1e | ||
|  | c277f85c2a | ||
|  | 035653c7fa | ||
|  | 6358bc3deb | ||
|  | 9f5066f982 | 
							
								
								
									
										679
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										679
									
								
								Cargo.lock
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,6 +1,6 @@ | |||||||
| [package] | [package] | ||||||
| name = "reminder-rs" | name = "reminder-rs" | ||||||
| version = "1.7.13" | version = "1.7.24" | ||||||
| authors = ["Jude Southworth <judesouthworth@pm.me>"] | authors = ["Jude Southworth <judesouthworth@pm.me>"] | ||||||
| edition = "2021" | edition = "2021" | ||||||
| license = "AGPL-3.0 only" | license = "AGPL-3.0 only" | ||||||
| @@ -30,11 +30,11 @@ secrecy = "0.8.0" | |||||||
| futures = "0.3.30" | futures = "0.3.30" | ||||||
| prometheus = "0.13.3" | prometheus = "0.13.3" | ||||||
| rocket = { version = "0.5.0", features = ["tls", "secrets", "json"] } | rocket = { version = "0.5.0", features = ["tls", "secrets", "json"] } | ||||||
| rocket_dyn_templates = { version = "0.1.0", features = ["tera"] } | rocket_dyn_templates = { version = "0.2.0", features = ["tera"] } | ||||||
| serenity = { version = "0.12", default-features = false, features = ["builder", "cache", "client", "gateway", "http", "model", "utils", "rustls_backend"] } | serenity = { version = "0.12", default-features = false, features = ["builder", "cache", "client", "gateway", "http", "model", "utils", "rustls_backend"] } | ||||||
| oauth2 = "4" | oauth2 = "4" | ||||||
| csv = "1.2" | csv = "1.2" | ||||||
| axum = "0.7" | sd-notify = "0.4.1" | ||||||
|  |  | ||||||
| [dependencies.extract_derive] | [dependencies.extract_derive] | ||||||
| path = "extract_derive" | path = "extract_derive" | ||||||
|   | |||||||
| @@ -136,9 +136,9 @@ CREATE TABLE reminders ( | |||||||
|     set_by INT UNSIGNED, |     set_by INT UNSIGNED, | ||||||
|  |  | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (message_id) REFERENCES messages(id) ON DELETE RESTRICT, |     CONSTRAINT `reminder_message_fk` FOREIGN KEY (message_id) REFERENCES messages(id) ON DELETE RESTRICT, | ||||||
|     FOREIGN KEY (channel_id) REFERENCES channels(id) ON DELETE CASCADE, |     CONSTRAINT `reminder_channel_fk` FOREIGN KEY (channel_id) REFERENCES channels(id) ON DELETE CASCADE, | ||||||
|     FOREIGN KEY (set_by) REFERENCES users(id) ON DELETE SET NULL |     CONSTRAINT `reminder_user_fk` FOREIGN KEY (set_by) REFERENCES users(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
|  |  | ||||||
| CREATE TRIGGER message_cleanup AFTER DELETE ON reminders | CREATE TRIGGER message_cleanup AFTER DELETE ON reminders | ||||||
| @@ -157,9 +157,9 @@ CREATE TABLE todos ( | |||||||
|     value VARCHAR(2000) NOT NULL, |     value VARCHAR(2000) NOT NULL, | ||||||
|  |  | ||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|     FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE SET NULL, |     CONSTRAINT todos_ibfk_5 FOREIGN KEY (user_id) REFERENCES users(id) ON DELETE SET NULL, | ||||||
|     FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE, |     CONSTRAINT todos_ibfk_4 FOREIGN KEY (guild_id) REFERENCES guilds(id) ON DELETE CASCADE, | ||||||
|     FOREIGN KEY (channel_id) REFERENCES channels(id) ON DELETE SET NULL |     CONSTRAINT todos_ibfk_3 FOREIGN KEY (channel_id) REFERENCES channels(id) ON DELETE SET NULL | ||||||
| ); | ); | ||||||
|  |  | ||||||
| CREATE TABLE command_restrictions ( | CREATE TABLE command_restrictions ( | ||||||
|   | |||||||
| @@ -46,7 +46,7 @@ CREATE TABLE reminders_new ( | |||||||
|     PRIMARY KEY (id), |     PRIMARY KEY (id), | ||||||
|  |  | ||||||
|     FOREIGN KEY (`channel_id`) REFERENCES channels (`id`) ON DELETE CASCADE, |     FOREIGN KEY (`channel_id`) REFERENCES channels (`id`) ON DELETE CASCADE, | ||||||
|     FOREIGN KEY (`set_by`) REFERENCES users (`id`) ON DELETE SET NULL |     CONSTRAINT `reminders_ibfk_2` FOREIGN KEY (`set_by`) REFERENCES `users` (`id`) ON DELETE SET NULL | ||||||
|  |  | ||||||
|     # disallow having a reminder as restartable if it has no interval |     # disallow having a reminder as restartable if it has no interval | ||||||
|     -- , CONSTRAINT restartable_interval_mutex CHECK (`restartable` = 0 OR `interval` IS NULL) |     -- , CONSTRAINT restartable_interval_mutex CHECK (`restartable` = 0 OR `interval` IS NULL) | ||||||
|   | |||||||
							
								
								
									
										27
									
								
								migrations/20240630150936_dashboard_preferences.sql
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								migrations/20240630150936_dashboard_preferences.sql
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | |||||||
|  | SET FOREIGN_KEY_CHECKS=0; | ||||||
|  |  | ||||||
|  | -- Tables no longer needed as old dashboard is decomm. | ||||||
|  | DROP TABLE guild_users; | ||||||
|  | DROP TABLE events; | ||||||
|  |  | ||||||
|  | ALTER TABLE users ADD COLUMN `reset_inputs_on_create` BOOLEAN NOT NULL DEFAULT 0; | ||||||
|  | ALTER TABLE users ADD COLUMN `use_browser_timezone` BOOLEAN NOT NULL DEFAULT 1; | ||||||
|  | ALTER TABLE users ADD COLUMN `dashboard_color_scheme` ENUM('system', 'light', 'dark') NOT NULL DEFAULT 'system'; | ||||||
|  |  | ||||||
|  | ALTER TABLE users DROP COLUMN `language`; | ||||||
|  | ALTER TABLE users DROP COLUMN `patreon`; | ||||||
|  | ALTER TABLE users DROP COLUMN `name`; | ||||||
|  |  | ||||||
|  | ALTER TABLE todos DROP CONSTRAINT todos_ibfk_5, MODIFY COLUMN user_id BIGINT UNSIGNED; | ||||||
|  | UPDATE todos SET user_id = (SELECT user FROM users WHERE id = user_id); | ||||||
|  | ALTER TABLE todos ADD CONSTRAINT todos_user_fk FOREIGN KEY (user_id) REFERENCES users(user); | ||||||
|  |  | ||||||
|  | ALTER TABLE reminders DROP CONSTRAINT reminders_ibfk_2, MODIFY COLUMN set_by BIGINT UNSIGNED; | ||||||
|  | UPDATE reminders SET set_by = (SELECT user FROM users WHERE id = set_by); | ||||||
|  | ALTER TABLE reminders ADD CONSTRAINT reminder_user_fk FOREIGN KEY (set_by) REFERENCES users(user); | ||||||
|  |  | ||||||
|  | ALTER TABLE users DROP PRIMARY KEY, CHANGE id id INT UNSIGNED, ADD PRIMARY KEY (`user`); | ||||||
|  | ALTER TABLE users DROP COLUMN `id`; | ||||||
|  | ALTER TABLE users RENAME COLUMN `user` TO `id`; | ||||||
|  |  | ||||||
|  | SET FOREIGN_KEY_CHECKS=1; | ||||||
| @@ -1,12 +1,22 @@ | |||||||
| server { | server { | ||||||
|     server_name www.reminder-bot.com; |     server_name www.reminder-bot.com; | ||||||
| 
 | 
 | ||||||
|     return 301 $scheme://reminder-bot.com$request_uri; |     return 301 https://reminder-bot.com$request_uri; | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| server { | server { | ||||||
|     listen 80; |     listen 80; | ||||||
|     server_name reminder-bot.com; |     server_name beta.reminder-bot.com; | ||||||
|  | 
 | ||||||
|  |     return 301 https://reminder-bot.com$request_uri; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | server { | ||||||
|  |     listen 443 ssl; | ||||||
|  |     server_name beta.reminder-bot.com; | ||||||
|  | 
 | ||||||
|  |     ssl_certificate /etc/letsencrypt/live/beta.reminder-bot.com/fullchain.pem; | ||||||
|  |     ssl_certificate_key /etc/letsencrypt/live/beta.reminder-bot.com/privkey.pem; | ||||||
| 
 | 
 | ||||||
|     return 301 https://reminder-bot.com$request_uri; |     return 301 https://reminder-bot.com$request_uri; | ||||||
| } | } | ||||||
| @@ -25,6 +35,8 @@ server { | |||||||
|     proxy_buffers 4 256k; |     proxy_buffers 4 256k; | ||||||
|     proxy_busy_buffers_size 256k; |     proxy_busy_buffers_size 256k; | ||||||
| 
 | 
 | ||||||
|  |     client_max_body_size 10M; | ||||||
|  | 
 | ||||||
|     location / { |     location / { | ||||||
|         proxy_pass http://localhost:18920; |         proxy_pass http://localhost:18920; | ||||||
|         proxy_redirect off; |         proxy_redirect off; | ||||||
| @@ -22,7 +22,6 @@ | |||||||
| 	<title>Reminder Bot | Dashboard</title> | 	<title>Reminder Bot | Dashboard</title> | ||||||
|  |  | ||||||
| 	<!-- styles --> | 	<!-- styles --> | ||||||
| 	<link rel="stylesheet" href="/static/css/bulma.min.css"> |  | ||||||
| 	<link rel="stylesheet" href="/static/css/fa.css"> | 	<link rel="stylesheet" href="/static/css/fa.css"> | ||||||
| 	<link rel="stylesheet" href="/static/css/font.css"> | 	<link rel="stylesheet" href="/static/css/font.css"> | ||||||
| 	<link rel="stylesheet" href="/static/css/style.css"> | 	<link rel="stylesheet" href="/static/css/style.css"> | ||||||
|   | |||||||
							
								
								
									
										1321
									
								
								reminder-dashboard/package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										1321
									
								
								reminder-dashboard/package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -1,9 +1,27 @@ | |||||||
| import axios from "axios"; | import axios from "axios"; | ||||||
|  |  | ||||||
|  | enum ColorScheme { | ||||||
|  |     System = "system", | ||||||
|  |     Dark = "dark", | ||||||
|  |     Light = "light", | ||||||
|  | } | ||||||
|  |  | ||||||
| type UserInfo = { | type UserInfo = { | ||||||
|     name: string; |     name: string; | ||||||
|     patreon: boolean; |     patreon: boolean; | ||||||
|     timezone: string | null; |     preferences: { | ||||||
|  |         timezone: string | null; | ||||||
|  |         reset_inputs_on_create: boolean; | ||||||
|  |         dashboard_color_scheme: ColorScheme; | ||||||
|  |         use_browser_timezone: boolean; | ||||||
|  |     }; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | export type UpdateUserInfo = { | ||||||
|  |     timezone?: string; | ||||||
|  |     reset_inputs_on_create?: boolean; | ||||||
|  |     dashboard_color_scheme?: ColorScheme; | ||||||
|  |     use_browser_timezone?: boolean; | ||||||
| }; | }; | ||||||
|  |  | ||||||
| export type GuildInfo = { | export type GuildInfo = { | ||||||
| @@ -109,7 +127,7 @@ export const fetchUserInfo = () => ({ | |||||||
| }); | }); | ||||||
|  |  | ||||||
| export const patchUserInfo = () => ({ | export const patchUserInfo = () => ({ | ||||||
|     mutationFn: (timezone: string) => axios.patch(`/dashboard/api/user`, { timezone }), |     mutationFn: (userInfo: UpdateUserInfo) => axios.patch(`/dashboard/api/user`, userInfo), | ||||||
| }); | }); | ||||||
|  |  | ||||||
| export const fetchUserGuilds = () => ({ | export const fetchUserGuilds = () => ({ | ||||||
|   | |||||||
| @@ -7,6 +7,10 @@ import { useGuild } from "./useGuild"; | |||||||
| export const Mentions = ({ input }) => { | export const Mentions = ({ input }) => { | ||||||
|     const guild = useGuild(); |     const guild = useGuild(); | ||||||
|  |  | ||||||
|  |     return <>{guild && <_Mentions guild={guild} input={input} />}</>; | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const _Mentions = ({ guild, input }) => { | ||||||
|     const { data: roles } = useQuery(fetchGuildRoles(guild)); |     const { data: roles } = useQuery(fetchGuildRoles(guild)); | ||||||
|     const { data: channels } = useQuery(fetchGuildChannels(guild)); |     const { data: channels } = useQuery(fetchGuildChannels(guild)); | ||||||
|     const { data: emojis } = useQuery(fetchGuildEmojis(guild)); |     const { data: emojis } = useQuery(fetchGuildEmojis(guild)); | ||||||
| @@ -17,7 +21,7 @@ export const Mentions = ({ input }) => { | |||||||
|                 { |                 { | ||||||
|                     trigger: "@", |                     trigger: "@", | ||||||
|                     values: (roles || []) |                     values: (roles || []) | ||||||
|                         .filter((role) => role.name === "@everyone") |                         .filter((role) => role.name !== "@everyone") | ||||||
|                         .map(({ id, name }) => ({ key: name, value: id })), |                         .map(({ id, name }) => ({ key: name, value: id })), | ||||||
|                     allowSpaces: true, |                     allowSpaces: true, | ||||||
|                     selectTemplate: (item) => `<@&${item.original.value}>`, |                     selectTemplate: (item) => `<@&${item.original.value}>`, | ||||||
|   | |||||||
| @@ -1,15 +1,27 @@ | |||||||
| import { createContext } from "preact"; | import { createContext } from "preact"; | ||||||
| import { useContext } from "preact/compat"; | import { useContext } from "preact/compat"; | ||||||
| import { useState } from "preact/hooks"; | import { useEffect, useState } from "preact/hooks"; | ||||||
| import { DateTime } from "luxon"; | import { DateTime } from "luxon"; | ||||||
|  | import { fetchUserInfo } from "../../api"; | ||||||
|  | import { useQuery } from "react-query"; | ||||||
|  |  | ||||||
| type TTimezoneContext = [string, (tz: string) => void]; | type TTimezoneContext = [string, (tz: string) => void]; | ||||||
|  |  | ||||||
| const TimezoneContext = createContext(["UTC", () => {}] as TTimezoneContext); | const TimezoneContext = createContext(["UTC", () => {}] as TTimezoneContext); | ||||||
|  |  | ||||||
| export const TimezoneProvider = ({ children }) => { | export const TimezoneProvider = ({ children }) => { | ||||||
|  |     const { data } = useQuery({ ...fetchUserInfo() }); | ||||||
|  |  | ||||||
|     const [timezone, setTimezone] = useState(DateTime.now().zoneName); |     const [timezone, setTimezone] = useState(DateTime.now().zoneName); | ||||||
|  |  | ||||||
|  |     useEffect(() => { | ||||||
|  |         setTimezone( | ||||||
|  |             data === undefined || data.preferences.use_browser_timezone | ||||||
|  |                 ? DateTime.now().zoneName | ||||||
|  |                 : data.preferences.timezone, | ||||||
|  |         ); | ||||||
|  |     }, [data]); | ||||||
|  |  | ||||||
|     return ( |     return ( | ||||||
|         <TimezoneContext.Provider value={[timezone, setTimezone]}> |         <TimezoneContext.Provider value={[timezone, setTimezone]}> | ||||||
|             {children} |             {children} | ||||||
|   | |||||||
| @@ -12,12 +12,17 @@ import { GuildTodos } from "../Guild/GuildTodos"; | |||||||
| export function App() { | export function App() { | ||||||
|     const queryClient = new QueryClient(); |     const queryClient = new QueryClient(); | ||||||
|  |  | ||||||
|  |     let scheme = "light"; | ||||||
|  |     // if (window.matchMedia("(prefers-color-scheme: dark)").matches) { | ||||||
|  |     //     scheme = "dark"; | ||||||
|  |     // } | ||||||
|  |  | ||||||
|     return ( |     return ( | ||||||
|         <TimezoneProvider> |         <QueryClientProvider client={queryClient}> | ||||||
|             <FlashProvider> |             <TimezoneProvider> | ||||||
|                 <QueryClientProvider client={queryClient}> |                 <FlashProvider> | ||||||
|                     <Router base={"/dashboard"}> |                     <Router base={"/dashboard"}> | ||||||
|                         <div class="columns is-gapless dashboard-frame"> |                         <div class={`columns is-gapless dashboard-frame scheme-${scheme}`}> | ||||||
|                             <Sidebar /> |                             <Sidebar /> | ||||||
|                             <div class="column is-main-content"> |                             <div class="column is-main-content"> | ||||||
|                                 <div style={{ margin: "0 12px 12px 12px" }}> |                                 <div style={{ margin: "0 12px 12px 12px" }}> | ||||||
| @@ -47,8 +52,8 @@ export function App() { | |||||||
|                             </div> |                             </div> | ||||||
|                         </div> |                         </div> | ||||||
|                     </Router> |                     </Router> | ||||||
|                 </QueryClientProvider> |                 </FlashProvider> | ||||||
|             </FlashProvider> |             </TimezoneProvider> | ||||||
|         </TimezoneProvider> |         </QueryClientProvider> | ||||||
|     ); |     ); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,21 +1,28 @@ | |||||||
| import { LoadTemplate } from "../LoadTemplate"; | import { LoadTemplate } from "../LoadTemplate"; | ||||||
| import { useReminder } from "../ReminderContext"; | import { useReminder } from "../ReminderContext"; | ||||||
| import { useMutation, useQueryClient } from "react-query"; | import { useMutation, useQuery, useQueryClient } from "react-query"; | ||||||
| import { postGuildReminder, postGuildTemplate, postUserReminder } from "../../../api"; | import { | ||||||
|  |     fetchUserInfo, | ||||||
|  |     postGuildReminder, | ||||||
|  |     postGuildTemplate, | ||||||
|  |     postUserReminder, | ||||||
|  | } from "../../../api"; | ||||||
| import { useState } from "preact/hooks"; | import { useState } from "preact/hooks"; | ||||||
| import { ICON_FLASH_TIME } from "../../../consts"; | import { ICON_FLASH_TIME } from "../../../consts"; | ||||||
| import { useFlash } from "../../App/FlashContext"; | import { useFlash } from "../../App/FlashContext"; | ||||||
| import { useGuild } from "../../App/useGuild"; | import { useGuild } from "../../App/useGuild"; | ||||||
|  | import { defaultReminder } from "../CreateReminder"; | ||||||
|  |  | ||||||
| export const CreateButtonRow = () => { | export const CreateButtonRow = () => { | ||||||
|     const guild = useGuild(); |     const guild = useGuild(); | ||||||
|     const [reminder] = useReminder(); |     const [reminder, setReminder] = useReminder(); | ||||||
|  |  | ||||||
|     const [recentlyCreated, setRecentlyCreated] = useState(false); |     const [recentlyCreated, setRecentlyCreated] = useState(false); | ||||||
|     const [templateRecentlyCreated, setTemplateRecentlyCreated] = useState(false); |     const [templateRecentlyCreated, setTemplateRecentlyCreated] = useState(false); | ||||||
|  |  | ||||||
|     const flash = useFlash(); |     const flash = useFlash(); | ||||||
|     const queryClient = useQueryClient(); |     const queryClient = useQueryClient(); | ||||||
|  |     const { data: userInfo } = useQuery({ ...fetchUserInfo() }); | ||||||
|     const mutation = useMutation({ |     const mutation = useMutation({ | ||||||
|         ...(guild ? postGuildReminder(guild) : postUserReminder()), |         ...(guild ? postGuildReminder(guild) : postUserReminder()), | ||||||
|         onError: (error) => { |         onError: (error) => { | ||||||
| @@ -44,6 +51,9 @@ export const CreateButtonRow = () => { | |||||||
|                         queryKey: ["USER_REMINDERS"], |                         queryKey: ["USER_REMINDERS"], | ||||||
|                     }); |                     }); | ||||||
|                 } |                 } | ||||||
|  |                 if (userInfo.preferences.reset_inputs_on_create) { | ||||||
|  |                     setReminder(() => defaultReminder()); | ||||||
|  |                 } | ||||||
|                 setRecentlyCreated(true); |                 setRecentlyCreated(true); | ||||||
|                 setTimeout(() => { |                 setTimeout(() => { | ||||||
|                     setRecentlyCreated(false); |                     setRecentlyCreated(false); | ||||||
|   | |||||||
| @@ -12,7 +12,6 @@ export const EditButtonRow = () => { | |||||||
|     const [reminder, setReminder] = useReminder(); |     const [reminder, setReminder] = useReminder(); | ||||||
|  |  | ||||||
|     const [recentlySaved, setRecentlySaved] = useState(false); |     const [recentlySaved, setRecentlySaved] = useState(false); | ||||||
|  |  | ||||||
|     const iconFlashTimeout = useRef(0); |     const iconFlashTimeout = useRef(0); | ||||||
|  |  | ||||||
|     const flash = useFlash(); |     const flash = useFlash(); | ||||||
|   | |||||||
| @@ -10,9 +10,8 @@ import { useQuery } from "react-query"; | |||||||
| import "./styles.scss"; | import "./styles.scss"; | ||||||
| import { useGuild } from "../App/useGuild"; | import { useGuild } from "../App/useGuild"; | ||||||
| import { DEFAULT_COLOR } from "./Embed"; | import { DEFAULT_COLOR } from "./Embed"; | ||||||
| import { useTimezone } from "../App/TimezoneProvider"; |  | ||||||
|  |  | ||||||
| function defaultReminder(): Reminder { | export function defaultReminder(): Reminder { | ||||||
|     return { |     return { | ||||||
|         attachment: null, |         attachment: null, | ||||||
|         attachment_name: null, |         attachment_name: null, | ||||||
| @@ -46,6 +45,37 @@ function defaultReminder(): Reminder { | |||||||
| export const CreateReminder = () => { | export const CreateReminder = () => { | ||||||
|     const guild = useGuild(); |     const guild = useGuild(); | ||||||
|  |  | ||||||
|  |     if (guild) { | ||||||
|  |         return <_Guild guild={guild} />; | ||||||
|  |     } else { | ||||||
|  |         return <_User />; | ||||||
|  |     } | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const _User = () => { | ||||||
|  |     const [reminder, setReminder] = useState(defaultReminder()); | ||||||
|  |     const [collapsed, setCollapsed] = useState(false); | ||||||
|  |  | ||||||
|  |     return ( | ||||||
|  |         <ReminderContext.Provider value={[reminder, setReminder]}> | ||||||
|  |             <div class={collapsed ? "reminderContent is-collapsed" : "reminderContent"}> | ||||||
|  |                 <TopBar | ||||||
|  |                     isCreating={true} | ||||||
|  |                     toggleCollapsed={() => { | ||||||
|  |                         setCollapsed(!collapsed); | ||||||
|  |                     }} | ||||||
|  |                 /> | ||||||
|  |                 <div class="columns reminder-settings"> | ||||||
|  |                     <Message /> | ||||||
|  |                     <Settings /> | ||||||
|  |                 </div> | ||||||
|  |                 <CreateButtonRow /> | ||||||
|  |             </div> | ||||||
|  |         </ReminderContext.Provider> | ||||||
|  |     ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const _Guild = ({ guild }) => { | ||||||
|     const [reminder, setReminder] = useState(defaultReminder()); |     const [reminder, setReminder] = useState(defaultReminder()); | ||||||
|     const [collapsed, setCollapsed] = useState(false); |     const [collapsed, setCollapsed] = useState(false); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -2,7 +2,7 @@ import { ChannelSelector } from "./ChannelSelector"; | |||||||
| import { DateTime } from "luxon"; | import { DateTime } from "luxon"; | ||||||
| import { IntervalSelector } from "./IntervalSelector"; | import { IntervalSelector } from "./IntervalSelector"; | ||||||
| import { useQuery } from "react-query"; | import { useQuery } from "react-query"; | ||||||
| import { fetchUserInfo } from "../../api"; | import { fetchGuildInfo, fetchUserInfo } from "../../api"; | ||||||
| import { useReminder } from "./ReminderContext"; | import { useReminder } from "./ReminderContext"; | ||||||
| import { Attachment } from "./Attachment"; | import { Attachment } from "./Attachment"; | ||||||
| import { TTS } from "./TTS"; | import { TTS } from "./TTS"; | ||||||
| @@ -11,11 +11,12 @@ import { useGuild } from "../App/useGuild"; | |||||||
|  |  | ||||||
| export const Settings = () => { | export const Settings = () => { | ||||||
|     const guild = useGuild(); |     const guild = useGuild(); | ||||||
|     const { isSuccess: userFetched, data: userInfo } = useQuery(fetchUserInfo()); |     const { isSuccess: userFetched, data: userInfo } = useQuery({ ...fetchUserInfo() }); | ||||||
|  |     const { isSuccess: guildFetched, data: guildInfo } = useQuery({ ...fetchGuildInfo(guild) }); | ||||||
|  |  | ||||||
|     const [reminder, setReminder] = useReminder(); |     const [reminder, setReminder] = useReminder(); | ||||||
|  |  | ||||||
|     if (!userFetched) { |     if (!userFetched || !guildFetched) { | ||||||
|         return <></>; |         return <></>; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -59,7 +60,13 @@ export const Settings = () => { | |||||||
|  |  | ||||||
|             <div class="collapses split-controls"> |             <div class="collapses split-controls"> | ||||||
|                 <div> |                 <div> | ||||||
|                     <div class={userInfo.patreon ? "patreon-only" : "patreon-only is-locked"}> |                     <div | ||||||
|  |                         class={ | ||||||
|  |                             userInfo.patreon || guildInfo.patreon | ||||||
|  |                                 ? "patreon-only" | ||||||
|  |                                 : "patreon-only is-locked" | ||||||
|  |                         } | ||||||
|  |                     > | ||||||
|                         <div class="patreon-invert foreground"> |                         <div class="patreon-invert foreground"> | ||||||
|                             Intervals available on <a href="https://patreon.com/jellywx">Patreon</a>{" "} |                             Intervals available on <a href="https://patreon.com/jellywx">Patreon</a>{" "} | ||||||
|                             or{" "} |                             or{" "} | ||||||
|   | |||||||
| @@ -16,28 +16,28 @@ export const TimeInput = ({ defaultValue, onInput }) => { | |||||||
|     const ref = useRef(null); |     const ref = useRef(null); | ||||||
|  |  | ||||||
|     const [timezone] = useTimezone(); |     const [timezone] = useTimezone(); | ||||||
|     const [time, setTime] = useState( |     const [localTime, setLocalTime] = useState( | ||||||
|         defaultValue ? DateTime.fromISO(defaultValue, { zone: "UTC" }) : null, |         defaultValue ? DateTime.fromISO(defaultValue, { zone: "UTC" }).setZone(timezone) : null, | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     const updateTime = useCallback( |     const updateTime = useCallback( | ||||||
|         (upd: TimeUpdate) => { |         (upd: TimeUpdate) => { | ||||||
|             if (upd === null) { |             if (upd === null) { | ||||||
|                 setTime(null); |                 setLocalTime(null); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             let newTime = time; |             let newTime = localTime; | ||||||
|             if (newTime === null) { |             if (newTime === null) { | ||||||
|                 newTime = DateTime.now().setZone("UTC"); |                 newTime = DateTime.now().setZone(timezone); | ||||||
|             } |             } | ||||||
|             setTime(newTime.setZone(timezone).set(upd).setZone("UTC")); |             setLocalTime(newTime.set(upd)); | ||||||
|         }, |         }, | ||||||
|         [time, timezone], |         [localTime, timezone], | ||||||
|     ); |     ); | ||||||
|  |  | ||||||
|     useEffect(() => { |     useEffect(() => { | ||||||
|         onInput(time?.setZone("UTC").toFormat("yyyy-LL-dd'T'HH:mm:ss")); |         onInput(localTime?.setZone("UTC").toFormat("yyyy-LL-dd'T'HH:mm:ss")); | ||||||
|     }, [time]); |     }, [localTime]); | ||||||
|  |  | ||||||
|     const flash = useFlash(); |     const flash = useFlash(); | ||||||
|  |  | ||||||
| @@ -51,14 +51,14 @@ export const TimeInput = ({ defaultValue, onInput }) => { | |||||||
|                     let dt = DateTime.fromISO(pasteValue, { zone: timezone }); |                     let dt = DateTime.fromISO(pasteValue, { zone: timezone }); | ||||||
|  |  | ||||||
|                     if (dt.isValid) { |                     if (dt.isValid) { | ||||||
|                         setTime(dt); |                         setLocalTime(dt); | ||||||
|                         return; |                         return; | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|                     dt = DateTime.fromSQL(pasteValue); |                     dt = DateTime.fromSQL(pasteValue); | ||||||
|  |  | ||||||
|                     if (dt.isValid) { |                     if (dt.isValid) { | ||||||
|                         setTime(dt); |                         setLocalTime(dt); | ||||||
|                         return; |                         return; | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
| @@ -83,8 +83,8 @@ export const TimeInput = ({ defaultValue, onInput }) => { | |||||||
|                             maxlength={4} |                             maxlength={4} | ||||||
|                             placeholder="YYYY" |                             placeholder="YYYY" | ||||||
|                             value={ |                             value={ | ||||||
|                                 time |                                 localTime | ||||||
|                                     ? time.setZone(timezone).year.toLocaleString("en-US", { |                                     ? localTime.year.toLocaleString("en-US", { | ||||||
|                                           minimumIntegerDigits: 4, |                                           minimumIntegerDigits: 4, | ||||||
|                                           useGrouping: false, |                                           useGrouping: false, | ||||||
|                                       }) |                                       }) | ||||||
| @@ -114,8 +114,8 @@ export const TimeInput = ({ defaultValue, onInput }) => { | |||||||
|                             maxlength={2} |                             maxlength={2} | ||||||
|                             placeholder="MM" |                             placeholder="MM" | ||||||
|                             value={ |                             value={ | ||||||
|                                 time |                                 localTime | ||||||
|                                     ? time.setZone(timezone).month.toLocaleString("en-US", { |                                     ? localTime.month.toLocaleString("en-US", { | ||||||
|                                           minimumIntegerDigits: 2, |                                           minimumIntegerDigits: 2, | ||||||
|                                       }) |                                       }) | ||||||
|                                     : "" |                                     : "" | ||||||
| @@ -144,10 +144,10 @@ export const TimeInput = ({ defaultValue, onInput }) => { | |||||||
|                             maxlength={2} |                             maxlength={2} | ||||||
|                             placeholder="DD" |                             placeholder="DD" | ||||||
|                             value={ |                             value={ | ||||||
|                                 time |                                 localTime | ||||||
|                                     ? time |                                     ? localTime.day.toLocaleString("en-US", { | ||||||
|                                           .setZone(timezone) |                                           minimumIntegerDigits: 2, | ||||||
|                                           .day.toLocaleString("en-US", { minimumIntegerDigits: 2 }) |                                       }) | ||||||
|                                     : "" |                                     : "" | ||||||
|                             } |                             } | ||||||
|                             onBlur={(ev) => { |                             onBlur={(ev) => { | ||||||
| @@ -173,10 +173,10 @@ export const TimeInput = ({ defaultValue, onInput }) => { | |||||||
|                             maxlength={2} |                             maxlength={2} | ||||||
|                             placeholder="hh" |                             placeholder="hh" | ||||||
|                             value={ |                             value={ | ||||||
|                                 time |                                 localTime | ||||||
|                                     ? time |                                     ? localTime.hour.toLocaleString("en-US", { | ||||||
|                                           .setZone(timezone) |                                           minimumIntegerDigits: 2, | ||||||
|                                           .hour.toLocaleString("en-US", { minimumIntegerDigits: 2 }) |                                       }) | ||||||
|                                     : "" |                                     : "" | ||||||
|                             } |                             } | ||||||
|                             onBlur={(ev) => { |                             onBlur={(ev) => { | ||||||
| @@ -203,8 +203,8 @@ export const TimeInput = ({ defaultValue, onInput }) => { | |||||||
|                             maxlength={2} |                             maxlength={2} | ||||||
|                             placeholder="mm" |                             placeholder="mm" | ||||||
|                             value={ |                             value={ | ||||||
|                                 time |                                 localTime | ||||||
|                                     ? time.setZone(timezone).minute.toLocaleString("en-US", { |                                     ? localTime.minute.toLocaleString("en-US", { | ||||||
|                                           minimumIntegerDigits: 2, |                                           minimumIntegerDigits: 2, | ||||||
|                                       }) |                                       }) | ||||||
|                                     : "" |                                     : "" | ||||||
| @@ -233,8 +233,8 @@ export const TimeInput = ({ defaultValue, onInput }) => { | |||||||
|                             maxlength={2} |                             maxlength={2} | ||||||
|                             placeholder="ss" |                             placeholder="ss" | ||||||
|                             value={ |                             value={ | ||||||
|                                 time |                                 localTime | ||||||
|                                     ? time.setZone(timezone).second.toLocaleString("en-US", { |                                     ? localTime.second.toLocaleString("en-US", { | ||||||
|                                           minimumIntegerDigits: 2, |                                           minimumIntegerDigits: 2, | ||||||
|                                       }) |                                       }) | ||||||
|                                     : "" |                                     : "" | ||||||
| @@ -276,15 +276,17 @@ export const TimeInput = ({ defaultValue, onInput }) => { | |||||||
|                 type="datetime-local" |                 type="datetime-local" | ||||||
|                 step="1" |                 step="1" | ||||||
|                 value={ |                 value={ | ||||||
|                     time |                     localTime | ||||||
|                         ? time.toFormat("yyyy-LL-dd'T'HH:mm:ss") |                         ? localTime.toFormat("yyyy-LL-dd'T'HH:mm:ss") | ||||||
|                         : DateTime.now().toFormat("yyyy-LL-dd'T'HH:mm:ss") |                         : DateTime.now().setZone(timezone).toFormat("yyyy-LL-dd'T'HH:mm:ss") | ||||||
|                 } |                 } | ||||||
|                 ref={ref} |                 ref={ref} | ||||||
|                 onInput={(ev) => { |                 onInput={(ev) => { | ||||||
|                     ev.currentTarget.value === "" |                     ev.currentTarget.value === "" | ||||||
|                         ? updateTime(null) |                         ? updateTime(null) | ||||||
|                         : setTime(DateTime.fromISO(ev.currentTarget.value, { zone: "UTC" })); |                         : setLocalTime( | ||||||
|  |                               DateTime.fromISO(ev.currentTarget.value, { zone: timezone }), | ||||||
|  |                           ); | ||||||
|                 }} |                 }} | ||||||
|             ></input> |             ></input> | ||||||
|         </> |         </> | ||||||
|   | |||||||
| @@ -26,3 +26,11 @@ | |||||||
| textarea.autoresize { | textarea.autoresize { | ||||||
|     resize: vertical !important; |     resize: vertical !important; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | div.reminderContent { | ||||||
|  |     margin-top: 10px; | ||||||
|  |     margin-bottom: 10px; | ||||||
|  |     padding: 14px; | ||||||
|  |     background-color: #f5f5f5; | ||||||
|  |     border-radius: 8px; | ||||||
|  | } | ||||||
|   | |||||||
| @@ -8,6 +8,7 @@ import { fetchUserGuilds, fetchUserInfo, GuildInfo } from "../../api"; | |||||||
| import { TimezonePicker } from "../TimezonePicker"; | import { TimezonePicker } from "../TimezonePicker"; | ||||||
| import "./styles.scss"; | import "./styles.scss"; | ||||||
| import { Link, useLocation } from "wouter"; | import { Link, useLocation } from "wouter"; | ||||||
|  | import { UserPreferences } from "../UserPreferences"; | ||||||
|  |  | ||||||
| type ContentProps = { | type ContentProps = { | ||||||
|     guilds: GuildInfo[]; |     guilds: GuildInfo[]; | ||||||
| @@ -24,7 +25,7 @@ const SidebarContent = ({ guilds }: ContentProps) => { | |||||||
|                 <Brand /> |                 <Brand /> | ||||||
|             </a> |             </a> | ||||||
|             <Wave /> |             <Wave /> | ||||||
|             <aside class="menu"> |             <aside class="menu theme-dark"> | ||||||
|                 <ul class="menu-list"> |                 <ul class="menu-list"> | ||||||
|                     <li> |                     <li> | ||||||
|                         <Link |                         <Link | ||||||
| @@ -44,7 +45,6 @@ const SidebarContent = ({ guilds }: ContentProps) => { | |||||||
|                     style={{ |                     style={{ | ||||||
|                         position: "sticky", |                         position: "sticky", | ||||||
|                         bottom: "0px", |                         bottom: "0px", | ||||||
|                         backgroundColor: "rgb(54, 54, 54)", |  | ||||||
|                     }} |                     }} | ||||||
|                 > |                 > | ||||||
|                     <p class="menu-label">Options</p> |                     <p class="menu-label">Options</p> | ||||||
| @@ -52,6 +52,7 @@ const SidebarContent = ({ guilds }: ContentProps) => { | |||||||
|                         <li> |                         <li> | ||||||
|                             <div id="bottom-sidebar"></div> |                             <div id="bottom-sidebar"></div> | ||||||
|                             <TimezonePicker /> |                             <TimezonePicker /> | ||||||
|  |                             <UserPreferences /> | ||||||
|                             <a href="/login/discord/logout"> |                             <a href="/login/discord/logout"> | ||||||
|                                 <span class="icon"> |                                 <span class="icon"> | ||||||
|                                     <i class="fas fa-sign-out"></i> |                                     <i class="fas fa-sign-out"></i> | ||||||
|   | |||||||
| @@ -39,7 +39,7 @@ aside.menu { | |||||||
| } | } | ||||||
|  |  | ||||||
| .dashboard-sidebar { | .dashboard-sidebar { | ||||||
|     display: flex; |     display: flex !important; | ||||||
|     flex-direction: column; |     flex-direction: column; | ||||||
|     background-color: #363636; |     background-color: #363636; | ||||||
|     width: 230px !important; |     width: 230px !important; | ||||||
| @@ -49,3 +49,19 @@ aside.menu { | |||||||
| .aside-footer { | .aside-footer { | ||||||
|     justify-self: end; |     justify-self: end; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .menu a { | ||||||
|  |     color: #fff !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .menu a:hover:not(.is-active) { | ||||||
|  |     color: #424242 !important; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .menu .menu-label { | ||||||
|  |     color: #bbb; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .menu { | ||||||
|  |     padding-left: 4px; | ||||||
|  | } | ||||||
|   | |||||||
| @@ -53,10 +53,10 @@ export const TimezonePicker = () => { | |||||||
|  |  | ||||||
| const TimezoneModal = ({ setModalOpen }) => { | const TimezoneModal = ({ setModalOpen }) => { | ||||||
|     const browserTimezone = DateTime.now().zoneName; |     const browserTimezone = DateTime.now().zoneName; | ||||||
|     const [selectedZone, setSelectedZone] = useTimezone(); |     const [selectedZone, setTimezone] = useTimezone(); | ||||||
|  |  | ||||||
|     const queryClient = useQueryClient(); |     const queryClient = useQueryClient(); | ||||||
|     const { isLoading, isError, data } = useQuery(fetchUserInfo()); |     const { isLoading, isError, data } = useQuery({ ...fetchUserInfo() }); | ||||||
|     const userInfoMutation = useMutation({ |     const userInfoMutation = useMutation({ | ||||||
|         ...patchUserInfo(), |         ...patchUserInfo(), | ||||||
|         onSuccess: () => { |         onSuccess: () => { | ||||||
| @@ -79,7 +79,7 @@ const TimezoneModal = ({ setModalOpen }) => { | |||||||
|                         {isLoading ? ( |                         {isLoading ? ( | ||||||
|                             <i className="fas fa-cog fa-spin"></i> |                             <i className="fas fa-cog fa-spin"></i> | ||||||
|                         ) : ( |                         ) : ( | ||||||
|                             <TimezoneDisplay timezone={data.timezone || "UTC"}></TimezoneDisplay> |                             <TimezoneDisplay timezone={data.preferences.timezone || "UTC"} /> | ||||||
|                         )} |                         )} | ||||||
|                     </> |                     </> | ||||||
|                 )} |                 )} | ||||||
| @@ -87,34 +87,16 @@ const TimezoneModal = ({ setModalOpen }) => { | |||||||
|             <br></br> |             <br></br> | ||||||
|             <div class="has-text-centered"> |             <div class="has-text-centered"> | ||||||
|                 <button |                 <button | ||||||
|                     class="button is-success" |                     class="button is-success is-outlined" | ||||||
|                     style={{ |                     id="update-bot-timezone" | ||||||
|                         margin: "2px", |  | ||||||
|                     }} |  | ||||||
|                     id="set-browser-timezone" |  | ||||||
|                     onClick={() => { |  | ||||||
|                         setSelectedZone(browserTimezone); |  | ||||||
|                     }} |  | ||||||
|                 > |  | ||||||
|                     <span>Use Browser Timezone</span>{" "} |  | ||||||
|                     <span class="icon"> |  | ||||||
|                         <i class="fab fa-firefox-browser"></i> |  | ||||||
|                     </span> |  | ||||||
|                 </button> |  | ||||||
|                 <button |  | ||||||
|                     class="button is-success" |  | ||||||
|                     id="set-bot-timezone" |  | ||||||
|                     style={{ |                     style={{ | ||||||
|                         margin: "2px", |                         margin: "2px", | ||||||
|                     }} |                     }} | ||||||
|                     onClick={() => { |                     onClick={() => { | ||||||
|                         setSelectedZone(data.timezone); |                         userInfoMutation.mutate({ timezone: browserTimezone }); | ||||||
|                     }} |                     }} | ||||||
|                 > |                 > | ||||||
|                     <span>Use Bot Timezone</span>{" "} |                     Set bot timezone | ||||||
|                     <span class="icon"> |  | ||||||
|                         <i class="fab fa-discord"></i> |  | ||||||
|                     </span> |  | ||||||
|                 </button> |                 </button> | ||||||
|                 <button |                 <button | ||||||
|                     class="button is-success is-outlined" |                     class="button is-success is-outlined" | ||||||
| @@ -123,10 +105,17 @@ const TimezoneModal = ({ setModalOpen }) => { | |||||||
|                         margin: "2px", |                         margin: "2px", | ||||||
|                     }} |                     }} | ||||||
|                     onClick={() => { |                     onClick={() => { | ||||||
|                         userInfoMutation.mutate(browserTimezone); |                         userInfoMutation.mutate({ | ||||||
|  |                             use_browser_timezone: !data.preferences.use_browser_timezone, | ||||||
|  |                         }); | ||||||
|  |                         setTimezone( | ||||||
|  |                             data.preferences.use_browser_timezone | ||||||
|  |                                 ? data.preferences.timezone | ||||||
|  |                                 : DateTime.now().zoneName, | ||||||
|  |                         ); | ||||||
|                     }} |                     }} | ||||||
|                 > |                 > | ||||||
|                     Set Bot Timezone |                     Use {data.preferences.use_browser_timezone ? "bot" : "browser"} timezone | ||||||
|                 </button> |                 </button> | ||||||
|             </div> |             </div> | ||||||
|         </Modal> |         </Modal> | ||||||
|   | |||||||
| @@ -11,12 +11,7 @@ enum Sort { | |||||||
| } | } | ||||||
|  |  | ||||||
| export const UserReminders = () => { | export const UserReminders = () => { | ||||||
|     const { |     const { isSuccess, isFetching, isFetched, data: reminders } = useQuery(fetchUserReminders()); | ||||||
|         isSuccess, |  | ||||||
|         isFetching, |  | ||||||
|         isFetched, |  | ||||||
|         data: guildReminders, |  | ||||||
|     } = useQuery(fetchUserReminders()); |  | ||||||
|  |  | ||||||
|     const [collapsed, setCollapsed] = useState(false); |     const [collapsed, setCollapsed] = useState(false); | ||||||
|     const [sort, setSort] = useState(Sort.Time); |     const [sort, setSort] = useState(Sort.Time); | ||||||
| @@ -85,7 +80,7 @@ export const UserReminders = () => { | |||||||
|  |  | ||||||
|                 <div id={"guildReminders"} className={isFetching ? "loading" : ""}> |                 <div id={"guildReminders"} className={isFetching ? "loading" : ""}> | ||||||
|                     {isSuccess && |                     {isSuccess && | ||||||
|                         guildReminders |                         reminders | ||||||
|                             .sort((r1, r2) => { |                             .sort((r1, r2) => { | ||||||
|                                 if (sort === Sort.Time) { |                                 if (sort === Sort.Time) { | ||||||
|                                     return r1.utc_time > r2.utc_time ? 1 : -1; |                                     return r1.utc_time > r2.utc_time ? 1 : -1; | ||||||
|   | |||||||
							
								
								
									
										128
									
								
								reminder-dashboard/src/components/UserPreferences/index.tsx
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								reminder-dashboard/src/components/UserPreferences/index.tsx
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,128 @@ | |||||||
|  | import { Modal } from "../Modal"; | ||||||
|  | import { fetchUserInfo, patchUserInfo, UpdateUserInfo } from "../../api"; | ||||||
|  | import { useMutation, useQuery, useQueryClient } from "react-query"; | ||||||
|  | import { useRef, useState } from "preact/hooks"; | ||||||
|  | import { ICON_FLASH_TIME } from "../../consts"; | ||||||
|  | import { useFlash } from "../App/FlashContext"; | ||||||
|  |  | ||||||
|  | export const UserPreferences = () => { | ||||||
|  |     const [modalOpen, setModalOpen] = useState(false); | ||||||
|  |  | ||||||
|  |     return ( | ||||||
|  |         <> | ||||||
|  |             <a | ||||||
|  |                 role={"button"} | ||||||
|  |                 onClick={() => { | ||||||
|  |                     setModalOpen(true); | ||||||
|  |                 }} | ||||||
|  |             > | ||||||
|  |                 <span class="icon"> | ||||||
|  |                     <i class="fas fa-cog"></i> | ||||||
|  |                 </span>{" "} | ||||||
|  |                 Preferences | ||||||
|  |             </a> | ||||||
|  |             {modalOpen && <PreferencesModal setModalOpen={setModalOpen} />} | ||||||
|  |         </> | ||||||
|  |     ); | ||||||
|  | }; | ||||||
|  |  | ||||||
|  | const PreferencesModal = ({ setModalOpen }) => { | ||||||
|  |     const flash = useFlash(); | ||||||
|  |     const queryClient = useQueryClient(); | ||||||
|  |     const { isLoading, isSuccess, isError, data } = useQuery({ ...fetchUserInfo() }); | ||||||
|  |     const userInfoMutation = useMutation({ | ||||||
|  |         ...patchUserInfo(), | ||||||
|  |         onError: (error) => { | ||||||
|  |             flash({ | ||||||
|  |                 message: `An error occurred: ${error}`, | ||||||
|  |                 type: "error", | ||||||
|  |             }); | ||||||
|  |         }, | ||||||
|  |         onSuccess: () => { | ||||||
|  |             if (iconFlashTimeout.current !== null) { | ||||||
|  |                 clearTimeout(iconFlashTimeout.current); | ||||||
|  |             } | ||||||
|  |             setRecentlySaved(true); | ||||||
|  |             iconFlashTimeout.current = setTimeout(() => { | ||||||
|  |                 setRecentlySaved(false); | ||||||
|  |             }, ICON_FLASH_TIME); | ||||||
|  |  | ||||||
|  |             queryClient.invalidateQueries(["USER_INFO"]).then(() => setUpdatedSettings({})); | ||||||
|  |         }, | ||||||
|  |     }); | ||||||
|  |  | ||||||
|  |     const resetInputsRef = useRef(null as HTMLInputElement | null); | ||||||
|  |  | ||||||
|  |     const [recentlySaved, setRecentlySaved] = useState(false); | ||||||
|  |     const iconFlashTimeout = useRef(0); | ||||||
|  |  | ||||||
|  |     const [updatedSettings, setUpdatedSettings] = useState({} as UpdateUserInfo); | ||||||
|  |  | ||||||
|  |     if (isError) { | ||||||
|  |         return ( | ||||||
|  |             <Modal setModalOpen={setModalOpen} title={"Preferences"}> | ||||||
|  |                 <span>An error occurred loading user preferences</span> | ||||||
|  |             </Modal> | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return ( | ||||||
|  |         <Modal title={"Preferences"} setModalOpen={setModalOpen}> | ||||||
|  |             <div style={{ display: "flex", flexDirection: "row", alignContent: "center" }}> | ||||||
|  |                 <label> | ||||||
|  |                     <div class={"is-inline-block"}> | ||||||
|  |                         {isLoading && <i class={"fa fa-spinner"} />} | ||||||
|  |                         {isSuccess && ( | ||||||
|  |                             <input | ||||||
|  |                                 type={"checkbox"} | ||||||
|  |                                 checked={ | ||||||
|  |                                     updatedSettings.reset_inputs_on_create === undefined | ||||||
|  |                                         ? data.preferences.reset_inputs_on_create | ||||||
|  |                                         : updatedSettings.reset_inputs_on_create | ||||||
|  |                                 } | ||||||
|  |                                 ref={resetInputsRef} | ||||||
|  |                                 onInput={() => | ||||||
|  |                                     setUpdatedSettings((s) => ({ | ||||||
|  |                                         ...s, | ||||||
|  |                                         reset_inputs_on_create: resetInputsRef.current.checked, | ||||||
|  |                                     })) | ||||||
|  |                                 } | ||||||
|  |                             /> | ||||||
|  |                         )} | ||||||
|  |                     </div> | ||||||
|  |                     <div class={"is-inline-block"} style={{ marginLeft: "6px" }}> | ||||||
|  |                         Reset reminder inputs when creating a reminder | ||||||
|  |                     </div> | ||||||
|  |                 </label> | ||||||
|  |             </div> | ||||||
|  |             <br></br> | ||||||
|  |             <div class="has-text-centered"> | ||||||
|  |                 <button | ||||||
|  |                     class="button is-success is-outlined" | ||||||
|  |                     style={{ | ||||||
|  |                         margin: "2px", | ||||||
|  |                     }} | ||||||
|  |                     onClick={() => { | ||||||
|  |                         userInfoMutation.mutate({ ...updatedSettings }); | ||||||
|  |                     }} | ||||||
|  |                     disabled={userInfoMutation.isLoading} | ||||||
|  |                 > | ||||||
|  |                     <span>Save</span> | ||||||
|  |                     {userInfoMutation.isLoading ? ( | ||||||
|  |                         <span class="icon"> | ||||||
|  |                             <i class="fas fa-spin fa-cog"></i> | ||||||
|  |                         </span> | ||||||
|  |                     ) : recentlySaved ? ( | ||||||
|  |                         <span class="icon"> | ||||||
|  |                             <i class="fas fa-check"></i> | ||||||
|  |                         </span> | ||||||
|  |                     ) : ( | ||||||
|  |                         <span class="icon"> | ||||||
|  |                             <i class="fas fa-save"></i> | ||||||
|  |                         </span> | ||||||
|  |                     )} | ||||||
|  |                 </button> | ||||||
|  |             </div> | ||||||
|  |         </Modal> | ||||||
|  |     ); | ||||||
|  | }; | ||||||
| @@ -1,3 +1,5 @@ | |||||||
|  | @import "node_modules/bulma/bulma"; | ||||||
|  |  | ||||||
| /* override styles for when the div is collapsed */ | /* override styles for when the div is collapsed */ | ||||||
| div.reminderContent.is-collapsed .column.discord-frame { | div.reminderContent.is-collapsed .column.discord-frame { | ||||||
|     display: none; |     display: none; | ||||||
| @@ -47,3 +49,11 @@ div.reminderContent.is-collapsed .hide-box { | |||||||
| div.reminderContent.is-collapsed .hide-box i { | div.reminderContent.is-collapsed .hide-box i { | ||||||
|     transform: rotate(90deg); |     transform: rotate(90deg); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | .button.is-success:not(.is-outlined) { | ||||||
|  |     color: white; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | .button.is-outlined.is-success:not(.is-focused, :hover, :focus) { | ||||||
|  |     background-color: white; | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										13
									
								
								reminder-dashboard/src/vars.scss
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								reminder-dashboard/src/vars.scss
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,13 @@ | |||||||
|  | $primary-background-light: #ffffff; | ||||||
|  | $secondary-background-light: #f5f5f5; | ||||||
|  | $contrast-background-light: #363636; | ||||||
|  | $primary-text-light: #4a4a4a; | ||||||
|  | $secondary-text-light: #4a4a4a; | ||||||
|  | $contrast-text-light: #ffffff; | ||||||
|  |  | ||||||
|  | $primary-background-dark: #363636; | ||||||
|  | $secondary-background-dark: #424242; | ||||||
|  | $contrast-background-dark: #242424; | ||||||
|  | $primary-text-dark: #ffffff; | ||||||
|  | $secondary-text-dark: #ffffff; | ||||||
|  | $contrast-text-dark: #ffffff; | ||||||
| @@ -22,9 +22,9 @@ impl Recordable for Options { | |||||||
|                 CreateEmbed::new() |                 CreateEmbed::new() | ||||||
|                     .title("Confirmations ephemeral") |                     .title("Confirmations ephemeral") | ||||||
|                     .description(concat!( |                     .description(concat!( | ||||||
|                     "Reminder confirmations will be sent privately, and removed when your client", |                         "Reminder and todo confirmations will be sent privately, and removed when ", | ||||||
|                     " restarts." |                         "your client restarts." | ||||||
|                 )) |                     )) | ||||||
|                     .color(*THEME_COLOR), |                     .color(*THEME_COLOR), | ||||||
|             ), |             ), | ||||||
|         ) |         ) | ||||||
|   | |||||||
| @@ -22,8 +22,8 @@ impl Recordable for Options { | |||||||
|                 CreateEmbed::new() |                 CreateEmbed::new() | ||||||
|                     .title("Confirmations public") |                     .title("Confirmations public") | ||||||
|                     .description(concat!( |                     .description(concat!( | ||||||
|                         "Reminder confirmations will be sent as regular messages, and won't be ", |                         "Reminder and todo confirmations will be sent as regular messages, and", | ||||||
|                         "removed automatically." |                         " won't be removed automatically." | ||||||
|                     )) |                     )) | ||||||
|                     .color(*THEME_COLOR), |                     .color(*THEME_COLOR), | ||||||
|             ), |             ), | ||||||
|   | |||||||
| @@ -1,3 +1,4 @@ | |||||||
|  | use poise::CreateReply; | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
| @@ -33,7 +34,13 @@ impl Recordable for Options { | |||||||
|         .await |         .await | ||||||
|         .unwrap(); |         .unwrap(); | ||||||
|  |  | ||||||
|         ctx.say("Item added to todo list").await?; |         let ephemeral = ctx | ||||||
|  |             .guild_data() | ||||||
|  |             .await | ||||||
|  |             .map_or(false, |gr| gr.map_or(false, |g| g.ephemeral_confirmations)); | ||||||
|  |  | ||||||
|  |         ctx.send(CreateReply::default().content("Item added to todo list").ephemeral(ephemeral)) | ||||||
|  |             .await?; | ||||||
|  |  | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,6 +1,8 @@ | |||||||
|  | use poise::CreateReply; | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
|  |     models::CtxData, | ||||||
|     utils::{Extract, Recordable}, |     utils::{Extract, Recordable}, | ||||||
|     Context, Error, |     Context, Error, | ||||||
| }; | }; | ||||||
| @@ -26,7 +28,13 @@ impl Recordable for Options { | |||||||
|         .await |         .await | ||||||
|         .unwrap(); |         .unwrap(); | ||||||
|  |  | ||||||
|         ctx.say("Item added to todo list").await?; |         let ephemeral = ctx | ||||||
|  |             .guild_data() | ||||||
|  |             .await | ||||||
|  |             .map_or(false, |gr| gr.map_or(false, |g| g.ephemeral_confirmations)); | ||||||
|  |  | ||||||
|  |         ctx.send(CreateReply::default().content("Item added to todo list").ephemeral(ephemeral)) | ||||||
|  |             .await?; | ||||||
|  |  | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,6 +1,8 @@ | |||||||
|  | use poise::CreateReply; | ||||||
| use serde::{Deserialize, Serialize}; | use serde::{Deserialize, Serialize}; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
|  |     models::CtxData, | ||||||
|     utils::{Extract, Recordable}, |     utils::{Extract, Recordable}, | ||||||
|     Context, Error, |     Context, Error, | ||||||
| }; | }; | ||||||
| @@ -15,10 +17,7 @@ impl Recordable for Options { | |||||||
|         sqlx::query!( |         sqlx::query!( | ||||||
|             " |             " | ||||||
|             INSERT INTO todos (user_id, value) |             INSERT INTO todos (user_id, value) | ||||||
|             VALUES ( |             VALUES (?, ?) | ||||||
|                 (SELECT id FROM users WHERE user = ?), |  | ||||||
|                 ? |  | ||||||
|             ) |  | ||||||
|             ", |             ", | ||||||
|             ctx.author().id.get(), |             ctx.author().id.get(), | ||||||
|             self.task |             self.task | ||||||
| @@ -27,7 +26,13 @@ impl Recordable for Options { | |||||||
|         .await |         .await | ||||||
|         .unwrap(); |         .unwrap(); | ||||||
|  |  | ||||||
|         ctx.say("Item added to todo list").await?; |         let ephemeral = ctx | ||||||
|  |             .guild_data() | ||||||
|  |             .await | ||||||
|  |             .map_or(false, |gr| gr.map_or(false, |g| g.ephemeral_confirmations)); | ||||||
|  |  | ||||||
|  |         ctx.send(CreateReply::default().content("Item added to todo list").ephemeral(ephemeral)) | ||||||
|  |             .await?; | ||||||
|  |  | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -14,8 +14,7 @@ impl Recordable for Options { | |||||||
|         let values = sqlx::query!( |         let values = sqlx::query!( | ||||||
|             " |             " | ||||||
|             SELECT todos.id, value FROM todos |             SELECT todos.id, value FROM todos | ||||||
|             INNER JOIN users ON todos.user_id = users.id |             WHERE user_id = ? | ||||||
|             WHERE users.user = ? |  | ||||||
|             ", |             ", | ||||||
|             ctx.author().id.get(), |             ctx.author().id.get(), | ||||||
|         ) |         ) | ||||||
|   | |||||||
| @@ -191,8 +191,7 @@ impl ComponentDataModel { | |||||||
|                         sqlx::query!( |                         sqlx::query!( | ||||||
|                             " |                             " | ||||||
|                             SELECT todos.id, value FROM todos |                             SELECT todos.id, value FROM todos | ||||||
|                             INNER JOIN users ON todos.user_id = users.id |                             WHERE user_id = ? | ||||||
|                             WHERE users.user = ? |  | ||||||
|                             ", |                             ", | ||||||
|                             uid, |                             uid, | ||||||
|                         ) |                         ) | ||||||
| @@ -282,21 +281,51 @@ impl ComponentDataModel { | |||||||
|                         .await |                         .await | ||||||
|                         .unwrap(); |                         .unwrap(); | ||||||
|  |  | ||||||
|                         let values = sqlx::query!( |                         let values = if let Some(uid) = selector.user_id { | ||||||
|                             // fucking braindead mysql use <=> instead of = for null comparison |                             sqlx::query!( | ||||||
|                             " |                                 " | ||||||
|                             SELECT id, value FROM todos WHERE user_id <=> ? AND channel_id <=> ? AND guild_id <=> ? |                                 SELECT todos.id, value FROM todos | ||||||
|  |                                 WHERE user_id = ? | ||||||
|  |                                 ", | ||||||
|  |                                 uid, | ||||||
|  |                             ) | ||||||
|  |                             .fetch_all(&data.database) | ||||||
|  |                             .await | ||||||
|  |                             .unwrap() | ||||||
|  |                             .iter() | ||||||
|  |                             .map(|row| (row.id as usize, row.value.clone())) | ||||||
|  |                             .collect::<Vec<(usize, String)>>() | ||||||
|  |                         } else if let Some(cid) = selector.channel_id { | ||||||
|  |                             sqlx::query!( | ||||||
|  |                                 " | ||||||
|  |                             SELECT todos.id, value FROM todos | ||||||
|  |                             INNER JOIN channels ON todos.channel_id = channels.id | ||||||
|  |                             WHERE channels.channel = ? | ||||||
|                             ", |                             ", | ||||||
|                             selector.user_id, |                                 cid, | ||||||
|                             selector.channel_id, |                             ) | ||||||
|                             selector.guild_id, |                             .fetch_all(&data.database) | ||||||
|                         ) |                             .await | ||||||
|                         .fetch_all(&data.database) |                             .unwrap() | ||||||
|                         .await |                             .iter() | ||||||
|                         .unwrap() |                             .map(|row| (row.id as usize, row.value.clone())) | ||||||
|                         .iter() |                             .collect::<Vec<(usize, String)>>() | ||||||
|                         .map(|row| (row.id as usize, row.value.clone())) |                         } else { | ||||||
|                         .collect::<Vec<(usize, String)>>(); |                             sqlx::query!( | ||||||
|  |                                 " | ||||||
|  |                             SELECT todos.id, value FROM todos | ||||||
|  |                             INNER JOIN guilds ON todos.guild_id = guilds.id | ||||||
|  |                             WHERE guilds.guild = ? | ||||||
|  |                             ", | ||||||
|  |                                 selector.guild_id, | ||||||
|  |                             ) | ||||||
|  |                             .fetch_all(&data.database) | ||||||
|  |                             .await | ||||||
|  |                             .unwrap() | ||||||
|  |                             .iter() | ||||||
|  |                             .map(|row| (row.id as usize, row.value.clone())) | ||||||
|  |                             .collect::<Vec<(usize, String)>>() | ||||||
|  |                         }; | ||||||
|  |  | ||||||
|                         let resp = show_todo_page( |                         let resp = show_todo_page( | ||||||
|                             &values, |                             &values, | ||||||
|   | |||||||
| @@ -1,6 +1,6 @@ | |||||||
| use poise::{ | use poise::{ | ||||||
|     serenity_prelude as serenity, |     serenity_prelude as serenity, | ||||||
|     serenity_prelude::{ActivityData, CreateEmbed, CreateMessage, FullEvent}, |     serenity_prelude::{CreateEmbed, CreateMessage, FullEvent}, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| use crate::{ | use crate::{ | ||||||
| @@ -13,9 +13,6 @@ pub async fn listener( | |||||||
|     data: &Data, |     data: &Data, | ||||||
| ) -> Result<(), Error> { | ) -> Result<(), Error> { | ||||||
|     match event { |     match event { | ||||||
|         FullEvent::Ready { .. } => { |  | ||||||
|             ctx.set_activity(Some(ActivityData::watching("for /remind"))); |  | ||||||
|         } |  | ||||||
|         FullEvent::ChannelDelete { channel, .. } => { |         FullEvent::ChannelDelete { channel, .. } => { | ||||||
|             sqlx::query!("DELETE FROM channels WHERE channel = ?", channel.id.get()) |             sqlx::query!("DELETE FROM channels WHERE channel = ?", channel.id.get()) | ||||||
|                 .execute(&data.database) |                 .execute(&data.database) | ||||||
|   | |||||||
							
								
								
									
										60
									
								
								src/hooks.rs
									
									
									
									
									
								
							
							
						
						
									
										60
									
								
								src/hooks.rs
									
									
									
									
									
								
							| @@ -1,4 +1,4 @@ | |||||||
| use poise::{serenity_prelude::model::channel::Channel, CommandInteractionType, CreateReply}; | use poise::{CommandInteractionType, CreateReply}; | ||||||
|  |  | ||||||
| use crate::{consts::MACRO_MAX_COMMANDS, models::command_macro::RecordedCommand, Context, Error}; | use crate::{consts::MACRO_MAX_COMMANDS, models::command_macro::RecordedCommand, Context, Error}; | ||||||
|  |  | ||||||
| @@ -58,6 +58,10 @@ async fn macro_check(ctx: Context<'_>) -> bool { | |||||||
|  |  | ||||||
| async fn check_self_permissions(ctx: Context<'_>) -> bool { | async fn check_self_permissions(ctx: Context<'_>) -> bool { | ||||||
|     let user_id = ctx.serenity_context().cache.current_user().id; |     let user_id = ctx.serenity_context().cache.current_user().id; | ||||||
|  |     let app_permissions = match ctx { | ||||||
|  |         Context::Application(app_ctx) => app_ctx.interaction.app_permissions, | ||||||
|  |         _ => None, | ||||||
|  |     }; | ||||||
|  |  | ||||||
|     match ctx.guild().map(|g| g.to_owned()) { |     match ctx.guild().map(|g| g.to_owned()) { | ||||||
|         Some(guild) => { |         Some(guild) => { | ||||||
| @@ -66,42 +70,34 @@ async fn check_self_permissions(ctx: Context<'_>) -> bool { | |||||||
|                 .await |                 .await | ||||||
|                 .map_or(false, |m| m.permissions(&ctx).map_or(false, |p| p.manage_webhooks())); |                 .map_or(false, |m| m.permissions(&ctx).map_or(false, |p| p.manage_webhooks())); | ||||||
|  |  | ||||||
|             let (view_channel, send_messages, embed_links) = ctx |             if let Some(permissions) = app_permissions { | ||||||
|                 .channel_id() |                 return if permissions.send_messages() | ||||||
|                 .to_channel(&ctx) |                     && permissions.embed_links() | ||||||
|                 .await |                     && manage_webhooks | ||||||
|                 .ok() |                 { | ||||||
|                 .and_then(|c| { |                     true | ||||||
|                     if let Channel::Guild(channel) = c { |                 } else { | ||||||
|                         let perms = channel.permissions_for_user(&ctx, user_id).ok()?; |                     let _ = ctx | ||||||
|  |                         .send(CreateReply::default().content(format!( | ||||||
|  |                             "The bot appears to be missing some permissions: | ||||||
|  |  | ||||||
|                         Some((perms.view_channel(), perms.send_messages(), perms.embed_links())) |  | ||||||
|                     } else { |  | ||||||
|                         None |  | ||||||
|                     } |  | ||||||
|                 }) |  | ||||||
|                 .unwrap_or((false, false, false)); |  | ||||||
|  |  | ||||||
|             if manage_webhooks && send_messages && embed_links { |  | ||||||
|                 true |  | ||||||
|             } else { |  | ||||||
|                 let _ = ctx |  | ||||||
|                     .send(CreateReply::default().content(format!( |  | ||||||
|                         "Please ensure the bot has the correct permissions: |  | ||||||
|  |  | ||||||
| {}     **View Channel** |  | ||||||
| {}     **Send Message** | {}     **Send Message** | ||||||
| {}     **Embed Links** | {}     **Embed Links** | ||||||
| {}     **Manage Webhooks**", | {}     **Manage Webhooks** | ||||||
|                         if view_channel { "✅" } else { "❌" }, |  | ||||||
|                         if send_messages { "✅" } else { "❌" }, |  | ||||||
|                         if embed_links { "✅" } else { "❌" }, |  | ||||||
|                         if manage_webhooks { "✅" } else { "❌" }, |  | ||||||
|                     ))) |  | ||||||
|                     .await; |  | ||||||
|  |  | ||||||
|                 false | Please check the bot's roles, and any channel overrides. Alternatively, giving the bot | ||||||
|  | \"Administrator\" will bypass permission checks", | ||||||
|  |                             if permissions.send_messages() { "✅" } else { "❌" }, | ||||||
|  |                             if permissions.embed_links() { "✅" } else { "❌" }, | ||||||
|  |                             if manage_webhooks { "✅" } else { "❌" }, | ||||||
|  |                         ))) | ||||||
|  |                         .await; | ||||||
|  |  | ||||||
|  |                     false | ||||||
|  |                 }; | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  |             manage_webhooks | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         None => { |         None => { | ||||||
|   | |||||||
| @@ -39,6 +39,7 @@ use poise::serenity_prelude::{ | |||||||
|     }, |     }, | ||||||
|     ClientBuilder, |     ClientBuilder, | ||||||
| }; | }; | ||||||
|  | use serenity::all::ActivityData; | ||||||
| use sqlx::{MySql, Pool}; | use sqlx::{MySql, Pool}; | ||||||
| use tokio::sync::{broadcast, broadcast::Sender, RwLock}; | use tokio::sync::{broadcast, broadcast::Sender, RwLock}; | ||||||
|  |  | ||||||
| @@ -212,7 +213,6 @@ async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | |||||||
|  |  | ||||||
|     // Start metrics |     // Start metrics | ||||||
|     init_metrics(); |     init_metrics(); | ||||||
|     tokio::spawn(async { metrics::serve().await }); |  | ||||||
|  |  | ||||||
|     let database = |     let database = | ||||||
|         Pool::connect(&env::var("DATABASE_URL").expect("No database URL provided")).await.unwrap(); |         Pool::connect(&env::var("DATABASE_URL").expect("No database URL provided")).await.unwrap(); | ||||||
| @@ -284,8 +284,10 @@ async fn _main(tx: Sender<()>) -> Result<(), Box<dyn StdError + Send + Sync>> { | |||||||
|         .options(options) |         .options(options) | ||||||
|         .build(); |         .build(); | ||||||
|  |  | ||||||
|     let mut client = |     let mut client = ClientBuilder::new(&discord_token, GatewayIntents::GUILDS) | ||||||
|         ClientBuilder::new(&discord_token, GatewayIntents::GUILDS).framework(framework).await?; |         .framework(framework) | ||||||
|  |         .activity(ActivityData::watching("for /remind")) | ||||||
|  |         .await?; | ||||||
|  |  | ||||||
|     client.start_autosharded().await?; |     client.start_autosharded().await?; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,6 +1,4 @@ | |||||||
| use axum::{routing::get, Router}; |  | ||||||
| use lazy_static::lazy_static; | use lazy_static::lazy_static; | ||||||
| use log::warn; |  | ||||||
| use prometheus::{IntCounterVec, Opts, Registry}; | use prometheus::{IntCounterVec, Opts, Registry}; | ||||||
|  |  | ||||||
| lazy_static! { | lazy_static! { | ||||||
| @@ -26,21 +24,3 @@ pub fn init_metrics() { | |||||||
|     REGISTRY.register(Box::new(REMINDER_FAIL_COUNTER.clone())).unwrap(); |     REGISTRY.register(Box::new(REMINDER_FAIL_COUNTER.clone())).unwrap(); | ||||||
|     REGISTRY.register(Box::new(COMMAND_COUNTER.clone())).unwrap(); |     REGISTRY.register(Box::new(COMMAND_COUNTER.clone())).unwrap(); | ||||||
| } | } | ||||||
|  |  | ||||||
| pub async fn serve() { |  | ||||||
|     let app = Router::new().route("/metrics", get(metrics)); |  | ||||||
|  |  | ||||||
|     let listener = tokio::net::TcpListener::bind("localhost:31756").await.unwrap(); |  | ||||||
|     axum::serve(listener, app).await.unwrap(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async fn metrics() -> String { |  | ||||||
|     let encoder = prometheus::TextEncoder::new(); |  | ||||||
|     let res_custom = encoder.encode_to_string(®ISTRY.gather()); |  | ||||||
|  |  | ||||||
|     res_custom.unwrap_or_else(|e| { |  | ||||||
|         warn!("Error encoding metrics: {:?}", e); |  | ||||||
|  |  | ||||||
|         String::new() |  | ||||||
|     }) |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -55,7 +55,7 @@ pub struct ReminderBuilder { | |||||||
|     tts: bool, |     tts: bool, | ||||||
|     attachment_name: Option<String>, |     attachment_name: Option<String>, | ||||||
|     attachment: Option<Vec<u8>>, |     attachment: Option<Vec<u8>>, | ||||||
|     set_by: Option<u32>, |     set_by: Option<u64>, | ||||||
| } | } | ||||||
|  |  | ||||||
| impl ReminderBuilder { | impl ReminderBuilder { | ||||||
| @@ -132,7 +132,7 @@ pub struct MultiReminderBuilder<'a> { | |||||||
|     interval: Option<Interval>, |     interval: Option<Interval>, | ||||||
|     expires: Option<NaiveDateTime>, |     expires: Option<NaiveDateTime>, | ||||||
|     content: Content, |     content: Content, | ||||||
|     set_by: Option<u32>, |     set_by: Option<u64>, | ||||||
|     ctx: &'a Context<'a>, |     ctx: &'a Context<'a>, | ||||||
|     guild_id: Option<GuildId>, |     guild_id: Option<GuildId>, | ||||||
| } | } | ||||||
|   | |||||||
| @@ -85,17 +85,13 @@ impl Reminder { | |||||||
|                 reminders.enabled, |                 reminders.enabled, | ||||||
|                 reminders.content, |                 reminders.content, | ||||||
|                 reminders.embed_description, |                 reminders.embed_description, | ||||||
|                 users.user AS set_by |                 reminders.set_by | ||||||
|             FROM |             FROM | ||||||
|                 reminders |                 reminders | ||||||
|             INNER JOIN |             INNER JOIN | ||||||
|                 channels |                 channels | ||||||
|             ON |             ON | ||||||
|                 reminders.channel_id = channels.id |                 reminders.channel_id = channels.id | ||||||
|             LEFT JOIN |  | ||||||
|                 users |  | ||||||
|             ON |  | ||||||
|                 reminders.set_by = users.id |  | ||||||
|             WHERE |             WHERE | ||||||
|                 reminders.uid = ? |                 reminders.uid = ? | ||||||
|             ", |             ", | ||||||
| @@ -122,17 +118,13 @@ impl Reminder { | |||||||
|                 reminders.enabled, |                 reminders.enabled, | ||||||
|                 reminders.content, |                 reminders.content, | ||||||
|                 reminders.embed_description, |                 reminders.embed_description, | ||||||
|                 users.user AS set_by |                 reminders.set_by | ||||||
|             FROM |             FROM | ||||||
|                 reminders |                 reminders | ||||||
|             INNER JOIN |             INNER JOIN | ||||||
|                 channels |                 channels | ||||||
|             ON |             ON | ||||||
|                 reminders.channel_id = channels.id |                 reminders.channel_id = channels.id | ||||||
|             LEFT JOIN |  | ||||||
|                 users |  | ||||||
|             ON |  | ||||||
|                 reminders.set_by = users.id |  | ||||||
|             WHERE |             WHERE | ||||||
|                 reminders.id = ? |                 reminders.id = ? | ||||||
|             ", |             ", | ||||||
| @@ -166,17 +158,13 @@ impl Reminder { | |||||||
|                 reminders.enabled, |                 reminders.enabled, | ||||||
|                 reminders.content, |                 reminders.content, | ||||||
|                 reminders.embed_description, |                 reminders.embed_description, | ||||||
|                 users.user AS set_by |                 reminders.set_by | ||||||
|             FROM |             FROM | ||||||
|                 reminders |                 reminders | ||||||
|             INNER JOIN |             INNER JOIN | ||||||
|                 channels |                 channels | ||||||
|             ON |             ON | ||||||
|                 reminders.channel_id = channels.id |                 reminders.channel_id = channels.id | ||||||
|             LEFT JOIN |  | ||||||
|                 users |  | ||||||
|             ON |  | ||||||
|                 reminders.set_by = users.id |  | ||||||
|             WHERE |             WHERE | ||||||
|                 `status` = 'pending' AND |                 `status` = 'pending' AND | ||||||
|                 channels.channel = ? AND |                 channels.channel = ? AND | ||||||
| @@ -230,17 +218,13 @@ impl Reminder { | |||||||
|                             reminders.enabled, |                             reminders.enabled, | ||||||
|                             reminders.content, |                             reminders.content, | ||||||
|                             reminders.embed_description, |                             reminders.embed_description, | ||||||
|                             users.user AS set_by |                             reminders.set_by | ||||||
|                         FROM |                         FROM | ||||||
|                             reminders |                             reminders | ||||||
|                         LEFT JOIN |                         LEFT JOIN | ||||||
|                             channels |                             channels | ||||||
|                         ON |                         ON | ||||||
|                             channels.id = reminders.channel_id |                             channels.id = reminders.channel_id | ||||||
|                         LEFT JOIN |  | ||||||
|                             users |  | ||||||
|                         ON |  | ||||||
|                             reminders.set_by = users.id |  | ||||||
|                         WHERE |                         WHERE | ||||||
|                             `status` = 'pending' AND |                             `status` = 'pending' AND | ||||||
|                             FIND_IN_SET(channels.channel, ?) |                             FIND_IN_SET(channels.channel, ?) | ||||||
| @@ -266,17 +250,13 @@ impl Reminder { | |||||||
|                             reminders.enabled, |                             reminders.enabled, | ||||||
|                             reminders.content, |                             reminders.content, | ||||||
|                             reminders.embed_description, |                             reminders.embed_description, | ||||||
|                             users.user AS set_by |                             reminders.set_by | ||||||
|                         FROM |                         FROM | ||||||
|                             reminders |                             reminders | ||||||
|                         LEFT JOIN |                         LEFT JOIN | ||||||
|                             channels |                             channels | ||||||
|                         ON |                         ON | ||||||
|                             channels.id = reminders.channel_id |                             channels.id = reminders.channel_id | ||||||
|                         LEFT JOIN |  | ||||||
|                             users |  | ||||||
|                         ON |  | ||||||
|                             reminders.set_by = users.id |  | ||||||
|                         WHERE |                         WHERE | ||||||
|                             `status` = 'pending' AND |                             `status` = 'pending' AND | ||||||
|                             channels.guild_id = (SELECT id FROM guilds WHERE guild = ?) |                             channels.guild_id = (SELECT id FROM guilds WHERE guild = ?) | ||||||
| @@ -303,20 +283,16 @@ impl Reminder { | |||||||
|                     reminders.enabled, |                     reminders.enabled, | ||||||
|                     reminders.content, |                     reminders.content, | ||||||
|                     reminders.embed_description, |                     reminders.embed_description, | ||||||
|                     users.user AS set_by |                     reminders.set_by | ||||||
|                 FROM |                 FROM | ||||||
|                     reminders |                     reminders | ||||||
|                 INNER JOIN |                 INNER JOIN | ||||||
|                     channels |                     channels | ||||||
|                 ON |                 ON | ||||||
|                     channels.id = reminders.channel_id |                     channels.id = reminders.channel_id | ||||||
|                 LEFT JOIN |  | ||||||
|                     users |  | ||||||
|                 ON |  | ||||||
|                     reminders.set_by = users.id |  | ||||||
|                 WHERE |                 WHERE | ||||||
|                     `status` = 'pending' AND |                     `status` = 'pending' AND | ||||||
|                     channels.id = (SELECT dm_channel FROM users WHERE user = ?) |                     channels.id = (SELECT dm_channel FROM users WHERE id = ?) | ||||||
|                 ", |                 ", | ||||||
|                 user.get() |                 user.get() | ||||||
|             ) |             ) | ||||||
|   | |||||||
| @@ -6,9 +6,7 @@ use sqlx::MySqlPool; | |||||||
| use crate::consts::LOCAL_TIMEZONE; | use crate::consts::LOCAL_TIMEZONE; | ||||||
|  |  | ||||||
| pub struct UserData { | pub struct UserData { | ||||||
|     pub id: u32, |     pub id: u64, | ||||||
|     #[allow(dead_code)] |  | ||||||
|     pub user: u64, |  | ||||||
|     pub dm_channel: u32, |     pub dm_channel: u32, | ||||||
|     pub timezone: String, |     pub timezone: String, | ||||||
|     pub allowed_dm: bool, |     pub allowed_dm: bool, | ||||||
| @@ -23,7 +21,9 @@ impl UserData { | |||||||
|  |  | ||||||
|         match sqlx::query!( |         match sqlx::query!( | ||||||
|             " |             " | ||||||
|             SELECT IFNULL(timezone, 'UTC') AS timezone FROM users WHERE user = ? |             SELECT IFNULL(timezone, 'UTC') AS timezone | ||||||
|  |             FROM users | ||||||
|  |             WHERE id = ? | ||||||
|             ", |             ", | ||||||
|             user_id |             user_id | ||||||
|         ) |         ) | ||||||
| @@ -48,7 +48,9 @@ impl UserData { | |||||||
|         match sqlx::query_as_unchecked!( |         match sqlx::query_as_unchecked!( | ||||||
|             Self, |             Self, | ||||||
|             " |             " | ||||||
| SELECT id, user, dm_channel, IF(timezone IS NULL, ?, timezone) AS timezone, allowed_dm FROM users WHERE user = ? |             SELECT id, dm_channel, IF(timezone IS NULL, ?, timezone) AS timezone, allowed_dm | ||||||
|  |             FROM users | ||||||
|  |             WHERE id = ? | ||||||
|             ", |             ", | ||||||
|             *LOCAL_TIMEZONE, |             *LOCAL_TIMEZONE, | ||||||
|             user_id.get() |             user_id.get() | ||||||
| @@ -64,7 +66,8 @@ SELECT id, user, dm_channel, IF(timezone IS NULL, ?, timezone) AS timezone, allo | |||||||
|  |  | ||||||
|                 sqlx::query!( |                 sqlx::query!( | ||||||
|                     " |                     " | ||||||
| INSERT IGNORE INTO channels (channel) VALUES (?) |                     INSERT IGNORE INTO channels (channel) | ||||||
|  |                     VALUES (?) | ||||||
|                     ", |                     ", | ||||||
|                     dm_channel.id.get() |                     dm_channel.id.get() | ||||||
|                 ) |                 ) | ||||||
| @@ -73,7 +76,8 @@ INSERT IGNORE INTO channels (channel) VALUES (?) | |||||||
|  |  | ||||||
|                 sqlx::query!( |                 sqlx::query!( | ||||||
|                     " |                     " | ||||||
| INSERT INTO users (name, user, dm_channel, timezone) VALUES ('', ?, (SELECT id FROM channels WHERE channel = ?), ?) |                     INSERT INTO users (id, dm_channel, timezone) | ||||||
|  |                     VALUES (?, (SELECT id FROM channels WHERE channel = ?), ?) | ||||||
|                     ", |                     ", | ||||||
|                     user_id.get(), |                     user_id.get(), | ||||||
|                     dm_channel.id.get(), |                     dm_channel.id.get(), | ||||||
| @@ -85,7 +89,9 @@ INSERT INTO users (name, user, dm_channel, timezone) VALUES ('', ?, (SELECT id F | |||||||
|                 Ok(sqlx::query_as_unchecked!( |                 Ok(sqlx::query_as_unchecked!( | ||||||
|                     Self, |                     Self, | ||||||
|                     " |                     " | ||||||
| SELECT id, user, dm_channel, timezone, allowed_dm FROM users WHERE user = ? |                     SELECT id, dm_channel, timezone, allowed_dm | ||||||
|  |                     FROM users | ||||||
|  |                     WHERE id = ? | ||||||
|                     ", |                     ", | ||||||
|                     user_id.get() |                     user_id.get() | ||||||
|                 ) |                 ) | ||||||
| @@ -104,7 +110,9 @@ SELECT id, user, dm_channel, timezone, allowed_dm FROM users WHERE user = ? | |||||||
|     pub async fn commit_changes(&self, pool: &MySqlPool) { |     pub async fn commit_changes(&self, pool: &MySqlPool) { | ||||||
|         sqlx::query!( |         sqlx::query!( | ||||||
|             " |             " | ||||||
| UPDATE users SET timezone = ?, allowed_dm = ? WHERE id = ? |             UPDATE users | ||||||
|  |             SET timezone = ?, allowed_dm = ? | ||||||
|  |             WHERE id = ? | ||||||
|             ", |             ", | ||||||
|             self.timezone, |             self.timezone, | ||||||
|             self.allowed_dm, |             self.allowed_dm, | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ use std::env; | |||||||
|  |  | ||||||
| use log::{info, warn}; | use log::{info, warn}; | ||||||
| use poise::serenity_prelude::client::Context; | use poise::serenity_prelude::client::Context; | ||||||
|  | use sd_notify::{self, NotifyState}; | ||||||
| use sqlx::{Executor, MySql}; | use sqlx::{Executor, MySql}; | ||||||
| use tokio::{ | use tokio::{ | ||||||
|     sync::broadcast::Receiver, |     sync::broadcast::Receiver, | ||||||
| @@ -33,6 +34,15 @@ async fn _initialize(ctx: Context, pool: impl Executor<'_, Database = Database> | |||||||
|         .flatten() |         .flatten() | ||||||
|         .unwrap_or(10); |         .unwrap_or(10); | ||||||
|  |  | ||||||
|  |     let mut watchdog_interval = 0; | ||||||
|  |     let watchdog = sd_notify::watchdog_enabled(false, &mut watchdog_interval); | ||||||
|  |  | ||||||
|  |     if watchdog { | ||||||
|  |         warn!("Watchdog enabled. Don't die!"); | ||||||
|  |     } else { | ||||||
|  |         warn!("No watchdog running") | ||||||
|  |     } | ||||||
|  |  | ||||||
|     loop { |     loop { | ||||||
|         let sleep_to = Instant::now() + Duration::from_secs(remind_interval); |         let sleep_to = Instant::now() + Duration::from_secs(remind_interval); | ||||||
|         let reminders = sender::Reminder::fetch_reminders(pool).await; |         let reminders = sender::Reminder::fetch_reminders(pool).await; | ||||||
| @@ -42,9 +52,11 @@ async fn _initialize(ctx: Context, pool: impl Executor<'_, Database = Database> | |||||||
|  |  | ||||||
|             for reminder in reminders { |             for reminder in reminders { | ||||||
|                 reminder.send(pool, ctx.clone()).await; |                 reminder.send(pool, ctx.clone()).await; | ||||||
|  |                 let _ = sd_notify::notify(false, &[NotifyState::Watchdog]); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         sleep_until(sleep_to).await; |         sleep_until(sleep_to).await; | ||||||
|  |         let _ = sd_notify::notify(false, &[NotifyState::Watchdog]); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										1
									
								
								src/web/fairings/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/web/fairings/mod.rs
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1 @@ | |||||||
|  | pub mod metrics; | ||||||
| @@ -2,9 +2,10 @@ mod consts; | |||||||
| #[macro_use] | #[macro_use] | ||||||
| mod macros; | mod macros; | ||||||
| mod catchers; | mod catchers; | ||||||
|  | mod fairings; | ||||||
| mod guards; | mod guards; | ||||||
| mod metrics; |  | ||||||
| mod routes; | mod routes; | ||||||
|  |  | ||||||
| pub mod string { | pub mod string { | ||||||
|     use std::{fmt::Display, str::FromStr}; |     use std::{fmt::Display, str::FromStr}; | ||||||
|  |  | ||||||
| @@ -79,7 +80,7 @@ use sqlx::{MySql, Pool}; | |||||||
|  |  | ||||||
| use crate::web::{ | use crate::web::{ | ||||||
|     consts::{CNC_GUILD, DISCORD_OAUTH_AUTHORIZE, DISCORD_OAUTH_TOKEN, SUBSCRIPTION_ROLES}, |     consts::{CNC_GUILD, DISCORD_OAUTH_AUTHORIZE, DISCORD_OAUTH_TOKEN, SUBSCRIPTION_ROLES}, | ||||||
|     metrics::MetricProducer, |     fairings::metrics::MetricProducer, | ||||||
| }; | }; | ||||||
|  |  | ||||||
| type Database = MySql; | type Database = MySql; | ||||||
| @@ -149,6 +150,7 @@ pub async fn initialize( | |||||||
|                 routes::report::report_error, |                 routes::report::report_error, | ||||||
|                 routes::return_to_same_site, |                 routes::return_to_same_site, | ||||||
|                 routes::terms, |                 routes::terms, | ||||||
|  |                 routes::metrics, | ||||||
|             ], |             ], | ||||||
|         ) |         ) | ||||||
|         .mount( |         .mount( | ||||||
| @@ -177,6 +179,8 @@ pub async fn initialize( | |||||||
|         .mount( |         .mount( | ||||||
|             "/dashboard", |             "/dashboard", | ||||||
|             routes![ |             routes![ | ||||||
|  |                 routes::dashboard::reminders_redirect, | ||||||
|  |                 routes::dashboard::todos_redirect, | ||||||
|                 routes::dashboard::dashboard, |                 routes::dashboard::dashboard, | ||||||
|                 routes::dashboard::dashboard_home, |                 routes::dashboard::dashboard_home, | ||||||
|                 routes::dashboard::api::delete_reminder, |                 routes::dashboard::api::delete_reminder, | ||||||
|   | |||||||
| @@ -16,17 +16,15 @@ pub async fn get_guild_roles(id: u64, cookies: &CookieJar<'_>, ctx: &State<Conte | |||||||
|     offline!(Ok(json!(vec![RoleInfo { name: "@everyone".to_string(), id: "1".to_string() }]))); |     offline!(Ok(json!(vec![RoleInfo { name: "@everyone".to_string(), id: "1".to_string() }]))); | ||||||
|     check_authorization(cookies, ctx.inner(), id).await?; |     check_authorization(cookies, ctx.inner(), id).await?; | ||||||
|  |  | ||||||
|     let roles_res = ctx.cache.guild_roles(id); |     let roles_res = ctx.cache.guild(id).map(|g| { | ||||||
|  |         g.roles | ||||||
|  |             .iter() | ||||||
|  |             .map(|(_, r)| RoleInfo { id: r.id.to_string(), name: r.name.to_string() }) | ||||||
|  |             .collect::<Vec<RoleInfo>>() | ||||||
|  |     }); | ||||||
|  |  | ||||||
|     match roles_res { |     match roles_res { | ||||||
|         Some(roles) => { |         Some(roles) => Ok(json!(roles)), | ||||||
|             let roles = roles |  | ||||||
|                 .iter() |  | ||||||
|                 .map(|(_, r)| RoleInfo { id: r.id.to_string(), name: r.name.to_string() }) |  | ||||||
|                 .collect::<Vec<RoleInfo>>(); |  | ||||||
|  |  | ||||||
|             Ok(json!(roles)) |  | ||||||
|         } |  | ||||||
|         None => { |         None => { | ||||||
|             warn!("Could not fetch roles from {}", id); |             warn!("Could not fetch roles from {}", id); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ use std::env; | |||||||
|  |  | ||||||
| use chrono_tz::Tz; | use chrono_tz::Tz; | ||||||
| pub use guilds::*; | pub use guilds::*; | ||||||
|  | use log::warn; | ||||||
| pub use reminders::*; | pub use reminders::*; | ||||||
| use rocket::{ | use rocket::{ | ||||||
|     get, |     get, | ||||||
| @@ -21,16 +22,29 @@ use serenity::{ | |||||||
| }; | }; | ||||||
| use sqlx::{MySql, Pool}; | use sqlx::{MySql, Pool}; | ||||||
|  |  | ||||||
|  | use crate::web::guards::transaction::Transaction; | ||||||
|  |  | ||||||
|  | #[derive(Serialize)] | ||||||
|  | struct UserPreferences { | ||||||
|  |     timezone: String, | ||||||
|  |     use_browser_timezone: bool, | ||||||
|  |     dashboard_color_scheme: String, | ||||||
|  |     reset_inputs_on_create: bool, | ||||||
|  | } | ||||||
|  |  | ||||||
| #[derive(Serialize)] | #[derive(Serialize)] | ||||||
| struct UserInfo { | struct UserInfo { | ||||||
|     name: String, |     name: String, | ||||||
|     patreon: bool, |     patreon: bool, | ||||||
|     timezone: Option<String>, |     preferences: UserPreferences, | ||||||
| } | } | ||||||
|  |  | ||||||
| #[derive(Deserialize)] | #[derive(Deserialize)] | ||||||
| pub struct UpdateUser { | pub struct UpdateUserPreferences { | ||||||
|     timezone: String, |     timezone: Option<String>, | ||||||
|  |     use_browser_timezone: Option<bool>, | ||||||
|  |     dashboard_color_scheme: Option<String>, | ||||||
|  |     reset_inputs_on_create: Option<bool>, | ||||||
| } | } | ||||||
|  |  | ||||||
| #[get("/api/user")] | #[get("/api/user")] | ||||||
| @@ -39,7 +53,16 @@ pub async fn get_user_info( | |||||||
|     ctx: &State<Context>, |     ctx: &State<Context>, | ||||||
|     pool: &State<Pool<MySql>>, |     pool: &State<Pool<MySql>>, | ||||||
| ) -> JsonValue { | ) -> JsonValue { | ||||||
|     offline!(json!(UserInfo { name: "Discord".to_string(), patreon: true, timezone: None })); |     offline!(json!(UserInfo { | ||||||
|  |         name: "Discord".to_string(), | ||||||
|  |         patreon: true, | ||||||
|  |         preferences: UserPreferences { | ||||||
|  |             timezone: "UTC".to_string(), | ||||||
|  |             use_browser_timezone: true, | ||||||
|  |             dashboard_color_scheme: "system".to_string(), | ||||||
|  |             reset_inputs_on_create: false, | ||||||
|  |         } | ||||||
|  |     })); | ||||||
|  |  | ||||||
|     if let Some(user_id) = |     if let Some(user_id) = | ||||||
|         cookies.get_private("userid").map(|u| u.value().parse::<u64>().ok()).flatten() |         cookies.get_private("userid").map(|u| u.value().parse::<u64>().ok()).flatten() | ||||||
| @@ -48,13 +71,34 @@ pub async fn get_user_info( | |||||||
|             .member(&ctx.inner(), user_id) |             .member(&ctx.inner(), user_id) | ||||||
|             .await; |             .await; | ||||||
|  |  | ||||||
|         let timezone = sqlx::query!( |         let preferences = sqlx::query!( | ||||||
|             "SELECT IFNULL(timezone, 'UTC') AS timezone FROM users WHERE user = ?", |             " | ||||||
|  |             SELECT | ||||||
|  |                 IFNULL(timezone, 'UTC') AS timezone, | ||||||
|  |                 use_browser_timezone, | ||||||
|  |                 dashboard_color_scheme, | ||||||
|  |                 reset_inputs_on_create | ||||||
|  |             FROM users | ||||||
|  |             WHERE id = ? | ||||||
|  |             ", | ||||||
|             user_id |             user_id | ||||||
|         ) |         ) | ||||||
|         .fetch_one(pool.inner()) |         .fetch_one(pool.inner()) | ||||||
|         .await |         .await | ||||||
|         .map_or(None, |q| Some(q.timezone)); |         .map_or( | ||||||
|  |             UserPreferences { | ||||||
|  |                 timezone: "UTC".to_string(), | ||||||
|  |                 use_browser_timezone: false, | ||||||
|  |                 dashboard_color_scheme: "system".to_string(), | ||||||
|  |                 reset_inputs_on_create: false, | ||||||
|  |             }, | ||||||
|  |             |q| UserPreferences { | ||||||
|  |                 timezone: q.timezone, | ||||||
|  |                 use_browser_timezone: q.use_browser_timezone != 0, | ||||||
|  |                 dashboard_color_scheme: q.dashboard_color_scheme, | ||||||
|  |                 reset_inputs_on_create: q.reset_inputs_on_create != 0, | ||||||
|  |             }, | ||||||
|  |         ); | ||||||
|  |  | ||||||
|         let user_info = UserInfo { |         let user_info = UserInfo { | ||||||
|             name: cookies |             name: cookies | ||||||
| @@ -65,7 +109,7 @@ pub async fn get_user_info( | |||||||
|                     .roles |                     .roles | ||||||
|                     .contains(&RoleId::new(env::var("PATREON_ROLE_ID").unwrap().parse().unwrap())) |                     .contains(&RoleId::new(env::var("PATREON_ROLE_ID").unwrap().parse().unwrap())) | ||||||
|             }), |             }), | ||||||
|             timezone, |             preferences, | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|         json!(user_info) |         json!(user_info) | ||||||
| @@ -74,27 +118,86 @@ pub async fn get_user_info( | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| #[patch("/api/user", data = "<user>")] | #[patch("/api/user", data = "<preferences>")] | ||||||
| pub async fn update_user_info( | pub async fn update_user_info( | ||||||
|     cookies: &CookieJar<'_>, |     cookies: &CookieJar<'_>, | ||||||
|     user: Json<UpdateUser>, |     preferences: Json<UpdateUserPreferences>, | ||||||
|     pool: &State<Pool<MySql>>, |     mut transaction: Transaction<'_>, | ||||||
| ) -> JsonValue { | ) -> JsonValue { | ||||||
|     if let Some(user_id) = |     if let Some(user_id) = | ||||||
|         cookies.get_private("userid").map(|u| u.value().parse::<u64>().ok()).flatten() |         cookies.get_private("userid").map(|u| u.value().parse::<u64>().ok()).flatten() | ||||||
|     { |     { | ||||||
|         if user.timezone.parse::<Tz>().is_ok() { |         if let Some(timezone) = &preferences.timezone { | ||||||
|  |             if timezone.parse::<Tz>().is_ok() { | ||||||
|  |                 let _ = sqlx::query!( | ||||||
|  |                     " | ||||||
|  |                     UPDATE users | ||||||
|  |                     SET timezone = ? | ||||||
|  |                     WHERE id = ? | ||||||
|  |                     ", | ||||||
|  |                     timezone, | ||||||
|  |                     user_id, | ||||||
|  |                 ) | ||||||
|  |                 .execute(transaction.executor()) | ||||||
|  |                 .await; | ||||||
|  |             } else { | ||||||
|  |                 return json!({"error": "Timezone not recognised"}); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if let Some(dashboard_color_scheme) = &preferences.dashboard_color_scheme { | ||||||
|  |             if vec!["system", "light", "dark"].contains(&dashboard_color_scheme.as_str()) { | ||||||
|  |                 let _ = sqlx::query!( | ||||||
|  |                     " | ||||||
|  |                     UPDATE users | ||||||
|  |                     SET dashboard_color_scheme = ? | ||||||
|  |                     WHERE id = ? | ||||||
|  |                     ", | ||||||
|  |                     dashboard_color_scheme, | ||||||
|  |                     user_id, | ||||||
|  |                 ) | ||||||
|  |                 .execute(transaction.executor()) | ||||||
|  |                 .await; | ||||||
|  |             } else { | ||||||
|  |                 return json!({"error": "Color scheme not recognised"}); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if let Some(reset_inputs_on_create) = &preferences.reset_inputs_on_create { | ||||||
|             let _ = sqlx::query!( |             let _ = sqlx::query!( | ||||||
|                 "UPDATE users SET timezone = ? WHERE user = ?", |                 " | ||||||
|                 user.timezone, |                 UPDATE users | ||||||
|  |                 SET reset_inputs_on_create = ? | ||||||
|  |                 WHERE id = ? | ||||||
|  |                 ", | ||||||
|  |                 reset_inputs_on_create, | ||||||
|                 user_id, |                 user_id, | ||||||
|             ) |             ) | ||||||
|             .execute(pool.inner()) |             .execute(transaction.executor()) | ||||||
|             .await; |             .await; | ||||||
|  |         } | ||||||
|  |  | ||||||
|             json!({}) |         if let Some(use_browser_timezone) = &preferences.use_browser_timezone { | ||||||
|         } else { |             let _ = sqlx::query!( | ||||||
|             json!({"error": "Timezone not recognized"}) |                 " | ||||||
|  |                 UPDATE users | ||||||
|  |                 SET use_browser_timezone = ? | ||||||
|  |                 WHERE id = ? | ||||||
|  |                 ", | ||||||
|  |                 use_browser_timezone, | ||||||
|  |                 user_id, | ||||||
|  |             ) | ||||||
|  |             .execute(transaction.executor()) | ||||||
|  |             .await; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         match transaction.commit().await { | ||||||
|  |             Ok(_) => json!({}), | ||||||
|  |             Err(e) => { | ||||||
|  |                 warn!("Error updating user preferences for {}: {:?}", user_id, e); | ||||||
|  |  | ||||||
|  |                 json!({"error": "Couldn't update preferences"}) | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|     } else { |     } else { | ||||||
|         json!({"error": "Not authorized"}) |         json!({"error": "Not authorized"}) | ||||||
|   | |||||||
| @@ -593,18 +593,11 @@ pub(crate) async fn create_reminder( | |||||||
| } | } | ||||||
|  |  | ||||||
| fn check_channel_matches_guild(ctx: &Context, channel_id: ChannelId, guild_id: GuildId) -> bool { | fn check_channel_matches_guild(ctx: &Context, channel_id: ChannelId, guild_id: GuildId) -> bool { | ||||||
|     // validate channel |     return match ctx.cache.guild(guild_id) { | ||||||
|     let channel = channel_id.to_channel_cached(&ctx.cache); |         Some(guild) => guild.channels.get(&channel_id).is_some(), | ||||||
|     let channel_exists = channel.is_some(); |  | ||||||
|  |  | ||||||
|     if !channel_exists { |         None => false, | ||||||
|         return false; |     }; | ||||||
|     } |  | ||||||
|  |  | ||||||
|     let channel_matches_guild = |  | ||||||
|         channel.map_or(false, |c| c.guild(&ctx.cache).map_or(false, |g| g.id == guild_id)); |  | ||||||
|  |  | ||||||
|     channel_matches_guild |  | ||||||
| } | } | ||||||
|  |  | ||||||
| async fn create_database_channel( | async fn create_database_channel( | ||||||
| @@ -697,6 +690,18 @@ pub enum DashboardPage { | |||||||
|     NotConfigured(Template), |     NotConfigured(Template), | ||||||
| } | } | ||||||
|  |  | ||||||
|  | // Legacy route to maintain compatibility with old dashboard routing | ||||||
|  | #[get("/?<id>")] | ||||||
|  | pub async fn reminders_redirect(id: &str) -> Redirect { | ||||||
|  |     Redirect::to(format!("/dashboard/{}/reminders", id)) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Legacy route to maintain compatibility with old dashboard routing | ||||||
|  | #[get("/todo?<id>")] | ||||||
|  | pub async fn todos_redirect(id: &str) -> Redirect { | ||||||
|  |     Redirect::to(format!("/dashboard/{}/todos", id)) | ||||||
|  | } | ||||||
|  |  | ||||||
| #[get("/")] | #[get("/")] | ||||||
| pub async fn dashboard_home(cookies: &CookieJar<'_>) -> DashboardPage { | pub async fn dashboard_home(cookies: &CookieJar<'_>) -> DashboardPage { | ||||||
|     if cookies.get_private("userid").is_some() { |     if cookies.get_private("userid").is_some() { | ||||||
|   | |||||||
| @@ -2,11 +2,14 @@ pub mod dashboard; | |||||||
| pub mod login; | pub mod login; | ||||||
| pub mod report; | pub mod report; | ||||||
|  |  | ||||||
| use std::collections::HashMap; | use std::{collections::HashMap, net::IpAddr}; | ||||||
|  |  | ||||||
|  | use log::warn; | ||||||
| use rocket::{get, request::FlashMessage, serde::json::Value as JsonValue}; | use rocket::{get, request::FlashMessage, serde::json::Value as JsonValue}; | ||||||
| use rocket_dyn_templates::Template; | use rocket_dyn_templates::Template; | ||||||
|  |  | ||||||
|  | use crate::metrics::REGISTRY; | ||||||
|  |  | ||||||
| pub type JsonResult = Result<JsonValue, JsonValue>; | pub type JsonResult = Result<JsonValue, JsonValue>; | ||||||
|  |  | ||||||
| #[get("/")] | #[get("/")] | ||||||
| @@ -107,3 +110,19 @@ pub async fn help_iemanager() -> Template { | |||||||
|     let map: HashMap<&str, String> = HashMap::new(); |     let map: HashMap<&str, String> = HashMap::new(); | ||||||
|     Template::render("support/iemanager", &map) |     Template::render("support/iemanager", &map) | ||||||
| } | } | ||||||
|  |  | ||||||
|  | #[get("/metrics")] | ||||||
|  | pub async fn metrics(client_ip: IpAddr) -> String { | ||||||
|  |     if !client_ip.is_loopback() { | ||||||
|  |         String::new() | ||||||
|  |     } else { | ||||||
|  |         let encoder = prometheus::TextEncoder::new(); | ||||||
|  |         let res_custom = encoder.encode_to_string(®ISTRY.gather()); | ||||||
|  |  | ||||||
|  |         res_custom.unwrap_or_else(|e| { | ||||||
|  |             warn!("Error encoding metrics: {:?}", e); | ||||||
|  |  | ||||||
|  |             String::new() | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | } | ||||||
|   | |||||||
| @@ -100,14 +100,6 @@ div.split-controls { | |||||||
|     flex-basis: 50%; |     flex-basis: 50%; | ||||||
| } | } | ||||||
|  |  | ||||||
| div.reminderContent { |  | ||||||
|     margin-top: 10px; |  | ||||||
|     margin-bottom: 10px; |  | ||||||
|     padding: 14px; |  | ||||||
|     background-color: #f5f5f5; |  | ||||||
|     border-radius: 8px; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .left-pad { | .left-pad { | ||||||
|     padding-left: 1rem; |     padding-left: 1rem; | ||||||
|     padding-right: 0.2rem; |     padding-right: 0.2rem; | ||||||
| @@ -214,18 +206,6 @@ div.dashboard-frame { | |||||||
|     min-width: auto; |     min-width: auto; | ||||||
| } | } | ||||||
|  |  | ||||||
| .menu a { |  | ||||||
|     color: #fff; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .menu .menu-label { |  | ||||||
|     color: #bbb; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .menu { |  | ||||||
|     padding-left: 4px; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .dashboard-navbar { | .dashboard-navbar { | ||||||
|     background-color: #8fb677 !important; |     background-color: #8fb677 !important; | ||||||
|     position: absolute; |     position: absolute; | ||||||
| @@ -680,14 +660,6 @@ div.reminderError .reminderMessage { | |||||||
|     margin: 0 12px 12px 12px; |     margin: 0 12px 12px 12px; | ||||||
| } | } | ||||||
|  |  | ||||||
| .button.is-success:not(.is-outlined) { |  | ||||||
|     color: white; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .button.is-outlined.is-success { |  | ||||||
|     background-color: white; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| a.switch-pane { | a.switch-pane { | ||||||
|     white-space: nowrap; |     white-space: nowrap; | ||||||
|     overflow: hidden; |     overflow: hidden; | ||||||
| @@ -733,27 +705,3 @@ a.switch-pane.is-active ~ .guild-submenu { | |||||||
| .is-locked .field:last-of-type { | .is-locked .field:last-of-type { | ||||||
|     display: none; |     display: none; | ||||||
| } | } | ||||||
|  |  | ||||||
| .stat-row { |  | ||||||
|     display: flex; |  | ||||||
|     flex-direction: row; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .stat-box { |  | ||||||
|     flex-grow: 1; |  | ||||||
|     border-radius: 6px; |  | ||||||
|     background-color: #fcfcfc; |  | ||||||
|     border-color: #efefef; |  | ||||||
|     border-style: solid; |  | ||||||
|     border-width: 1px; |  | ||||||
|     margin: 4px; |  | ||||||
|     padding: 4px; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .figure { |  | ||||||
|     text-align: center; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| .figure-num { |  | ||||||
|     font-size: 2rem; |  | ||||||
| } |  | ||||||
|   | |||||||
| @@ -7,8 +7,9 @@ Type=simple | |||||||
| ExecStart=/usr/bin/reminder-rs | ExecStart=/usr/bin/reminder-rs | ||||||
| WorkingDirectory=/etc/reminder-rs | WorkingDirectory=/etc/reminder-rs | ||||||
| Restart=always | Restart=always | ||||||
| RestartSec=4 | RestartSec=10 | ||||||
| Environment="reminder_rs=warn,postman=warn" | Environment="RUST_LOG=warn,rocket=info,reminder_rs=debug,postman=debug" | ||||||
|  | WatchdogSec=120 | ||||||
|  |  | ||||||
| [Install] | [Install] | ||||||
| WantedBy=multi-user.target | WantedBy=multi-user.target | ||||||
|   | |||||||
| @@ -4,6 +4,7 @@ | |||||||
| <html lang="EN"> | <html lang="EN"> | ||||||
| <head> | <head> | ||||||
|     <meta name="description" content="The most powerful Discord Reminders Bot"> |     <meta name="description" content="The most powerful Discord Reminders Bot"> | ||||||
|  |     <meta name="keywords" content="discord,discord bot,reminders,reminders bot,discord reminders,discord automation,discord messages"> | ||||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0"> |     <meta name="viewport" content="width=device-width, initial-scale=1.0"> | ||||||
|     <meta charset="UTF-8"> |     <meta charset="UTF-8"> | ||||||
|     <meta name="yandex-verification" content="bb77b8681eb64a90" /> |     <meta name="yandex-verification" content="bb77b8681eb64a90" /> | ||||||
|   | |||||||
| @@ -1,389 +0,0 @@ | |||||||
| <!DOCTYPE html> |  | ||||||
| <html lang="EN"> |  | ||||||
| <head> |  | ||||||
|     <script src="/static/js/reporter.js" type="application/javascript"></script> |  | ||||||
|  |  | ||||||
|     <meta name="description" content="The most powerful Discord Reminders Bot"> |  | ||||||
|     <meta name="viewport" content="width=device-width, initial-scale=1.0"> |  | ||||||
|     <meta charset="UTF-8"> |  | ||||||
|     <meta name="yandex-verification" content="bb77b8681eb64a90"/> |  | ||||||
|     <meta name="google-site-verification" content="7h7UVTeEe0AOzHiH3cFtsqMULYGN-zCZdMT_YCkW1Ho"/> |  | ||||||
|     <!-- <meta http-equiv="Content-Security-Policy" content="default-src 'self'; img-src *; font-src fonts.gstatic.com 'self'"> --> |  | ||||||
|  |  | ||||||
|     <!-- favicon --> |  | ||||||
|     <link rel="apple-touch-icon" sizes="180x180" |  | ||||||
|           href="/static/favicon/apple-touch-icon.png"> |  | ||||||
|     <link rel="icon" type="image/png" sizes="32x32" |  | ||||||
|           href="/static/favicon/favicon-32x32.png"> |  | ||||||
|     <link rel="icon" type="image/png" sizes="16x16" |  | ||||||
|           href="/static/favicon/favicon-16x16.png"> |  | ||||||
|     <link rel="manifest" href="/static/favicon/site.webmanifest"> |  | ||||||
|     <meta name="msapplication-TileColor" content="#da532c"> |  | ||||||
|     <meta name="theme-color" content="#ffffff"> |  | ||||||
|  |  | ||||||
|     <title>Reminder Bot | Dashboard</title> |  | ||||||
|  |  | ||||||
|     <!-- styles --> |  | ||||||
|     <link rel="stylesheet" href="/static/css/bulma.min.css"> |  | ||||||
|     <link rel="stylesheet" href="/static/css/fa.css"> |  | ||||||
|     <link rel="stylesheet" href="/static/css/font.css"> |  | ||||||
|     <link rel="stylesheet" href="/static/css/style.css?v{{ version }}"> |  | ||||||
|     <link rel="stylesheet" href="/static/css/dtsel.css"> |  | ||||||
|     <link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/tributejs/5.1.3/tribute.css" integrity="sha512-GnwBnXd+ZGO9CdP343MUr0jCcJXCr++JVtQRnllexRW2IDq4Zvrh/McTQjooAKnSUbXZ7wamp7AQSweTnfMVoA==" crossorigin="anonymous" referrerpolicy="no-referrer" /> |  | ||||||
|  |  | ||||||
|     <script src="/static/js/luxon.min.js"></script> |  | ||||||
|     <script src="https://cdnjs.cloudflare.com/ajax/libs/tributejs/5.1.3/tribute.min.js" integrity="sha512-KJYWC7RKz/Abtsu1QXd7VJ1IJua7P7GTpl3IKUqfa21Otg2opvRYmkui/CXBC6qeDYCNlQZ7c+7JfDXnKdILUA==" crossorigin="anonymous" referrerpolicy="no-referrer"></script> |  | ||||||
| </head> |  | ||||||
| <body> |  | ||||||
| <nav class="navbar is-spaced is-size-4 is-hidden-desktop dashboard-navbar" role="navigation" |  | ||||||
|      aria-label="main navigation"> |  | ||||||
|     <div class="navbar-brand"> |  | ||||||
|         <a class="navbar-item" href="/"> |  | ||||||
|             <figure class="image"> |  | ||||||
|                 <img width="28px" height="28px" src="/static/img/logo_nobg.webp" alt="Reminder Bot Logo"> |  | ||||||
|             </figure> |  | ||||||
|         </a> |  | ||||||
|  |  | ||||||
|         <p class="navbar-item pageTitle"> |  | ||||||
|         </p> |  | ||||||
|  |  | ||||||
|         <a role="button" class="dashboard-burger navbar-burger is-right" aria-label="menu" aria-expanded="false" |  | ||||||
|            data-target="mobileSidebar"> |  | ||||||
|             <span aria-hidden="true"></span> |  | ||||||
|             <span aria-hidden="true"></span> |  | ||||||
|             <span aria-hidden="true"></span> |  | ||||||
|         </a> |  | ||||||
|     </div> |  | ||||||
| </nav> |  | ||||||
|  |  | ||||||
| <div id="loader" class="is-hidden hero is-fullheight"> |  | ||||||
|     <div class="hero-body"> |  | ||||||
|         <div class="container has-text-centered"> |  | ||||||
|             <p class="title"> |  | ||||||
|                 <i class="fas fa-cog fa-spin"></i> |  | ||||||
|             </p> |  | ||||||
|             <p class="subtitle"> |  | ||||||
|                 <strong>Loading...</strong> |  | ||||||
|             </p> |  | ||||||
|         </div> |  | ||||||
|     </div> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <!-- dead image used to check which other images are dead --> |  | ||||||
| <img src="" id="dead"> |  | ||||||
|  |  | ||||||
| <div class="notification is-danger flash-message" id="errors"> |  | ||||||
|     <span class="icon"><i class="far fa-exclamation-circle"></i></span> <span class="error-message"></span> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="notification is-success flash-message" id="success"> |  | ||||||
|     <span class="icon"><i class="far fa-check"></i></span> <span class="success-message"></span> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="modal" id="addImageModal"> |  | ||||||
|     <div class="modal-background"></div> |  | ||||||
|     <div class="modal-card"> |  | ||||||
|         <header class="modal-card-head"> |  | ||||||
|             <label class="modal-card-title" for="urlInput">Enter Image URL</label> |  | ||||||
|             <button class="delete close-modal" aria-label="close"></button> |  | ||||||
|         </header> |  | ||||||
|         <section class="modal-card-body"> |  | ||||||
|             <input class="input" id="urlInput" placeholder="Image URL..."> |  | ||||||
|         </section> |  | ||||||
|         <footer class="modal-card-foot"> |  | ||||||
|             <button class="button is-success" id="setImgUrl">Save</button> |  | ||||||
|             <button class="button close-modal">Cancel</button> |  | ||||||
|         </footer> |  | ||||||
|     </div> |  | ||||||
|     <button class="modal-close is-large close-modal" aria-label="close"></button> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="modal" id="pickColorModal"> |  | ||||||
|     <div class="modal-background"></div> |  | ||||||
|     <div class="modal-card"> |  | ||||||
|         <header class="modal-card-head"> |  | ||||||
|             <label class="modal-card-title" for="colorInput">Select Color</label> |  | ||||||
|             <button class="delete close-modal" aria-label="close"></button> |  | ||||||
|         </header> |  | ||||||
|         <section class="modal-card-body"> |  | ||||||
|             <div class="colorpicker-container"> |  | ||||||
|                 <div id="colorpicker"></div> |  | ||||||
|             </div> |  | ||||||
|             <input class="input" id="colorInput"> |  | ||||||
|         </section> |  | ||||||
|         <footer class="modal-card-foot"> |  | ||||||
|             <button class="button is-success">Save</button> |  | ||||||
|             <button class="button close-modal">Cancel</button> |  | ||||||
|         </footer> |  | ||||||
|     </div> |  | ||||||
|     <button class="modal-close is-large close-modal" aria-label="close"></button> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="modal" id="chooseTimezoneModal"> |  | ||||||
|     <div class="modal-background"></div> |  | ||||||
|     <div class="modal-card"> |  | ||||||
|         <header class="modal-card-head"> |  | ||||||
|             <label class="modal-card-title" for="urlInput">Update Timezone <a href="/help/timezone"><span><i class="fa fa-question-circle"></i></span></a></label> |  | ||||||
|             <button class="delete close-modal" aria-label="close"></button> |  | ||||||
|         </header> |  | ||||||
|         <section class="modal-card-body"> |  | ||||||
|             <p> |  | ||||||
|                 Your configured timezone is: <strong><span class="set-timezone">%browsertimezone%</span></strong> (<span class="set-time">HH:mm</span>) |  | ||||||
|                 <br> |  | ||||||
|                 <br> |  | ||||||
|                 Your browser timezone is: <strong><span class="browser-timezone">%browsertimezone%</span></strong> (<span class="browser-time">HH:mm</span>) |  | ||||||
|                 <br> |  | ||||||
|                 Your bot timezone is: <strong><span class="bot-timezone">%bottimezone%</span></strong> (<span class="bot-time">HH:mm</span>) |  | ||||||
|             </p> |  | ||||||
|             <br> |  | ||||||
|             <div class="has-text-centered"> |  | ||||||
|                 <button class="button is-success close-modal" id="set-browser-timezone">Use Browser Timezone</button> |  | ||||||
|                 <button class="button is-link close-modal" id="set-bot-timezone">Use Bot Timezone</button> |  | ||||||
|                 <button class="button is-warning close-modal" id="update-bot-timezone">Set Bot Timezone</button> |  | ||||||
|             </div> |  | ||||||
|         </section> |  | ||||||
|     </div> |  | ||||||
|     <button class="modal-close is-large close-modal" aria-label="close"></button> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="modal" id="chooseTemplateModal"> |  | ||||||
|     <div class="modal-background"></div> |  | ||||||
|     <div class="modal-card"> |  | ||||||
|         <header class="modal-card-head"> |  | ||||||
|             <label class="modal-card-title" for="urlInput">Load Template</label> |  | ||||||
|             <button class="delete close-modal" aria-label="close"></button> |  | ||||||
|         </header> |  | ||||||
|         <section class="modal-card-body"> |  | ||||||
|             <div class="control has-icons-left"> |  | ||||||
|                 <div class="select is-fullwidth"> |  | ||||||
|                     <select id="templateSelect"> |  | ||||||
|                     </select> |  | ||||||
|                 </div> |  | ||||||
|                 <div class="icon is-small is-left"> |  | ||||||
|                     <i class="fas fa-file-spreadsheet"></i> |  | ||||||
|                 </div> |  | ||||||
|             </div> |  | ||||||
|             <br> |  | ||||||
|             <div class="has-text-centered"> |  | ||||||
|                 <button class="button is-success close-modal" id="load-template">Load Template</button> |  | ||||||
|                 <button class="button is-danger" id="delete-template">Delete</button> |  | ||||||
|             </div> |  | ||||||
|         </section> |  | ||||||
|     </div> |  | ||||||
|     <button class="modal-close is-large close-modal" aria-label="close"></button> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="modal" id="dataManagerModal"> |  | ||||||
|     <div class="modal-background"></div> |  | ||||||
|     <div class="modal-card"> |  | ||||||
|         <header class="modal-card-head"> |  | ||||||
|             <label class="modal-card-title" for="urlInput">Import/Export Manager <a href="/help/iemanager"><span><i class="fa fa-question-circle"></i></span></a></label> |  | ||||||
|             <button class="delete close-modal" aria-label="close"></button> |  | ||||||
|         </header> |  | ||||||
|         <section class="modal-card-body"> |  | ||||||
|             <div class="control"> |  | ||||||
|                 <div class="field"> |  | ||||||
|                     <label> |  | ||||||
|                         <input type="radio" class="default-width" name="exportSelect" value="reminders" checked> |  | ||||||
|                         Reminders |  | ||||||
|                     </label> |  | ||||||
|                 </div> |  | ||||||
|             </div> |  | ||||||
|             <br> |  | ||||||
|             <div class="has-text-centered"> |  | ||||||
|                 <div style="color: red"> |  | ||||||
|                     Please first read the <a href="/help/iemanager">support page</a> |  | ||||||
|                 </div> |  | ||||||
|                 <button class="button is-success is-outlined" id="import-data">Import Data</button> |  | ||||||
|                 <button class="button is-success" id="export-data">Export Data</button> |  | ||||||
|             </div> |  | ||||||
|             <a id="downloader" download="export.csv" class="is-hidden"></a> |  | ||||||
|             <input id="uploader" type="file" hidden></input> |  | ||||||
|         </section> |  | ||||||
|     </div> |  | ||||||
|     <button class="modal-close is-large close-modal" aria-label="close"></button> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="modal" id="deleteReminderModal"> |  | ||||||
|     <div class="modal-background"></div> |  | ||||||
|     <div class="modal-card"> |  | ||||||
|         <header class="modal-card-head"> |  | ||||||
|             <label class="modal-card-title">Delete Reminder</label> |  | ||||||
|             <button class="delete close-modal" aria-label="close"></button> |  | ||||||
|         </header> |  | ||||||
|         <section class="modal-card-body"> |  | ||||||
|             <p> |  | ||||||
|                 This reminder will be permanently deleted. Are you sure? |  | ||||||
|             </p> |  | ||||||
|             <br> |  | ||||||
|             <div class="has-text-centered"> |  | ||||||
|                 <button class="button is-danger" id="delete-reminder-confirm">Delete</button> |  | ||||||
|                 <button class="button is-light close-modal">Cancel</button> |  | ||||||
|             </div> |  | ||||||
|         </section> |  | ||||||
|     </div> |  | ||||||
|     <button class="modal-close is-large close-modal" aria-label="close"></button> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <div class="columns is-gapless dashboard-frame"> |  | ||||||
|     <div class="column is-2 is-sidebar-menu dashboard-sidebar is-hidden-touch"> |  | ||||||
|         <a href="/"> |  | ||||||
|             <div class="brand"> |  | ||||||
|                 <img src="/static/img/logo_nobg.webp" alt="Reminder bot logo" |  | ||||||
|                      width="52px" height="52px" |  | ||||||
|                      class="dashboard-brand"> |  | ||||||
|             </div> |  | ||||||
|         </a> |  | ||||||
|         <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1440 160"> |  | ||||||
|             <g transform="scale(1, 0.5)"> |  | ||||||
|                 <path fill="#8fb677" fill-opacity="1" |  | ||||||
|                       d="M0,192L60,170.7C120,149,240,107,360,96C480,85,600,107,720,138.7C840,171,960,213,1080,197.3C1200,181,1320,107,1380,69.3L1440,32L1440,0L1380,0C1320,0,1200,0,1080,0C960,0,840,0,720,0C600,0,480,0,360,0C240,0,120,0,60,0L0,0Z"></path> |  | ||||||
|             </g> |  | ||||||
|         </svg> |  | ||||||
|         <aside class="menu"> |  | ||||||
|             <p class="menu-label"> |  | ||||||
|                 Servers |  | ||||||
|             </p> |  | ||||||
|             <ul class="menu-list guildList"> |  | ||||||
|  |  | ||||||
|             </ul> |  | ||||||
|             <div class="aside-footer"> |  | ||||||
|                 <p class="menu-label"> |  | ||||||
|                     Options |  | ||||||
|                 </p> |  | ||||||
|                 <ul class="menu-list"> |  | ||||||
|                     <li> |  | ||||||
|                         <a class="show-modal" data-modal="dataManagerModal"> |  | ||||||
|                             <span class="icon"><i class="fas fa-exchange"></i></span> Import/Export |  | ||||||
|                         </a> |  | ||||||
|                         <a class="show-modal" data-modal="chooseTimezoneModal"> |  | ||||||
|                             <span class="icon"><i class="fas fa-map-marked"></i></span> Timezone |  | ||||||
|                         </a> |  | ||||||
|                         <a href="/login/discord/logout"> |  | ||||||
|                             <span class="icon"><i class="fas fa-sign-out"></i></span> Log out |  | ||||||
|                         </a> |  | ||||||
|                         <a href="https://discord.jellywx.com" class="feedback"> |  | ||||||
|                             <span class="icon"><i class="fab fa-discord"></i></span> Give feedback |  | ||||||
|                         </a> |  | ||||||
|                     </li> |  | ||||||
|                 </ul> |  | ||||||
|             </div> |  | ||||||
|         </aside> |  | ||||||
|     </div> |  | ||||||
|  |  | ||||||
|     <div class="dashboard-sidebar mobile-sidebar is-hidden-desktop" id="mobileSidebar"> |  | ||||||
|         <a href="/"> |  | ||||||
|             <div class="brand"> |  | ||||||
|                 <img src="/static/img/logo_nobg.webp" alt="Reminder bot logo" |  | ||||||
|                      class="dashboard-brand"> |  | ||||||
|             </div> |  | ||||||
|         </a> |  | ||||||
|         <svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 1440 160"> |  | ||||||
|             <g transform="scale(1, 0.5)"> |  | ||||||
|                 <path fill="#8fb677" fill-opacity="1" |  | ||||||
|                       d="M0,192L60,170.7C120,149,240,107,360,96C480,85,600,107,720,138.7C840,171,960,213,1080,197.3C1200,181,1320,107,1380,69.3L1440,32L1440,0L1380,0C1320,0,1200,0,1080,0C960,0,840,0,720,0C600,0,480,0,360,0C240,0,120,0,60,0L0,0Z"></path> |  | ||||||
|             </g> |  | ||||||
|         </svg> |  | ||||||
|         <aside class="menu"> |  | ||||||
|             <p class="menu-label"> |  | ||||||
|                 Servers |  | ||||||
|             </p> |  | ||||||
|             <ul class="menu-list guildList"> |  | ||||||
|  |  | ||||||
|             </ul> |  | ||||||
|             <div class="aside-footer"> |  | ||||||
|                 <p class="menu-label"> |  | ||||||
|                     Settings |  | ||||||
|                 </p> |  | ||||||
|                 <ul class="menu-list"> |  | ||||||
|                     <li> |  | ||||||
|                         <a class="show-modal" data-modal="dataManagerModal"> |  | ||||||
|                             <span class="icon"><i class="fas fa-exchange"></i></span> Import/Export |  | ||||||
|                         </a> |  | ||||||
|                         <a class="show-modal" data-modal="chooseTimezoneModal"> |  | ||||||
|                             <span class="icon"><i class="fas fa-map-marked"></i></span> Timezone |  | ||||||
|                         </a> |  | ||||||
|                         <a href="/login/discord/logout"> |  | ||||||
|                             <span class="icon"><i class="fas fa-sign-out"></i></span> Log out |  | ||||||
|                         </a> |  | ||||||
|                         <a href="https://discord.jellywx.com/" class="feedback"> |  | ||||||
|                             <span class="icon"><i class="fab fa-discord"></i></span> Give feedback |  | ||||||
|                         </a> |  | ||||||
|                     </li> |  | ||||||
|                 </ul> |  | ||||||
|             </div> |  | ||||||
|         </aside> |  | ||||||
|     </div> |  | ||||||
|  |  | ||||||
|     <!-- main content --> |  | ||||||
|     <div class="column is-main-content"> |  | ||||||
|         <p class="title pageTitle"></p> |  | ||||||
|         <section id="welcome"> |  | ||||||
|             <div class="has-text-centered"> |  | ||||||
|                 <p class="title">Welcome!</p> |  | ||||||
|                 <p class="subtitle is-hidden-touch">Select an option from the side to get started</p> |  | ||||||
|                 <p class="subtitle is-hidden-desktop">Press the <span class="icon"><i class="fal fa-bars"></i></span> to get started</p> |  | ||||||
|             </div> |  | ||||||
|         </section> |  | ||||||
|         <section id="reminders" class="is-hidden"> |  | ||||||
|             {% include "reminder_dashboard/reminder_dashboard" %} |  | ||||||
|         </section> |  | ||||||
|         <section id="reminder-errors" class="is-hidden"> |  | ||||||
|             {% include "reminder_dashboard/reminder_errors" %} |  | ||||||
|         </section> |  | ||||||
|         <section id="guild-error" class="is-hidden"> |  | ||||||
|             {% include "reminder_dashboard/guild_error" %} |  | ||||||
|         </section> |  | ||||||
|         <section id="user-error" class="is-hidden"> |  | ||||||
|             {% include "reminder_dashboard/user_error" %} |  | ||||||
|         </section> |  | ||||||
|     </div> |  | ||||||
|     <!-- /main content --> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <template id="embedFieldTemplate"> |  | ||||||
|     <div data-inlined="1" class="embed-field-box"> |  | ||||||
|         <div class="is-flex"> |  | ||||||
|             <label> |  | ||||||
|                 <span class="is-sr-only">Field Title</span> |  | ||||||
|                 <textarea class="discord-field-title field-input message-input autoresize" |  | ||||||
|                           placeholder="Field Title..." rows="1" |  | ||||||
|                           maxlength="256" name="embed_field_title[]"></textarea> |  | ||||||
|             </label> |  | ||||||
|             <button class="button is-small inline-btn"> |  | ||||||
|                 <span class="is-sr-only">Toggle field inline</span><i class="fas fa-arrows-h"></i> |  | ||||||
|             </button> |  | ||||||
|         </div> |  | ||||||
|  |  | ||||||
|         <label> |  | ||||||
|             <span class="is-sr-only">Field Value</span> |  | ||||||
|             <textarea |  | ||||||
|                     class="discord-field-value field-input message-input autoresize" |  | ||||||
|                     placeholder="Field Value..." |  | ||||||
|                     maxlength="1024" name="embed_field_value[]" |  | ||||||
|                     rows="1"></textarea> |  | ||||||
|         </label> |  | ||||||
|     </div> |  | ||||||
| </template> |  | ||||||
|  |  | ||||||
| <template id="guildListEntry"> |  | ||||||
|     <li> |  | ||||||
|         <a class="switch-pane" data-pane="guild"> |  | ||||||
|             <span class="icon"><i class="fas fa-map-pin"></i></span> <span class="guild-name">%guildname%</span> |  | ||||||
|         </a> |  | ||||||
|     </li> |  | ||||||
| </template> |  | ||||||
|  |  | ||||||
| <template id="guildReminder"> |  | ||||||
|     {% include "reminder_dashboard/guild_reminder" %} |  | ||||||
| </template> |  | ||||||
|  |  | ||||||
| <script src="/static/js/iro.js"></script> |  | ||||||
| <script src="/static/js/dtsel.js"></script> |  | ||||||
|  |  | ||||||
| <script src="/static/js/interval.js?v{{ version }}"></script> |  | ||||||
| <script src="/static/js/timezone.js?v{{ version }}" defer></script> |  | ||||||
| <script src="/static/js/main.js?v{{ version }}" defer></script> |  | ||||||
|  |  | ||||||
| </body> |  | ||||||
| </html> |  | ||||||
| @@ -1,17 +0,0 @@ | |||||||
| <div class="hero is-fullheight"> |  | ||||||
|     <div class="hero-body"> |  | ||||||
|         <div class="container has-text-centered"> |  | ||||||
|             <p class="title"> |  | ||||||
|                 We couldn't get this server's data |  | ||||||
|             </p> |  | ||||||
|             <p class="subtitle"> |  | ||||||
|                 Please check Reminder Bot is in the server, and has correct permissions. |  | ||||||
|             </p> |  | ||||||
|             <a class="button is-size-4 is-rounded is-success" href="https://invite.reminder-bot.com"> |  | ||||||
|                 <p class="is-size-4"> |  | ||||||
|                     <span>Add to Server</span> <span class="icon"><i class="fas fa-chevron-right"></i></span> |  | ||||||
|                 </p> |  | ||||||
|             </a> |  | ||||||
|         </div> |  | ||||||
|     </div> |  | ||||||
| </div> |  | ||||||
| @@ -1,269 +0,0 @@ | |||||||
| <div class="reminderContent {% if creating %}creator{% endif %}"> |  | ||||||
|     <div class="columns is-mobile column reminder-topbar"> |  | ||||||
|         {% if not creating %} |  | ||||||
|         <div class="invert-collapses channel-bar"> |  | ||||||
|             #channel |  | ||||||
|         </div> |  | ||||||
|         {% endif %} |  | ||||||
|         <div class="name-bar"> |  | ||||||
|             <div class="field"> |  | ||||||
|                 <div class="control"> |  | ||||||
|                     <label class="label sr-only">Reminder Name</label> |  | ||||||
|                     <input class="input" type="text" name="name" placeholder="Reminder Name" maxlength="100"> |  | ||||||
|                 </div> |  | ||||||
|             </div> |  | ||||||
|         </div> |  | ||||||
|         <div class="hide-button-bar"> |  | ||||||
|             <button class="button hide-box"> |  | ||||||
|                 <span class="is-sr-only">Hide reminder</span><i class="fas fa-chevron-down"></i> |  | ||||||
|             </button> |  | ||||||
|         </div> |  | ||||||
|     </div> |  | ||||||
|     <div class="columns reminder-settings"> |  | ||||||
|         <div class="column discord-frame"> |  | ||||||
|             <article class="media"> |  | ||||||
|                 <figure class="media-left"> |  | ||||||
|                     <p class="image is-32x32 customizable"> |  | ||||||
|                         <a> |  | ||||||
|                             <img class="is-rounded avatar" src="/static/img/bg.webp" alt="Image for discord avatar"> |  | ||||||
|                         </a> |  | ||||||
|                     </p> |  | ||||||
|                 </figure> |  | ||||||
|                 <div class="media-content"> |  | ||||||
|                     <div class="content"> |  | ||||||
|                         <div class="discord-message-header"> |  | ||||||
|                             <label class="is-sr-only">Username Override</label> |  | ||||||
|                             <input class="discord-username message-input" placeholder="Username Override" |  | ||||||
|                                    maxlength="32" name="username"> |  | ||||||
|                         </div> |  | ||||||
|                         <label class="is-sr-only">Message</label> |  | ||||||
|                         <textarea class="message-input autoresize discord-content" |  | ||||||
|                                   placeholder="Message Content..." |  | ||||||
|                                   maxlength="2000" name="content" rows="1"></textarea> |  | ||||||
|  |  | ||||||
|                         <div class="discord-embed"> |  | ||||||
|                             <div class="embed-body"> |  | ||||||
|                                 <button class="change-color button is-rounded is-small"> |  | ||||||
|                                     <span class="is-sr-only">Choose embed color</span><i class="fas fa-eye-dropper"></i> |  | ||||||
|                                 </button> |  | ||||||
|                                 <div class="a"> |  | ||||||
|                                     <div class="embed-author-box"> |  | ||||||
|                                         <div class="a"> |  | ||||||
|                                             <p class="image is-24x24 customizable"> |  | ||||||
|                                                 <a> |  | ||||||
|                                                     <img class="is-rounded embed_author_url" src="/static/img/bg.webp" alt="Image for embed author"> |  | ||||||
|                                                 </a> |  | ||||||
|                                             </p> |  | ||||||
|                                         </div> |  | ||||||
|  |  | ||||||
|                                         <div class="b"> |  | ||||||
|                                             <label class="is-sr-only" for="embedAuthor">Embed Author</label> |  | ||||||
|                                             <textarea |  | ||||||
|                                                     class="discord-embed-author message-input autoresize" |  | ||||||
|                                                     placeholder="Embed Author..." rows="1" maxlength="256" |  | ||||||
|                                                     name="embed_author"></textarea> |  | ||||||
|                                         </div> |  | ||||||
|                                     </div> |  | ||||||
|  |  | ||||||
|                                     <label class="is-sr-only" for="embedTitle">Embed Title</label> |  | ||||||
|                                     <textarea class="discord-title message-input  autoresize" |  | ||||||
|                                               placeholder="Embed Title..." |  | ||||||
|                                               maxlength="256" rows="1" |  | ||||||
|                                               name="embed_title"></textarea> |  | ||||||
|                                     <br> |  | ||||||
|                                     <label class="is-sr-only" for="embedDescription">Embed Description</label> |  | ||||||
|                                     <textarea class="discord-description message-input autoresize " |  | ||||||
|                                               placeholder="Embed Description..." |  | ||||||
|                                               maxlength="4096" name="embed_description" |  | ||||||
|                                               rows="1"></textarea> |  | ||||||
|                                     <br> |  | ||||||
|  |  | ||||||
|                                     <div class="embed-multifield-box"> |  | ||||||
|                                         <div data-inlined="1" class="embed-field-box"> |  | ||||||
|                                             <label class="is-sr-only" for="embedFieldTitle">Field Title</label> |  | ||||||
|                                             <div class="is-flex"> |  | ||||||
|                                                 <textarea class="discord-field-title field-input message-input autoresize" |  | ||||||
|                                                           placeholder="Field Title..." rows="1" |  | ||||||
|                                                           maxlength="256" name="embed_field_title[]"></textarea> |  | ||||||
|                                                 <button class="button is-small inline-btn"> |  | ||||||
|                                                     <span class="is-sr-only">Toggle field inline</span><i class="fas fa-arrows-h"></i> |  | ||||||
|                                                 </button> |  | ||||||
|                                             </div> |  | ||||||
|  |  | ||||||
|                                             <label class="is-sr-only" for="embedFieldValue">Field Value</label> |  | ||||||
|                                             <textarea |  | ||||||
|                                                     class="discord-field-value field-input message-input autoresize " |  | ||||||
|                                                     placeholder="Field Value..." |  | ||||||
|                                                     maxlength="1024" name="embed_field_value[]" |  | ||||||
|                                                     rows="1"></textarea> |  | ||||||
|                                         </div> |  | ||||||
|                                     </div> |  | ||||||
|                                 </div> |  | ||||||
|  |  | ||||||
|                                 <div class="b"> |  | ||||||
|                                     <p class="image thumbnail customizable"> |  | ||||||
|                                         <a> |  | ||||||
|                                             <img class="embed_thumbnail_url" src="/static/img/bg.webp" alt="Square thumbnail embedded image"> |  | ||||||
|                                         </a> |  | ||||||
|                                     </p> |  | ||||||
|                                 </div> |  | ||||||
|                             </div> |  | ||||||
|  |  | ||||||
|                             <p class="image is-400x300 customizable"> |  | ||||||
|                                 <a> |  | ||||||
|                                     <img class="embed_image_url" src="/static/img/bg.webp" alt="Large embedded image"> |  | ||||||
|                                 </a> |  | ||||||
|                             </p> |  | ||||||
|  |  | ||||||
|                             <div class="embed-footer-box"> |  | ||||||
|                                 <p class="image is-20x20 customizable"> |  | ||||||
|                                     <a> |  | ||||||
|                                         <img class="is-rounded embed_footer_url" src="/static/img/bg.webp" alt="Footer profile-like image"> |  | ||||||
|                                     </a> |  | ||||||
|                                 </p> |  | ||||||
|  |  | ||||||
|                                 <label class="is-sr-only" for="embedFooter">Embed Footer text</label> |  | ||||||
|                                 <textarea class="discord-embed-footer message-input autoresize " |  | ||||||
|                                           placeholder="Embed Footer..." |  | ||||||
|                                           maxlength="2048" name="embed_footer" rows="1"></textarea> |  | ||||||
|                             </div> |  | ||||||
|                         </div> |  | ||||||
|                     </div> |  | ||||||
|                 </div> |  | ||||||
|             </article> |  | ||||||
|         </div> |  | ||||||
|         <div class="column settings"> |  | ||||||
|             <div class="field channel-field"> |  | ||||||
|                 <div class="collapses"> |  | ||||||
|                     <label class="label" for="channelOption">Channel*</label> |  | ||||||
|                 </div> |  | ||||||
|                 <div class="control has-icons-left"> |  | ||||||
|                     <div class="select"> |  | ||||||
|                         <select name="channel" class="channel-selector"> |  | ||||||
|                         </select> |  | ||||||
|                     </div> |  | ||||||
|                     <div class="icon is-small is-left"> |  | ||||||
|                         <i class="fas fa-hashtag"></i> |  | ||||||
|                     </div> |  | ||||||
|                 </div> |  | ||||||
|             </div> |  | ||||||
|  |  | ||||||
|             <div class="field"> |  | ||||||
|                 <div class="control"> |  | ||||||
|                     <label class="label collapses"> |  | ||||||
|                         Time* |  | ||||||
|                         <input class="input prefill-now" type="datetime-local" step="1" name="time"> |  | ||||||
|                     </label> |  | ||||||
|                 </div> |  | ||||||
|             </div> |  | ||||||
|  |  | ||||||
|             <div class="collapses split-controls"> |  | ||||||
|                 <div> |  | ||||||
|                     <div class="patreon-only"> |  | ||||||
|                         <div class="patreon-invert foreground"> |  | ||||||
|                             Intervals available on <a href="https://patreon.com/jellywx">Patreon</a> or <a href="https://gitea.jellypro.xyz/jude/reminder-bot">self-hosting</a> |  | ||||||
|                         </div> |  | ||||||
|                         <div class="field"> |  | ||||||
|                             <label class="label">Interval <a class="foreground" href="/help/intervals"><i class="fas fa-question-circle"></i></a></label> |  | ||||||
|                             <div class="control intervalSelector"> |  | ||||||
|                                 <div class="input interval-group"> |  | ||||||
|                                     <div class="interval-group-left"> |  | ||||||
|                                         <span class="no-break"> |  | ||||||
|                                             <label> |  | ||||||
|                                                 <span class="is-sr-only">Interval months</span> |  | ||||||
|                                                 <input class="w2" type="text" pattern="\d*" name="interval_months" maxlength="2" placeholder=""> <span class="half-rem"></span> months, <span class="half-rem"></span> |  | ||||||
|                                             </label> |  | ||||||
|                                             <label> |  | ||||||
|                                                 <span class="is-sr-only">Interval days</span> |  | ||||||
|                                                 <input class="w3" type="text" pattern="\d*" name="interval_days" maxlength="4" placeholder=""> <span class="half-rem"></span> days, <span class="half-rem"></span> |  | ||||||
|                                             </label> |  | ||||||
|                                         </span> |  | ||||||
|                                         <span class="no-break"> |  | ||||||
|                                             <label> |  | ||||||
|                                                 <span class="is-sr-only">Interval hours</span> |  | ||||||
|                                                 <input class="w2" type="text" pattern="\d*" name="interval_hours" maxlength="2" placeholder="HH">: |  | ||||||
|                                             </label> |  | ||||||
|                                             <label> |  | ||||||
|                                                 <span class="is-sr-only">Interval minutes</span> |  | ||||||
|                                                 <input class="w2" type="text" pattern="\d*" name="interval_minutes" maxlength="2" placeholder="MM">: |  | ||||||
|                                             </label> |  | ||||||
|                                             <label> |  | ||||||
|                                                 <span class="is-sr-only">Interval seconds</span> |  | ||||||
|                                                 <input class="w2" type="text" pattern="\d*" name="interval_seconds" maxlength="2" placeholder="SS"> |  | ||||||
|                                             </label> |  | ||||||
|                                         </span> |  | ||||||
|                                     </div> |  | ||||||
|                                     <button class="clear"><span class="is-sr-only">Clear interval</span><span class="icon"><i class="fas fa-trash"></i></span></button> |  | ||||||
|                                 </div> |  | ||||||
|                             </div> |  | ||||||
|                         </div> |  | ||||||
|  |  | ||||||
|                         <div class="field"> |  | ||||||
|                             <div class="control"> |  | ||||||
|                                 <label class="label"> |  | ||||||
|                                     Expiration |  | ||||||
|                                     <input class="input" type="datetime-local" step="1" name="expiration"> |  | ||||||
|                                 </label> |  | ||||||
|                             </div> |  | ||||||
|                         </div> |  | ||||||
|                     </div> |  | ||||||
|  |  | ||||||
|                     <div class="columns is-mobile tts-row"> |  | ||||||
|                         <div class="column has-text-centered"> |  | ||||||
|                             <div class="is-boxed"> |  | ||||||
|                                 <label class="label">Enable TTS <input type="checkbox" name="tts"></label> |  | ||||||
|                             </div> |  | ||||||
|                         </div> |  | ||||||
|                         <div class="column has-text-centered"> |  | ||||||
|                             <div class="file is-small is-boxed"> |  | ||||||
|                                 <label class="file-label"> |  | ||||||
|                                     <input class="file-input" type="file" name="attachment"> |  | ||||||
|                                     <span class="file-cta"> |  | ||||||
|                                         <span class="file-label"> |  | ||||||
|                                             Add Attachment |  | ||||||
|                                         </span> |  | ||||||
|                                         <span class="file-icon"> |  | ||||||
|                                             <i class="fas fa-upload"></i> |  | ||||||
|                                         </span> |  | ||||||
|                                     </span> |  | ||||||
|                                 </label> |  | ||||||
|                             </div> |  | ||||||
|                         </div> |  | ||||||
|                     </div> |  | ||||||
|                 </div> |  | ||||||
|             </div> |  | ||||||
|         </div> |  | ||||||
|     </div> |  | ||||||
|     {% if creating %} |  | ||||||
|         <div class="button-row"> |  | ||||||
|             <div class="button-row-reminder"> |  | ||||||
|                 <button class="button is-success" id="createReminder"> |  | ||||||
|                     <span>Create Reminder</span> <span class="icon"><i class="fas fa-sparkles"></i></span> |  | ||||||
|                 </button> |  | ||||||
|             </div> |  | ||||||
|             <div class="button-row-template"> |  | ||||||
|                 <div> |  | ||||||
|                     <button class="button is-success is-outlined" id="createTemplate"> |  | ||||||
|                         <span>Create Template</span> <span class="icon"><i class="fas fa-file-spreadsheet"></i></span> |  | ||||||
|                     </button> |  | ||||||
|                 </div> |  | ||||||
|                 <div> |  | ||||||
|                     <button class="button is-outlined show-modal is-pulled-right" data-modal="chooseTemplateModal"> |  | ||||||
|                         Load Template |  | ||||||
|                     </button> |  | ||||||
|                 </div> |  | ||||||
|             </div> |  | ||||||
|         </div> |  | ||||||
|     {% else %} |  | ||||||
|         <div class="button-row-edit"> |  | ||||||
|             <button class="button is-success save-btn"> |  | ||||||
|                 <span>Save</span> <span class="icon"><i class="fas fa-save"></i></span> |  | ||||||
|             </button> |  | ||||||
|             <button class="button is-warning disable-enable"> |  | ||||||
|             </button> |  | ||||||
|             <button class="button is-danger delete-reminder"> |  | ||||||
|                 Delete |  | ||||||
|             </button> |  | ||||||
|         </div> |  | ||||||
|     {% endif %} |  | ||||||
| </div> |  | ||||||
| @@ -1,52 +0,0 @@ | |||||||
| <div class="create-reminder"> |  | ||||||
|     <strong>Create Reminder</strong> |  | ||||||
|     <div id="reminderCreator"> |  | ||||||
|         {% set creating = true %} |  | ||||||
|         {% include "reminder_dashboard/guild_reminder" %} |  | ||||||
|         {% set creating = false %} |  | ||||||
|     </div> |  | ||||||
|     <br> |  | ||||||
|  |  | ||||||
|     <div class="field"> |  | ||||||
|         <div class="columns is-mobile"> |  | ||||||
|             <div class="column"> |  | ||||||
|                 <strong>Reminders</strong> |  | ||||||
|             </div> |  | ||||||
|             <div class="column is-narrow"> |  | ||||||
|                 <div class="control has-icons-left"> |  | ||||||
|                     <div class="select is-small"> |  | ||||||
|                         <select id="orderBy"> |  | ||||||
|                             <option value="time" selected>Time</option> |  | ||||||
|                             <option value="name">Name</option> |  | ||||||
|                             <option value="channel">Channel</option> |  | ||||||
|                         </select> |  | ||||||
|                     </div> |  | ||||||
|                     <div class="icon is-small is-left"> |  | ||||||
|                         <i class="fas fa-sort-amount-down"></i> |  | ||||||
|                     </div> |  | ||||||
|                 </div> |  | ||||||
|             </div> |  | ||||||
|             <div class="column is-narrow"> |  | ||||||
|                 <div class="control has-icons-left"> |  | ||||||
|                     <div class="select is-small"> |  | ||||||
|                         <select id="expandAll"> |  | ||||||
|                             <option value="" selected></option> |  | ||||||
|                             <option value="expand">Expand All</option> |  | ||||||
|                             <option value="collapse">Collapse All</option> |  | ||||||
|                         </select> |  | ||||||
|                     </div> |  | ||||||
|                     <div class="icon is-small is-left"> |  | ||||||
|                         <i class="fas fa-expand-arrows"></i> |  | ||||||
|                     </div> |  | ||||||
|                 </div> |  | ||||||
|             </div> |  | ||||||
|         </div> |  | ||||||
|     </div> |  | ||||||
|  |  | ||||||
|     <div id="guildReminders"> |  | ||||||
|  |  | ||||||
|     </div> |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <script src="/static/js/sort.js"></script> |  | ||||||
| <script src="/static/js/expand.js"></script> |  | ||||||
| @@ -1,5 +0,0 @@ | |||||||
| <div> |  | ||||||
|  |  | ||||||
| </div> |  | ||||||
|  |  | ||||||
| <!--<script src="/static/js/reminder_errors.js"></script>--> |  | ||||||
| @@ -1,12 +0,0 @@ | |||||||
| <div class="hero is-fullheight"> |  | ||||||
|     <div class="hero-body"> |  | ||||||
|         <div class="container has-text-centered"> |  | ||||||
|             <p class="title"> |  | ||||||
|                 You do not have permissions for this server |  | ||||||
|             </p> |  | ||||||
|             <p class="subtitle"> |  | ||||||
|                 Ask an admin to grant you the "Manage Messages" permission. |  | ||||||
|             </p> |  | ||||||
|         </div> |  | ||||||
|     </div> |  | ||||||
| </div> |  | ||||||
		Reference in New Issue
	
	Block a user