diff --git a/resources/scripts/.eslintrc.yml b/resources/scripts/.eslintrc.yml index b95cb2916..3d5a324c1 100644 --- a/resources/scripts/.eslintrc.yml +++ b/resources/scripts/.eslintrc.yml @@ -36,6 +36,9 @@ rules: comma-dangle: - error - always-multiline + array-bracket-spacing: + - warn + - always "react-hooks/rules-of-hooks": - error "react-hooks/exhaustive-deps": 0 @@ -45,6 +48,8 @@ rules: "@typescript-eslint/no-unused-vars": 0 "@typescript-eslint/no-explicit-any": 0 "@typescript-eslint/no-non-null-assertion": 0 + # @todo this would be nice to have, but don't want to deal with the warning spam at the moment. + "@typescript-eslint/explicit-module-boundary-types": 0 no-restricted-imports: - error - paths: diff --git a/resources/scripts/TransitionRouter.tsx b/resources/scripts/TransitionRouter.tsx index 78fd164d1..abe421d41 100644 --- a/resources/scripts/TransitionRouter.tsx +++ b/resources/scripts/TransitionRouter.tsx @@ -2,15 +2,11 @@ import React from 'react'; import { Route } from 'react-router'; import { CSSTransition, TransitionGroup } from 'react-transition-group'; -type Props = Readonly<{ - children: React.ReactNode; -}>; - -export default ({ children }: Props) => ( +const TransitionRouter: React.FC = ({ children }) => ( ( - +
{children}
@@ -19,3 +15,5 @@ export default ({ children }: Props) => ( )} /> ); + +export default TransitionRouter; diff --git a/resources/scripts/components/App.tsx b/resources/scripts/components/App.tsx index 6a726fda3..a90786ae0 100644 --- a/resources/scripts/components/App.tsx +++ b/resources/scripts/components/App.tsx @@ -8,9 +8,9 @@ import ServerRouter from '@/routers/ServerRouter'; import AuthenticationRouter from '@/routers/AuthenticationRouter'; import { Provider } from 'react-redux'; import { SiteSettings } from '@/state/settings'; -import { DefaultTheme, ThemeProvider } from 'styled-components'; import ProgressBar from '@/components/elements/ProgressBar'; import NotFound from '@/components/screens/NotFound'; +import tw from 'twin.macro'; interface ExtendedWindow extends Window { SiteConfiguration?: SiteSettings; @@ -28,16 +28,6 @@ interface ExtendedWindow extends Window { }; } -const theme: DefaultTheme = { - breakpoints: { - xs: 0, - sm: 576, - md: 768, - lg: 992, - xl: 1200, - }, -}; - const App = () => { const { PterodactylUser, SiteConfiguration } = (window as ExtendedWindow); if (PterodactylUser && !store.getState().user.data) { @@ -58,23 +48,21 @@ const App = () => { } return ( - - - - -
- - - - - - - - -
-
-
-
+ + + +
+ + + + + + + + +
+
+
); }; diff --git a/resources/scripts/components/FlashMessageRender.tsx b/resources/scripts/components/FlashMessageRender.tsx index cb0b37026..0af9d47e8 100644 --- a/resources/scripts/components/FlashMessageRender.tsx +++ b/resources/scripts/components/FlashMessageRender.tsx @@ -6,10 +6,9 @@ import { ApplicationStore } from '@/state'; type Props = Readonly<{ byKey?: string; spacerClass?: string; - className?: string; }>; -export default ({ className, spacerClass, byKey }: Props) => { +export default ({ spacerClass, byKey }: Props) => { const flashes = useStoreState((state: State) => state.flashes.items); let filtered = flashes; @@ -22,11 +21,11 @@ export default ({ className, spacerClass, byKey }: Props) => { } return ( -
+
{ filtered.map((flash, index) => ( - {index > 0 &&
} + {index > 0 &&
} {flash.message} diff --git a/resources/scripts/components/auth/LoginFormContainer.tsx b/resources/scripts/components/auth/LoginFormContainer.tsx index afc8093fc..541763dc0 100644 --- a/resources/scripts/components/auth/LoginFormContainer.tsx +++ b/resources/scripts/components/auth/LoginFormContainer.tsx @@ -1,8 +1,9 @@ import React, { forwardRef } from 'react'; import { Form } from 'formik'; -import styled from 'styled-components'; +import styled from 'styled-components/macro'; import { breakpoint } from 'styled-components-breakpoint'; import FlashMessageRender from '@/components/FlashMessageRender'; +import tw from 'twin.macro'; type Props = React.DetailedHTMLProps, HTMLFormElement> & { title?: string; diff --git a/resources/scripts/components/dashboard/AccountOverviewContainer.tsx b/resources/scripts/components/dashboard/AccountOverviewContainer.tsx index 4b052e850..e98ddd4a6 100644 --- a/resources/scripts/components/dashboard/AccountOverviewContainer.tsx +++ b/resources/scripts/components/dashboard/AccountOverviewContainer.tsx @@ -3,9 +3,10 @@ import ContentBox from '@/components/elements/ContentBox'; import UpdatePasswordForm from '@/components/dashboard/forms/UpdatePasswordForm'; import UpdateEmailAddressForm from '@/components/dashboard/forms/UpdateEmailAddressForm'; import ConfigureTwoFactorForm from '@/components/dashboard/forms/ConfigureTwoFactorForm'; -import styled from 'styled-components'; -import { breakpoint } from 'styled-components-breakpoint'; import PageContentBlock from '@/components/elements/PageContentBlock'; +import tw from 'twin.macro'; +import { breakpoint } from '@/theme'; +import styled from 'styled-components/macro'; const Container = styled.div` ${tw`flex flex-wrap my-10`}; @@ -31,13 +32,13 @@ export default () => { - + diff --git a/resources/scripts/components/dashboard/search/SearchModal.tsx b/resources/scripts/components/dashboard/search/SearchModal.tsx index 9c2a4eb9c..9c1777df3 100644 --- a/resources/scripts/components/dashboard/search/SearchModal.tsx +++ b/resources/scripts/components/dashboard/search/SearchModal.tsx @@ -11,7 +11,8 @@ import { Server } from '@/api/server/getServer'; import { ApplicationStore } from '@/state'; import { httpErrorToHuman } from '@/api/http'; import { Link } from 'react-router-dom'; -import styled from 'styled-components'; +import styled from 'styled-components/macro'; +import tw from 'twin.macro'; type Props = RequiredModalProps; @@ -102,7 +103,7 @@ export default ({ ...props }: Props) => { {servers.length > 0 && -
+
{ servers.map(server => ( { onClick={() => props.onDismissed()} >
-

{server.name}

-

+

{server.name}

+

{ server.allocations.filter(alloc => alloc.default).map(allocation => ( {allocation.alias || allocation.ip}:{allocation.port} @@ -120,8 +121,8 @@ export default ({ ...props }: Props) => { }

-
- +
+ {server.node}
diff --git a/resources/scripts/components/elements/AceEditor.tsx b/resources/scripts/components/elements/AceEditor.tsx index 3b01307fa..4fc71cb3f 100644 --- a/resources/scripts/components/elements/AceEditor.tsx +++ b/resources/scripts/components/elements/AceEditor.tsx @@ -3,7 +3,7 @@ import useRouter from 'use-react-router'; import { ServerContext } from '@/state/server'; import ace, { Editor } from 'brace'; import getFileContents from '@/api/server/files/getFileContents'; -import styled from 'styled-components'; +import styled from 'styled-components/macro'; // @ts-ignore require('brace/ext/modelist'); @@ -113,7 +113,7 @@ export default ({ style, initialContent, initialModePath, fetchContent, onConten return (
-
+
{(label || description) && -
+
{label && + > + {label} + } {description &&

diff --git a/resources/scripts/components/screens/ScreenBlock.tsx b/resources/scripts/components/screens/ScreenBlock.tsx index cff1c5501..f20836493 100644 --- a/resources/scripts/components/screens/ScreenBlock.tsx +++ b/resources/scripts/components/screens/ScreenBlock.tsx @@ -4,7 +4,8 @@ import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; import { faArrowLeft } from '@fortawesome/free-solid-svg-icons/faArrowLeft'; import { faSyncAlt } from '@fortawesome/free-solid-svg-icons/faSyncAlt'; import classNames from 'classnames'; -import styled from 'styled-components'; +import styled from 'styled-components/macro'; +import tw from 'twin.macro'; interface BaseProps { title: string; @@ -42,10 +43,10 @@ const ActionButton = styled.button` export default ({ title, image, message, onBack, onRetry }: Props) => ( -

-
+
+
{(typeof onBack === 'function' || typeof onRetry === 'function') && -
+
onRetry ? onRetry() : (onBack ? onBack() : null)} className={classNames('btn btn-primary', { 'hover:spin': !!onRetry })} @@ -54,9 +55,9 @@ export default ({ title, image, message, onBack, onRetry }: Props) => (
} - -

{title}

-

+ +

{title}

+

{message}

diff --git a/resources/scripts/components/screens/ServerError.tsx b/resources/scripts/components/screens/ServerError.tsx index 42e82e5a5..7851cf72f 100644 --- a/resources/scripts/components/screens/ServerError.tsx +++ b/resources/scripts/components/screens/ServerError.tsx @@ -1,5 +1,5 @@ import React from 'react'; -import styled from 'styled-components'; +import styled from 'styled-components/macro'; import ScreenBlock from '@/components/screens/ScreenBlock'; interface Props { diff --git a/resources/scripts/components/server/Console.tsx b/resources/scripts/components/server/Console.tsx index 54d6e9620..5deb90a6d 100644 --- a/resources/scripts/components/server/Console.tsx +++ b/resources/scripts/components/server/Console.tsx @@ -3,10 +3,11 @@ import { ITerminalOptions, Terminal } from 'xterm'; import * as TerminalFit from 'xterm/lib/addons/fit/fit'; import SpinnerOverlay from '@/components/elements/SpinnerOverlay'; import { ServerContext } from '@/state/server'; -import styled from 'styled-components'; +import styled from 'styled-components/macro'; import Can from '@/components/elements/Can'; import { usePermissions } from '@/plugins/usePermissions'; import classNames from 'classnames'; +import tw from 'twin.macro'; const theme = { background: 'transparent', diff --git a/resources/scripts/components/server/users/EditSubuserModal.tsx b/resources/scripts/components/server/users/EditSubuserModal.tsx index 104684f23..e37ca42d6 100644 --- a/resources/scripts/components/server/users/EditSubuserModal.tsx +++ b/resources/scripts/components/server/users/EditSubuserModal.tsx @@ -8,7 +8,7 @@ import { Actions, useStoreActions, useStoreState } from 'easy-peasy'; import { ApplicationStore } from '@/state'; import TitledGreyBox from '@/components/elements/TitledGreyBox'; import Checkbox from '@/components/elements/Checkbox'; -import styled from 'styled-components'; +import styled from 'styled-components/macro'; import classNames from 'classnames'; import createOrUpdateSubuser from '@/api/server/users/createOrUpdateSubuser'; import { ServerContext } from '@/state/server'; @@ -17,6 +17,7 @@ import FlashMessageRender from '@/components/FlashMessageRender'; import Can from '@/components/elements/Can'; import { usePermissions } from '@/plugins/usePermissions'; import { useDeepMemo } from '@/plugins/useDeepMemo'; +import tw from 'twin.macro'; type Props = { subuser?: Subuser; @@ -82,17 +83,17 @@ const EditSubuserModal = forwardRef(({ subuser, ...pr 'Create new subuser' } - + {(!user.rootAdmin && loggedInPermissions[0] !== '*') && -
-

+

+

Only permissions which your account is currently assigned may be selected when creating or modifying other users.

} {!subuser && -
+
(({ subuser, ...pr />
} -
+
{Object.keys(permissions).filter(key => key !== 'websocket').map((key, index) => ( -

{key}

+
+

{key}

{canEditUser && editablePermissions.indexOf(key) >= 0 && (({ subuser, ...pr } className={index !== 0 ? 'mt-4' : undefined} > -

+

{permissions[key].description}

{Object.keys(permissions[key].keys).map((pkey, index) => ( @@ -146,21 +147,21 @@ const EditSubuserModal = forwardRef(({ subuser, ...pr disabled: !canEditUser || editablePermissions.indexOf(`${key}.${pkey}`) < 0, })} > -
+
-
- +
+ {pkey} {permissions[key].keys[pkey].length > 0 && -

+

{permissions[key].keys[pkey]}

} @@ -171,7 +172,7 @@ const EditSubuserModal = forwardRef(({ subuser, ...pr ))}
-
+
diff --git a/resources/scripts/theme.ts b/resources/scripts/theme.ts index 8679ebaa7..9dd0b467d 100644 --- a/resources/scripts/theme.ts +++ b/resources/scripts/theme.ts @@ -3,8 +3,8 @@ import { BreakpointFunction, createBreakpoint } from 'styled-components-breakpoi type Breakpoints = 'xs' | 'sm' | 'md' | 'lg' | 'xl'; export const breakpoint: BreakpointFunction = createBreakpoint({ xs: 0, - sm: 640, + sm: 576, md: 768, - lg: 1024, - xl: 1280, + lg: 992, + xl: 1200, }); diff --git a/resources/styles/components/animations.css b/resources/styles/components/animations.css index bcbd60e40..3c3689198 100644 --- a/resources/styles/components/animations.css +++ b/resources/styles/components/animations.css @@ -22,7 +22,7 @@ div.route-transition-group { @apply .relative; & section { - @apply .absolute .w-full .pin-t .pin-l; + @apply .absolute .w-full .top-0 .left-0; } } /*! purgecss end ignore */ diff --git a/resources/styles/components/modal.css b/resources/styles/components/modal.css index 550191c6b..0919d9ba1 100644 --- a/resources/styles/components/modal.css +++ b/resources/styles/components/modal.css @@ -11,7 +11,7 @@ } & > .modal-close-icon { - @apply .absolute .pin-r .p-2 .text-white .cursor-pointer .opacity-50; + @apply .absolute .right-0 .p-2 .text-white .cursor-pointer .opacity-50; transition: opacity 150ms linear, transform 150ms ease-in; top: -2rem; diff --git a/tailwind.config.js b/tailwind.config.js index ddce8410c..775ecbd27 100644 --- a/tailwind.config.js +++ b/tailwind.config.js @@ -34,9 +34,11 @@ module.exports = { 'monospace', ], }, - }, - extend: { colors: { + transparent: 'transparent', + black: 'hsl(210, 27%, 10%)', + white: '#ffffff', + 'basically-white': '#fafafb', primary: { 50: 'hsl(202, 100%, 95%)', // lightest 100: 'hsl(204, 100%, 86%)', // lighter