Compare commits
197 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
ce0f27bb6e | ||
|
|
5c1a171832 | ||
|
|
fd8ab2f643 | ||
|
|
c750eed5ef | ||
|
|
c446bf1a1f | ||
|
|
0fdfa55067 | ||
|
|
c313eacefd | ||
|
|
649f7521bc | ||
|
|
7065b06c82 | ||
|
|
c4e01e482e | ||
|
|
e56c4b6b56 | ||
|
|
ce38163c6d | ||
|
|
0406d6d5ee | ||
|
|
4b92c52ea8 | ||
|
|
be770d282a | ||
|
|
0bf95483f9 | ||
|
|
60b88389a6 | ||
|
|
72832a5767 | ||
|
|
eb0313f551 | ||
|
|
c0ecc3d4c6 | ||
|
|
da7b478d81 | ||
|
|
2702c9a7cf | ||
|
|
3bda6c2b76 | ||
|
|
1a66bfb8be | ||
|
|
41be0e6362 | ||
|
|
e93a3a3b5f | ||
|
|
9945ef892e | ||
|
|
812de35149 | ||
|
|
035224b02b | ||
|
|
72aba9d8cd | ||
|
|
df7e833b84 | ||
|
|
aab1492934 | ||
|
|
1ae074db8f | ||
|
|
f21004e944 | ||
|
|
7c421cc52f | ||
|
|
d8e407ab22 | ||
|
|
37565284e6 | ||
|
|
b758df9f44 | ||
|
|
a735ae47c5 | ||
|
|
97d585dc17 | ||
|
|
7f3db9add1 | ||
|
|
6d6964f086 | ||
|
|
2a4012f73a | ||
|
|
9385315f03 | ||
|
|
ee824f0b27 | ||
|
|
792af504c7 | ||
|
|
cd3c062a24 | ||
|
|
a5f477c19b | ||
|
|
85164d79fc | ||
|
|
7aedc4111f | ||
|
|
d1f89847f5 | ||
|
|
57170847a1 | ||
|
|
45de715390 | ||
|
|
c29d6f58dd | ||
|
|
f0bae49830 | ||
|
|
c3ceae4dc6 | ||
|
|
d654fb39e5 | ||
|
|
7dc205fa66 | ||
|
|
91a249d953 | ||
|
|
356afda9c7 | ||
|
|
35f02a2296 | ||
|
|
16bcec0deb | ||
|
|
16ec57081b | ||
|
|
690f09fcf3 | ||
|
|
2f960169bb | ||
|
|
14a40d9f66 | ||
|
|
e5abd67f83 | ||
|
|
399ba7e2fc | ||
|
|
7780ae3d7a | ||
|
|
80d3f16473 | ||
|
|
a8c0dfcd0c | ||
|
|
6ee7d6ec8d | ||
|
|
544fae3808 | ||
|
|
4516dde1f4 | ||
|
|
a20c5f8d12 | ||
|
|
60e5c0d165 | ||
|
|
b7bf18250d | ||
|
|
93256b7a6a | ||
|
|
47a4437a01 | ||
|
|
92470c619e | ||
|
|
7cb3dfbd16 | ||
|
|
d69e4f41a1 | ||
|
|
4980254e89 | ||
|
|
5133286e04 | ||
|
|
ca2713a12c | ||
|
|
4981823c37 | ||
|
|
5d31e414f0 | ||
|
|
8ec2b9d0cd | ||
|
|
bd920dfc86 | ||
|
|
b5540a9958 | ||
|
|
778988de58 | ||
|
|
5b1437552d | ||
|
|
e8a8fbe6ac | ||
|
|
5c0a074219 | ||
|
|
58ec74bb68 | ||
|
|
6ac82bda40 | ||
|
|
2c6e86840a | ||
|
|
df85fc6b7d | ||
|
|
89804dafd1 | ||
|
|
98eaee1234 | ||
|
|
433edafddd | ||
|
|
e39d5741b6 | ||
|
|
21c08cbe63 | ||
|
|
ec92a1d89c | ||
|
|
0f2c5dbce2 | ||
|
|
8eae5a908c | ||
|
|
a37f0fdee6 | ||
|
|
08799aac18 | ||
|
|
06531e0fb8 | ||
|
|
0f56ead24f | ||
|
|
922caa76da | ||
|
|
0acb1f6b6d | ||
|
|
8d645ca404 | ||
|
|
a5c4f30f57 | ||
|
|
562a05adf5 | ||
|
|
de77e06b18 | ||
|
|
03c499d822 | ||
|
|
169d08f3b6 | ||
|
|
437807a9e0 | ||
|
|
4866fd74b5 | ||
|
|
426ba69afd | ||
|
|
74f87b570d | ||
|
|
fed5f6df52 | ||
|
|
5cc160473c | ||
|
|
4833157061 | ||
|
|
a0c8518d22 | ||
|
|
c0c816d3db | ||
|
|
ac47de72ee | ||
|
|
d631865f71 | ||
|
|
4ee6562e35 | ||
|
|
19f80b9b4c | ||
|
|
949deacd6d | ||
|
|
b0f4a91878 | ||
|
|
68f2e79056 | ||
|
|
43e68e1bbf | ||
|
|
5033323b7c | ||
|
|
7519b4a6b2 | ||
|
|
e6eedefec4 | ||
|
|
845d6a3d87 | ||
|
|
f75da289c2 | ||
|
|
063a6447c0 | ||
|
|
4dac730412 | ||
|
|
de6e0f645f | ||
|
|
b26ab50c8d | ||
|
|
423f8110b9 | ||
|
|
84ae49ed2a | ||
|
|
fb291c5411 | ||
|
|
901798055b | ||
|
|
d32d599098 | ||
|
|
76e02cf148 | ||
|
|
f19b4675ad | ||
|
|
4f1640b70a | ||
|
|
c1d17ec8b2 | ||
|
|
d2f1268520 | ||
|
|
b72afc2270 | ||
|
|
de0c625f88 | ||
|
|
29c9f3ecac | ||
|
|
a321095daf | ||
|
|
ced18da65a | ||
|
|
1a642ad7b4 | ||
|
|
838f196937 | ||
|
|
6af5166aa5 | ||
|
|
7935fb6616 | ||
|
|
ed567065b4 | ||
|
|
06035fb6f0 | ||
|
|
c1af0a087d | ||
|
|
6067c5dfcf | ||
|
|
bf7b9637f7 | ||
|
|
c552104413 | ||
|
|
6fd23cf6a0 | ||
|
|
e2f59383d6 | ||
|
|
8b92135a80 | ||
|
|
aef4a30512 | ||
|
|
ace8bd75e7 | ||
|
|
2e461b4e7a | ||
|
|
3f87e939c9 | ||
|
|
1d9dfc5102 | ||
|
|
80a94d3778 | ||
|
|
39d66faf4e | ||
|
|
c50e11c75b | ||
|
|
9a3ebb56cb | ||
|
|
1d1495453a | ||
|
|
26cfc485c2 | ||
|
|
83b4da282a | ||
|
|
ea972effb4 | ||
|
|
9686761c3d | ||
|
|
13a5a4a263 | ||
|
|
339919cfff | ||
|
|
2594a7caa5 | ||
|
|
2966be4fc4 | ||
|
|
5e21a7df9c | ||
|
|
64eb00f2ee | ||
|
|
00928ae709 | ||
|
|
bbb912479b | ||
|
|
5b16589360 | ||
|
|
39674fc769 | ||
|
|
bdaf70f26b |
@@ -2,5 +2,8 @@ Dockerfile
|
|||||||
.dockerignore
|
.dockerignore
|
||||||
node_modules
|
node_modules
|
||||||
npm-debug.log
|
npm-debug.log
|
||||||
README.md
|
*.md
|
||||||
.git
|
.git
|
||||||
|
.github
|
||||||
|
LICENSE
|
||||||
|
docs/
|
||||||
|
|||||||
7
.github/pull_request_template.md
vendored
7
.github/pull_request_template.md
vendored
@@ -14,10 +14,3 @@
|
|||||||
|
|
||||||
### Screenshot _(if applicable)_
|
### Screenshot _(if applicable)_
|
||||||
> If you've introduced any significant UI changes, please include a screenshot.
|
> If you've introduced any significant UI changes, please include a screenshot.
|
||||||
|
|
||||||
### Code Quality Checklist _(Please complete)_
|
|
||||||
- [ ] All changes are backwards compatible
|
|
||||||
- [ ] There are no (new) build warnings or errors
|
|
||||||
- [ ] _(If a new config option is added)_ Attribute is outlined in the schema and documented
|
|
||||||
- [ ] _(If a new dependency is added)_ Package is essential, and has been checked out for security or performance
|
|
||||||
- [ ] Bumps version, if new feature added
|
|
||||||
|
|||||||
2
.github/workflows/docker_dev.yml
vendored
2
.github/workflows/docker_dev.yml
vendored
@@ -16,7 +16,7 @@ on:
|
|||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
inputs:
|
inputs:
|
||||||
tags:
|
tags:
|
||||||
requierd: true
|
required: true
|
||||||
description: 'Tags to deploy to'
|
description: 'Tags to deploy to'
|
||||||
|
|
||||||
env:
|
env:
|
||||||
|
|||||||
@@ -2,12 +2,13 @@ FROM node:16-alpine
|
|||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
ENV NODE_ENV production
|
ENV NODE_ENV production
|
||||||
COPY /next.config.js ./
|
COPY /next.config.js ./
|
||||||
COPY /public ./public
|
COPY /public ./public
|
||||||
COPY /package.json ./package.json
|
COPY /package.json ./package.json
|
||||||
# Automatically leverage output traces to reduce image size. https://nextjs.org/docs/advanced-features/output-file-tracing
|
# Automatically leverage output traces to reduce image size. https://nextjs.org/docs/advanced-features/output-file-tracing
|
||||||
COPY /.next/standalone ./
|
COPY /.next/standalone ./
|
||||||
COPY /.next/static ./.next/static
|
COPY /.next/static ./.next/static
|
||||||
EXPOSE 7575
|
EXPOSE 7575
|
||||||
ENV PORT 7575
|
ENV PORT 7575
|
||||||
|
RUN apk add tzdata
|
||||||
VOLUME /app/data/configs
|
VOLUME /app/data/configs
|
||||||
CMD ["node", "server.js"]
|
CMD ["node", "server.js"]
|
||||||
|
|||||||
@@ -33,7 +33,7 @@ Homarr is a simple and lightweight homepage for your server, that helps you easi
|
|||||||
|
|
||||||
It integrates with the services you use to display information on the homepage (E.g. Show upcoming Sonarr/Radarr releases).
|
It integrates with the services you use to display information on the homepage (E.g. Show upcoming Sonarr/Radarr releases).
|
||||||
|
|
||||||
For a full list of integrations look at: [wiki/integrations](#).
|
For a full list of integrations look at: [wiki/integrations](https://github.com/ajnart/homarr/wiki/Integrations)
|
||||||
|
|
||||||
If you have any questions about Homarr or want to share information with us, please go to one of the following places:
|
If you have any questions about Homarr or want to share information with us, please go to one of the following places:
|
||||||
|
|
||||||
@@ -198,7 +198,4 @@ SOFTWARE.
|
|||||||
<i>Thank you for visiting! <b>For more information <a href="https://github.com/ajnart/homarr/wiki">read the wiki!</a></b></i>
|
<i>Thank you for visiting! <b>For more information <a href="https://github.com/ajnart/homarr/wiki">read the wiki!</a></b></i>
|
||||||
<br/>
|
<br/>
|
||||||
<br/>
|
<br/>
|
||||||
<a href="https://trackgit.com">
|
|
||||||
<img src="https://us-central1-trackgit-analytics.cloudfunctions.net/token/ping/l3khzc3a3pexzw5w5whl" alt="trackgit-views" />
|
|
||||||
</a>
|
|
||||||
</p>
|
</p>
|
||||||
|
|||||||
@@ -1,2 +1,2 @@
|
|||||||
export const REPO_URL = 'ajnart/homarr';
|
export const REPO_URL = 'ajnart/homarr';
|
||||||
export const CURRENT_VERSION = 'v0.6.0';
|
export const CURRENT_VERSION = 'v0.8.0';
|
||||||
|
|||||||
@@ -9,8 +9,6 @@ module.exports = withBundleAnalyzer({
|
|||||||
eslint: {
|
eslint: {
|
||||||
ignoreDuringBuilds: true,
|
ignoreDuringBuilds: true,
|
||||||
},
|
},
|
||||||
experimental: {
|
output: 'standalone',
|
||||||
outputStandalone: true,
|
|
||||||
},
|
|
||||||
basePath: env.BASE_URL,
|
basePath: env.BASE_URL,
|
||||||
});
|
});
|
||||||
|
|||||||
33
package.json
33
package.json
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "homarr",
|
"name": "homarr",
|
||||||
"version": "0.6.0",
|
"version": "0.8.0",
|
||||||
"description": "Homarr - A homepage for your server.",
|
"description": "Homarr - A homepage for your server.",
|
||||||
"repository": {
|
"repository": {
|
||||||
"type": "git",
|
"type": "git",
|
||||||
@@ -24,29 +24,31 @@
|
|||||||
"ci": "yarn test && yarn lint --fix && yarn typecheck && yarn prettier:write"
|
"ci": "yarn test && yarn lint --fix && yarn typecheck && yarn prettier:write"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@ctrl/deluge": "^4.0.0",
|
"@ctrl/deluge": "^4.1.0",
|
||||||
"@ctrl/qbittorrent": "^4.0.0",
|
"@ctrl/qbittorrent": "^4.0.0",
|
||||||
"@ctrl/shared-torrent": "^4.1.0",
|
"@ctrl/shared-torrent": "^4.1.0",
|
||||||
|
"@ctrl/transmission": "^4.1.1",
|
||||||
"@dnd-kit/core": "^6.0.1",
|
"@dnd-kit/core": "^6.0.1",
|
||||||
"@dnd-kit/sortable": "^7.0.0",
|
"@dnd-kit/sortable": "^7.0.0",
|
||||||
"@dnd-kit/utilities": "^3.2.0",
|
"@dnd-kit/utilities": "^3.2.0",
|
||||||
"@mantine/core": "^4.2.6",
|
"@mantine/core": "^4.2.8",
|
||||||
"@mantine/dates": "^4.2.6",
|
"@mantine/dates": "^4.2.8",
|
||||||
"@mantine/dropzone": "^4.2.6",
|
"@mantine/dropzone": "^4.2.8",
|
||||||
"@mantine/form": "^4.2.6",
|
"@mantine/form": "^4.2.8",
|
||||||
"@mantine/hooks": "^4.2.6",
|
"@mantine/hooks": "^4.2.8",
|
||||||
"@mantine/next": "^4.2.6",
|
"@mantine/next": "^4.2.8",
|
||||||
"@mantine/notifications": "^4.2.6",
|
"@mantine/notifications": "^4.2.8",
|
||||||
"@mantine/prism": "^4.2.6",
|
"@mantine/prism": "^4.2.8",
|
||||||
"@nivo/core": "^0.79.0",
|
"@nivo/core": "^0.79.0",
|
||||||
"@nivo/line": "^0.79.1",
|
"@nivo/line": "^0.79.1",
|
||||||
"@tabler/icons": "^1.68.0",
|
"@tabler/icons": "^1.68.0",
|
||||||
"axios": "^0.27.2",
|
"axios": "^0.27.2",
|
||||||
"cookies-next": "^2.0.4",
|
"cookies-next": "^2.1.1",
|
||||||
"dayjs": "^1.11.2",
|
"dayjs": "^1.11.3",
|
||||||
|
"dockerode": "^3.3.2",
|
||||||
"framer-motion": "^6.3.1",
|
"framer-motion": "^6.3.1",
|
||||||
"js-file-download": "^0.4.12",
|
"js-file-download": "^0.4.12",
|
||||||
"next": "12.1.6",
|
"next": "^12.2.0",
|
||||||
"prism-react-renderer": "^1.3.1",
|
"prism-react-renderer": "^1.3.1",
|
||||||
"react": "^17.0.1",
|
"react": "^17.0.1",
|
||||||
"react-dom": "^17.0.1",
|
"react-dom": "^17.0.1",
|
||||||
@@ -55,9 +57,10 @@
|
|||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@babel/core": "^7.17.8",
|
"@babel/core": "^7.17.8",
|
||||||
"@next/bundle-analyzer": "^12.1.4",
|
"@next/bundle-analyzer": "^12.2.0",
|
||||||
"@next/eslint-plugin-next": "^12.1.4",
|
"@next/eslint-plugin-next": "^12.2.0",
|
||||||
"@storybook/react": "^6.5.4",
|
"@storybook/react": "^6.5.4",
|
||||||
|
"@types/dockerode": "^3.3.9",
|
||||||
"@types/node": "^17.0.23",
|
"@types/node": "^17.0.23",
|
||||||
"@types/react": "17.0.43",
|
"@types/react": "17.0.43",
|
||||||
"@types/uuid": "^8.3.4",
|
"@types/uuid": "^8.3.4",
|
||||||
|
|||||||
@@ -1,24 +1,30 @@
|
|||||||
import {
|
import {
|
||||||
Modal,
|
ActionIcon,
|
||||||
|
Anchor,
|
||||||
|
Button,
|
||||||
Center,
|
Center,
|
||||||
Group,
|
Group,
|
||||||
TextInput,
|
|
||||||
Image,
|
Image,
|
||||||
Button,
|
|
||||||
Select,
|
|
||||||
LoadingOverlay,
|
LoadingOverlay,
|
||||||
ActionIcon,
|
Modal,
|
||||||
Tooltip,
|
MultiSelect,
|
||||||
Title,
|
ScrollArea,
|
||||||
Anchor,
|
Select,
|
||||||
|
Switch,
|
||||||
|
Tabs,
|
||||||
Text,
|
Text,
|
||||||
|
TextInput,
|
||||||
|
Title,
|
||||||
|
Tooltip,
|
||||||
} from '@mantine/core';
|
} from '@mantine/core';
|
||||||
import { useForm } from '@mantine/form';
|
import { useForm } from '@mantine/form';
|
||||||
import { useState } from 'react';
|
import { useDebouncedValue } from '@mantine/hooks';
|
||||||
import { IconApps as Apps } from '@tabler/icons';
|
import { IconApps as Apps } from '@tabler/icons';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
import { v4 as uuidv4 } from 'uuid';
|
import { v4 as uuidv4 } from 'uuid';
|
||||||
import { useConfig } from '../../tools/state';
|
import { useConfig } from '../../tools/state';
|
||||||
import { ServiceTypeList } from '../../tools/types';
|
import { ServiceTypeList, StatusCodes } from '../../tools/types';
|
||||||
|
import Tip from '../layout/Tip';
|
||||||
|
|
||||||
export function AddItemShelfButton(props: any) {
|
export function AddItemShelfButton(props: any) {
|
||||||
const [opened, setOpened] = useState(false);
|
const [opened, setOpened] = useState(false);
|
||||||
@@ -53,7 +59,8 @@ function MatchIcon(name: string, form: any) {
|
|||||||
fetch(
|
fetch(
|
||||||
`https://cdn.jsdelivr.net/gh/walkxhub/dashboard-icons/png/${name
|
`https://cdn.jsdelivr.net/gh/walkxhub/dashboard-icons/png/${name
|
||||||
.replace(/\s+/g, '-')
|
.replace(/\s+/g, '-')
|
||||||
.toLowerCase()}.png`
|
.toLowerCase()
|
||||||
|
.replace(/^dash\.$/, 'dashdot')}.png`
|
||||||
).then((res) => {
|
).then((res) => {
|
||||||
if (res.ok) {
|
if (res.ok) {
|
||||||
form.setFieldValue('icon', res.url);
|
form.setFieldValue('icon', res.url);
|
||||||
@@ -64,7 +71,7 @@ function MatchIcon(name: string, form: any) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function MatchService(name: string, form: any) {
|
function MatchService(name: string, form: any) {
|
||||||
const service = ServiceTypeList.find((s) => s === name);
|
const service = ServiceTypeList.find((s) => s.toLowerCase() === name.toLowerCase());
|
||||||
if (service) {
|
if (service) {
|
||||||
form.setFieldValue('type', service);
|
form.setFieldValue('type', service);
|
||||||
}
|
}
|
||||||
@@ -72,21 +79,24 @@ function MatchService(name: string, form: any) {
|
|||||||
|
|
||||||
function MatchPort(name: string, form: any) {
|
function MatchPort(name: string, form: any) {
|
||||||
const portmap = [
|
const portmap = [
|
||||||
{ name: 'qBittorrent', value: '8080' },
|
{ name: 'qbittorrent', value: '8080' },
|
||||||
{ name: 'Sonarr', value: '8989' },
|
{ name: 'sonarr', value: '8989' },
|
||||||
{ name: 'Radarr', value: '7878' },
|
{ name: 'radarr', value: '7878' },
|
||||||
{ name: 'Lidarr', value: '8686' },
|
{ name: 'lidarr', value: '8686' },
|
||||||
{ name: 'Readarr', value: '8686' },
|
{ name: 'readarr', value: '8787' },
|
||||||
{ name: 'Deluge', value: '8112' },
|
{ name: 'deluge', value: '8112' },
|
||||||
{ name: 'Transmission', value: '9091' },
|
{ name: 'transmission', value: '9091' },
|
||||||
|
{ name: 'dash.', value: '3001' },
|
||||||
];
|
];
|
||||||
// Match name with portmap key
|
// Match name with portmap key
|
||||||
const port = portmap.find((p) => p.name === name);
|
const port = portmap.find((p) => p.name === name.toLowerCase());
|
||||||
if (port) {
|
if (port) {
|
||||||
form.setFieldValue('url', `http://localhost:${port.value}`);
|
form.setFieldValue('url', `http://localhost:${port.value}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const DEFAULT_ICON = '/favicon.svg';
|
||||||
|
|
||||||
export function AddAppShelfItemForm(props: { setOpened: (b: boolean) => void } & any) {
|
export function AddAppShelfItemForm(props: { setOpened: (b: boolean) => void } & any) {
|
||||||
const { setOpened } = props;
|
const { setOpened } = props;
|
||||||
const { config, setConfig } = useConfig();
|
const { config, setConfig } = useConfig();
|
||||||
@@ -106,22 +116,21 @@ export function AddAppShelfItemForm(props: { setOpened: (b: boolean) => void } &
|
|||||||
type: props.type ?? 'Other',
|
type: props.type ?? 'Other',
|
||||||
category: props.category ?? undefined,
|
category: props.category ?? undefined,
|
||||||
name: props.name ?? '',
|
name: props.name ?? '',
|
||||||
icon: props.icon ?? '/favicon.svg',
|
icon: props.icon ?? DEFAULT_ICON,
|
||||||
url: props.url ?? '',
|
url: props.url ?? '',
|
||||||
apiKey: props.apiKey ?? (undefined as unknown as string),
|
apiKey: props.apiKey ?? (undefined as unknown as string),
|
||||||
username: props.username ?? (undefined as unknown as string),
|
username: props.username ?? (undefined as unknown as string),
|
||||||
password: props.password ?? (undefined as unknown as string),
|
password: props.password ?? (undefined as unknown as string),
|
||||||
|
openedUrl: props.openedUrl ?? (undefined as unknown as string),
|
||||||
|
status: props.status ?? ['200'],
|
||||||
|
newTab: props.newTab ?? true,
|
||||||
},
|
},
|
||||||
validate: {
|
validate: {
|
||||||
apiKey: () => null,
|
apiKey: () => null,
|
||||||
// Validate icon with a regex
|
// Validate icon with a regex
|
||||||
icon: (value: string) => {
|
icon: (value: string) =>
|
||||||
// Regex to match everything that ends with and icon extension
|
// Disable matching to allow any values
|
||||||
if (!value.match(/\.(png|jpg|jpeg|gif|svg)$/)) {
|
null,
|
||||||
return 'Please enter a valid icon URL';
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
},
|
|
||||||
// Validate url with a regex http/https
|
// Validate url with a regex http/https
|
||||||
url: (value: string) => {
|
url: (value: string) => {
|
||||||
try {
|
try {
|
||||||
@@ -131,9 +140,23 @@ export function AddAppShelfItemForm(props: { setOpened: (b: boolean) => void } &
|
|||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
|
status: (value: string[]) => {
|
||||||
|
if (!value.length) {
|
||||||
|
return 'Please select a status code';
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const [debounced, cancel] = useDebouncedValue(form.values.name, 250);
|
||||||
|
useEffect(() => {
|
||||||
|
if (form.values.name !== debounced || form.values.icon !== DEFAULT_ICON) return;
|
||||||
|
MatchIcon(form.values.name, form);
|
||||||
|
MatchService(form.values.name, form);
|
||||||
|
MatchPort(form.values.name, form);
|
||||||
|
}, [debounced]);
|
||||||
|
|
||||||
// Try to set const hostname to new URL(form.values.url).hostname)
|
// Try to set const hostname to new URL(form.values.url).hostname)
|
||||||
// If it fails, set it to the form.values.url
|
// If it fails, set it to the form.values.url
|
||||||
let hostname = form.values.url;
|
let hostname = form.values.url;
|
||||||
@@ -157,6 +180,12 @@ export function AddAppShelfItemForm(props: { setOpened: (b: boolean) => void } &
|
|||||||
</Center>
|
</Center>
|
||||||
<form
|
<form
|
||||||
onSubmit={form.onSubmit(() => {
|
onSubmit={form.onSubmit(() => {
|
||||||
|
if (JSON.stringify(form.values.status) === JSON.stringify(['200'])) {
|
||||||
|
form.values.status = undefined;
|
||||||
|
}
|
||||||
|
if (form.values.newTab === true) {
|
||||||
|
form.values.newTab = undefined;
|
||||||
|
}
|
||||||
// If service already exists, update it.
|
// If service already exists, update it.
|
||||||
if (config.services && config.services.find((s) => s.id === form.values.id)) {
|
if (config.services && config.services.find((s) => s.id === form.values.id)) {
|
||||||
setConfig({
|
setConfig({
|
||||||
@@ -181,133 +210,171 @@ export function AddAppShelfItemForm(props: { setOpened: (b: boolean) => void } &
|
|||||||
form.reset();
|
form.reset();
|
||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
<Group direction="column" grow>
|
<Tabs grow>
|
||||||
<TextInput
|
<Tabs.Tab label="Options">
|
||||||
required
|
<ScrollArea style={{ height: 500 }} scrollbarSize={4}>
|
||||||
label="Service name"
|
<Group direction="column" grow>
|
||||||
placeholder="Plex"
|
<TextInput
|
||||||
value={form.values.name}
|
required
|
||||||
onChange={(event) => {
|
label="Service name"
|
||||||
form.setFieldValue('name', event.currentTarget.value);
|
placeholder="Plex"
|
||||||
MatchIcon(event.currentTarget.value, form);
|
{...form.getInputProps('name')}
|
||||||
MatchService(event.currentTarget.value, form);
|
/>
|
||||||
MatchPort(event.currentTarget.value, form);
|
|
||||||
}}
|
|
||||||
error={form.errors.name && 'Invalid icon url'}
|
|
||||||
/>
|
|
||||||
|
|
||||||
<TextInput
|
<TextInput
|
||||||
required
|
required
|
||||||
label="Icon url"
|
label="Icon URL"
|
||||||
placeholder="https://i.gifer.com/ANPC.gif"
|
placeholder={DEFAULT_ICON}
|
||||||
{...form.getInputProps('icon')}
|
{...form.getInputProps('icon')}
|
||||||
/>
|
/>
|
||||||
<TextInput
|
<TextInput
|
||||||
required
|
required
|
||||||
label="Service url"
|
label="Service URL"
|
||||||
placeholder="http://localhost:7575"
|
placeholder="http://localhost:7575"
|
||||||
{...form.getInputProps('url')}
|
{...form.getInputProps('url')}
|
||||||
/>
|
/>
|
||||||
<Select
|
<TextInput
|
||||||
label="Service type"
|
label="On Click URL"
|
||||||
defaultValue="Other"
|
placeholder="http://sonarr.example.com"
|
||||||
placeholder="Pick one"
|
{...form.getInputProps('openedUrl')}
|
||||||
required
|
/>
|
||||||
searchable
|
<Select
|
||||||
data={ServiceTypeList}
|
label="Service type"
|
||||||
{...form.getInputProps('type')}
|
defaultValue="Other"
|
||||||
/>
|
placeholder="Pick one"
|
||||||
<Select
|
required
|
||||||
label="Category"
|
searchable
|
||||||
data={categoryList}
|
data={ServiceTypeList}
|
||||||
placeholder="Select a category or create a new one"
|
{...form.getInputProps('type')}
|
||||||
nothingFound="Nothing found"
|
/>
|
||||||
searchable
|
<Select
|
||||||
clearable
|
label="Category"
|
||||||
creatable
|
data={categoryList}
|
||||||
onClick={(e) => {
|
placeholder="Select a category or create a new one"
|
||||||
e.preventDefault();
|
nothingFound="Nothing found"
|
||||||
}}
|
searchable
|
||||||
getCreateLabel={(query) => `+ Create "${query}"`}
|
clearable
|
||||||
onCreate={(query) => {}}
|
creatable
|
||||||
{...form.getInputProps('category')}
|
onClick={(e) => {
|
||||||
/>
|
e.preventDefault();
|
||||||
<LoadingOverlay visible={isLoading} />
|
}}
|
||||||
{(form.values.type === 'Sonarr' ||
|
getCreateLabel={(query) => `+ Create "${query}"`}
|
||||||
form.values.type === 'Radarr' ||
|
onCreate={(query) => {}}
|
||||||
form.values.type === 'Lidarr' ||
|
{...form.getInputProps('category')}
|
||||||
form.values.type === 'Readarr') && (
|
/>
|
||||||
<>
|
<LoadingOverlay visible={isLoading} />
|
||||||
<TextInput
|
{(form.values.type === 'Sonarr' ||
|
||||||
|
form.values.type === 'Radarr' ||
|
||||||
|
form.values.type === 'Lidarr' ||
|
||||||
|
form.values.type === 'Readarr') && (
|
||||||
|
<>
|
||||||
|
<TextInput
|
||||||
|
required
|
||||||
|
label="API key"
|
||||||
|
placeholder="Your API key"
|
||||||
|
value={form.values.apiKey}
|
||||||
|
onChange={(event) => {
|
||||||
|
form.setFieldValue('apiKey', event.currentTarget.value);
|
||||||
|
}}
|
||||||
|
error={form.errors.apiKey && 'Invalid API key'}
|
||||||
|
/>
|
||||||
|
<Tip>
|
||||||
|
Get your API key{' '}
|
||||||
|
<Anchor
|
||||||
|
target="_blank"
|
||||||
|
weight="bold"
|
||||||
|
style={{ fontStyle: 'inherit', fontSize: 'inherit' }}
|
||||||
|
href={`${hostname}/settings/general`}
|
||||||
|
>
|
||||||
|
here.
|
||||||
|
</Anchor>
|
||||||
|
</Tip>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{form.values.type === 'qBittorrent' && (
|
||||||
|
<>
|
||||||
|
<TextInput
|
||||||
|
required
|
||||||
|
label="Username"
|
||||||
|
placeholder="admin"
|
||||||
|
value={form.values.username}
|
||||||
|
onChange={(event) => {
|
||||||
|
form.setFieldValue('username', event.currentTarget.value);
|
||||||
|
}}
|
||||||
|
error={form.errors.username && 'Invalid username'}
|
||||||
|
/>
|
||||||
|
<TextInput
|
||||||
|
required
|
||||||
|
label="Password"
|
||||||
|
placeholder="adminadmin"
|
||||||
|
value={form.values.password}
|
||||||
|
onChange={(event) => {
|
||||||
|
form.setFieldValue('password', event.currentTarget.value);
|
||||||
|
}}
|
||||||
|
error={form.errors.password && 'Invalid password'}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{form.values.type === 'Deluge' && (
|
||||||
|
<>
|
||||||
|
<TextInput
|
||||||
|
label="Password"
|
||||||
|
placeholder="password"
|
||||||
|
value={form.values.password}
|
||||||
|
onChange={(event) => {
|
||||||
|
form.setFieldValue('password', event.currentTarget.value);
|
||||||
|
}}
|
||||||
|
error={form.errors.password && 'Invalid password'}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
{form.values.type === 'Transmission' && (
|
||||||
|
<>
|
||||||
|
<TextInput
|
||||||
|
label="Username"
|
||||||
|
placeholder="admin"
|
||||||
|
value={form.values.username}
|
||||||
|
onChange={(event) => {
|
||||||
|
form.setFieldValue('username', event.currentTarget.value);
|
||||||
|
}}
|
||||||
|
error={form.errors.username && 'Invalid username'}
|
||||||
|
/>
|
||||||
|
<TextInput
|
||||||
|
label="Password"
|
||||||
|
placeholder="adminadmin"
|
||||||
|
value={form.values.password}
|
||||||
|
onChange={(event) => {
|
||||||
|
form.setFieldValue('password', event.currentTarget.value);
|
||||||
|
}}
|
||||||
|
error={form.errors.password && 'Invalid password'}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Group>
|
||||||
|
</ScrollArea>
|
||||||
|
</Tabs.Tab>
|
||||||
|
<Tabs.Tab label="Advanced Options">
|
||||||
|
<Group direction="column" grow>
|
||||||
|
<MultiSelect
|
||||||
required
|
required
|
||||||
label="API key"
|
label="HTTP Status Codes"
|
||||||
placeholder="Your API key"
|
data={StatusCodes}
|
||||||
value={form.values.apiKey}
|
placeholder="Select valid status codes"
|
||||||
onChange={(event) => {
|
clearButtonLabel="Clear selection"
|
||||||
form.setFieldValue('apiKey', event.currentTarget.value);
|
nothingFound="Nothing found"
|
||||||
}}
|
defaultValue={['200']}
|
||||||
error={form.errors.apiKey && 'Invalid API key'}
|
clearable
|
||||||
|
searchable
|
||||||
|
{...form.getInputProps('status')}
|
||||||
/>
|
/>
|
||||||
<Text
|
<Switch
|
||||||
style={{
|
label="Open service in new tab"
|
||||||
alignSelf: 'center',
|
defaultChecked={form.values.newTab}
|
||||||
fontSize: '0.75rem',
|
{...form.getInputProps('newTab')}
|
||||||
textAlign: 'center',
|
|
||||||
color: 'gray',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
Tip: Get your API key{' '}
|
|
||||||
<Anchor
|
|
||||||
target="_blank"
|
|
||||||
weight="bold"
|
|
||||||
style={{ fontStyle: 'inherit', fontSize: 'inherit' }}
|
|
||||||
href={`${hostname}/settings/general`}
|
|
||||||
>
|
|
||||||
here.
|
|
||||||
</Anchor>
|
|
||||||
</Text>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
{form.values.type === 'qBittorrent' && (
|
|
||||||
<>
|
|
||||||
<TextInput
|
|
||||||
required
|
|
||||||
label="Username"
|
|
||||||
placeholder="admin"
|
|
||||||
value={form.values.username}
|
|
||||||
onChange={(event) => {
|
|
||||||
form.setFieldValue('username', event.currentTarget.value);
|
|
||||||
}}
|
|
||||||
error={form.errors.username && 'Invalid username'}
|
|
||||||
/>
|
/>
|
||||||
<TextInput
|
</Group>
|
||||||
required
|
</Tabs.Tab>
|
||||||
label="Password"
|
</Tabs>
|
||||||
placeholder="adminadmin"
|
|
||||||
value={form.values.password}
|
|
||||||
onChange={(event) => {
|
|
||||||
form.setFieldValue('password', event.currentTarget.value);
|
|
||||||
}}
|
|
||||||
error={form.errors.password && 'Invalid password'}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
{form.values.type === 'Deluge' && (
|
|
||||||
<>
|
|
||||||
<TextInput
|
|
||||||
required
|
|
||||||
label="Password"
|
|
||||||
placeholder="deluge"
|
|
||||||
value={form.values.password}
|
|
||||||
onChange={(event) => {
|
|
||||||
form.setFieldValue('password', event.currentTarget.value);
|
|
||||||
}}
|
|
||||||
error={form.errors.password && 'Invalid password'}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</Group>
|
|
||||||
|
|
||||||
<Group grow position="center" mt="xl">
|
<Group grow position="center" mt="xl">
|
||||||
<Button type="submit">{props.message ?? 'Add service'}</Button>
|
<Button type="submit">{props.message ?? 'Add service'}</Button>
|
||||||
</Group>
|
</Group>
|
||||||
|
|||||||
@@ -1,28 +1,74 @@
|
|||||||
import React, { useState } from 'react';
|
import React, { useState } from 'react';
|
||||||
import { Grid, Group, Title } from '@mantine/core';
|
import { Accordion, createStyles, Grid, Group, Paper, useMantineColorScheme } from '@mantine/core';
|
||||||
import {
|
import {
|
||||||
closestCenter,
|
closestCenter,
|
||||||
DndContext,
|
DndContext,
|
||||||
DragOverlay,
|
DragOverlay,
|
||||||
MouseSensor,
|
MouseSensor,
|
||||||
|
TouchSensor,
|
||||||
useSensor,
|
useSensor,
|
||||||
useSensors,
|
useSensors,
|
||||||
} from '@dnd-kit/core';
|
} from '@dnd-kit/core';
|
||||||
import { arrayMove, SortableContext } from '@dnd-kit/sortable';
|
import { arrayMove, SortableContext } from '@dnd-kit/sortable';
|
||||||
|
import { useLocalStorage } from '@mantine/hooks';
|
||||||
import { useConfig } from '../../tools/state';
|
import { useConfig } from '../../tools/state';
|
||||||
|
|
||||||
import { SortableAppShelfItem, AppShelfItem } from './AppShelfItem';
|
import { SortableAppShelfItem, AppShelfItem } from './AppShelfItem';
|
||||||
import { ModuleWrapper } from '../modules/moduleWrapper';
|
import { ModuleMenu, ModuleWrapper } from '../modules/moduleWrapper';
|
||||||
import { DownloadsModule } from '../modules';
|
import { DownloadsModule } from '../modules';
|
||||||
|
import DownloadComponent from '../modules/downloads/DownloadsModule';
|
||||||
|
|
||||||
|
const useStyles = createStyles((theme, _params) => ({
|
||||||
|
item: {
|
||||||
|
overflow: 'hidden',
|
||||||
|
borderLeft: '3px solid transparent',
|
||||||
|
borderRight: '3px solid transparent',
|
||||||
|
borderBottom: '3px solid transparent',
|
||||||
|
borderRadius: '20px',
|
||||||
|
borderColor: theme.colorScheme === 'dark' ? theme.colors.dark[5] : theme.colors.gray[1],
|
||||||
|
marginTop: theme.spacing.md,
|
||||||
|
},
|
||||||
|
|
||||||
|
control: {
|
||||||
|
backgroundColor: theme.colorScheme === 'dark' ? theme.colors.dark[5] : theme.colors.gray[1],
|
||||||
|
borderRadius: theme.spacing.md,
|
||||||
|
|
||||||
|
'&:hover': {
|
||||||
|
backgroundColor: theme.colorScheme === 'dark' ? theme.colors.dark[5] : theme.colors.gray[1],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
content: {
|
||||||
|
margin: theme.spacing.md,
|
||||||
|
},
|
||||||
|
|
||||||
|
label: {
|
||||||
|
overflow: 'visible',
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
const AppShelf = (props: any) => {
|
const AppShelf = (props: any) => {
|
||||||
|
const { classes, cx } = useStyles(props);
|
||||||
|
const [toggledCategories, settoggledCategories] = useLocalStorage({
|
||||||
|
key: 'app-shelf-toggled',
|
||||||
|
// This is a bit of a hack to get the 5 first categories to be toggled on by default
|
||||||
|
defaultValue: { 0: true, 1: true, 2: true, 3: true, 4: true } as Record<string, boolean>,
|
||||||
|
});
|
||||||
const [activeId, setActiveId] = useState(null);
|
const [activeId, setActiveId] = useState(null);
|
||||||
const { config, setConfig } = useConfig();
|
const { config, setConfig } = useConfig();
|
||||||
|
const { colorScheme } = useMantineColorScheme();
|
||||||
|
|
||||||
const sensors = useSensors(
|
const sensors = useSensors(
|
||||||
|
useSensor(TouchSensor, {
|
||||||
|
activationConstraint: {
|
||||||
|
delay: 500,
|
||||||
|
tolerance: 5,
|
||||||
|
},
|
||||||
|
}),
|
||||||
useSensor(MouseSensor, {
|
useSensor(MouseSensor, {
|
||||||
// Require the mouse to move by 10 pixels before activating
|
// Require the mouse to move by 10 pixels before activating
|
||||||
activationConstraint: {
|
activationConstraint: {
|
||||||
delay: 250,
|
delay: 500,
|
||||||
tolerance: 5,
|
tolerance: 5,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
@@ -75,7 +121,14 @@ const AppShelf = (props: any) => {
|
|||||||
<SortableContext items={config.services}>
|
<SortableContext items={config.services}>
|
||||||
<Grid gutter="xl" align="center">
|
<Grid gutter="xl" align="center">
|
||||||
{filtered.map((service) => (
|
{filtered.map((service) => (
|
||||||
<Grid.Col key={service.id} span={6} xl={2} xs={4} sm={3} md={3}>
|
<Grid.Col
|
||||||
|
key={service.id}
|
||||||
|
span={6}
|
||||||
|
xl={config.settings.appCardWidth || 2}
|
||||||
|
xs={4}
|
||||||
|
sm={3}
|
||||||
|
md={3}
|
||||||
|
>
|
||||||
<SortableAppShelfItem service={service} key={service.id} id={service.id} />
|
<SortableAppShelfItem service={service} key={service.id} id={service.id} />
|
||||||
</Grid.Col>
|
</Grid.Col>
|
||||||
))}
|
))}
|
||||||
@@ -99,26 +152,49 @@ const AppShelf = (props: any) => {
|
|||||||
const noCategory = config.services.filter(
|
const noCategory = config.services.filter(
|
||||||
(e) => e.category === undefined || e.category === null
|
(e) => e.category === undefined || e.category === null
|
||||||
);
|
);
|
||||||
|
const downloadEnabled = config.modules?.[DownloadsModule.title]?.enabled ?? false;
|
||||||
|
// Create an item with 0: true, 1: true, 2: true... For each category
|
||||||
return (
|
return (
|
||||||
// Return one item for each category
|
// Return one item for each category
|
||||||
<Group grow direction="column">
|
<Group grow direction="column">
|
||||||
{categoryList.map((category) => (
|
<Accordion
|
||||||
<>
|
disableIconRotation
|
||||||
<Title order={3} key={category}>
|
classNames={classes}
|
||||||
{category}
|
order={2}
|
||||||
</Title>
|
iconPosition="right"
|
||||||
{item(category)}
|
multiple
|
||||||
</>
|
initialState={toggledCategories}
|
||||||
))}
|
onChange={(idx) => settoggledCategories(idx)}
|
||||||
{/* Return the item for all services without category */}
|
>
|
||||||
{noCategory && noCategory.length > 0 ? (
|
{categoryList.map((category, idx) => (
|
||||||
<>
|
<Accordion.Item key={category} label={category}>
|
||||||
<Title order={3}>Other</Title>
|
{item(category)}
|
||||||
{item()}
|
</Accordion.Item>
|
||||||
</>
|
))}
|
||||||
) : null}
|
{/* Return the item for all services without category */}
|
||||||
<ModuleWrapper mt="xl" module={DownloadsModule} />
|
{noCategory && noCategory.length > 0 ? (
|
||||||
|
<Accordion.Item key="Other" label="Other">
|
||||||
|
{item()}
|
||||||
|
</Accordion.Item>
|
||||||
|
) : null}
|
||||||
|
{downloadEnabled ? (
|
||||||
|
<Accordion.Item key="Downloads" label="Your downloads">
|
||||||
|
<Paper
|
||||||
|
p="lg"
|
||||||
|
radius="lg"
|
||||||
|
style={{
|
||||||
|
background: `rgba(${colorScheme === 'dark' ? '37, 38, 43,' : '255, 255, 255,'} \
|
||||||
|
${(config.settings.appOpacity || 100) / 100}`,
|
||||||
|
borderColor: `rgba(${colorScheme === 'dark' ? '37, 38, 43,' : '233, 236, 239,'} \
|
||||||
|
${(config.settings.appOpacity || 100) / 100}`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<ModuleMenu module={DownloadsModule} />
|
||||||
|
<DownloadComponent />
|
||||||
|
</Paper>
|
||||||
|
</Accordion.Item>
|
||||||
|
) : null}
|
||||||
|
</Accordion>
|
||||||
</Group>
|
</Group>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,13 @@
|
|||||||
import { Text, Card, Anchor, AspectRatio, Image, Center, createStyles } from '@mantine/core';
|
import {
|
||||||
|
Text,
|
||||||
|
Card,
|
||||||
|
Anchor,
|
||||||
|
AspectRatio,
|
||||||
|
Image,
|
||||||
|
Center,
|
||||||
|
createStyles,
|
||||||
|
useMantineColorScheme,
|
||||||
|
} from '@mantine/core';
|
||||||
import { motion } from 'framer-motion';
|
import { motion } from 'framer-motion';
|
||||||
import { useState } from 'react';
|
import { useState } from 'react';
|
||||||
import { useSortable } from '@dnd-kit/sortable';
|
import { useSortable } from '@dnd-kit/sortable';
|
||||||
@@ -6,6 +15,7 @@ import { CSS } from '@dnd-kit/utilities';
|
|||||||
import { serviceItem } from '../../tools/types';
|
import { serviceItem } from '../../tools/types';
|
||||||
import PingComponent from '../modules/ping/PingModule';
|
import PingComponent from '../modules/ping/PingModule';
|
||||||
import AppShelfMenu from './AppShelfMenu';
|
import AppShelfMenu from './AppShelfMenu';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
|
||||||
const useStyles = createStyles((theme) => ({
|
const useStyles = createStyles((theme) => ({
|
||||||
item: {
|
item: {
|
||||||
@@ -15,6 +25,9 @@ const useStyles = createStyles((theme) => ({
|
|||||||
boxShadow: `${theme.shadows.md} !important`,
|
boxShadow: `${theme.shadows.md} !important`,
|
||||||
transform: 'scale(1.05)',
|
transform: 'scale(1.05)',
|
||||||
},
|
},
|
||||||
|
[theme.fn.smallerThan('sm')]: {
|
||||||
|
WebkitUserSelect: 'none',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
}));
|
}));
|
||||||
|
|
||||||
@@ -38,7 +51,9 @@ export function SortableAppShelfItem(props: any) {
|
|||||||
export function AppShelfItem(props: any) {
|
export function AppShelfItem(props: any) {
|
||||||
const { service }: { service: serviceItem } = props;
|
const { service }: { service: serviceItem } = props;
|
||||||
const [hovering, setHovering] = useState(false);
|
const [hovering, setHovering] = useState(false);
|
||||||
const { classes, theme } = useStyles();
|
const { config } = useConfig();
|
||||||
|
const { colorScheme } = useMantineColorScheme();
|
||||||
|
const { classes } = useStyles();
|
||||||
return (
|
return (
|
||||||
<motion.div
|
<motion.div
|
||||||
animate={{
|
animate={{
|
||||||
@@ -54,11 +69,22 @@ export function AppShelfItem(props: any) {
|
|||||||
setHovering(false);
|
setHovering(false);
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Card withBorder radius="lg" shadow="md" className={classes.item}>
|
<Card
|
||||||
|
withBorder
|
||||||
|
radius="lg"
|
||||||
|
shadow="md"
|
||||||
|
className={classes.item}
|
||||||
|
style={{
|
||||||
|
background: `rgba(${colorScheme === 'dark' ? '37, 38, 43,' : '255, 255, 255,'} \
|
||||||
|
${(config.settings.appOpacity || 100) / 100}`,
|
||||||
|
borderColor: `rgba(${colorScheme === 'dark' ? '37, 38, 43,' : '233, 236, 239,'} \
|
||||||
|
${(config.settings.appOpacity || 100) / 100}`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
<Card.Section>
|
<Card.Section>
|
||||||
<Anchor
|
<Anchor
|
||||||
target="_blank"
|
target={service.newTab === false ? '_top' : '_blank'}
|
||||||
href={service.url}
|
href={service.openedUrl ? service.openedUrl : service.url}
|
||||||
style={{ color: 'inherit', fontStyle: 'inherit', fontSize: 'inherit' }}
|
style={{ color: 'inherit', fontStyle: 'inherit', fontSize: 'inherit' }}
|
||||||
>
|
>
|
||||||
<Text mt="sm" align="center" lineClamp={1} weight={550}>
|
<Text mt="sm" align="center" lineClamp={1} weight={550}>
|
||||||
@@ -101,12 +127,14 @@ export function AppShelfItem(props: any) {
|
|||||||
src={service.icon}
|
src={service.icon}
|
||||||
fit="contain"
|
fit="contain"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
window.open(service.url);
|
if (service.openedUrl) {
|
||||||
|
window.open(service.openedUrl, service.newTab === false ? '_top' : '_blank');
|
||||||
|
} else window.open(service.url, service.newTab === false ? '_top' : '_blank');
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</motion.i>
|
</motion.i>
|
||||||
</AspectRatio>
|
</AspectRatio>
|
||||||
<PingComponent url={service.url} />
|
<PingComponent url={service.url} status={service.status} />
|
||||||
</Card.Section>
|
</Card.Section>
|
||||||
</Center>
|
</Center>
|
||||||
</Card>
|
</Card>
|
||||||
|
|||||||
@@ -20,19 +20,7 @@ export default function AppShelfMenu(props: any) {
|
|||||||
onClose={() => setOpened(false)}
|
onClose={() => setOpened(false)}
|
||||||
title="Modify a service"
|
title="Modify a service"
|
||||||
>
|
>
|
||||||
<AddAppShelfItemForm
|
<AddAppShelfItemForm setOpened={setOpened} {...service} message="Save service" />
|
||||||
setOpened={setOpened}
|
|
||||||
name={service.name}
|
|
||||||
id={service.id}
|
|
||||||
category={service.category}
|
|
||||||
type={service.type}
|
|
||||||
url={service.url}
|
|
||||||
icon={service.icon}
|
|
||||||
apiKey={service.apiKey}
|
|
||||||
username={service.username}
|
|
||||||
password={service.password}
|
|
||||||
message="Save service"
|
|
||||||
/>
|
|
||||||
</Modal>
|
</Modal>
|
||||||
<Menu
|
<Menu
|
||||||
position="right"
|
position="right"
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { createStyles, Switch, Group, useMantineColorScheme, Kbd } from '@mantine/core';
|
import { createStyles, Switch, Group, useMantineColorScheme, Kbd } from '@mantine/core';
|
||||||
import { IconSun as Sun, IconMoonStars as MoonStars } from '@tabler/icons';
|
import { IconSun as Sun, IconMoonStars as MoonStars } from '@tabler/icons';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
|
||||||
const useStyles = createStyles((theme) => ({
|
const useStyles = createStyles((theme) => ({
|
||||||
root: {
|
root: {
|
||||||
@@ -29,6 +30,7 @@ const useStyles = createStyles((theme) => ({
|
|||||||
}));
|
}));
|
||||||
|
|
||||||
export function ColorSchemeSwitch() {
|
export function ColorSchemeSwitch() {
|
||||||
|
const { config } = useConfig();
|
||||||
const { colorScheme, toggleColorScheme } = useMantineColorScheme();
|
const { colorScheme, toggleColorScheme } = useMantineColorScheme();
|
||||||
const { classes, cx } = useStyles();
|
const { classes, cx } = useStyles();
|
||||||
|
|
||||||
|
|||||||
@@ -26,7 +26,10 @@ export default function ConfigChanger() {
|
|||||||
label="Config loader"
|
label="Config loader"
|
||||||
onChange={(e) => {
|
onChange={(e) => {
|
||||||
loadConfig(e ?? 'default');
|
loadConfig(e ?? 'default');
|
||||||
setCookies('config-name', e ?? 'default', { maxAge: 60 * 60 * 24 * 30 });
|
setCookies('config-name', e ?? 'default', {
|
||||||
|
maxAge: 60 * 60 * 24 * 30,
|
||||||
|
sameSite: 'strict',
|
||||||
|
});
|
||||||
}}
|
}}
|
||||||
data={
|
data={
|
||||||
// If config list is empty, return the current config
|
// If config list is empty, return the current config
|
||||||
|
|||||||
@@ -90,7 +90,10 @@ export default function LoadConfigComponent(props: any) {
|
|||||||
icon: <Check />,
|
icon: <Check />,
|
||||||
message: undefined,
|
message: undefined,
|
||||||
});
|
});
|
||||||
setCookies('config-name', newConfig.name, { maxAge: 60 * 60 * 24 * 30 });
|
setCookies('config-name', newConfig.name, {
|
||||||
|
maxAge: 60 * 60 * 24 * 30,
|
||||||
|
sameSite: 'strict',
|
||||||
|
});
|
||||||
const migratedConfig = migrateToIdConfig(newConfig);
|
const migratedConfig = migrateToIdConfig(newConfig);
|
||||||
setConfig(migratedConfig);
|
setConfig(migratedConfig);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -27,7 +27,7 @@ export default function SaveConfigComponent(props: any) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<Group>
|
<Group spacing="xs">
|
||||||
<Modal
|
<Modal
|
||||||
radius="md"
|
radius="md"
|
||||||
opened={opened}
|
opened={opened}
|
||||||
@@ -59,10 +59,11 @@ export default function SaveConfigComponent(props: any) {
|
|||||||
</Group>
|
</Group>
|
||||||
</form>
|
</form>
|
||||||
</Modal>
|
</Modal>
|
||||||
<Button leftIcon={<Download />} variant="outline" onClick={onClick}>
|
<Button size="xs" leftIcon={<Download />} variant="outline" onClick={onClick}>
|
||||||
Download config
|
Download config
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
|
size="xs"
|
||||||
leftIcon={<Trash />}
|
leftIcon={<Trash />}
|
||||||
variant="outline"
|
variant="outline"
|
||||||
onClick={() => {
|
onClick={() => {
|
||||||
@@ -93,7 +94,7 @@ export default function SaveConfigComponent(props: any) {
|
|||||||
>
|
>
|
||||||
Delete config
|
Delete config
|
||||||
</Button>
|
</Button>
|
||||||
<Button leftIcon={<Plus />} variant="outline" onClick={() => setOpened(true)}>
|
<Button size="xs" leftIcon={<Plus />} variant="outline" onClick={() => setOpened(true)}>
|
||||||
Save a copy
|
Save a copy
|
||||||
</Button>
|
</Button>
|
||||||
</Group>
|
</Group>
|
||||||
|
|||||||
182
src/components/Docker/ContainerActionBar.tsx
Normal file
182
src/components/Docker/ContainerActionBar.tsx
Normal file
@@ -0,0 +1,182 @@
|
|||||||
|
import { Button, Group, Modal, Title } from '@mantine/core';
|
||||||
|
import { useBooleanToggle } from '@mantine/hooks';
|
||||||
|
import { showNotification, updateNotification } from '@mantine/notifications';
|
||||||
|
import {
|
||||||
|
IconCheck,
|
||||||
|
IconPlayerPlay,
|
||||||
|
IconPlayerStop,
|
||||||
|
IconPlus,
|
||||||
|
IconRefresh,
|
||||||
|
IconRotateClockwise,
|
||||||
|
IconTrash,
|
||||||
|
IconX,
|
||||||
|
} from '@tabler/icons';
|
||||||
|
import axios from 'axios';
|
||||||
|
import Dockerode from 'dockerode';
|
||||||
|
import { tryMatchService } from '../../tools/addToHomarr';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
import { AddAppShelfItemForm } from '../AppShelf/AddAppShelfItem';
|
||||||
|
|
||||||
|
function sendDockerCommand(action: string, containerId: string, containerName: string) {
|
||||||
|
showNotification({
|
||||||
|
id: containerId,
|
||||||
|
loading: true,
|
||||||
|
title: `${action}ing container ${containerName.substring(1)}`,
|
||||||
|
message: undefined,
|
||||||
|
autoClose: false,
|
||||||
|
disallowClose: true,
|
||||||
|
});
|
||||||
|
axios.get(`/api/docker/container/${containerId}?action=${action}`).then((res) => {
|
||||||
|
setTimeout(() => {
|
||||||
|
if (res.data.success === true) {
|
||||||
|
updateNotification({
|
||||||
|
id: containerId,
|
||||||
|
title: `Container ${containerName} ${action}ed`,
|
||||||
|
message: `Your container was successfully ${action}ed`,
|
||||||
|
icon: <IconCheck />,
|
||||||
|
autoClose: 2000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (res.data.success === false) {
|
||||||
|
updateNotification({
|
||||||
|
id: containerId,
|
||||||
|
color: 'red',
|
||||||
|
title: 'There was an error with your container.',
|
||||||
|
message: undefined,
|
||||||
|
icon: <IconX />,
|
||||||
|
autoClose: 2000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, 500);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ContainerActionBarProps {
|
||||||
|
selected: Dockerode.ContainerInfo[];
|
||||||
|
reload: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function ContainerActionBar({ selected, reload }: ContainerActionBarProps) {
|
||||||
|
const { config, setConfig } = useConfig();
|
||||||
|
const [opened, setOpened] = useBooleanToggle(false);
|
||||||
|
return (
|
||||||
|
<Group>
|
||||||
|
<Modal
|
||||||
|
size="xl"
|
||||||
|
radius="md"
|
||||||
|
opened={opened}
|
||||||
|
onClose={() => setOpened(false)}
|
||||||
|
title="Add service"
|
||||||
|
>
|
||||||
|
<AddAppShelfItemForm
|
||||||
|
setOpened={setOpened}
|
||||||
|
{...tryMatchService(selected.at(0))}
|
||||||
|
message="Add service to homarr"
|
||||||
|
/>
|
||||||
|
</Modal>
|
||||||
|
<Button
|
||||||
|
leftIcon={<IconRotateClockwise />}
|
||||||
|
onClick={() =>
|
||||||
|
Promise.all(
|
||||||
|
selected.map((container) =>
|
||||||
|
sendDockerCommand('restart', container.Id, container.Names[0].substring(1))
|
||||||
|
)
|
||||||
|
).then(() => reload())
|
||||||
|
}
|
||||||
|
variant="light"
|
||||||
|
color="orange"
|
||||||
|
radius="md"
|
||||||
|
>
|
||||||
|
Restart
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
leftIcon={<IconPlayerStop />}
|
||||||
|
onClick={() =>
|
||||||
|
Promise.all(
|
||||||
|
selected.map((container) => {
|
||||||
|
if (
|
||||||
|
container.State === 'stopped' ||
|
||||||
|
container.State === 'created' ||
|
||||||
|
container.State === 'exited'
|
||||||
|
) {
|
||||||
|
return showNotification({
|
||||||
|
id: container.Id,
|
||||||
|
title: `Failed to stop ${container.Names[0].substring(1)}`,
|
||||||
|
message: "You can't stop a stopped container",
|
||||||
|
autoClose: 1000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return sendDockerCommand('stop', container.Id, container.Names[0].substring(1));
|
||||||
|
})
|
||||||
|
).then(() => reload())
|
||||||
|
}
|
||||||
|
variant="light"
|
||||||
|
color="red"
|
||||||
|
radius="md"
|
||||||
|
>
|
||||||
|
Stop
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
leftIcon={<IconPlayerPlay />}
|
||||||
|
onClick={() =>
|
||||||
|
Promise.all(
|
||||||
|
selected.map((container) =>
|
||||||
|
sendDockerCommand('start', container.Id, container.Names[0].substring(1))
|
||||||
|
)
|
||||||
|
).then(() => reload())
|
||||||
|
}
|
||||||
|
variant="light"
|
||||||
|
color="green"
|
||||||
|
radius="md"
|
||||||
|
>
|
||||||
|
Start
|
||||||
|
</Button>
|
||||||
|
<Button leftIcon={<IconRefresh />} onClick={() => reload()} variant="light" radius="md">
|
||||||
|
Refresh data
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
leftIcon={<IconPlus />}
|
||||||
|
color="indigo"
|
||||||
|
variant="light"
|
||||||
|
radius="md"
|
||||||
|
onClick={() => {
|
||||||
|
if (selected.length !== 1) {
|
||||||
|
showNotification({
|
||||||
|
autoClose: 5000,
|
||||||
|
title: <Title order={4}>Please only add one service at a time!</Title>,
|
||||||
|
color: 'red',
|
||||||
|
message: undefined,
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
setOpened(true);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Add to Homarr
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
leftIcon={<IconTrash />}
|
||||||
|
color="red"
|
||||||
|
variant="light"
|
||||||
|
radius="md"
|
||||||
|
onClick={() =>
|
||||||
|
Promise.all(
|
||||||
|
selected.map((container) => {
|
||||||
|
if (container.State === 'running') {
|
||||||
|
return showNotification({
|
||||||
|
id: container.Id,
|
||||||
|
title: `Failed to delete ${container.Names[0].substring(1)}`,
|
||||||
|
message: "You can't delete a running container",
|
||||||
|
autoClose: 1000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return sendDockerCommand('remove', container.Id, container.Names[0].substring(1));
|
||||||
|
})
|
||||||
|
).then(() => reload())
|
||||||
|
}
|
||||||
|
>
|
||||||
|
Remove
|
||||||
|
</Button>
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
49
src/components/Docker/ContainerState.tsx
Normal file
49
src/components/Docker/ContainerState.tsx
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
import { Badge, BadgeVariant, MantineSize } from '@mantine/core';
|
||||||
|
import Dockerode from 'dockerode';
|
||||||
|
|
||||||
|
export interface ContainerStateProps {
|
||||||
|
state: Dockerode.ContainerInfo['State'];
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function ContainerState(props: ContainerStateProps) {
|
||||||
|
const { state } = props;
|
||||||
|
const options: {
|
||||||
|
size: MantineSize;
|
||||||
|
radius: MantineSize;
|
||||||
|
variant: BadgeVariant;
|
||||||
|
} = {
|
||||||
|
size: 'md',
|
||||||
|
radius: 'md',
|
||||||
|
variant: 'outline',
|
||||||
|
};
|
||||||
|
switch (state) {
|
||||||
|
case 'running': {
|
||||||
|
return (
|
||||||
|
<Badge color="green" {...options}>
|
||||||
|
Running
|
||||||
|
</Badge>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
case 'created': {
|
||||||
|
return (
|
||||||
|
<Badge color="cyan" {...options}>
|
||||||
|
Created
|
||||||
|
</Badge>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
case 'exited': {
|
||||||
|
return (
|
||||||
|
<Badge color="red" {...options}>
|
||||||
|
Stopped
|
||||||
|
</Badge>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
default: {
|
||||||
|
return (
|
||||||
|
<Badge color="purple" {...options}>
|
||||||
|
Unknown
|
||||||
|
</Badge>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
53
src/components/Docker/DockerDrawer.tsx
Normal file
53
src/components/Docker/DockerDrawer.tsx
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
import { ActionIcon, Drawer, Group, LoadingOverlay } from '@mantine/core';
|
||||||
|
import { IconBrandDocker } from '@tabler/icons';
|
||||||
|
import axios from 'axios';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
import Docker from 'dockerode';
|
||||||
|
import ContainerActionBar from './ContainerActionBar';
|
||||||
|
import DockerTable from './DockerTable';
|
||||||
|
|
||||||
|
export default function DockerDrawer(props: any) {
|
||||||
|
const [opened, setOpened] = useState(false);
|
||||||
|
const [containers, setContainers] = useState<Docker.ContainerInfo[]>([]);
|
||||||
|
const [selection, setSelection] = useState<Docker.ContainerInfo[]>([]);
|
||||||
|
const [visible, setVisible] = useState(false);
|
||||||
|
|
||||||
|
function reload() {
|
||||||
|
setVisible(true);
|
||||||
|
setTimeout(() => {
|
||||||
|
axios.get('/api/docker/containers').then((res) => {
|
||||||
|
setContainers(res.data);
|
||||||
|
setSelection([]);
|
||||||
|
setVisible(false);
|
||||||
|
});
|
||||||
|
}, 300);
|
||||||
|
}
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
reload();
|
||||||
|
}, []);
|
||||||
|
// Check if the user has at least one container
|
||||||
|
if (containers.length < 1) return null;
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Drawer opened={opened} onClose={() => setOpened(false)} padding="xl" size="full">
|
||||||
|
<ContainerActionBar selected={selection} reload={reload} />
|
||||||
|
<div style={{ position: 'relative' }}>
|
||||||
|
<LoadingOverlay transitionDuration={500} visible={visible} />
|
||||||
|
<DockerTable containers={containers} selection={selection} setSelection={setSelection} />
|
||||||
|
</div>
|
||||||
|
</Drawer>
|
||||||
|
<Group position="center">
|
||||||
|
<ActionIcon
|
||||||
|
variant="default"
|
||||||
|
radius="md"
|
||||||
|
size="xl"
|
||||||
|
color="blue"
|
||||||
|
onClick={() => setOpened(true)}
|
||||||
|
>
|
||||||
|
<IconBrandDocker />
|
||||||
|
</ActionIcon>
|
||||||
|
</Group>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
91
src/components/Docker/DockerMenu.tsx
Normal file
91
src/components/Docker/DockerMenu.tsx
Normal file
@@ -0,0 +1,91 @@
|
|||||||
|
import { Menu, Text, useMantineTheme } from '@mantine/core';
|
||||||
|
import { showNotification, updateNotification } from '@mantine/notifications';
|
||||||
|
import {
|
||||||
|
IconCheck,
|
||||||
|
IconCodePlus,
|
||||||
|
IconPlayerPlay,
|
||||||
|
IconPlayerStop,
|
||||||
|
IconRotateClockwise,
|
||||||
|
IconX,
|
||||||
|
} from '@tabler/icons';
|
||||||
|
import axios from 'axios';
|
||||||
|
import Dockerode from 'dockerode';
|
||||||
|
|
||||||
|
function sendNotification(action: string, containerId: string, containerName: string) {
|
||||||
|
showNotification({
|
||||||
|
id: 'load-data',
|
||||||
|
loading: true,
|
||||||
|
title: `${action}ing container ${containerName}`,
|
||||||
|
message: 'Your password is being checked...',
|
||||||
|
autoClose: false,
|
||||||
|
disallowClose: true,
|
||||||
|
});
|
||||||
|
axios.get(`/api/docker/container/${containerId}?action=${action}`).then((res) => {
|
||||||
|
setTimeout(() => {
|
||||||
|
if (res.data.success === true) {
|
||||||
|
updateNotification({
|
||||||
|
id: 'load-data',
|
||||||
|
title: 'Container restarted',
|
||||||
|
message: 'Your container was successfully restarted',
|
||||||
|
icon: <IconCheck />,
|
||||||
|
autoClose: 2000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (res.data.success === false) {
|
||||||
|
updateNotification({
|
||||||
|
id: 'load-data',
|
||||||
|
color: 'red',
|
||||||
|
title: 'There was an error restarting your container.',
|
||||||
|
message: 'Your container has encountered issues while restarting.',
|
||||||
|
icon: <IconX />,
|
||||||
|
autoClose: 2000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, 500);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function restart(container: Dockerode.ContainerInfo) {
|
||||||
|
sendNotification('restart', container.Id, container.Names[0]);
|
||||||
|
}
|
||||||
|
function stop(container: Dockerode.ContainerInfo) {
|
||||||
|
console.log('stoping container', container.Id);
|
||||||
|
}
|
||||||
|
function start(container: Dockerode.ContainerInfo) {
|
||||||
|
console.log('starting container', container.Id);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function DockerMenu(props: any) {
|
||||||
|
const { container }: { container: Dockerode.ContainerInfo } = props;
|
||||||
|
const theme = useMantineTheme();
|
||||||
|
if (container === undefined) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return (
|
||||||
|
<Menu shadow="lg" radius="md">
|
||||||
|
<Menu.Label>Actions</Menu.Label>
|
||||||
|
<Menu.Item icon={<IconRotateClockwise color="orange" />} onClick={() => restart(container)}>
|
||||||
|
<Text>Restart</Text>
|
||||||
|
</Menu.Item>
|
||||||
|
{container.State === 'running' ? (
|
||||||
|
<Menu.Item icon={<IconPlayerStop color="red" />}>
|
||||||
|
<Text>Stop</Text>
|
||||||
|
</Menu.Item>
|
||||||
|
) : (
|
||||||
|
<Menu.Item icon={<IconPlayerPlay color="green" />}>
|
||||||
|
<Text>Start</Text>
|
||||||
|
</Menu.Item>
|
||||||
|
)}
|
||||||
|
{/* <Menu.Item icon={<IconDownload color="blue" />}>
|
||||||
|
<Text>Pull latest image </Text>
|
||||||
|
</Menu.Item>
|
||||||
|
<Menu.Item icon={<IconFileText color="grey" />}>
|
||||||
|
<Text>Logs</Text>
|
||||||
|
</Menu.Item> */}
|
||||||
|
<Menu.Label>Homarr</Menu.Label>
|
||||||
|
<Menu.Item icon={<IconCodePlus color={theme.primaryColor} />}>
|
||||||
|
<Text>Add to Homarr</Text>
|
||||||
|
</Menu.Item>
|
||||||
|
</Menu>
|
||||||
|
);
|
||||||
|
}
|
||||||
90
src/components/Docker/DockerTable.tsx
Normal file
90
src/components/Docker/DockerTable.tsx
Normal file
@@ -0,0 +1,90 @@
|
|||||||
|
import { Table, Checkbox, Group, Badge, createStyles } from '@mantine/core';
|
||||||
|
import Dockerode from 'dockerode';
|
||||||
|
import ContainerState from './ContainerState';
|
||||||
|
|
||||||
|
const useStyles = createStyles((theme) => ({
|
||||||
|
rowSelected: {
|
||||||
|
backgroundColor:
|
||||||
|
theme.colorScheme === 'dark'
|
||||||
|
? theme.fn.rgba(theme.colors[theme.primaryColor][7], 0.2)
|
||||||
|
: theme.colors[theme.primaryColor][0],
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
export default function DockerTable({
|
||||||
|
containers,
|
||||||
|
selection,
|
||||||
|
setSelection,
|
||||||
|
}: {
|
||||||
|
setSelection: any;
|
||||||
|
containers: Dockerode.ContainerInfo[];
|
||||||
|
selection: Dockerode.ContainerInfo[];
|
||||||
|
}) {
|
||||||
|
const { classes, cx } = useStyles();
|
||||||
|
|
||||||
|
const toggleRow = (container: Dockerode.ContainerInfo) =>
|
||||||
|
setSelection((current: Dockerode.ContainerInfo[]) =>
|
||||||
|
current.includes(container) ? current.filter((c) => c !== container) : [...current, container]
|
||||||
|
);
|
||||||
|
const toggleAll = () =>
|
||||||
|
setSelection((current: any) =>
|
||||||
|
current.length === containers.length ? [] : containers.map((c) => c)
|
||||||
|
);
|
||||||
|
|
||||||
|
const rows = containers.map((element) => {
|
||||||
|
const selected = selection.includes(element);
|
||||||
|
return (
|
||||||
|
<tr key={element.Id} className={cx({ [classes.rowSelected]: selected })}>
|
||||||
|
<td>
|
||||||
|
<Checkbox
|
||||||
|
checked={selection.includes(element)}
|
||||||
|
onChange={() => toggleRow(element)}
|
||||||
|
transitionDuration={0}
|
||||||
|
/>
|
||||||
|
</td>
|
||||||
|
<td>{element.Names[0].replace('/', '')}</td>
|
||||||
|
<td>{element.Image}</td>
|
||||||
|
<td>
|
||||||
|
<Group>
|
||||||
|
{element.Ports.sort((a, b) => a.PrivatePort - b.PrivatePort)
|
||||||
|
.slice(-3)
|
||||||
|
.map((port) => (
|
||||||
|
<Badge key={port.PrivatePort} variant="outline">
|
||||||
|
{port.PrivatePort}:{port.PublicPort}
|
||||||
|
</Badge>
|
||||||
|
))}
|
||||||
|
{element.Ports.length > 3 && (
|
||||||
|
<Badge variant="filled">{element.Ports.length - 3} more</Badge>
|
||||||
|
)}
|
||||||
|
</Group>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<ContainerState state={element.State} />
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Table captionSide="bottom" highlightOnHover sx={{ minWidth: 800 }} verticalSpacing="sm">
|
||||||
|
<caption>your docker containers</caption>
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th style={{ width: 40 }}>
|
||||||
|
<Checkbox
|
||||||
|
onChange={toggleAll}
|
||||||
|
checked={selection.length === containers.length}
|
||||||
|
indeterminate={selection.length > 0 && selection.length !== containers.length}
|
||||||
|
transitionDuration={0}
|
||||||
|
/>
|
||||||
|
</th>
|
||||||
|
<th>Name</th>
|
||||||
|
<th>Image</th>
|
||||||
|
<th>Ports</th>
|
||||||
|
<th>State</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>{rows}</tbody>
|
||||||
|
</Table>
|
||||||
|
);
|
||||||
|
}
|
||||||
65
src/components/Settings/AdvancedSettings.tsx
Normal file
65
src/components/Settings/AdvancedSettings.tsx
Normal file
@@ -0,0 +1,65 @@
|
|||||||
|
import { TextInput, Group, Button } from '@mantine/core';
|
||||||
|
import { useForm } from '@mantine/form';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
import { ColorSelector } from './ColorSelector';
|
||||||
|
import { OpacitySelector } from './OpacitySelector';
|
||||||
|
import { AppCardWidthSelector } from './AppCardWidthSelector';
|
||||||
|
import { ShadeSelector } from './ShadeSelector';
|
||||||
|
|
||||||
|
export default function TitleChanger() {
|
||||||
|
const { config, setConfig } = useConfig();
|
||||||
|
|
||||||
|
const form = useForm({
|
||||||
|
initialValues: {
|
||||||
|
title: config.settings.title,
|
||||||
|
logo: config.settings.logo,
|
||||||
|
favicon: config.settings.favicon,
|
||||||
|
background: config.settings.background,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const saveChanges = (values: {
|
||||||
|
title?: string;
|
||||||
|
logo?: string;
|
||||||
|
favicon?: string;
|
||||||
|
background?: string;
|
||||||
|
}) => {
|
||||||
|
setConfig({
|
||||||
|
...config,
|
||||||
|
settings: {
|
||||||
|
...config.settings,
|
||||||
|
title: values.title,
|
||||||
|
logo: values.logo,
|
||||||
|
favicon: values.favicon,
|
||||||
|
background: values.background,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group direction="column" grow mb="lg">
|
||||||
|
<form onSubmit={form.onSubmit((values) => saveChanges(values))}>
|
||||||
|
<Group grow direction="column">
|
||||||
|
<TextInput label="Page title" placeholder="Homarr 🦞" {...form.getInputProps('title')} />
|
||||||
|
<TextInput label="Logo" placeholder="/img/logo.png" {...form.getInputProps('logo')} />
|
||||||
|
<TextInput
|
||||||
|
label="Favicon"
|
||||||
|
placeholder="/favicon.svg"
|
||||||
|
{...form.getInputProps('favicon')}
|
||||||
|
/>
|
||||||
|
<TextInput
|
||||||
|
label="Background"
|
||||||
|
placeholder="/img/background.png"
|
||||||
|
{...form.getInputProps('background')}
|
||||||
|
/>
|
||||||
|
<Button type="submit">Save</Button>
|
||||||
|
</Group>
|
||||||
|
</form>
|
||||||
|
<ColorSelector type="primary" />
|
||||||
|
<ColorSelector type="secondary" />
|
||||||
|
<ShadeSelector />
|
||||||
|
<OpacitySelector />
|
||||||
|
<AppCardWidthSelector />
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
32
src/components/Settings/AppCardWidthSelector.tsx
Normal file
32
src/components/Settings/AppCardWidthSelector.tsx
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { Group, Text, Slider } from '@mantine/core';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
|
||||||
|
export function AppCardWidthSelector() {
|
||||||
|
const { config, setConfig } = useConfig();
|
||||||
|
|
||||||
|
const setappCardWidth = (appCardWidth: number) => {
|
||||||
|
setConfig({
|
||||||
|
...config,
|
||||||
|
settings: {
|
||||||
|
...config.settings,
|
||||||
|
appCardWidth,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group direction="column" spacing="xs" grow>
|
||||||
|
<Text>App Width</Text>
|
||||||
|
<Slider
|
||||||
|
label={null}
|
||||||
|
defaultValue={config.settings.appCardWidth}
|
||||||
|
step={0.2}
|
||||||
|
min={0.8}
|
||||||
|
max={2}
|
||||||
|
styles={{ markLabel: { fontSize: 'xx-small' } }}
|
||||||
|
onChange={(value) => setappCardWidth(value)}
|
||||||
|
/>
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
96
src/components/Settings/ColorSelector.tsx
Normal file
96
src/components/Settings/ColorSelector.tsx
Normal file
@@ -0,0 +1,96 @@
|
|||||||
|
import React, { useState } from 'react';
|
||||||
|
import { ColorSwatch, Group, Popover, Text, useMantineTheme } from '@mantine/core';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
import { useColorTheme } from '../../tools/color';
|
||||||
|
|
||||||
|
interface ColorControlProps {
|
||||||
|
type: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ColorSelector({ type }: ColorControlProps) {
|
||||||
|
const { config, setConfig } = useConfig();
|
||||||
|
const [opened, setOpened] = useState(false);
|
||||||
|
|
||||||
|
const { primaryColor, secondaryColor, setPrimaryColor, setSecondaryColor } = useColorTheme();
|
||||||
|
|
||||||
|
const theme = useMantineTheme();
|
||||||
|
const colors = Object.keys(theme.colors).map((color) => ({
|
||||||
|
swatch: theme.colors[color][6],
|
||||||
|
color,
|
||||||
|
}));
|
||||||
|
|
||||||
|
const configColor = type === 'primary' ? primaryColor : secondaryColor;
|
||||||
|
|
||||||
|
const setConfigColor = (color: string) => {
|
||||||
|
if (type === 'primary') {
|
||||||
|
setPrimaryColor(color);
|
||||||
|
setConfig({
|
||||||
|
...config,
|
||||||
|
settings: {
|
||||||
|
...config.settings,
|
||||||
|
primaryColor: color,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
setSecondaryColor(color);
|
||||||
|
setConfig({
|
||||||
|
...config,
|
||||||
|
settings: {
|
||||||
|
...config.settings,
|
||||||
|
secondaryColor: color,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const swatches = colors.map(({ color, swatch }) => (
|
||||||
|
<ColorSwatch
|
||||||
|
component="button"
|
||||||
|
type="button"
|
||||||
|
onClick={() => setConfigColor(color)}
|
||||||
|
key={color}
|
||||||
|
color={swatch}
|
||||||
|
size={22}
|
||||||
|
style={{ color: theme.white, cursor: 'pointer' }}
|
||||||
|
/>
|
||||||
|
));
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group direction="row" spacing={3}>
|
||||||
|
<Popover
|
||||||
|
opened={opened}
|
||||||
|
onClose={() => setOpened(false)}
|
||||||
|
transitionDuration={0}
|
||||||
|
target={
|
||||||
|
<ColorSwatch
|
||||||
|
component="button"
|
||||||
|
type="button"
|
||||||
|
color={theme.colors[configColor][6]}
|
||||||
|
onClick={() => setOpened((o) => !o)}
|
||||||
|
size={22}
|
||||||
|
style={{ display: 'block', cursor: 'pointer' }}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
styles={{
|
||||||
|
root: {
|
||||||
|
marginRight: theme.spacing.xs,
|
||||||
|
},
|
||||||
|
body: {
|
||||||
|
width: 152,
|
||||||
|
backgroundColor: theme.colorScheme === 'dark' ? theme.colors.dark[8] : theme.white,
|
||||||
|
},
|
||||||
|
arrow: {
|
||||||
|
backgroundColor: theme.colorScheme === 'dark' ? theme.colors.dark[8] : theme.white,
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
position="bottom"
|
||||||
|
placement="end"
|
||||||
|
withArrow
|
||||||
|
arrowSize={3}
|
||||||
|
>
|
||||||
|
<Group spacing="xs">{swatches}</Group>
|
||||||
|
</Popover>
|
||||||
|
<Text>{type[0].toUpperCase() + type.slice(1)} color</Text>
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
86
src/components/Settings/CommonSettings.tsx
Normal file
86
src/components/Settings/CommonSettings.tsx
Normal file
@@ -0,0 +1,86 @@
|
|||||||
|
import { Group, Text, SegmentedControl, TextInput } from '@mantine/core';
|
||||||
|
import { useState } from 'react';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
import { ColorSchemeSwitch } from '../ColorSchemeToggle/ColorSchemeSwitch';
|
||||||
|
import { WidgetsPositionSwitch } from '../WidgetsPositionSwitch/WidgetsPositionSwitch';
|
||||||
|
import ConfigChanger from '../Config/ConfigChanger';
|
||||||
|
import SaveConfigComponent from '../Config/SaveConfig';
|
||||||
|
import ModuleEnabler from './ModuleEnabler';
|
||||||
|
import Tip from '../layout/Tip';
|
||||||
|
|
||||||
|
export default function CommonSettings(args: any) {
|
||||||
|
const { config, setConfig } = useConfig();
|
||||||
|
|
||||||
|
const matches = [
|
||||||
|
{ label: 'Google', value: 'https://google.com/search?q=' },
|
||||||
|
{ label: 'DuckDuckGo', value: 'https://duckduckgo.com/?q=' },
|
||||||
|
{ label: 'Bing', value: 'https://bing.com/search?q=' },
|
||||||
|
{ label: 'Custom', value: 'Custom' },
|
||||||
|
];
|
||||||
|
|
||||||
|
const [customSearchUrl, setCustomSearchUrl] = useState(config.settings.searchUrl);
|
||||||
|
const [searchUrl, setSearchUrl] = useState(
|
||||||
|
matches.find((match) => match.value === config.settings.searchUrl)?.value ?? 'Custom'
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group direction="column" grow mb="lg">
|
||||||
|
<Group grow direction="column" spacing={0}>
|
||||||
|
<Text>Search engine</Text>
|
||||||
|
<Tip>
|
||||||
|
Use the prefixes <b>!yt</b> and <b>!t</b> in front of your query to search on YouTube or
|
||||||
|
for a Torrent respectively.
|
||||||
|
</Tip>
|
||||||
|
<SegmentedControl
|
||||||
|
fullWidth
|
||||||
|
mb="sm"
|
||||||
|
title="Search engine"
|
||||||
|
value={
|
||||||
|
// Match config.settings.searchUrl with a key in the matches array
|
||||||
|
searchUrl
|
||||||
|
}
|
||||||
|
onChange={
|
||||||
|
// Set config.settings.searchUrl to the value of the selected item
|
||||||
|
(e) => {
|
||||||
|
setSearchUrl(e);
|
||||||
|
setConfig({
|
||||||
|
...config,
|
||||||
|
settings: {
|
||||||
|
...config.settings,
|
||||||
|
searchUrl: e,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
data={matches}
|
||||||
|
/>
|
||||||
|
{searchUrl === 'Custom' && (
|
||||||
|
<>
|
||||||
|
<Tip>%s can be used as a placeholder for the query.</Tip>
|
||||||
|
<TextInput
|
||||||
|
label="Query URL"
|
||||||
|
placeholder="Custom query URL"
|
||||||
|
value={customSearchUrl}
|
||||||
|
onChange={(event) => {
|
||||||
|
setCustomSearchUrl(event.currentTarget.value);
|
||||||
|
setConfig({
|
||||||
|
...config,
|
||||||
|
settings: {
|
||||||
|
...config.settings,
|
||||||
|
searchUrl: event.currentTarget.value,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Group>
|
||||||
|
<ColorSchemeSwitch />
|
||||||
|
<WidgetsPositionSwitch />
|
||||||
|
<ModuleEnabler />
|
||||||
|
<ConfigChanger />
|
||||||
|
<SaveConfigComponent />
|
||||||
|
<Tip>Upload your config file by dragging and dropping it onto the page!</Tip>
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
44
src/components/Settings/Credits.tsx
Normal file
44
src/components/Settings/Credits.tsx
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
import { Group, ActionIcon, Anchor, Text } from '@mantine/core';
|
||||||
|
import { IconBrandDiscord, IconBrandGithub } from '@tabler/icons';
|
||||||
|
import { CURRENT_VERSION } from '../../../data/constants';
|
||||||
|
|
||||||
|
export default function Credits(props: any) {
|
||||||
|
return (
|
||||||
|
<Group position="center" direction="row" mr="xs">
|
||||||
|
<Group spacing={0}>
|
||||||
|
<ActionIcon<'a'> component="a" href="https://github.com/ajnart/homarr" size="lg">
|
||||||
|
<IconBrandGithub size={18} />
|
||||||
|
</ActionIcon>
|
||||||
|
<Text
|
||||||
|
style={{
|
||||||
|
position: 'relative',
|
||||||
|
fontSize: '0.90rem',
|
||||||
|
color: 'gray',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{CURRENT_VERSION}
|
||||||
|
</Text>
|
||||||
|
</Group>
|
||||||
|
<Group spacing={1}>
|
||||||
|
<Text
|
||||||
|
style={{
|
||||||
|
fontSize: '0.90rem',
|
||||||
|
textAlign: 'center',
|
||||||
|
color: 'gray',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Made with ❤️ by @
|
||||||
|
<Anchor
|
||||||
|
href="https://github.com/ajnart"
|
||||||
|
style={{ color: 'inherit', fontStyle: 'inherit', fontSize: 'inherit' }}
|
||||||
|
>
|
||||||
|
ajnart
|
||||||
|
</Anchor>
|
||||||
|
</Text>
|
||||||
|
<ActionIcon<'a'> component="a" href="https://discord.gg/aCsmEV5RgA" size="lg">
|
||||||
|
<IconBrandDiscord size={18} />
|
||||||
|
</ActionIcon>
|
||||||
|
</Group>
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
import { Group, Switch } from '@mantine/core';
|
import { Checkbox, Group, SimpleGrid, Title } from '@mantine/core';
|
||||||
import * as Modules from '../modules';
|
import * as Modules from '../modules';
|
||||||
import { useConfig } from '../../tools/state';
|
import { useConfig } from '../../tools/state';
|
||||||
|
|
||||||
@@ -7,26 +7,29 @@ export default function ModuleEnabler(props: any) {
|
|||||||
const modules = Object.values(Modules).map((module) => module);
|
const modules = Object.values(Modules).map((module) => module);
|
||||||
return (
|
return (
|
||||||
<Group direction="column">
|
<Group direction="column">
|
||||||
{modules.map((module) => (
|
<Title order={4}>Module enabler</Title>
|
||||||
<Switch
|
<SimpleGrid cols={2} spacing="md">
|
||||||
key={module.title}
|
{modules.map((module) => (
|
||||||
size="md"
|
<Checkbox
|
||||||
checked={config.modules?.[module.title]?.enabled ?? false}
|
key={module.title}
|
||||||
label={`Enable ${module.title}`}
|
size="md"
|
||||||
onChange={(e) => {
|
checked={config.modules?.[module.title]?.enabled ?? false}
|
||||||
setConfig({
|
label={`${module.title}`}
|
||||||
...config,
|
onChange={(e) => {
|
||||||
modules: {
|
setConfig({
|
||||||
...config.modules,
|
...config,
|
||||||
[module.title]: {
|
modules: {
|
||||||
...config.modules?.[module.title],
|
...config.modules,
|
||||||
enabled: e.currentTarget.checked,
|
[module.title]: {
|
||||||
|
...config.modules?.[module.title],
|
||||||
|
enabled: e.currentTarget.checked,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
},
|
});
|
||||||
});
|
}}
|
||||||
}}
|
/>
|
||||||
/>
|
))}
|
||||||
))}
|
</SimpleGrid>
|
||||||
</Group>
|
</Group>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
44
src/components/Settings/OpacitySelector.tsx
Normal file
44
src/components/Settings/OpacitySelector.tsx
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { Group, Text, Slider } from '@mantine/core';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
|
||||||
|
export function OpacitySelector() {
|
||||||
|
const { config, setConfig } = useConfig();
|
||||||
|
|
||||||
|
const MARKS = [
|
||||||
|
{ value: 10, label: '10' },
|
||||||
|
{ value: 20, label: '20' },
|
||||||
|
{ value: 30, label: '30' },
|
||||||
|
{ value: 40, label: '40' },
|
||||||
|
{ value: 50, label: '50' },
|
||||||
|
{ value: 60, label: '60' },
|
||||||
|
{ value: 70, label: '70' },
|
||||||
|
{ value: 80, label: '80' },
|
||||||
|
{ value: 90, label: '90' },
|
||||||
|
{ value: 100, label: '100' },
|
||||||
|
];
|
||||||
|
|
||||||
|
const setConfigOpacity = (opacity: number) => {
|
||||||
|
setConfig({
|
||||||
|
...config,
|
||||||
|
settings: {
|
||||||
|
...config.settings,
|
||||||
|
appOpacity: opacity,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group direction="column" spacing="xs" grow>
|
||||||
|
<Text>App Opacity</Text>
|
||||||
|
<Slider
|
||||||
|
defaultValue={config.settings.appOpacity || 100}
|
||||||
|
step={10}
|
||||||
|
min={10}
|
||||||
|
marks={MARKS}
|
||||||
|
styles={{ markLabel: { fontSize: 'xx-small' } }}
|
||||||
|
onChange={(value) => setConfigOpacity(value)}
|
||||||
|
/>
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,131 +1,25 @@
|
|||||||
import {
|
import { ActionIcon, Title, Tooltip, Drawer, Tabs, ScrollArea } from '@mantine/core';
|
||||||
ActionIcon,
|
import { useHotkeys } from '@mantine/hooks';
|
||||||
Group,
|
|
||||||
Title,
|
|
||||||
Text,
|
|
||||||
Tooltip,
|
|
||||||
SegmentedControl,
|
|
||||||
TextInput,
|
|
||||||
Drawer,
|
|
||||||
Anchor,
|
|
||||||
} from '@mantine/core';
|
|
||||||
import { useColorScheme, useHotkeys } from '@mantine/hooks';
|
|
||||||
import { useState } from 'react';
|
import { useState } from 'react';
|
||||||
import { IconBrandGithub as BrandGithub, IconSettings } from '@tabler/icons';
|
import { IconSettings } from '@tabler/icons';
|
||||||
import { CURRENT_VERSION } from '../../../data/constants';
|
import AdvancedSettings from './AdvancedSettings';
|
||||||
import { useConfig } from '../../tools/state';
|
import CommonSettings from './CommonSettings';
|
||||||
import { ColorSchemeSwitch } from '../ColorSchemeToggle/ColorSchemeSwitch';
|
import Credits from './Credits';
|
||||||
import ConfigChanger from '../Config/ConfigChanger';
|
|
||||||
import SaveConfigComponent from '../Config/SaveConfig';
|
|
||||||
import ModuleEnabler from './ModuleEnabler';
|
|
||||||
|
|
||||||
function SettingsMenu(props: any) {
|
function SettingsMenu(props: any) {
|
||||||
const { config, setConfig } = useConfig();
|
|
||||||
const colorScheme = useColorScheme();
|
|
||||||
const { current, latest } = props;
|
|
||||||
|
|
||||||
const matches = [
|
|
||||||
{ label: 'Google', value: 'https://google.com/search?q=' },
|
|
||||||
{ label: 'DuckDuckGo', value: 'https://duckduckgo.com/?q=' },
|
|
||||||
{ label: 'Bing', value: 'https://bing.com/search?q=' },
|
|
||||||
{ label: 'Custom', value: 'Custom' },
|
|
||||||
];
|
|
||||||
|
|
||||||
const [customSearchUrl, setCustomSearchUrl] = useState(config.settings.searchUrl);
|
|
||||||
const [searchUrl, setSearchUrl] = useState(
|
|
||||||
matches.find((match) => match.value === config.settings.searchUrl)?.value ?? 'Custom'
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Group direction="column" grow>
|
<Tabs grow>
|
||||||
<Group grow direction="column" spacing={0}>
|
<Tabs.Tab data-autofocus label="Common">
|
||||||
<Text>Search engine</Text>
|
<ScrollArea style={{ height: '78vh' }} offsetScrollbars>
|
||||||
<SegmentedControl
|
<CommonSettings />
|
||||||
fullWidth
|
</ScrollArea>
|
||||||
title="Search engine"
|
</Tabs.Tab>
|
||||||
value={
|
<Tabs.Tab label="Customizations">
|
||||||
// Match config.settings.searchUrl with a key in the matches array
|
<ScrollArea style={{ height: '78vh' }} offsetScrollbars>
|
||||||
searchUrl
|
<AdvancedSettings />
|
||||||
}
|
</ScrollArea>
|
||||||
onChange={
|
</Tabs.Tab>
|
||||||
// Set config.settings.searchUrl to the value of the selected item
|
</Tabs>
|
||||||
(e) => {
|
|
||||||
setSearchUrl(e);
|
|
||||||
setConfig({
|
|
||||||
...config,
|
|
||||||
settings: {
|
|
||||||
...config.settings,
|
|
||||||
searchUrl: e,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
data={matches}
|
|
||||||
/>
|
|
||||||
{searchUrl === 'Custom' && (
|
|
||||||
<TextInput
|
|
||||||
label="Query URL"
|
|
||||||
placeholder="Custom query url"
|
|
||||||
value={customSearchUrl}
|
|
||||||
onChange={(event) => {
|
|
||||||
setCustomSearchUrl(event.currentTarget.value);
|
|
||||||
setConfig({
|
|
||||||
...config,
|
|
||||||
settings: {
|
|
||||||
...config.settings,
|
|
||||||
searchUrl: event.currentTarget.value,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</Group>
|
|
||||||
<ModuleEnabler />
|
|
||||||
<ColorSchemeSwitch />
|
|
||||||
<ConfigChanger />
|
|
||||||
<SaveConfigComponent />
|
|
||||||
<Text
|
|
||||||
style={{
|
|
||||||
alignSelf: 'center',
|
|
||||||
fontSize: '0.75rem',
|
|
||||||
textAlign: 'center',
|
|
||||||
color: 'gray',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
Tip: You can upload your config file by dragging and dropping it onto the page!
|
|
||||||
</Text>
|
|
||||||
<Group position="center" direction="row" mr="xs">
|
|
||||||
<Group spacing={0}>
|
|
||||||
<ActionIcon<'a'> component="a" href="https://github.com/ajnart/homarr" size="lg">
|
|
||||||
<BrandGithub size={18} />
|
|
||||||
</ActionIcon>
|
|
||||||
<Text
|
|
||||||
style={{
|
|
||||||
position: 'relative',
|
|
||||||
fontSize: '0.90rem',
|
|
||||||
color: 'gray',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
{CURRENT_VERSION}
|
|
||||||
</Text>
|
|
||||||
</Group>
|
|
||||||
<Text
|
|
||||||
style={{
|
|
||||||
fontSize: '0.90rem',
|
|
||||||
textAlign: 'center',
|
|
||||||
color: 'gray',
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
Made with ❤️ by @
|
|
||||||
<Anchor
|
|
||||||
href="https://github.com/ajnart"
|
|
||||||
style={{ color: 'inherit', fontStyle: 'inherit', fontSize: 'inherit' }}
|
|
||||||
>
|
|
||||||
ajnart
|
|
||||||
</Anchor>
|
|
||||||
</Text>
|
|
||||||
</Group>
|
|
||||||
</Group>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -136,14 +30,15 @@ export function SettingsMenuButton(props: any) {
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Drawer
|
<Drawer
|
||||||
size="auto"
|
size="xl"
|
||||||
padding="xl"
|
padding="lg"
|
||||||
position="right"
|
position="right"
|
||||||
title={<Title order={3}>Settings</Title>}
|
title={<Title order={5}>Settings</Title>}
|
||||||
opened={props.opened || opened}
|
opened={props.opened || opened}
|
||||||
onClose={() => setOpened(false)}
|
onClose={() => setOpened(false)}
|
||||||
>
|
>
|
||||||
<SettingsMenu />
|
<SettingsMenu />
|
||||||
|
<Credits />
|
||||||
</Drawer>
|
</Drawer>
|
||||||
<ActionIcon
|
<ActionIcon
|
||||||
variant="default"
|
variant="default"
|
||||||
|
|||||||
97
src/components/Settings/ShadeSelector.tsx
Normal file
97
src/components/Settings/ShadeSelector.tsx
Normal file
@@ -0,0 +1,97 @@
|
|||||||
|
import React, { useState } from 'react';
|
||||||
|
import { ColorSwatch, Group, Popover, Text, useMantineTheme, MantineTheme } from '@mantine/core';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
import { useColorTheme } from '../../tools/color';
|
||||||
|
|
||||||
|
export function ShadeSelector() {
|
||||||
|
const { config, setConfig } = useConfig();
|
||||||
|
const [opened, setOpened] = useState(false);
|
||||||
|
|
||||||
|
const { primaryColor, secondaryColor, primaryShade, setPrimaryShade } = useColorTheme();
|
||||||
|
|
||||||
|
const theme = useMantineTheme();
|
||||||
|
const primaryShades = theme.colors[primaryColor].map((s, i) => ({
|
||||||
|
swatch: theme.colors[primaryColor][i],
|
||||||
|
shade: i as MantineTheme['primaryShade'],
|
||||||
|
}));
|
||||||
|
const secondaryShades = theme.colors[secondaryColor].map((s, i) => ({
|
||||||
|
swatch: theme.colors[secondaryColor][i],
|
||||||
|
shade: i as MantineTheme['primaryShade'],
|
||||||
|
}));
|
||||||
|
|
||||||
|
const setConfigShade = (shade: MantineTheme['primaryShade']) => {
|
||||||
|
setPrimaryShade(shade);
|
||||||
|
setConfig({
|
||||||
|
...config,
|
||||||
|
settings: {
|
||||||
|
...config.settings,
|
||||||
|
primaryShade: shade,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const primarySwatches = primaryShades.map(({ swatch, shade }) => (
|
||||||
|
<ColorSwatch
|
||||||
|
component="button"
|
||||||
|
type="button"
|
||||||
|
onClick={() => setConfigShade(shade)}
|
||||||
|
key={Number(shade)}
|
||||||
|
color={swatch}
|
||||||
|
size={22}
|
||||||
|
style={{ color: theme.white, cursor: 'pointer' }}
|
||||||
|
/>
|
||||||
|
));
|
||||||
|
|
||||||
|
const secondarySwatches = secondaryShades.map(({ swatch, shade }) => (
|
||||||
|
<ColorSwatch
|
||||||
|
component="button"
|
||||||
|
type="button"
|
||||||
|
onClick={() => setConfigShade(shade)}
|
||||||
|
key={Number(shade)}
|
||||||
|
color={swatch}
|
||||||
|
size={22}
|
||||||
|
style={{ color: theme.white, cursor: 'pointer' }}
|
||||||
|
/>
|
||||||
|
));
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group direction="row" spacing={3}>
|
||||||
|
<Popover
|
||||||
|
opened={opened}
|
||||||
|
onClose={() => setOpened(false)}
|
||||||
|
transitionDuration={0}
|
||||||
|
target={
|
||||||
|
<ColorSwatch
|
||||||
|
component="button"
|
||||||
|
type="button"
|
||||||
|
color={theme.colors[primaryColor][Number(primaryShade)]}
|
||||||
|
onClick={() => setOpened((o) => !o)}
|
||||||
|
size={22}
|
||||||
|
style={{ display: 'block', cursor: 'pointer' }}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
styles={{
|
||||||
|
root: {
|
||||||
|
marginRight: theme.spacing.xs,
|
||||||
|
},
|
||||||
|
body: {
|
||||||
|
backgroundColor: theme.colorScheme === 'dark' ? theme.colors.dark[8] : theme.white,
|
||||||
|
},
|
||||||
|
arrow: {
|
||||||
|
backgroundColor: theme.colorScheme === 'dark' ? theme.colors.dark[8] : theme.white,
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
position="bottom"
|
||||||
|
placement="end"
|
||||||
|
withArrow
|
||||||
|
arrowSize={3}
|
||||||
|
>
|
||||||
|
<Group direction="column" spacing="xs">
|
||||||
|
<Group spacing="xs">{primarySwatches}</Group>
|
||||||
|
<Group spacing="xs">{secondarySwatches}</Group>
|
||||||
|
</Group>
|
||||||
|
</Popover>
|
||||||
|
<Text>Shade</Text>
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -0,0 +1,60 @@
|
|||||||
|
import React, { useState } from 'react';
|
||||||
|
import { createStyles, Switch, Group } from '@mantine/core';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
|
||||||
|
const useStyles = createStyles((theme) => ({
|
||||||
|
root: {
|
||||||
|
position: 'relative',
|
||||||
|
'& *': {
|
||||||
|
cursor: 'pointer',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
|
||||||
|
icon: {
|
||||||
|
pointerEvents: 'none',
|
||||||
|
position: 'absolute',
|
||||||
|
zIndex: 1,
|
||||||
|
top: 3,
|
||||||
|
},
|
||||||
|
|
||||||
|
iconLight: {
|
||||||
|
left: 4,
|
||||||
|
color: theme.white,
|
||||||
|
},
|
||||||
|
|
||||||
|
iconDark: {
|
||||||
|
right: 4,
|
||||||
|
color: theme.colors.gray[6],
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
export function WidgetsPositionSwitch() {
|
||||||
|
const { config, setConfig } = useConfig();
|
||||||
|
const { classes, cx } = useStyles();
|
||||||
|
const defaultPosition = config?.settings?.widgetPosition || 'right';
|
||||||
|
const [widgetPosition, setWidgetPosition] = useState(defaultPosition);
|
||||||
|
const toggleWidgetPosition = () => {
|
||||||
|
const position = widgetPosition === 'right' ? 'left' : 'right';
|
||||||
|
setWidgetPosition(position);
|
||||||
|
setConfig({
|
||||||
|
...config,
|
||||||
|
settings: {
|
||||||
|
...config.settings,
|
||||||
|
widgetPosition: position,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Group>
|
||||||
|
<div className={classes.root}>
|
||||||
|
<Switch
|
||||||
|
checked={widgetPosition === 'left'}
|
||||||
|
onChange={() => toggleWidgetPosition()}
|
||||||
|
size="md"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
Position widgets on left
|
||||||
|
</Group>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,33 +1,36 @@
|
|||||||
import { Aside as MantineAside, Group } from '@mantine/core';
|
import { Aside as MantineAside, createStyles } from '@mantine/core';
|
||||||
import {
|
import Widgets from './Widgets';
|
||||||
WeatherModule,
|
|
||||||
DateModule,
|
const useStyles = createStyles((theme) => ({
|
||||||
CalendarModule,
|
hide: {
|
||||||
TotalDownloadsModule,
|
[theme.fn.smallerThan('xs')]: {
|
||||||
SystemModule,
|
display: 'none',
|
||||||
} from '../modules';
|
},
|
||||||
import { ModuleWrapper } from '../modules/moduleWrapper';
|
},
|
||||||
|
burger: {
|
||||||
|
[theme.fn.largerThan('sm')]: {
|
||||||
|
display: 'none',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
export default function Aside(props: any) {
|
export default function Aside(props: any) {
|
||||||
|
const { classes, cx } = useStyles();
|
||||||
return (
|
return (
|
||||||
<MantineAside
|
<MantineAside
|
||||||
pr="md"
|
pr="md"
|
||||||
hiddenBreakpoint="md"
|
hiddenBreakpoint="sm"
|
||||||
hidden
|
hidden
|
||||||
|
className={cx(classes.hide)}
|
||||||
style={{
|
style={{
|
||||||
border: 'none',
|
border: 'none',
|
||||||
|
background: 'none',
|
||||||
}}
|
}}
|
||||||
width={{
|
width={{
|
||||||
base: 'auto',
|
base: 'auto',
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Group my="sm" grow direction="column" style={{ width: 300 }}>
|
<Widgets />
|
||||||
<ModuleWrapper module={CalendarModule} />
|
|
||||||
<ModuleWrapper module={TotalDownloadsModule} />
|
|
||||||
<ModuleWrapper module={WeatherModule} />
|
|
||||||
<ModuleWrapper module={DateModule} />
|
|
||||||
<ModuleWrapper module={SystemModule} />
|
|
||||||
</Group>
|
|
||||||
</MantineAside>
|
</MantineAside>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
20
src/components/layout/Background.tsx
Normal file
20
src/components/layout/Background.tsx
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
import { Global } from '@mantine/core';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
|
||||||
|
export function Background() {
|
||||||
|
const { config } = useConfig();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Global
|
||||||
|
styles={{
|
||||||
|
body: {
|
||||||
|
minHeight: '100vh',
|
||||||
|
backgroundImage: `url('${config.settings.background}')` || '',
|
||||||
|
backgroundPosition: 'center center',
|
||||||
|
backgroundSize: 'cover',
|
||||||
|
backgroundRepeat: 'no-repeat',
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
import React, { useEffect } from 'react';
|
import React, { useEffect } from 'react';
|
||||||
import { createStyles, Footer as FooterComponent } from '@mantine/core';
|
import { createStyles, Footer as FooterComponent } from '@mantine/core';
|
||||||
import { showNotification } from '@mantine/notifications';
|
import { showNotification } from '@mantine/notifications';
|
||||||
import { CURRENT_VERSION, REPO_URL } from '../../../data/constants';
|
|
||||||
import { IconAlertCircle as AlertCircle } from '@tabler/icons';
|
import { IconAlertCircle as AlertCircle } from '@tabler/icons';
|
||||||
|
import { CURRENT_VERSION, REPO_URL } from '../../../data/constants';
|
||||||
|
|
||||||
const useStyles = createStyles((theme) => ({
|
const useStyles = createStyles((theme) => ({
|
||||||
footer: {
|
footer: {
|
||||||
|
|||||||
@@ -1,9 +1,29 @@
|
|||||||
import React from 'react';
|
import {
|
||||||
import { createStyles, Header as Head, Group, Box } from '@mantine/core';
|
ActionIcon,
|
||||||
import { Logo } from './Logo';
|
Box,
|
||||||
import SearchBar from '../modules/search/SearchModule';
|
Burger,
|
||||||
|
createStyles,
|
||||||
|
Drawer,
|
||||||
|
Group,
|
||||||
|
Header as Head,
|
||||||
|
ScrollArea,
|
||||||
|
Title,
|
||||||
|
Transition,
|
||||||
|
} from '@mantine/core';
|
||||||
|
import { useBooleanToggle } from '@mantine/hooks';
|
||||||
import { AddItemShelfButton } from '../AppShelf/AddAppShelfItem';
|
import { AddItemShelfButton } from '../AppShelf/AddAppShelfItem';
|
||||||
|
import {
|
||||||
|
CalendarModule,
|
||||||
|
DateModule,
|
||||||
|
TotalDownloadsModule,
|
||||||
|
WeatherModule,
|
||||||
|
DashdotModule,
|
||||||
|
} from '../modules';
|
||||||
|
import { ModuleWrapper } from '../modules/moduleWrapper';
|
||||||
|
import DockerDrawer from '../Docker/DockerDrawer';
|
||||||
|
import SearchBar from '../modules/search/SearchModule';
|
||||||
import { SettingsMenuButton } from '../Settings/SettingsMenu';
|
import { SettingsMenuButton } from '../Settings/SettingsMenu';
|
||||||
|
import { Logo } from './Logo';
|
||||||
|
|
||||||
const HEADER_HEIGHT = 60;
|
const HEADER_HEIGHT = 60;
|
||||||
|
|
||||||
@@ -13,21 +33,71 @@ const useStyles = createStyles((theme) => ({
|
|||||||
display: 'none',
|
display: 'none',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
burger: {
|
||||||
|
[theme.fn.largerThan('sm')]: {
|
||||||
|
display: 'none',
|
||||||
|
},
|
||||||
|
},
|
||||||
}));
|
}));
|
||||||
|
|
||||||
export function Header(props: any) {
|
export function Header(props: any) {
|
||||||
|
const [opened, toggleOpened] = useBooleanToggle(false);
|
||||||
const { classes, cx } = useStyles();
|
const { classes, cx } = useStyles();
|
||||||
|
const [hidden, toggleHidden] = useBooleanToggle(true);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Head height="auto">
|
<Head height="auto">
|
||||||
<Group m="xs" position="apart">
|
<Group p="xs" position="apart">
|
||||||
<Box className={classes.hide}>
|
<Box className={classes.hide}>
|
||||||
<Logo style={{ fontSize: 22 }} />
|
<Logo style={{ fontSize: 22 }} />
|
||||||
</Box>
|
</Box>
|
||||||
<Group noWrap>
|
<Group noWrap>
|
||||||
<SearchBar />
|
<SearchBar />
|
||||||
|
<DockerDrawer />
|
||||||
<SettingsMenuButton />
|
<SettingsMenuButton />
|
||||||
<AddItemShelfButton />
|
<AddItemShelfButton />
|
||||||
|
<ActionIcon className={classes.burger} variant="default" radius="md" size="xl">
|
||||||
|
<Burger
|
||||||
|
opened={!hidden}
|
||||||
|
onClick={(_) => {
|
||||||
|
toggleHidden();
|
||||||
|
toggleOpened();
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</ActionIcon>
|
||||||
|
<Drawer
|
||||||
|
size="auto"
|
||||||
|
padding="xl"
|
||||||
|
position="right"
|
||||||
|
hidden={hidden}
|
||||||
|
title={<Title order={3}>Modules</Title>}
|
||||||
|
opened
|
||||||
|
onClose={() => {
|
||||||
|
toggleHidden();
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Transition
|
||||||
|
mounted={opened}
|
||||||
|
transition="pop-top-right"
|
||||||
|
duration={300}
|
||||||
|
timingFunction="ease"
|
||||||
|
onExit={() => toggleOpened()}
|
||||||
|
>
|
||||||
|
{(styles) => (
|
||||||
|
<div style={styles}>
|
||||||
|
<ScrollArea offsetScrollbars style={{ height: '90vh' }}>
|
||||||
|
<Group my="sm" grow direction="column" style={{ width: 300 }}>
|
||||||
|
<ModuleWrapper module={CalendarModule} />
|
||||||
|
<ModuleWrapper module={TotalDownloadsModule} />
|
||||||
|
<ModuleWrapper module={WeatherModule} />
|
||||||
|
<ModuleWrapper module={DateModule} />
|
||||||
|
<ModuleWrapper module={DashdotModule} />
|
||||||
|
</Group>
|
||||||
|
</ScrollArea>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</Transition>
|
||||||
|
</Drawer>
|
||||||
</Group>
|
</Group>
|
||||||
</Group>
|
</Group>
|
||||||
</Head>
|
</Head>
|
||||||
|
|||||||
14
src/components/layout/HeaderConfig.tsx
Normal file
14
src/components/layout/HeaderConfig.tsx
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import Head from 'next/head';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
|
||||||
|
export function HeaderConfig(props: any) {
|
||||||
|
const { config } = useConfig();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Head>
|
||||||
|
<title>{config.settings.title || 'Homarr 🦞'}</title>
|
||||||
|
<link rel="shortcut icon" href={config.settings.favicon || '/favicon.svg'} />
|
||||||
|
</Head>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -2,6 +2,10 @@ import { AppShell, createStyles } from '@mantine/core';
|
|||||||
import { Header } from './Header';
|
import { Header } from './Header';
|
||||||
import { Footer } from './Footer';
|
import { Footer } from './Footer';
|
||||||
import Aside from './Aside';
|
import Aside from './Aside';
|
||||||
|
import Navbar from './Navbar';
|
||||||
|
import { HeaderConfig } from './HeaderConfig';
|
||||||
|
import { Background } from './Background';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
|
||||||
const useStyles = createStyles((theme) => ({
|
const useStyles = createStyles((theme) => ({
|
||||||
main: {},
|
main: {},
|
||||||
@@ -9,8 +13,18 @@ const useStyles = createStyles((theme) => ({
|
|||||||
|
|
||||||
export default function Layout({ children, style }: any) {
|
export default function Layout({ children, style }: any) {
|
||||||
const { classes, cx } = useStyles();
|
const { classes, cx } = useStyles();
|
||||||
|
const { config } = useConfig();
|
||||||
|
const widgetPosition = config?.settings?.widgetPosition === 'left';
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<AppShell aside={<Aside />} header={<Header />} footer={<Footer links={[]} />}>
|
<AppShell
|
||||||
|
header={<Header />}
|
||||||
|
navbar={widgetPosition ? <Navbar /> : <></>}
|
||||||
|
aside={widgetPosition ? <></> : <Aside />}
|
||||||
|
footer={<Footer links={[]} />}
|
||||||
|
>
|
||||||
|
<HeaderConfig />
|
||||||
|
<Background />
|
||||||
<main
|
<main
|
||||||
className={cx(classes.main)}
|
className={cx(classes.main)}
|
||||||
style={{
|
style={{
|
||||||
|
|||||||
@@ -1,13 +1,18 @@
|
|||||||
import { Group, Image, Text } from '@mantine/core';
|
import { Group, Image, Text } from '@mantine/core';
|
||||||
import { NextLink } from '@mantine/next';
|
import { NextLink } from '@mantine/next';
|
||||||
import * as React from 'react';
|
import * as React from 'react';
|
||||||
|
import { useColorTheme } from '../../tools/color';
|
||||||
|
import { useConfig } from '../../tools/state';
|
||||||
|
|
||||||
export function Logo({ style }: any) {
|
export function Logo({ style }: any) {
|
||||||
|
const { config } = useConfig();
|
||||||
|
const { primaryColor, secondaryColor } = useColorTheme();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Group spacing="xs">
|
<Group spacing="xs">
|
||||||
<Image
|
<Image
|
||||||
width={50}
|
width={50}
|
||||||
src="/imgs/logo.png"
|
src={config.settings.logo || '/imgs/logo.png'}
|
||||||
style={{
|
style={{
|
||||||
position: 'relative',
|
position: 'relative',
|
||||||
}}
|
}}
|
||||||
@@ -23,9 +28,13 @@ export function Logo({ style }: any) {
|
|||||||
sx={style}
|
sx={style}
|
||||||
weight="bold"
|
weight="bold"
|
||||||
variant="gradient"
|
variant="gradient"
|
||||||
gradient={{ from: 'red', to: 'orange', deg: 145 }}
|
gradient={{
|
||||||
|
from: primaryColor,
|
||||||
|
to: secondaryColor,
|
||||||
|
deg: 145,
|
||||||
|
}}
|
||||||
>
|
>
|
||||||
Homarr
|
{config.settings.title || 'Homarr'}
|
||||||
</Text>
|
</Text>
|
||||||
</NextLink>
|
</NextLink>
|
||||||
</Group>
|
</Group>
|
||||||
|
|||||||
@@ -1,24 +1,37 @@
|
|||||||
import { Group, Navbar as MantineNavbar } from '@mantine/core';
|
import { createStyles, Navbar as MantineNavbar } from '@mantine/core';
|
||||||
import { WeatherModule, DateModule } from '../modules';
|
import Widgets from './Widgets';
|
||||||
import { ModuleWrapper } from '../modules/moduleWrapper';
|
|
||||||
|
const useStyles = createStyles((theme) => ({
|
||||||
|
hide: {
|
||||||
|
[theme.fn.smallerThan('xs')]: {
|
||||||
|
display: 'none',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
burger: {
|
||||||
|
[theme.fn.largerThan('sm')]: {
|
||||||
|
display: 'none',
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
export default function Navbar() {
|
export default function Navbar() {
|
||||||
|
const { classes, cx } = useStyles();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<MantineNavbar
|
<MantineNavbar
|
||||||
hiddenBreakpoint="lg"
|
pl="md"
|
||||||
|
hiddenBreakpoint="sm"
|
||||||
hidden
|
hidden
|
||||||
|
className={cx(classes.hide)}
|
||||||
style={{
|
style={{
|
||||||
border: 'none',
|
border: 'none',
|
||||||
|
background: 'none',
|
||||||
}}
|
}}
|
||||||
width={{
|
width={{
|
||||||
base: 'auto',
|
base: 'auto',
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Group mt="sm" direction="column" align="center">
|
<Widgets />
|
||||||
<ModuleWrapper module={DateModule} />
|
|
||||||
<ModuleWrapper module={WeatherModule} />
|
|
||||||
<ModuleWrapper module={WeatherModule} />
|
|
||||||
</Group>
|
|
||||||
</MantineNavbar>
|
</MantineNavbar>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
19
src/components/layout/Tip.tsx
Normal file
19
src/components/layout/Tip.tsx
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
import { Text } from '@mantine/core';
|
||||||
|
|
||||||
|
interface TipProps {
|
||||||
|
children: React.ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function Tip(props: TipProps) {
|
||||||
|
return (
|
||||||
|
<Text
|
||||||
|
style={{
|
||||||
|
fontSize: '0.75rem',
|
||||||
|
color: 'gray',
|
||||||
|
marginBottom: '0.5rem',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
Tip: {props.children}
|
||||||
|
</Text>
|
||||||
|
);
|
||||||
|
}
|
||||||
23
src/components/layout/Widgets.tsx
Normal file
23
src/components/layout/Widgets.tsx
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import { Group } from '@mantine/core';
|
||||||
|
import { useMediaQuery } from '@mantine/hooks';
|
||||||
|
import { CalendarModule, DateModule, TotalDownloadsModule, WeatherModule } from '../modules';
|
||||||
|
import { DashdotModule } from '../modules/dash.';
|
||||||
|
import { ModuleWrapper } from '../modules/moduleWrapper';
|
||||||
|
|
||||||
|
export default function Widgets(props: any) {
|
||||||
|
const matches = useMediaQuery('(min-width: 800px)');
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{matches && (
|
||||||
|
<Group my="sm" grow direction="column" style={{ width: 300 }}>
|
||||||
|
<ModuleWrapper module={CalendarModule} />
|
||||||
|
<ModuleWrapper module={TotalDownloadsModule} />
|
||||||
|
<ModuleWrapper module={WeatherModule} />
|
||||||
|
<ModuleWrapper module={DateModule} />
|
||||||
|
<ModuleWrapper module={DashdotModule} />
|
||||||
|
</Group>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,5 +1,13 @@
|
|||||||
/* eslint-disable react/no-children-prop */
|
/* eslint-disable react/no-children-prop */
|
||||||
import { Box, Divider, Indicator, Popover, ScrollArea } from '@mantine/core';
|
import {
|
||||||
|
Box,
|
||||||
|
Divider,
|
||||||
|
Indicator,
|
||||||
|
Popover,
|
||||||
|
ScrollArea,
|
||||||
|
createStyles,
|
||||||
|
useMantineTheme,
|
||||||
|
} from '@mantine/core';
|
||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useState } from 'react';
|
||||||
import { Calendar } from '@mantine/dates';
|
import { Calendar } from '@mantine/dates';
|
||||||
import { IconCalendar as CalendarIcon } from '@tabler/icons';
|
import { IconCalendar as CalendarIcon } from '@tabler/icons';
|
||||||
@@ -13,6 +21,7 @@ import {
|
|||||||
ReadarrMediaDisplay,
|
ReadarrMediaDisplay,
|
||||||
} from '../common';
|
} from '../common';
|
||||||
import { serviceItem } from '../../../tools/types';
|
import { serviceItem } from '../../../tools/types';
|
||||||
|
import { useColorTheme } from '../../../tools/color';
|
||||||
|
|
||||||
export const CalendarModule: IModule = {
|
export const CalendarModule: IModule = {
|
||||||
title: 'Calendar',
|
title: 'Calendar',
|
||||||
@@ -20,18 +29,35 @@ export const CalendarModule: IModule = {
|
|||||||
'A calendar module for displaying upcoming releases. It interacts with the Sonarr and Radarr API.',
|
'A calendar module for displaying upcoming releases. It interacts with the Sonarr and Radarr API.',
|
||||||
icon: CalendarIcon,
|
icon: CalendarIcon,
|
||||||
component: CalendarComponent,
|
component: CalendarComponent,
|
||||||
|
options: {
|
||||||
|
sundaystart: {
|
||||||
|
name: 'Start the week on Sunday',
|
||||||
|
value: false,
|
||||||
|
},
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function CalendarComponent(props: any) {
|
export default function CalendarComponent(props: any) {
|
||||||
const { config } = useConfig();
|
const { config } = useConfig();
|
||||||
|
const theme = useMantineTheme();
|
||||||
|
const { secondaryColor } = useColorTheme();
|
||||||
|
const useStyles = createStyles((theme) => ({
|
||||||
|
weekend: {
|
||||||
|
color: `${secondaryColor} !important`,
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
const [sonarrMedias, setSonarrMedias] = useState([] as any);
|
const [sonarrMedias, setSonarrMedias] = useState([] as any);
|
||||||
const [lidarrMedias, setLidarrMedias] = useState([] as any);
|
const [lidarrMedias, setLidarrMedias] = useState([] as any);
|
||||||
const [radarrMedias, setRadarrMedias] = useState([] as any);
|
const [radarrMedias, setRadarrMedias] = useState([] as any);
|
||||||
const [readarrMedias, setReadarrMedias] = useState([] as any);
|
const [readarrMedias, setReadarrMedias] = useState([] as any);
|
||||||
const sonarrService = config.services.filter((service) => service.type === 'Sonarr').at(0);
|
const sonarrServices = config.services.filter((service) => service.type === 'Sonarr');
|
||||||
const radarrService = config.services.filter((service) => service.type === 'Radarr').at(0);
|
const radarrServices = config.services.filter((service) => service.type === 'Radarr');
|
||||||
const lidarrService = config.services.filter((service) => service.type === 'Lidarr').at(0);
|
const lidarrServices = config.services.filter((service) => service.type === 'Lidarr');
|
||||||
const readarrService = config.services.filter((service) => service.type === 'Readarr').at(0);
|
const readarrServices = config.services.filter((service) => service.type === 'Readarr');
|
||||||
|
const today = new Date();
|
||||||
|
|
||||||
|
const { classes, cx } = useStyles();
|
||||||
|
|
||||||
function getMedias(service: serviceItem | undefined, type: string) {
|
function getMedias(service: serviceItem | undefined, type: string) {
|
||||||
if (!service || !service.apiKey) {
|
if (!service || !service.apiKey) {
|
||||||
@@ -41,18 +67,87 @@ export default function CalendarComponent(props: any) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
// Filter only sonarr and radarr services
|
// Create each Sonarr service and get the medias
|
||||||
|
const currentSonarrMedias: any[] = [];
|
||||||
// Get the url and apiKey for all Sonarr and Radarr services
|
Promise.all(
|
||||||
getMedias(sonarrService, 'sonarr').then((res) => setSonarrMedias(res.data));
|
sonarrServices.map((service) =>
|
||||||
getMedias(radarrService, 'radarr').then((res) => setRadarrMedias(res.data));
|
getMedias(service, 'sonarr')
|
||||||
getMedias(lidarrService, 'lidarr').then((res) => setLidarrMedias(res.data));
|
.then((res) => {
|
||||||
getMedias(readarrService, 'readarr').then((res) => setReadarrMedias(res.data));
|
currentSonarrMedias.push(...res.data);
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
currentSonarrMedias.push([]);
|
||||||
|
})
|
||||||
|
)
|
||||||
|
).then(() => {
|
||||||
|
setSonarrMedias(currentSonarrMedias);
|
||||||
|
});
|
||||||
|
const currentRadarrMedias: any[] = [];
|
||||||
|
Promise.all(
|
||||||
|
radarrServices.map((service) =>
|
||||||
|
getMedias(service, 'radarr')
|
||||||
|
.then((res) => {
|
||||||
|
currentRadarrMedias.push(...res.data);
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
currentRadarrMedias.push([]);
|
||||||
|
})
|
||||||
|
)
|
||||||
|
).then(() => {
|
||||||
|
setRadarrMedias(currentRadarrMedias);
|
||||||
|
});
|
||||||
|
const currentLidarrMedias: any[] = [];
|
||||||
|
Promise.all(
|
||||||
|
lidarrServices.map((service) =>
|
||||||
|
getMedias(service, 'lidarr')
|
||||||
|
.then((res) => {
|
||||||
|
currentLidarrMedias.push(...res.data);
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
currentLidarrMedias.push([]);
|
||||||
|
})
|
||||||
|
)
|
||||||
|
).then(() => {
|
||||||
|
setLidarrMedias(currentLidarrMedias);
|
||||||
|
});
|
||||||
|
const currentReadarrMedias: any[] = [];
|
||||||
|
Promise.all(
|
||||||
|
readarrServices.map((service) =>
|
||||||
|
getMedias(service, 'readarr')
|
||||||
|
.then((res) => {
|
||||||
|
currentReadarrMedias.push(...res.data);
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
currentReadarrMedias.push([]);
|
||||||
|
})
|
||||||
|
)
|
||||||
|
).then(() => {
|
||||||
|
setReadarrMedias(currentReadarrMedias);
|
||||||
|
});
|
||||||
}, [config.services]);
|
}, [config.services]);
|
||||||
|
|
||||||
|
const weekStartsAtSunday =
|
||||||
|
(config?.modules?.[CalendarModule.title]?.options?.sundaystart?.value as boolean) ?? false;
|
||||||
return (
|
return (
|
||||||
<Calendar
|
<Calendar
|
||||||
|
firstDayOfWeek={weekStartsAtSunday ? 'sunday' : 'monday'}
|
||||||
onChange={(day: any) => {}}
|
onChange={(day: any) => {}}
|
||||||
|
dayStyle={(date) =>
|
||||||
|
date.getDay() === today.getDay() && date.getDate() === today.getDate()
|
||||||
|
? {
|
||||||
|
backgroundColor:
|
||||||
|
theme.colorScheme === 'dark' ? theme.colors.dark[5] : theme.colors.gray[0],
|
||||||
|
}
|
||||||
|
: {}
|
||||||
|
}
|
||||||
|
styles={{
|
||||||
|
calendarHeader: {
|
||||||
|
marginRight: 15,
|
||||||
|
marginLeft: 15,
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
allowLevelChange={false}
|
||||||
|
dayClassName={(date, modifiers) => cx({ [classes.weekend]: modifiers.weekend })}
|
||||||
renderDay={(renderdate) => (
|
renderDay={(renderdate) => (
|
||||||
<DayComponent
|
<DayComponent
|
||||||
renderdate={renderdate}
|
renderdate={renderdate}
|
||||||
@@ -81,23 +176,20 @@ function DayComponent(props: any) {
|
|||||||
|
|
||||||
const readarrFiltered = readarrmedias.filter((media: any) => {
|
const readarrFiltered = readarrmedias.filter((media: any) => {
|
||||||
const date = new Date(media.releaseDate);
|
const date = new Date(media.releaseDate);
|
||||||
return date.getDate() === day && date.getMonth() === renderdate.getMonth();
|
return date.toDateString() === renderdate.toDateString();
|
||||||
});
|
});
|
||||||
|
|
||||||
const lidarrFiltered = lidarrmedias.filter((media: any) => {
|
const lidarrFiltered = lidarrmedias.filter((media: any) => {
|
||||||
const date = new Date(media.releaseDate);
|
const date = new Date(media.releaseDate);
|
||||||
// Return true if the date is renerdate without counting hours and minutes
|
return date.toDateString() === renderdate.toDateString();
|
||||||
return date.getDate() === day && date.getMonth() === renderdate.getMonth();
|
|
||||||
});
|
});
|
||||||
const sonarrFiltered = sonarrmedias.filter((media: any) => {
|
const sonarrFiltered = sonarrmedias.filter((media: any) => {
|
||||||
const date = new Date(media.airDate);
|
const date = new Date(media.airDateUtc);
|
||||||
// Return true if the date is renerdate without counting hours and minutes
|
return date.toDateString() === renderdate.toDateString();
|
||||||
return date.getDate() === day && date.getMonth() === renderdate.getMonth();
|
|
||||||
});
|
});
|
||||||
const radarrFiltered = radarrmedias.filter((media: any) => {
|
const radarrFiltered = radarrmedias.filter((media: any) => {
|
||||||
const date = new Date(media.inCinemas);
|
const date = new Date(media.inCinemas);
|
||||||
// Return true if the date is renerdate without counting hours and minutes
|
return date.toDateString() === renderdate.toDateString();
|
||||||
return date.getDate() === day && date.getMonth() === renderdate.getMonth();
|
|
||||||
});
|
});
|
||||||
if (
|
if (
|
||||||
sonarrFiltered.length === 0 &&
|
sonarrFiltered.length === 0 &&
|
||||||
@@ -167,7 +259,7 @@ function DayComponent(props: any) {
|
|||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<Popover
|
<Popover
|
||||||
position="left"
|
position="bottom"
|
||||||
radius="lg"
|
radius="lg"
|
||||||
shadow="xl"
|
shadow="xl"
|
||||||
transition="pop"
|
transition="pop"
|
||||||
@@ -176,7 +268,7 @@ function DayComponent(props: any) {
|
|||||||
boxShadow: '0 0 14px 14px rgba(0, 0, 0, 0.1), 0 14px 11px rgba(0, 0, 0, 0.1)',
|
boxShadow: '0 0 14px 14px rgba(0, 0, 0, 0.1), 0 14px 11px rgba(0, 0, 0, 0.1)',
|
||||||
},
|
},
|
||||||
}}
|
}}
|
||||||
width={700}
|
width="auto"
|
||||||
onClose={() => setOpened(false)}
|
onClose={() => setOpened(false)}
|
||||||
opened={opened}
|
opened={opened}
|
||||||
target={day}
|
target={day}
|
||||||
@@ -197,12 +289,18 @@ function DayComponent(props: any) {
|
|||||||
{index < radarrFiltered.length - 1 && <Divider variant="dashed" my="xl" />}
|
{index < radarrFiltered.length - 1 && <Divider variant="dashed" my="xl" />}
|
||||||
</React.Fragment>
|
</React.Fragment>
|
||||||
))}
|
))}
|
||||||
|
{sonarrFiltered.length > 0 && lidarrFiltered.length > 0 && (
|
||||||
|
<Divider variant="dashed" my="xl" />
|
||||||
|
)}
|
||||||
{lidarrFiltered.map((media: any, index: number) => (
|
{lidarrFiltered.map((media: any, index: number) => (
|
||||||
<React.Fragment key={index}>
|
<React.Fragment key={index}>
|
||||||
<LidarrMediaDisplay media={media} />
|
<LidarrMediaDisplay media={media} />
|
||||||
{index < lidarrFiltered.length - 1 && <Divider variant="dashed" my="xl" />}
|
{index < lidarrFiltered.length - 1 && <Divider variant="dashed" my="xl" />}
|
||||||
</React.Fragment>
|
</React.Fragment>
|
||||||
))}
|
))}
|
||||||
|
{lidarrFiltered.length > 0 && readarrFiltered.length > 0 && (
|
||||||
|
<Divider variant="dashed" my="xl" />
|
||||||
|
)}
|
||||||
{readarrFiltered.map((media: any, index: number) => (
|
{readarrFiltered.map((media: any, index: number) => (
|
||||||
<React.Fragment key={index}>
|
<React.Fragment key={index}>
|
||||||
<ReadarrMediaDisplay media={media} />
|
<ReadarrMediaDisplay media={media} />
|
||||||
|
|||||||
@@ -1,4 +1,15 @@
|
|||||||
import { Image, Group, Title, Badge, Text, ActionIcon, Anchor, ScrollArea } from '@mantine/core';
|
import {
|
||||||
|
Image,
|
||||||
|
Group,
|
||||||
|
Title,
|
||||||
|
Badge,
|
||||||
|
Text,
|
||||||
|
ActionIcon,
|
||||||
|
Anchor,
|
||||||
|
ScrollArea,
|
||||||
|
createStyles,
|
||||||
|
} from '@mantine/core';
|
||||||
|
import { useMediaQuery } from '@mantine/hooks';
|
||||||
import { IconLink as Link } from '@tabler/icons';
|
import { IconLink as Link } from '@tabler/icons';
|
||||||
import { useConfig } from '../../../tools/state';
|
import { useConfig } from '../../../tools/state';
|
||||||
import { serviceItem } from '../../../tools/types';
|
import { serviceItem } from '../../../tools/types';
|
||||||
@@ -14,13 +25,25 @@ export interface IMedia {
|
|||||||
episodeNumber?: number;
|
episodeNumber?: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const useStyles = createStyles((theme) => ({
|
||||||
|
overview: {
|
||||||
|
[theme.fn.largerThan('sm')]: {
|
||||||
|
width: 400,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
export function MediaDisplay(props: { media: IMedia }) {
|
export function MediaDisplay(props: { media: IMedia }) {
|
||||||
const { media }: { media: IMedia } = props;
|
const { media }: { media: IMedia } = props;
|
||||||
|
const { classes, cx } = useStyles();
|
||||||
|
const phone = useMediaQuery('(min-width: 800px)');
|
||||||
return (
|
return (
|
||||||
<Group position="apart">
|
<Group position="apart">
|
||||||
<Text>
|
<Text>
|
||||||
{media.poster && (
|
{media.poster && (
|
||||||
<Image
|
<Image
|
||||||
|
width={phone ? 250 : 100}
|
||||||
|
height={phone ? 400 : 160}
|
||||||
style={{
|
style={{
|
||||||
float: 'right',
|
float: 'right',
|
||||||
}}
|
}}
|
||||||
@@ -28,12 +51,10 @@ export function MediaDisplay(props: { media: IMedia }) {
|
|||||||
fit="cover"
|
fit="cover"
|
||||||
src={media.poster}
|
src={media.poster}
|
||||||
alt={media.title}
|
alt={media.title}
|
||||||
width={250}
|
|
||||||
height={400}
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
<Group direction="column">
|
<Group direction="column" style={{ minWidth: phone ? 450 : '65vw' }}>
|
||||||
<Group noWrap mr="sm" style={{ minWidth: 400 }}>
|
<Group noWrap mr="sm" className={classes.overview}>
|
||||||
<Title order={3}>{media.title}</Title>
|
<Title order={3}>{media.title}</Title>
|
||||||
{media.imdbId && (
|
{media.imdbId && (
|
||||||
<Anchor href={`https://www.imdb.com/title/${media.imdbId}`} target="_blank">
|
<Anchor href={`https://www.imdb.com/title/${media.imdbId}`} target="_blank">
|
||||||
@@ -65,9 +86,9 @@ export function MediaDisplay(props: { media: IMedia }) {
|
|||||||
)}
|
)}
|
||||||
</Group>
|
</Group>
|
||||||
<Group direction="column" position="apart">
|
<Group direction="column" position="apart">
|
||||||
<ScrollArea style={{ height: 250 }}>{media.overview}</ScrollArea>
|
<ScrollArea style={{ height: 280, maxWidth: 700 }}>{media.overview}</ScrollArea>
|
||||||
<Group align="center" position="center" spacing="xs">
|
<Group align="center" position="center" spacing="xs">
|
||||||
{media.genres.map((genre: string, i: number) => (
|
{media.genres.slice(-5).map((genre: string, i: number) => (
|
||||||
<Badge size="sm" key={i}>
|
<Badge size="sm" key={i}>
|
||||||
{genre}
|
{genre}
|
||||||
</Badge>
|
</Badge>
|
||||||
|
|||||||
233
src/components/modules/dash./DashdotModule.tsx
Normal file
233
src/components/modules/dash./DashdotModule.tsx
Normal file
@@ -0,0 +1,233 @@
|
|||||||
|
import { createStyles, useMantineColorScheme, useMantineTheme } from '@mantine/core';
|
||||||
|
import { IconCalendar as CalendarIcon } from '@tabler/icons';
|
||||||
|
import axios from 'axios';
|
||||||
|
import { useEffect, useState } from 'react';
|
||||||
|
import { useConfig } from '../../../tools/state';
|
||||||
|
import { serviceItem } from '../../../tools/types';
|
||||||
|
import { IModule } from '../modules';
|
||||||
|
|
||||||
|
const asModule = <T extends IModule>(t: T) => t;
|
||||||
|
export const DashdotModule = asModule({
|
||||||
|
title: 'Dash.',
|
||||||
|
description: 'A module for displaying the graphs of your running Dash. instance.',
|
||||||
|
icon: CalendarIcon,
|
||||||
|
component: DashdotComponent,
|
||||||
|
options: {
|
||||||
|
cpuMultiView: {
|
||||||
|
name: 'CPU Multi-Core View',
|
||||||
|
value: false,
|
||||||
|
},
|
||||||
|
storageMultiView: {
|
||||||
|
name: 'Storage Multi-Drive View',
|
||||||
|
value: false,
|
||||||
|
},
|
||||||
|
useCompactView: {
|
||||||
|
name: 'Use Compact View',
|
||||||
|
value: false,
|
||||||
|
},
|
||||||
|
graphs: {
|
||||||
|
name: 'Graphs',
|
||||||
|
value: ['CPU', 'RAM', 'Storage', 'Network'],
|
||||||
|
options: ['CPU', 'RAM', 'Storage', 'Network', 'GPU'],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const useStyles = createStyles((theme, _params) => ({
|
||||||
|
heading: {
|
||||||
|
marginTop: 0,
|
||||||
|
marginBottom: 10,
|
||||||
|
},
|
||||||
|
table: {
|
||||||
|
display: 'table',
|
||||||
|
},
|
||||||
|
tableRow: {
|
||||||
|
display: 'table-row',
|
||||||
|
},
|
||||||
|
tableLabel: {
|
||||||
|
display: 'table-cell',
|
||||||
|
paddingRight: 10,
|
||||||
|
},
|
||||||
|
tableValue: {
|
||||||
|
display: 'table-cell',
|
||||||
|
whiteSpace: 'pre-wrap',
|
||||||
|
paddingBottom: 5,
|
||||||
|
},
|
||||||
|
graphsContainer: {
|
||||||
|
display: 'flex',
|
||||||
|
flexDirection: 'row',
|
||||||
|
flexWrap: 'wrap',
|
||||||
|
rowGap: 10,
|
||||||
|
columnGap: 10,
|
||||||
|
},
|
||||||
|
iframe: {
|
||||||
|
flex: '1 0 auto',
|
||||||
|
maxWidth: '100%',
|
||||||
|
height: '140px',
|
||||||
|
borderRadius: theme.radius.lg,
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
const bpsPrettyPrint = (bits?: number) =>
|
||||||
|
!bits
|
||||||
|
? '-'
|
||||||
|
: bits > 1000 * 1000 * 1000
|
||||||
|
? `${(bits / 1000 / 1000 / 1000).toFixed(1)} Gb/s`
|
||||||
|
: bits > 1000 * 1000
|
||||||
|
? `${(bits / 1000 / 1000).toFixed(1)} Mb/s`
|
||||||
|
: bits > 1000
|
||||||
|
? `${(bits / 1000).toFixed(1)} Kb/s`
|
||||||
|
: `${bits.toFixed(1)} b/s`;
|
||||||
|
|
||||||
|
const bytePrettyPrint = (byte: number): string =>
|
||||||
|
byte > 1024 * 1024 * 1024
|
||||||
|
? `${(byte / 1024 / 1024 / 1024).toFixed(1)} GiB`
|
||||||
|
: byte > 1024 * 1024
|
||||||
|
? `${(byte / 1024 / 1024).toFixed(1)} MiB`
|
||||||
|
: byte > 1024
|
||||||
|
? `${(byte / 1024).toFixed(1)} KiB`
|
||||||
|
: `${byte.toFixed(1)} B`;
|
||||||
|
|
||||||
|
const useJson = (service: serviceItem | undefined, url: string) => {
|
||||||
|
const [data, setData] = useState<any | undefined>();
|
||||||
|
|
||||||
|
const doRequest = async () => {
|
||||||
|
try {
|
||||||
|
const resp = await axios.get(url, { baseURL: service?.url });
|
||||||
|
|
||||||
|
setData(resp.data);
|
||||||
|
// eslint-disable-next-line no-empty
|
||||||
|
} catch (e) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (service?.url) {
|
||||||
|
doRequest();
|
||||||
|
}
|
||||||
|
}, [service?.url]);
|
||||||
|
|
||||||
|
return data;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function DashdotComponent() {
|
||||||
|
const { config } = useConfig();
|
||||||
|
const theme = useMantineTheme();
|
||||||
|
const { classes } = useStyles();
|
||||||
|
const { colorScheme } = useMantineColorScheme();
|
||||||
|
|
||||||
|
const dashConfig = config.modules?.[DashdotModule.title]
|
||||||
|
.options as typeof DashdotModule['options'];
|
||||||
|
const isCompact = dashConfig?.useCompactView?.value ?? false;
|
||||||
|
const dashdotService = config.services.filter((service) => service.type === 'Dash.')[0];
|
||||||
|
|
||||||
|
const enabledGraphs = dashConfig?.graphs?.value ?? ['CPU', 'RAM', 'Storage', 'Network'];
|
||||||
|
const cpuEnabled = enabledGraphs.includes('CPU');
|
||||||
|
const storageEnabled = enabledGraphs.includes('Storage');
|
||||||
|
const ramEnabled = enabledGraphs.includes('RAM');
|
||||||
|
const networkEnabled = enabledGraphs.includes('Network');
|
||||||
|
const gpuEnabled = enabledGraphs.includes('GPU');
|
||||||
|
|
||||||
|
const info = useJson(dashdotService, '/info');
|
||||||
|
const storageLoad = useJson(dashdotService, '/load/storage');
|
||||||
|
|
||||||
|
const totalUsed =
|
||||||
|
(storageLoad?.layout as any[])?.reduce((acc, curr) => (curr.load ?? 0) + acc, 0) ?? 0;
|
||||||
|
const totalSize =
|
||||||
|
(info?.storage?.layout as any[])?.reduce((acc, curr) => (curr.size ?? 0) + acc, 0) ?? 0;
|
||||||
|
|
||||||
|
const graphs = [
|
||||||
|
{
|
||||||
|
name: 'CPU',
|
||||||
|
enabled: cpuEnabled,
|
||||||
|
params: {
|
||||||
|
multiView: dashConfig?.cpuMultiView?.value ?? false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Storage',
|
||||||
|
enabled: storageEnabled && !isCompact,
|
||||||
|
params: {
|
||||||
|
multiView: dashConfig?.storageMultiView?.value ?? false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'RAM',
|
||||||
|
enabled: ramEnabled,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'Network',
|
||||||
|
enabled: networkEnabled,
|
||||||
|
spanTwo: true,
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'GPU',
|
||||||
|
enabled: gpuEnabled,
|
||||||
|
spanTwo: true,
|
||||||
|
},
|
||||||
|
].filter((g) => g.enabled);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div>
|
||||||
|
<h2 className={classes.heading}>Dash.</h2>
|
||||||
|
|
||||||
|
{!dashdotService ? (
|
||||||
|
<p>No dash. service found. Please add one to your Homarr dashboard.</p>
|
||||||
|
) : !info ? (
|
||||||
|
<p>Cannot acquire information from dash. - are you running the latest version?</p>
|
||||||
|
) : (
|
||||||
|
<div className={classes.graphsContainer}>
|
||||||
|
<div className={classes.table}>
|
||||||
|
{storageEnabled && isCompact && (
|
||||||
|
<div className={classes.tableRow}>
|
||||||
|
<p className={classes.tableLabel}>Storage:</p>
|
||||||
|
<p className={classes.tableValue}>
|
||||||
|
{(totalUsed / (totalSize || 1)).toFixed(1)}%{'\n'}
|
||||||
|
{bytePrettyPrint(totalUsed)} / {bytePrettyPrint(totalSize)}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{networkEnabled && (
|
||||||
|
<div className={classes.tableRow}>
|
||||||
|
<p className={classes.tableLabel}>Network:</p>
|
||||||
|
<p className={classes.tableValue}>
|
||||||
|
{bpsPrettyPrint(info?.network?.speedUp)} Up{'\n'}
|
||||||
|
{bpsPrettyPrint(info?.network?.speedDown)} Down
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{graphs.map((graph) => (
|
||||||
|
<iframe
|
||||||
|
className={classes.iframe}
|
||||||
|
style={
|
||||||
|
isCompact
|
||||||
|
? {
|
||||||
|
width: graph.spanTwo ? '100%' : 'calc(50% - 5px)',
|
||||||
|
}
|
||||||
|
: undefined
|
||||||
|
}
|
||||||
|
key={graph.name}
|
||||||
|
title={graph.name}
|
||||||
|
src={`${
|
||||||
|
dashdotService.url
|
||||||
|
}?singleGraphMode=true&graph=${graph.name.toLowerCase()}&theme=${colorScheme}&surface=${(colorScheme ===
|
||||||
|
'dark'
|
||||||
|
? theme.colors.dark[7]
|
||||||
|
: theme.colors.gray[0]
|
||||||
|
).substring(1)}${isCompact ? '&gap=10' : '&gap=5'}&innerRadius=${theme.radius.lg}${
|
||||||
|
graph.params
|
||||||
|
? `&${Object.entries(graph.params)
|
||||||
|
.map(([key, value]) => `${key}=${value.toString()}`)
|
||||||
|
.join('&')}`
|
||||||
|
: ''
|
||||||
|
}`}
|
||||||
|
frameBorder="0"
|
||||||
|
allowTransparency
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
1
src/components/modules/dash./index.ts
Normal file
1
src/components/modules/dash./index.ts
Normal file
@@ -0,0 +1 @@
|
|||||||
|
export { DashdotModule } from './DashdotModule';
|
||||||
@@ -4,6 +4,7 @@ import { useEffect, useState } from 'react';
|
|||||||
import { IconClock as Clock } from '@tabler/icons';
|
import { IconClock as Clock } from '@tabler/icons';
|
||||||
import { useConfig } from '../../../tools/state';
|
import { useConfig } from '../../../tools/state';
|
||||||
import { IModule } from '../modules';
|
import { IModule } from '../modules';
|
||||||
|
import { useSetSafeInterval } from '../../../tools/hooks/useSetSafeInterval';
|
||||||
|
|
||||||
export const DateModule: IModule = {
|
export const DateModule: IModule = {
|
||||||
title: 'Date',
|
title: 'Date',
|
||||||
@@ -20,13 +21,14 @@ export const DateModule: IModule = {
|
|||||||
|
|
||||||
export default function DateComponent(props: any) {
|
export default function DateComponent(props: any) {
|
||||||
const [date, setDate] = useState(new Date());
|
const [date, setDate] = useState(new Date());
|
||||||
|
const setSafeInterval = useSetSafeInterval();
|
||||||
const { config } = useConfig();
|
const { config } = useConfig();
|
||||||
const isFullTime = config?.modules?.[DateModule.title]?.options?.full?.value ?? false;
|
const isFullTime = config?.modules?.[DateModule.title]?.options?.full?.value ?? true;
|
||||||
const formatString = isFullTime ? 'HH:mm' : 'h:mm A';
|
const formatString = isFullTime ? 'HH:mm' : 'h:mm A';
|
||||||
// Change date on minute change
|
// Change date on minute change
|
||||||
// Note: Using 10 000ms instead of 1000ms to chill a little :)
|
// Note: Using 10 000ms instead of 1000ms to chill a little :)
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setInterval(() => {
|
setSafeInterval(() => {
|
||||||
setDate(new Date());
|
setDate(new Date());
|
||||||
}, 1000 * 60);
|
}, 1000 * 60);
|
||||||
}, []);
|
}, []);
|
||||||
|
|||||||
@@ -1,11 +1,25 @@
|
|||||||
import { Table, Text, Tooltip, Title, Group, Progress, Skeleton, ScrollArea } from '@mantine/core';
|
import {
|
||||||
|
Table,
|
||||||
|
Text,
|
||||||
|
Tooltip,
|
||||||
|
Title,
|
||||||
|
Group,
|
||||||
|
Progress,
|
||||||
|
Skeleton,
|
||||||
|
ScrollArea,
|
||||||
|
Center,
|
||||||
|
Image,
|
||||||
|
} from '@mantine/core';
|
||||||
import { IconDownload as Download } from '@tabler/icons';
|
import { IconDownload as Download } from '@tabler/icons';
|
||||||
import { useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import { NormalizedTorrent } from '@ctrl/shared-torrent';
|
import { NormalizedTorrent } from '@ctrl/shared-torrent';
|
||||||
|
import { useViewportSize } from '@mantine/hooks';
|
||||||
import { IModule } from '../modules';
|
import { IModule } from '../modules';
|
||||||
import { useConfig } from '../../../tools/state';
|
import { useConfig } from '../../../tools/state';
|
||||||
import { AddItemShelfButton } from '../../AppShelf/AddAppShelfItem';
|
import { AddItemShelfButton } from '../../AppShelf/AddAppShelfItem';
|
||||||
|
import { useSetSafeInterval } from '../../../tools/hooks/useSetSafeInterval';
|
||||||
|
import { humanFileSize } from '../../../tools/humanFileSize';
|
||||||
|
|
||||||
export const DownloadsModule: IModule = {
|
export const DownloadsModule: IModule = {
|
||||||
title: 'Torrent',
|
title: 'Torrent',
|
||||||
@@ -22,36 +36,32 @@ export const DownloadsModule: IModule = {
|
|||||||
|
|
||||||
export default function DownloadComponent() {
|
export default function DownloadComponent() {
|
||||||
const { config } = useConfig();
|
const { config } = useConfig();
|
||||||
const qBittorrentService = config.services
|
const { height, width } = useViewportSize();
|
||||||
.filter((service) => service.type === 'qBittorrent')
|
const downloadServices =
|
||||||
.at(0);
|
config.services.filter(
|
||||||
const delugeService = config.services.filter((service) => service.type === 'Deluge').at(0);
|
(service) =>
|
||||||
|
service.type === 'qBittorrent' ||
|
||||||
|
service.type === 'Transmission' ||
|
||||||
|
service.type === 'Deluge'
|
||||||
|
) ?? [];
|
||||||
const hideComplete: boolean =
|
const hideComplete: boolean =
|
||||||
(config?.modules?.[DownloadsModule.title]?.options?.hidecomplete?.value as boolean) ?? false;
|
(config?.modules?.[DownloadsModule.title]?.options?.hidecomplete?.value as boolean) ?? false;
|
||||||
|
const [torrents, setTorrents] = useState<NormalizedTorrent[]>([]);
|
||||||
const [delugeTorrents, setDelugeTorrents] = useState<NormalizedTorrent[]>([]);
|
const setSafeInterval = useSetSafeInterval();
|
||||||
const [qBittorrentTorrents, setqBittorrentTorrents] = useState<NormalizedTorrent[]>([]);
|
const [isLoading, setIsLoading] = useState(true);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (qBittorrentService) {
|
setIsLoading(true);
|
||||||
setInterval(() => {
|
if (downloadServices.length === 0) return;
|
||||||
axios
|
setSafeInterval(() => {
|
||||||
.post('/api/modules/downloads?dlclient=qbit', { ...qBittorrentService })
|
// Send one request with each download service inside
|
||||||
.then((res) => {
|
axios.post('/api/modules/downloads', { config }).then((response) => {
|
||||||
setqBittorrentTorrents(res.data.torrents);
|
setTorrents(response.data);
|
||||||
});
|
setIsLoading(false);
|
||||||
}, 3000);
|
});
|
||||||
}
|
}, 5000);
|
||||||
if (delugeService) {
|
}, []);
|
||||||
setInterval(() => {
|
|
||||||
axios.post('/api/modules/downloads?dlclient=deluge', { ...delugeService }).then((res) => {
|
|
||||||
setDelugeTorrents(res.data.torrents);
|
|
||||||
});
|
|
||||||
}, 3000);
|
|
||||||
}
|
|
||||||
}, [config.modules]);
|
|
||||||
|
|
||||||
if (!qBittorrentService && !delugeService) {
|
if (downloadServices.length === 0) {
|
||||||
return (
|
return (
|
||||||
<Group direction="column">
|
<Group direction="column">
|
||||||
<Title order={3}>No supported download clients found!</Title>
|
<Title order={3}>No supported download clients found!</Title>
|
||||||
@@ -63,7 +73,7 @@ export default function DownloadComponent() {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (qBittorrentTorrents.length === 0 && delugeTorrents.length === 0) {
|
if (isLoading) {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Skeleton height={40} mt={10} />
|
<Skeleton height={40} mt={10} />
|
||||||
@@ -74,68 +84,106 @@ export default function DownloadComponent() {
|
|||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
const DEVICE_WIDTH = 576;
|
||||||
const ths = (
|
const ths = (
|
||||||
<tr>
|
<tr>
|
||||||
<th>Name</th>
|
<th>Name</th>
|
||||||
<th>Download</th>
|
<th>Size</th>
|
||||||
<th>Upload</th>
|
{width > 576 ? <th>Down</th> : ''}
|
||||||
|
{width > 576 ? <th>Up</th> : ''}
|
||||||
|
<th>ETA</th>
|
||||||
<th>Progress</th>
|
<th>Progress</th>
|
||||||
</tr>
|
</tr>
|
||||||
);
|
);
|
||||||
// Loop over qBittorrent torrents merging with deluge torrents
|
// Convert Seconds to readable format.
|
||||||
const torrents: NormalizedTorrent[] = [];
|
function calculateETA(givenSeconds: number) {
|
||||||
delugeTorrents.forEach((delugeTorrent) =>
|
// If its superior than one day return > 1 day
|
||||||
torrents.push({ ...delugeTorrent, progress: delugeTorrent.progress / 100 })
|
if (givenSeconds > 86400) {
|
||||||
);
|
return '> 1 day';
|
||||||
qBittorrentTorrents.forEach((torrent) => torrents.push(torrent));
|
|
||||||
const rows = torrents.map((torrent) => {
|
|
||||||
if (torrent.progress === 1 && hideComplete) {
|
|
||||||
return [];
|
|
||||||
}
|
}
|
||||||
const downloadSpeed = torrent.downloadSpeed / 1024 / 1024;
|
// Transform the givenSeconds into a readable format. e.g. 1h 2m 3s
|
||||||
const uploadSpeed = torrent.uploadSpeed / 1024 / 1024;
|
const hours = Math.floor(givenSeconds / 3600);
|
||||||
return (
|
const minutes = Math.floor((givenSeconds % 3600) / 60);
|
||||||
<tr key={torrent.id}>
|
const seconds = Math.floor(givenSeconds % 60);
|
||||||
<td>
|
// Only show hours if it's greater than 0.
|
||||||
<Tooltip position="top" label={torrent.name}>
|
const hoursString = hours > 0 ? `${hours}h ` : '';
|
||||||
<Text
|
const minutesString = minutes > 0 ? `${minutes}m ` : '';
|
||||||
style={{
|
const secondsString = seconds > 0 ? `${seconds}s` : '';
|
||||||
maxWidth: '30vw',
|
return `${hoursString}${minutesString}${secondsString}`;
|
||||||
}}
|
}
|
||||||
size="xs"
|
// Loop over qBittorrent torrents merging with deluge torrents
|
||||||
lineClamp={1}
|
const rows = torrents
|
||||||
>
|
.filter((torrent) => !(torrent.progress === 1 && hideComplete))
|
||||||
{torrent.name}
|
.map((torrent) => {
|
||||||
</Text>
|
const downloadSpeed = torrent.downloadSpeed / 1024 / 1024;
|
||||||
</Tooltip>
|
const uploadSpeed = torrent.uploadSpeed / 1024 / 1024;
|
||||||
</td>
|
const size = torrent.totalSelected;
|
||||||
<td>
|
return (
|
||||||
<Text size="xs">{downloadSpeed > 0 ? `${downloadSpeed.toFixed(1)} Mb/s` : '-'}</Text>
|
<tr key={torrent.id}>
|
||||||
</td>
|
<td>
|
||||||
<td>
|
<Tooltip position="top" label={torrent.name}>
|
||||||
<Text size="xs">{uploadSpeed > 0 ? `${uploadSpeed.toFixed(1)} Mb/s` : '-'}</Text>
|
<Text
|
||||||
</td>
|
style={{
|
||||||
<td>
|
maxWidth: '30vw',
|
||||||
<Text>{(torrent.progress * 100).toFixed(1)}%</Text>
|
}}
|
||||||
<Progress
|
size="xs"
|
||||||
radius="lg"
|
lineClamp={1}
|
||||||
color={torrent.progress === 1 ? 'green' : 'blue'}
|
>
|
||||||
value={torrent.progress * 100}
|
{torrent.name}
|
||||||
size="lg"
|
</Text>
|
||||||
/>
|
</Tooltip>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
<td>
|
||||||
);
|
<Text size="xs">{humanFileSize(size)}</Text>
|
||||||
});
|
</td>
|
||||||
|
{width > 576 ? (
|
||||||
|
<td>
|
||||||
|
<Text size="xs">{downloadSpeed > 0 ? `${downloadSpeed.toFixed(1)} Mb/s` : '-'}</Text>
|
||||||
|
</td>
|
||||||
|
) : (
|
||||||
|
''
|
||||||
|
)}
|
||||||
|
{width > 576 ? (
|
||||||
|
<td>
|
||||||
|
<Text size="xs">{uploadSpeed > 0 ? `${uploadSpeed.toFixed(1)} Mb/s` : '-'}</Text>
|
||||||
|
</td>
|
||||||
|
) : (
|
||||||
|
''
|
||||||
|
)}
|
||||||
|
<td>
|
||||||
|
<Text size="xs">{torrent.eta <= 0 ? '∞' : calculateETA(torrent.eta)}</Text>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<Text>{(torrent.progress * 100).toFixed(1)}%</Text>
|
||||||
|
<Progress
|
||||||
|
radius="lg"
|
||||||
|
color={
|
||||||
|
torrent.progress === 1 ? 'green' : torrent.state === 'paused' ? 'yellow' : 'blue'
|
||||||
|
}
|
||||||
|
value={torrent.progress * 100}
|
||||||
|
size="lg"
|
||||||
|
/>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
const easteregg = (
|
||||||
|
<Center style={{ display: 'flex', alignItems: 'center', justifyContent: 'center' }}>
|
||||||
|
<Image fit="cover" height={300} src="https://danjohnvelasco.github.io/images/empty.png" />
|
||||||
|
</Center>
|
||||||
|
);
|
||||||
return (
|
return (
|
||||||
<Group noWrap grow direction="column">
|
<Group noWrap grow direction="column" mt="xl">
|
||||||
<Title order={4}>Your torrents</Title>
|
|
||||||
<ScrollArea sx={{ height: 300 }}>
|
<ScrollArea sx={{ height: 300 }}>
|
||||||
<Table highlightOnHover>
|
{rows.length > 0 ? (
|
||||||
<thead>{ths}</thead>
|
<Table highlightOnHover>
|
||||||
<tbody>{rows}</tbody>
|
<thead>{ths}</thead>
|
||||||
</Table>
|
<tbody>{rows}</tbody>
|
||||||
|
</Table>
|
||||||
|
) : (
|
||||||
|
easteregg
|
||||||
|
)}
|
||||||
</ScrollArea>
|
</ScrollArea>
|
||||||
</Group>
|
</Group>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -8,39 +8,9 @@ import { Datum, ResponsiveLine } from '@nivo/line';
|
|||||||
import { useListState } from '@mantine/hooks';
|
import { useListState } from '@mantine/hooks';
|
||||||
import { AddItemShelfButton } from '../../AppShelf/AddAppShelfItem';
|
import { AddItemShelfButton } from '../../AppShelf/AddAppShelfItem';
|
||||||
import { useConfig } from '../../../tools/state';
|
import { useConfig } from '../../../tools/state';
|
||||||
|
import { humanFileSize } from '../../../tools/humanFileSize';
|
||||||
import { IModule } from '../modules';
|
import { IModule } from '../modules';
|
||||||
|
import { useSetSafeInterval } from '../../../tools/hooks/useSetSafeInterval';
|
||||||
/**
|
|
||||||
* Format bytes as human-readable text.
|
|
||||||
*
|
|
||||||
* @param bytes Number of bytes.
|
|
||||||
* @param si True to use metric (SI) units, aka powers of 1000. False to use
|
|
||||||
* binary (IEC), aka powers of 1024.
|
|
||||||
* @param dp Number of decimal places to display.
|
|
||||||
*
|
|
||||||
* @return Formatted string.
|
|
||||||
*/
|
|
||||||
function humanFileSize(initialBytes: number, si = true, dp = 1) {
|
|
||||||
const thresh = si ? 1000 : 1024;
|
|
||||||
let bytes = initialBytes;
|
|
||||||
|
|
||||||
if (Math.abs(bytes) < thresh) {
|
|
||||||
return `${bytes} B`;
|
|
||||||
}
|
|
||||||
|
|
||||||
const units = si
|
|
||||||
? ['kb', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB']
|
|
||||||
: ['KiB', 'MiB', 'GiB', 'TiB', 'PiB', 'EiB', 'ZiB', 'YiB'];
|
|
||||||
let u = -1;
|
|
||||||
const r = 10 ** dp;
|
|
||||||
|
|
||||||
do {
|
|
||||||
bytes /= thresh;
|
|
||||||
u += 1;
|
|
||||||
} while (Math.round(Math.abs(bytes) * r) / r >= thresh && u < units.length - 1);
|
|
||||||
|
|
||||||
return `${bytes.toFixed(dp)} ${units[u]}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const TotalDownloadsModule: IModule = {
|
export const TotalDownloadsModule: IModule = {
|
||||||
title: 'Download Speed',
|
title: 'Download Speed',
|
||||||
@@ -56,42 +26,29 @@ interface torrentHistory {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function TotalDownloadsComponent() {
|
export default function TotalDownloadsComponent() {
|
||||||
|
const setSafeInterval = useSetSafeInterval();
|
||||||
const { config } = useConfig();
|
const { config } = useConfig();
|
||||||
const qBittorrentService = config.services
|
const downloadServices =
|
||||||
.filter((service) => service.type === 'qBittorrent')
|
config.services.filter(
|
||||||
.at(0);
|
(service) =>
|
||||||
const delugeService = config.services.filter((service) => service.type === 'Deluge').at(0);
|
service.type === 'qBittorrent' ||
|
||||||
|
service.type === 'Transmission' ||
|
||||||
|
service.type === 'Deluge'
|
||||||
|
) ?? [];
|
||||||
|
|
||||||
const [delugeTorrents, setDelugeTorrents] = useState<NormalizedTorrent[]>([]);
|
|
||||||
const [torrentHistory, torrentHistoryHandlers] = useListState<torrentHistory>([]);
|
const [torrentHistory, torrentHistoryHandlers] = useListState<torrentHistory>([]);
|
||||||
const [qBittorrentTorrents, setqBittorrentTorrents] = useState<NormalizedTorrent[]>([]);
|
const [torrents, setTorrents] = useState<NormalizedTorrent[]>([]);
|
||||||
|
|
||||||
const torrents: NormalizedTorrent[] = [];
|
|
||||||
delugeTorrents.forEach((delugeTorrent) =>
|
|
||||||
torrents.push({ ...delugeTorrent, progress: delugeTorrent.progress / 100 })
|
|
||||||
);
|
|
||||||
qBittorrentTorrents.forEach((torrent) => torrents.push(torrent));
|
|
||||||
|
|
||||||
const totalDownloadSpeed = torrents.reduce((acc, torrent) => acc + torrent.downloadSpeed, 0);
|
const totalDownloadSpeed = torrents.reduce((acc, torrent) => acc + torrent.downloadSpeed, 0);
|
||||||
const totalUploadSpeed = torrents.reduce((acc, torrent) => acc + torrent.uploadSpeed, 0);
|
const totalUploadSpeed = torrents.reduce((acc, torrent) => acc + torrent.uploadSpeed, 0);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const interval = setInterval(() => {
|
if (downloadServices.length === 0) return;
|
||||||
// Get the current download speed of qBittorrent.
|
setSafeInterval(() => {
|
||||||
if (qBittorrentService) {
|
axios.post('/api/modules/downloads', { config }).then((response) => {
|
||||||
axios
|
setTorrents(response.data);
|
||||||
.post('/api/modules/downloads?dlclient=qbit', { ...qBittorrentService })
|
});
|
||||||
.then((res) => {
|
|
||||||
setqBittorrentTorrents(res.data.torrents);
|
|
||||||
});
|
|
||||||
if (delugeService) {
|
|
||||||
axios.post('/api/modules/downloads?dlclient=deluge', { ...delugeService }).then((res) => {
|
|
||||||
setDelugeTorrents(res.data.torrents);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, 1000);
|
}, 1000);
|
||||||
}, [config.modules]);
|
}, [config.services]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
torrentHistoryHandlers.append({
|
torrentHistoryHandlers.append({
|
||||||
@@ -101,7 +58,7 @@ export default function TotalDownloadsComponent() {
|
|||||||
});
|
});
|
||||||
}, [totalDownloadSpeed, totalUploadSpeed]);
|
}, [totalDownloadSpeed, totalUploadSpeed]);
|
||||||
|
|
||||||
if (!qBittorrentService && !delugeService) {
|
if (downloadServices.length === 0) {
|
||||||
return (
|
return (
|
||||||
<Group direction="column">
|
<Group direction="column">
|
||||||
<Title order={4}>No supported download clients found!</Title>
|
<Title order={4}>No supported download clients found!</Title>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
export * from './date';
|
|
||||||
export * from './calendar';
|
export * from './calendar';
|
||||||
export * from './search';
|
export * from './dash.';
|
||||||
export * from './ping';
|
export * from './date';
|
||||||
export * from './weather';
|
|
||||||
export * from './downloads';
|
export * from './downloads';
|
||||||
export * from './system';
|
export * from './ping';
|
||||||
|
export * from './search';
|
||||||
|
export * from './weather';
|
||||||
|
|||||||
@@ -1,10 +1,18 @@
|
|||||||
import { Button, Card, Group, Menu, Switch, TextInput, useMantineTheme } from '@mantine/core';
|
import {
|
||||||
|
Button,
|
||||||
|
Card,
|
||||||
|
Group,
|
||||||
|
Menu,
|
||||||
|
MultiSelect,
|
||||||
|
Switch,
|
||||||
|
TextInput,
|
||||||
|
useMantineColorScheme,
|
||||||
|
} from '@mantine/core';
|
||||||
import { useConfig } from '../../tools/state';
|
import { useConfig } from '../../tools/state';
|
||||||
import { IModule } from './modules';
|
import { IModule } from './modules';
|
||||||
|
|
||||||
function getItems(module: IModule) {
|
function getItems(module: IModule) {
|
||||||
const { config, setConfig } = useConfig();
|
const { config, setConfig } = useConfig();
|
||||||
const enabledModules = config.modules ?? {};
|
|
||||||
const items: JSX.Element[] = [];
|
const items: JSX.Element[] = [];
|
||||||
if (module.options) {
|
if (module.options) {
|
||||||
const keys = Object.keys(module.options);
|
const keys = Object.keys(module.options);
|
||||||
@@ -15,6 +23,38 @@ function getItems(module: IModule) {
|
|||||||
types.forEach((type, index) => {
|
types.forEach((type, index) => {
|
||||||
const optionName = `${module.title}.${keys[index]}`;
|
const optionName = `${module.title}.${keys[index]}`;
|
||||||
const moduleInConfig = config.modules?.[module.title];
|
const moduleInConfig = config.modules?.[module.title];
|
||||||
|
if (type === 'object') {
|
||||||
|
items.push(
|
||||||
|
<MultiSelect
|
||||||
|
label={module.options?.[keys[index]].name}
|
||||||
|
data={module.options?.[keys[index]].options ?? []}
|
||||||
|
defaultValue={
|
||||||
|
(moduleInConfig?.options?.[keys[index]]?.value as string[]) ??
|
||||||
|
(values[index].value as string[]) ??
|
||||||
|
[]
|
||||||
|
}
|
||||||
|
searchable
|
||||||
|
onChange={(value) => {
|
||||||
|
setConfig({
|
||||||
|
...config,
|
||||||
|
modules: {
|
||||||
|
...config.modules,
|
||||||
|
[module.title]: {
|
||||||
|
...moduleInConfig,
|
||||||
|
options: {
|
||||||
|
...moduleInConfig?.options,
|
||||||
|
[keys[index]]: {
|
||||||
|
...moduleInConfig?.options?.[keys[index]],
|
||||||
|
value,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
}
|
||||||
if (type === 'string') {
|
if (type === 'string') {
|
||||||
items.push(
|
items.push(
|
||||||
<form
|
<form
|
||||||
@@ -44,7 +84,11 @@ function getItems(module: IModule) {
|
|||||||
id={optionName}
|
id={optionName}
|
||||||
name={optionName}
|
name={optionName}
|
||||||
label={values[index].name}
|
label={values[index].name}
|
||||||
defaultValue={(moduleInConfig?.options?.[keys[index]]?.value as string) ?? ''}
|
defaultValue={
|
||||||
|
(moduleInConfig?.options?.[keys[index]]?.value as string) ??
|
||||||
|
(values[index].value as string) ??
|
||||||
|
''
|
||||||
|
}
|
||||||
onChange={(e) => {}}
|
onChange={(e) => {}}
|
||||||
/>
|
/>
|
||||||
|
|
||||||
@@ -59,7 +103,9 @@ function getItems(module: IModule) {
|
|||||||
<Switch
|
<Switch
|
||||||
defaultChecked={
|
defaultChecked={
|
||||||
// Set default checked to the value of the option if it exists
|
// Set default checked to the value of the option if it exists
|
||||||
(moduleInConfig?.options?.[keys[index]]?.value as boolean) ?? false
|
(moduleInConfig?.options?.[keys[index]]?.value as boolean) ??
|
||||||
|
(values[index].value as boolean) ??
|
||||||
|
false
|
||||||
}
|
}
|
||||||
key={keys[index]}
|
key={keys[index]}
|
||||||
onClick={(e) => {
|
onClick={(e) => {
|
||||||
@@ -91,18 +137,50 @@ function getItems(module: IModule) {
|
|||||||
|
|
||||||
export function ModuleWrapper(props: any) {
|
export function ModuleWrapper(props: any) {
|
||||||
const { module }: { module: IModule } = props;
|
const { module }: { module: IModule } = props;
|
||||||
|
const { colorScheme } = useMantineColorScheme();
|
||||||
const { config, setConfig } = useConfig();
|
const { config, setConfig } = useConfig();
|
||||||
const enabledModules = config.modules ?? {};
|
const enabledModules = config.modules ?? {};
|
||||||
// Remove 'Module' from enabled modules titles
|
// Remove 'Module' from enabled modules titles
|
||||||
const isShown = enabledModules[module.title]?.enabled ?? false;
|
const isShown = enabledModules[module.title]?.enabled ?? false;
|
||||||
const theme = useMantineTheme();
|
|
||||||
const items: JSX.Element[] = getItems(module);
|
|
||||||
|
|
||||||
if (!isShown) {
|
if (!isShown) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card {...props} hidden={!isShown} withBorder radius="lg" shadow="sm">
|
<Card
|
||||||
|
{...props}
|
||||||
|
hidden={!isShown}
|
||||||
|
withBorder
|
||||||
|
radius="lg"
|
||||||
|
shadow="sm"
|
||||||
|
style={{
|
||||||
|
background: `rgba(${colorScheme === 'dark' ? '37, 38, 43,' : '255, 255, 255,'} \
|
||||||
|
${(config.settings.appOpacity || 100) / 100}`,
|
||||||
|
borderColor: `rgba(${colorScheme === 'dark' ? '37, 38, 43,' : '233, 236, 239,'} \
|
||||||
|
${(config.settings.appOpacity || 100) / 100}`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<ModuleMenu
|
||||||
|
module={module}
|
||||||
|
styles={{
|
||||||
|
root: {
|
||||||
|
position: 'absolute',
|
||||||
|
top: 12,
|
||||||
|
right: 12,
|
||||||
|
},
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<module.component />
|
||||||
|
</Card>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function ModuleMenu(props: any) {
|
||||||
|
const { module, styles } = props;
|
||||||
|
const items: JSX.Element[] = getItems(module);
|
||||||
|
return (
|
||||||
|
<>
|
||||||
{module.options && (
|
{module.options && (
|
||||||
<Menu
|
<Menu
|
||||||
size="lg"
|
size="lg"
|
||||||
@@ -112,9 +190,7 @@ export function ModuleWrapper(props: any) {
|
|||||||
position="left"
|
position="left"
|
||||||
styles={{
|
styles={{
|
||||||
root: {
|
root: {
|
||||||
position: 'absolute',
|
...props?.styles?.root,
|
||||||
top: 15,
|
|
||||||
right: 15,
|
|
||||||
},
|
},
|
||||||
body: {
|
body: {
|
||||||
// Add shadow and elevation to the body
|
// Add shadow and elevation to the body
|
||||||
@@ -128,7 +204,6 @@ export function ModuleWrapper(props: any) {
|
|||||||
))}
|
))}
|
||||||
</Menu>
|
</Menu>
|
||||||
)}
|
)}
|
||||||
<module.component />
|
</>
|
||||||
</Card>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -16,5 +16,6 @@ interface Option {
|
|||||||
|
|
||||||
export interface OptionValues {
|
export interface OptionValues {
|
||||||
name: string;
|
name: string;
|
||||||
value: boolean | string;
|
value: boolean | string | string[];
|
||||||
|
options?: string[];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,6 +11,8 @@ const service: serviceItem = {
|
|||||||
name: 'YouTube',
|
name: 'YouTube',
|
||||||
icon: 'https://cdn.jsdelivr.net/gh/walkxhub/dashboard-icons/png/youtube.png',
|
icon: 'https://cdn.jsdelivr.net/gh/walkxhub/dashboard-icons/png/youtube.png',
|
||||||
url: 'https://youtube.com/',
|
url: 'https://youtube.com/',
|
||||||
|
status: ['200'],
|
||||||
|
newTab: false,
|
||||||
};
|
};
|
||||||
|
|
||||||
export const Default = (args: any) => <PingComponent service={service} />;
|
export const Default = (args: any) => <PingComponent service={service} />;
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { Indicator, Tooltip } from '@mantine/core';
|
import { Indicator, Tooltip } from '@mantine/core';
|
||||||
import axios from 'axios';
|
import axios, { AxiosResponse } from 'axios';
|
||||||
import { motion } from 'framer-motion';
|
import { motion } from 'framer-motion';
|
||||||
import { useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
import { IconPlug as Plug } from '@tabler/icons';
|
import { IconPlug as Plug } from '@tabler/icons';
|
||||||
@@ -19,20 +19,39 @@ export default function PingComponent(props: any) {
|
|||||||
|
|
||||||
const { url }: { url: string } = props;
|
const { url }: { url: string } = props;
|
||||||
const [isOnline, setOnline] = useState<State>('loading');
|
const [isOnline, setOnline] = useState<State>('loading');
|
||||||
|
const [response, setResponse] = useState(500);
|
||||||
const exists = config.modules?.[PingModule.title]?.enabled ?? false;
|
const exists = config.modules?.[PingModule.title]?.enabled ?? false;
|
||||||
|
|
||||||
|
function statusCheck(response: AxiosResponse) {
|
||||||
|
const { status }: { status: string[] } = props;
|
||||||
|
//Default Status
|
||||||
|
let acceptableStatus = ['200'];
|
||||||
|
if (status !== undefined && status.length) {
|
||||||
|
acceptableStatus = status;
|
||||||
|
}
|
||||||
|
// Checks if reported status is in acceptable status array
|
||||||
|
if (acceptableStatus.indexOf(response.status.toString()) >= 0) {
|
||||||
|
setOnline('online');
|
||||||
|
setResponse(response.status);
|
||||||
|
} else {
|
||||||
|
setOnline('down');
|
||||||
|
setResponse(response.status);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!exists) {
|
if (!exists) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
axios
|
axios
|
||||||
.get('/api/modules/ping', { params: { url } })
|
.get('/api/modules/ping', { params: { url } })
|
||||||
.then(() => {
|
.then((response) => {
|
||||||
setOnline('online');
|
statusCheck(response);
|
||||||
})
|
})
|
||||||
.catch(() => {
|
.catch((error) => {
|
||||||
setOnline('down');
|
statusCheck(error.response);
|
||||||
});
|
});
|
||||||
}, []);
|
}, [config.modules?.[PingModule.title]?.enabled]);
|
||||||
if (!exists) {
|
if (!exists) {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
@@ -40,7 +59,13 @@ export default function PingComponent(props: any) {
|
|||||||
<Tooltip
|
<Tooltip
|
||||||
radius="lg"
|
radius="lg"
|
||||||
style={{ position: 'absolute', bottom: 20, right: 20 }}
|
style={{ position: 'absolute', bottom: 20, right: 20 }}
|
||||||
label={isOnline === 'loading' ? 'Loading...' : isOnline === 'online' ? 'Online' : 'Offline'}
|
label={
|
||||||
|
isOnline === 'loading'
|
||||||
|
? 'Loading...'
|
||||||
|
: isOnline === 'online'
|
||||||
|
? `Online - ${response}`
|
||||||
|
: `Offline - ${response}`
|
||||||
|
}
|
||||||
>
|
>
|
||||||
<motion.div
|
<motion.div
|
||||||
animate={{
|
animate={{
|
||||||
|
|||||||
@@ -1,11 +1,12 @@
|
|||||||
import { TextInput, Kbd, createStyles, Text, Popover } from '@mantine/core';
|
import { Kbd, createStyles, Text, Popover, Autocomplete, Tooltip } from '@mantine/core';
|
||||||
import { useForm, useHotkeys } from '@mantine/hooks';
|
import { useDebouncedValue, useForm, useHotkeys } from '@mantine/hooks';
|
||||||
import { useRef, useState } from 'react';
|
import { useEffect, useRef, useState } from 'react';
|
||||||
import {
|
import {
|
||||||
IconSearch as Search,
|
IconSearch as Search,
|
||||||
IconBrandYoutube as BrandYoutube,
|
IconBrandYoutube as BrandYoutube,
|
||||||
IconDownload as Download,
|
IconDownload as Download,
|
||||||
} from '@tabler/icons';
|
} from '@tabler/icons';
|
||||||
|
import axios from 'axios';
|
||||||
import { useConfig } from '../../../tools/state';
|
import { useConfig } from '../../../tools/state';
|
||||||
import { IModule } from '../modules';
|
import { IModule } from '../modules';
|
||||||
|
|
||||||
@@ -32,8 +33,22 @@ export default function SearchBar(props: any) {
|
|||||||
const [icon, setIcon] = useState(<Search />);
|
const [icon, setIcon] = useState(<Search />);
|
||||||
const queryUrl = config.settings.searchUrl ?? 'https://www.google.com/search?q=';
|
const queryUrl = config.settings.searchUrl ?? 'https://www.google.com/search?q=';
|
||||||
const textInput = useRef<HTMLInputElement>();
|
const textInput = useRef<HTMLInputElement>();
|
||||||
useHotkeys([['ctrl+K', () => textInput.current && textInput.current.focus()]]);
|
// Find a service with the type of 'Overseerr'
|
||||||
|
const form = useForm({
|
||||||
|
initialValues: {
|
||||||
|
query: '',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const [debounced, cancel] = useDebouncedValue(form.values.query, 250);
|
||||||
|
const [results, setResults] = useState<any[]>([]);
|
||||||
|
useEffect(() => {
|
||||||
|
if (form.values.query !== debounced || form.values.query === '') return;
|
||||||
|
axios
|
||||||
|
.get(`/api/modules/search?q=${form.values.query}`)
|
||||||
|
.then((res) => setResults(res.data ?? []));
|
||||||
|
}, [debounced]);
|
||||||
|
useHotkeys([['ctrl+K', () => textInput.current && textInput.current.focus()]]);
|
||||||
const { classes, cx } = useStyles();
|
const { classes, cx } = useStyles();
|
||||||
const rightSection = (
|
const rightSection = (
|
||||||
<div className={classes.hide}>
|
<div className={classes.hide}>
|
||||||
@@ -43,12 +58,6 @@ export default function SearchBar(props: any) {
|
|||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
|
||||||
const form = useForm({
|
|
||||||
initialValues: {
|
|
||||||
query: '',
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
// If enabled modules doesn't contain the module, return null
|
// If enabled modules doesn't contain the module, return null
|
||||||
// If module in enabled
|
// If module in enabled
|
||||||
|
|
||||||
@@ -57,6 +66,10 @@ export default function SearchBar(props: any) {
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const autocompleteData = results.map((result) => ({
|
||||||
|
label: result.phrase,
|
||||||
|
value: result.phrase,
|
||||||
|
}));
|
||||||
return (
|
return (
|
||||||
<form
|
<form
|
||||||
onChange={() => {
|
onChange={() => {
|
||||||
@@ -83,43 +96,32 @@ export default function SearchBar(props: any) {
|
|||||||
} else if (isTorrent) {
|
} else if (isTorrent) {
|
||||||
window.open(`https://bitsearch.to/search?q=${query.substring(3)}`);
|
window.open(`https://bitsearch.to/search?q=${query.substring(3)}`);
|
||||||
} else {
|
} else {
|
||||||
window.open(`${queryUrl}${values.query}`);
|
window.open(
|
||||||
|
`${
|
||||||
|
queryUrl.includes('%s')
|
||||||
|
? queryUrl.replace('%s', values.query)
|
||||||
|
: queryUrl + values.query
|
||||||
|
}`
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}, 20);
|
}, 20);
|
||||||
})}
|
})}
|
||||||
>
|
>
|
||||||
<Popover
|
<Autocomplete
|
||||||
opened={opened}
|
autoFocus
|
||||||
position="bottom"
|
variant="filled"
|
||||||
placement="start"
|
data={autocompleteData}
|
||||||
width={260}
|
icon={icon}
|
||||||
withArrow
|
ref={textInput}
|
||||||
|
rightSectionWidth={90}
|
||||||
|
rightSection={rightSection}
|
||||||
radius="md"
|
radius="md"
|
||||||
trapFocus={false}
|
size="md"
|
||||||
transition="pop-bottom-right"
|
styles={{ rightSection: { pointerEvents: 'none' } }}
|
||||||
onFocusCapture={() => setOpened(true)}
|
placeholder="Search the web..."
|
||||||
onBlurCapture={() => setOpened(false)}
|
{...props}
|
||||||
target={
|
{...form.getInputProps('query')}
|
||||||
<TextInput
|
/>
|
||||||
variant="filled"
|
|
||||||
icon={icon}
|
|
||||||
ref={textInput}
|
|
||||||
rightSectionWidth={90}
|
|
||||||
rightSection={rightSection}
|
|
||||||
radius="md"
|
|
||||||
size="md"
|
|
||||||
styles={{ rightSection: { pointerEvents: 'none' } }}
|
|
||||||
placeholder="Search the web..."
|
|
||||||
{...props}
|
|
||||||
{...form.getInputProps('query')}
|
|
||||||
/>
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<Text>
|
|
||||||
Tip: Use the prefixes <b>!yt</b> and <b>!t</b> in front of your query to search on YouTube
|
|
||||||
or for a Torrent respectively.
|
|
||||||
</Text>
|
|
||||||
</Popover>
|
|
||||||
</form>
|
</form>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,16 +1,11 @@
|
|||||||
import {
|
import { Center, Group, RingProgress, Title, useMantineTheme } from '@mantine/core';
|
||||||
Center,
|
|
||||||
Group,
|
|
||||||
RingProgress,
|
|
||||||
Title,
|
|
||||||
useMantineTheme,
|
|
||||||
} from '@mantine/core';
|
|
||||||
import { IconCpu } from '@tabler/icons';
|
import { IconCpu } from '@tabler/icons';
|
||||||
import { useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import si from 'systeminformation';
|
import si from 'systeminformation';
|
||||||
import { useListState } from '@mantine/hooks';
|
import { useListState } from '@mantine/hooks';
|
||||||
import { IModule } from '../modules';
|
import { IModule } from '../modules';
|
||||||
|
import { useSetSafeInterval } from '../../../tools/hooks/useSetSafeInterval';
|
||||||
|
|
||||||
export const SystemModule: IModule = {
|
export const SystemModule: IModule = {
|
||||||
title: 'System info',
|
title: 'System info',
|
||||||
@@ -28,13 +23,13 @@ interface ApiResponse {
|
|||||||
|
|
||||||
export default function SystemInfo(args: any) {
|
export default function SystemInfo(args: any) {
|
||||||
const [data, setData] = useState<ApiResponse>();
|
const [data, setData] = useState<ApiResponse>();
|
||||||
|
const setSafeInterval = useSetSafeInterval();
|
||||||
// Refresh data every second
|
// Refresh data every second
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setInterval(() => {
|
setSafeInterval(() => {
|
||||||
axios.get('/api/modules/systeminfo').then((res) => setData(res.data));
|
axios.get('/api/modules/systeminfo').then((res) => setData(res.data));
|
||||||
}, 1000);
|
}, 1000);
|
||||||
}, [args]);
|
}, []);
|
||||||
|
|
||||||
// Update data every time data changes
|
// Update data every time data changes
|
||||||
const [cpuLoadHistory, cpuLoadHistoryHandlers] =
|
const [cpuLoadHistory, cpuLoadHistoryHandlers] =
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { Group, Space, Title, Tooltip } from '@mantine/core';
|
import { Group, Space, Title, Tooltip, Skeleton } from '@mantine/core';
|
||||||
import axios from 'axios';
|
import axios from 'axios';
|
||||||
import { useEffect, useState } from 'react';
|
import { useEffect, useState } from 'react';
|
||||||
import {
|
import {
|
||||||
@@ -18,7 +18,7 @@ import { IModule } from '../modules';
|
|||||||
import { WeatherResponse } from './WeatherInterface';
|
import { WeatherResponse } from './WeatherInterface';
|
||||||
|
|
||||||
export const WeatherModule: IModule = {
|
export const WeatherModule: IModule = {
|
||||||
title: 'Weather (beta)',
|
title: 'Weather',
|
||||||
description: 'Look up the current weather in your location',
|
description: 'Look up the current weather in your location',
|
||||||
icon: Sun,
|
icon: Sun,
|
||||||
component: WeatherComponent,
|
component: WeatherComponent,
|
||||||
@@ -29,7 +29,7 @@ export const WeatherModule: IModule = {
|
|||||||
},
|
},
|
||||||
location: {
|
location: {
|
||||||
name: 'Current location',
|
name: 'Current location',
|
||||||
value: '',
|
value: 'Paris',
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
@@ -135,7 +135,7 @@ export default function WeatherComponent(props: any) {
|
|||||||
const { config } = useConfig();
|
const { config } = useConfig();
|
||||||
const [weather, setWeather] = useState({} as WeatherResponse);
|
const [weather, setWeather] = useState({} as WeatherResponse);
|
||||||
const cityInput: string =
|
const cityInput: string =
|
||||||
(config?.modules?.[WeatherModule.title]?.options?.location?.value as string) ?? '';
|
(config?.modules?.[WeatherModule.title]?.options?.location?.value as string) ?? 'Paris';
|
||||||
const isFahrenheit: boolean =
|
const isFahrenheit: boolean =
|
||||||
(config?.modules?.[WeatherModule.title]?.options?.freedomunit?.value as boolean) ?? false;
|
(config?.modules?.[WeatherModule.title]?.options?.freedomunit?.value as boolean) ?? false;
|
||||||
|
|
||||||
@@ -157,10 +157,21 @@ export default function WeatherComponent(props: any) {
|
|||||||
});
|
});
|
||||||
}, [cityInput]);
|
}, [cityInput]);
|
||||||
if (!weather.current_weather) {
|
if (!weather.current_weather) {
|
||||||
return null;
|
return (
|
||||||
|
<>
|
||||||
|
<Skeleton height={40} width={100} mb="xl" />
|
||||||
|
<Group noWrap direction="row">
|
||||||
|
<Skeleton height={50} circle />
|
||||||
|
<Group>
|
||||||
|
<Skeleton height={25} width={70} mr="lg" />
|
||||||
|
<Skeleton height={25} width={70} />
|
||||||
|
</Group>
|
||||||
|
</Group>
|
||||||
|
</>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
function usePerferedUnit(value: number): string {
|
function usePerferedUnit(value: number): string {
|
||||||
return isFahrenheit ? `${(value * (9 / 5)).toFixed(1)}°F` : `${value.toFixed(1)}°C`;
|
return isFahrenheit ? `${(value * (9 / 5) + 32).toFixed(1)}°F` : `${value.toFixed(1)}°C`;
|
||||||
}
|
}
|
||||||
return (
|
return (
|
||||||
<Group p="sm" spacing="xs" direction="column">
|
<Group p="sm" spacing="xs" direction="column">
|
||||||
|
|||||||
15
src/middleware.ts
Normal file
15
src/middleware.ts
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
import { NextFetchEvent, NextRequest, NextResponse } from 'next/server';
|
||||||
|
|
||||||
|
export function middleware(req: NextRequest, ev: NextFetchEvent) {
|
||||||
|
const ok = req.cookies.get('password') === process.env.PASSWORD;
|
||||||
|
const url = req.nextUrl.clone();
|
||||||
|
if (
|
||||||
|
!ok &&
|
||||||
|
url.pathname !== '/login' &&
|
||||||
|
process.env.PASSWORD &&
|
||||||
|
url.pathname !== '/api/configs/tryPassword'
|
||||||
|
) {
|
||||||
|
url.pathname = '/login';
|
||||||
|
}
|
||||||
|
return NextResponse.rewrite(url);
|
||||||
|
}
|
||||||
@@ -6,6 +6,7 @@ import AppShelf from '../components/AppShelf/AppShelf';
|
|||||||
import LoadConfigComponent from '../components/Config/LoadConfig';
|
import LoadConfigComponent from '../components/Config/LoadConfig';
|
||||||
import { Config } from '../tools/types';
|
import { Config } from '../tools/types';
|
||||||
import { useConfig } from '../tools/state';
|
import { useConfig } from '../tools/state';
|
||||||
|
import Layout from '../components/layout/Layout';
|
||||||
|
|
||||||
export async function getServerSideProps(
|
export async function getServerSideProps(
|
||||||
context: GetServerSidePropsContext
|
context: GetServerSidePropsContext
|
||||||
@@ -46,9 +47,9 @@ export default function HomePage(props: any) {
|
|||||||
setConfig(initialConfig);
|
setConfig(initialConfig);
|
||||||
}, [initialConfig]);
|
}, [initialConfig]);
|
||||||
return (
|
return (
|
||||||
<>
|
<Layout>
|
||||||
<AppShelf />
|
<AppShelf />
|
||||||
<LoadConfigComponent />
|
<LoadConfigComponent />
|
||||||
</>
|
</Layout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,18 +3,30 @@ import { useState } from 'react';
|
|||||||
import { AppProps } from 'next/app';
|
import { AppProps } from 'next/app';
|
||||||
import { getCookie, setCookies } from 'cookies-next';
|
import { getCookie, setCookies } from 'cookies-next';
|
||||||
import Head from 'next/head';
|
import Head from 'next/head';
|
||||||
import { MantineProvider, ColorScheme, ColorSchemeProvider } from '@mantine/core';
|
import { MantineProvider, ColorScheme, ColorSchemeProvider, MantineTheme } from '@mantine/core';
|
||||||
import { NotificationsProvider } from '@mantine/notifications';
|
import { NotificationsProvider } from '@mantine/notifications';
|
||||||
import { useHotkeys } from '@mantine/hooks';
|
import { useHotkeys } from '@mantine/hooks';
|
||||||
import Layout from '../components/layout/Layout';
|
|
||||||
import { ConfigProvider } from '../tools/state';
|
import { ConfigProvider } from '../tools/state';
|
||||||
import { theme } from '../tools/theme';
|
import { theme } from '../tools/theme';
|
||||||
import { styles } from '../tools/styles';
|
import { styles } from '../tools/styles';
|
||||||
|
import { ColorTheme } from '../tools/color';
|
||||||
|
|
||||||
export default function App(props: AppProps & { colorScheme: ColorScheme }) {
|
export default function App(this: any, props: AppProps & { colorScheme: ColorScheme }) {
|
||||||
const { Component, pageProps } = props;
|
const { Component, pageProps } = props;
|
||||||
const [colorScheme, setColorScheme] = useState<ColorScheme>(props.colorScheme);
|
const [colorScheme, setColorScheme] = useState<ColorScheme>(props.colorScheme);
|
||||||
|
|
||||||
|
const [primaryColor, setPrimaryColor] = useState<MantineTheme['primaryColor']>('red');
|
||||||
|
const [secondaryColor, setSecondaryColor] = useState<MantineTheme['primaryColor']>('orange');
|
||||||
|
const [primaryShade, setPrimaryShade] = useState<MantineTheme['primaryShade']>(6);
|
||||||
|
const colorTheme = {
|
||||||
|
primaryColor,
|
||||||
|
secondaryColor,
|
||||||
|
setPrimaryColor,
|
||||||
|
setSecondaryColor,
|
||||||
|
primaryShade,
|
||||||
|
setPrimaryShade,
|
||||||
|
};
|
||||||
|
|
||||||
const toggleColorScheme = (value?: ColorScheme) => {
|
const toggleColorScheme = (value?: ColorScheme) => {
|
||||||
const nextColorScheme = value || (colorScheme === 'dark' ? 'light' : 'dark');
|
const nextColorScheme = value || (colorScheme === 'dark' ? 'light' : 'dark');
|
||||||
setColorScheme(nextColorScheme);
|
setColorScheme(nextColorScheme);
|
||||||
@@ -25,31 +37,31 @@ export default function App(props: AppProps & { colorScheme: ColorScheme }) {
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Head>
|
<Head>
|
||||||
<title>Homarr 🦞</title>
|
|
||||||
<meta name="viewport" content="minimum-scale=1, initial-scale=1, width=device-width" />
|
<meta name="viewport" content="minimum-scale=1, initial-scale=1, width=device-width" />
|
||||||
<link rel="shortcut icon" href="/favicon.svg" />
|
|
||||||
</Head>
|
</Head>
|
||||||
|
|
||||||
<ColorSchemeProvider colorScheme={colorScheme} toggleColorScheme={toggleColorScheme}>
|
<ColorSchemeProvider colorScheme={colorScheme} toggleColorScheme={toggleColorScheme}>
|
||||||
<MantineProvider
|
<ColorTheme.Provider value={colorTheme}>
|
||||||
theme={{
|
<MantineProvider
|
||||||
...theme,
|
theme={{
|
||||||
colorScheme,
|
...theme,
|
||||||
}}
|
primaryColor,
|
||||||
styles={{
|
primaryShade,
|
||||||
...styles,
|
colorScheme,
|
||||||
}}
|
}}
|
||||||
withGlobalStyles
|
styles={{
|
||||||
withNormalizeCSS
|
...styles,
|
||||||
>
|
}}
|
||||||
<NotificationsProvider limit={4} position="bottom-left">
|
withGlobalStyles
|
||||||
<ConfigProvider>
|
withNormalizeCSS
|
||||||
<Layout>
|
>
|
||||||
|
<NotificationsProvider limit={4} position="bottom-left">
|
||||||
|
<ConfigProvider>
|
||||||
<Component {...pageProps} />
|
<Component {...pageProps} />
|
||||||
</Layout>
|
</ConfigProvider>
|
||||||
</ConfigProvider>
|
</NotificationsProvider>
|
||||||
</NotificationsProvider>
|
</MantineProvider>
|
||||||
</MantineProvider>
|
</ColorTheme.Provider>
|
||||||
</ColorSchemeProvider>
|
</ColorSchemeProvider>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
|
|||||||
25
src/pages/api/configs/tryPassword.tsx
Normal file
25
src/pages/api/configs/tryPassword.tsx
Normal file
@@ -0,0 +1,25 @@
|
|||||||
|
import { NextApiRequest, NextApiResponse } from 'next';
|
||||||
|
|
||||||
|
function Post(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
const { tried } = req.body;
|
||||||
|
// Try to match the password with the PASSWORD env variable
|
||||||
|
if (tried === process.env.PASSWORD) {
|
||||||
|
return res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return res.status(200).json({
|
||||||
|
success: false,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async (req: NextApiRequest, res: NextApiResponse) => {
|
||||||
|
// Filter out if the reuqest is a POST or a GET
|
||||||
|
if (req.method === 'POST') {
|
||||||
|
return Post(req, res);
|
||||||
|
}
|
||||||
|
return res.status(405).json({
|
||||||
|
statusCode: 405,
|
||||||
|
message: 'Method not allowed',
|
||||||
|
});
|
||||||
|
};
|
||||||
66
src/pages/api/docker/container/[id].tsx
Normal file
66
src/pages/api/docker/container/[id].tsx
Normal file
@@ -0,0 +1,66 @@
|
|||||||
|
import { NextApiRequest, NextApiResponse } from 'next';
|
||||||
|
import Docker from 'dockerode';
|
||||||
|
|
||||||
|
const docker = new Docker();
|
||||||
|
|
||||||
|
async function Get(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
// Get the slug of the request
|
||||||
|
const { id } = req.query as { id: string };
|
||||||
|
const { action } = req.query;
|
||||||
|
// Get the action on the request (start, stop, restart)
|
||||||
|
if (action !== 'start' && action !== 'stop' && action !== 'restart' && action !== 'remove') {
|
||||||
|
return res.status(400).json({
|
||||||
|
statusCode: 400,
|
||||||
|
message: 'Invalid action',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (!id) {
|
||||||
|
return res.status(400).json({
|
||||||
|
message: 'Missing ID',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// Get the container with the ID
|
||||||
|
const container = docker.getContainer(id);
|
||||||
|
// Get the container info
|
||||||
|
container.inspect((err, data) => {
|
||||||
|
if (err) {
|
||||||
|
res.status(500).json({
|
||||||
|
message: err,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
try {
|
||||||
|
switch (action) {
|
||||||
|
case 'remove':
|
||||||
|
await container.remove();
|
||||||
|
break;
|
||||||
|
case 'start':
|
||||||
|
container.start();
|
||||||
|
break;
|
||||||
|
case 'stop':
|
||||||
|
container.stop();
|
||||||
|
break;
|
||||||
|
case 'restart':
|
||||||
|
container.restart();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} catch (err) {
|
||||||
|
res.status(500).json({
|
||||||
|
message: err,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
return res.status(200).json({
|
||||||
|
success: true,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async (req: NextApiRequest, res: NextApiResponse) => {
|
||||||
|
// Filter out if the reuqest is a Put or a GET
|
||||||
|
if (req.method === 'GET') {
|
||||||
|
return Get(req, res);
|
||||||
|
}
|
||||||
|
return res.status(405).json({
|
||||||
|
statusCode: 405,
|
||||||
|
message: 'Method not allowed',
|
||||||
|
});
|
||||||
|
};
|
||||||
21
src/pages/api/docker/containers.tsx
Normal file
21
src/pages/api/docker/containers.tsx
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
import { NextApiRequest, NextApiResponse } from 'next';
|
||||||
|
|
||||||
|
import Docker from 'dockerode';
|
||||||
|
|
||||||
|
const docker = new Docker();
|
||||||
|
|
||||||
|
async function Get(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
const containers = await docker.listContainers({ all: true });
|
||||||
|
return res.status(200).json(containers);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async (req: NextApiRequest, res: NextApiResponse) => {
|
||||||
|
// Filter out if the reuqest is a POST or a GET
|
||||||
|
if (req.method === 'GET') {
|
||||||
|
return Get(req, res);
|
||||||
|
}
|
||||||
|
return res.status(405).json({
|
||||||
|
statusCode: 405,
|
||||||
|
message: 'Method not allowed',
|
||||||
|
});
|
||||||
|
};
|
||||||
@@ -44,7 +44,10 @@ async function Post(req: NextApiRequest, res: NextApiResponse) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
// Get the origin URL
|
// Get the origin URL
|
||||||
const { origin } = new URL(service.url);
|
let { href: origin } = new URL(service.url);
|
||||||
|
if (origin.endsWith('/')) {
|
||||||
|
origin = origin.slice(0, -1);
|
||||||
|
}
|
||||||
const pined = `${origin}${url?.url}?apiKey=${service.apiKey}&end=${nextMonth}&start=${lastMonth}`;
|
const pined = `${origin}${url?.url}?apiKey=${service.apiKey}&end=${nextMonth}&start=${lastMonth}`;
|
||||||
const data = await axios.get(
|
const data = await axios.get(
|
||||||
`${origin}${url?.url}?apiKey=${service.apiKey}&end=${nextMonth}&start=${lastMonth}`
|
`${origin}${url?.url}?apiKey=${service.apiKey}&end=${nextMonth}&start=${lastMonth}`
|
||||||
|
|||||||
@@ -1,42 +1,59 @@
|
|||||||
import { Deluge } from '@ctrl/deluge';
|
import { Deluge } from '@ctrl/deluge';
|
||||||
import { QBittorrent } from '@ctrl/qbittorrent';
|
import { QBittorrent } from '@ctrl/qbittorrent';
|
||||||
|
import { NormalizedTorrent } from '@ctrl/shared-torrent';
|
||||||
|
import { Transmission } from '@ctrl/transmission';
|
||||||
import { NextApiRequest, NextApiResponse } from 'next';
|
import { NextApiRequest, NextApiResponse } from 'next';
|
||||||
|
import { Config } from '../../../tools/types';
|
||||||
|
|
||||||
async function Post(req: NextApiRequest, res: NextApiResponse) {
|
async function Post(req: NextApiRequest, res: NextApiResponse) {
|
||||||
// Get the type of service from the request url
|
// Get the type of service from the request url
|
||||||
const { dlclient } = req.query;
|
const { config }: { config: Config } = req.body;
|
||||||
const { body } = req;
|
const qBittorrentServices = config.services.filter((service) => service.type === 'qBittorrent');
|
||||||
// Get login, password and url from the body
|
const delugeServices = config.services.filter((service) => service.type === 'Deluge');
|
||||||
const { username, password, url } = body;
|
const transmissionServices = config.services.filter((service) => service.type === 'Transmission');
|
||||||
if (!dlclient || (!username && !password) || !url) {
|
|
||||||
return res.status(400).json({
|
const torrents: NormalizedTorrent[] = [];
|
||||||
error: 'Wrong request',
|
|
||||||
|
if (!qBittorrentServices && !delugeServices && !transmissionServices) {
|
||||||
|
return res.status(500).json({
|
||||||
|
statusCode: 500,
|
||||||
|
message: 'Missing services',
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
let client: Deluge | QBittorrent;
|
await Promise.all(
|
||||||
switch (dlclient) {
|
qBittorrentServices.map((service) =>
|
||||||
case 'qbit':
|
new QBittorrent({
|
||||||
client = new QBittorrent({
|
baseUrl: service.url,
|
||||||
baseUrl: new URL(url).origin,
|
username: service.username,
|
||||||
username,
|
password: service.password,
|
||||||
password,
|
})
|
||||||
});
|
.getAllData()
|
||||||
break;
|
.then((e) => torrents.push(...e.torrents))
|
||||||
case 'deluge':
|
)
|
||||||
client = new Deluge({
|
);
|
||||||
baseUrl: new URL(url).origin,
|
await Promise.all(
|
||||||
password,
|
delugeServices.map((service) =>
|
||||||
});
|
new Deluge({
|
||||||
break;
|
baseUrl: service.url,
|
||||||
default:
|
password: 'password' in service ? service.password : '',
|
||||||
return res.status(400).json({
|
})
|
||||||
error: 'Wrong request',
|
.getAllData()
|
||||||
});
|
.then((e) => torrents.push(...e.torrents))
|
||||||
}
|
)
|
||||||
const data = await client.getAllData();
|
);
|
||||||
res.status(200).json({
|
// Map transmissionServices
|
||||||
torrents: data.torrents,
|
await Promise.all(
|
||||||
});
|
transmissionServices.map((service) =>
|
||||||
|
new Transmission({
|
||||||
|
baseUrl: service.url,
|
||||||
|
username: 'username' in service ? service.username : '',
|
||||||
|
password: 'password' in service ? service.password : '',
|
||||||
|
})
|
||||||
|
.getAllData()
|
||||||
|
.then((e) => torrents.push(...e.torrents))
|
||||||
|
)
|
||||||
|
);
|
||||||
|
res.status(200).json(torrents);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default async (req: NextApiRequest, res: NextApiResponse) => {
|
export default async (req: NextApiRequest, res: NextApiResponse) => {
|
||||||
|
|||||||
@@ -7,10 +7,14 @@ async function Get(req: NextApiRequest, res: NextApiResponse) {
|
|||||||
await axios
|
await axios
|
||||||
.get(url as string)
|
.get(url as string)
|
||||||
.then((response) => {
|
.then((response) => {
|
||||||
res.status(200).json(response.data);
|
res.status(response.status).json(response.statusText);
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
res.status(500).json(error);
|
if (error.response) {
|
||||||
|
res.status(error.response.status).json(error.response.statusText);
|
||||||
|
} else {
|
||||||
|
res.status(500).json('Server Error');
|
||||||
|
}
|
||||||
});
|
});
|
||||||
// // Make a request to the URL
|
// // Make a request to the URL
|
||||||
// const response = await axios.get(url);
|
// const response = await axios.get(url);
|
||||||
|
|||||||
19
src/pages/api/modules/search.ts
Normal file
19
src/pages/api/modules/search.ts
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
import { NextApiRequest, NextApiResponse } from 'next';
|
||||||
|
|
||||||
|
async function Get(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
const { q } = req.query;
|
||||||
|
const response = await axios.get(`https://duckduckgo.com/ac/?q=${q}`);
|
||||||
|
res.status(200).json(response.data);
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async (req: NextApiRequest, res: NextApiResponse) => {
|
||||||
|
// Filter out if the reuqest is a POST or a GET
|
||||||
|
if (req.method === 'GET') {
|
||||||
|
return Get(req, res);
|
||||||
|
}
|
||||||
|
return res.status(405).json({
|
||||||
|
statusCode: 405,
|
||||||
|
message: 'Method not allowed',
|
||||||
|
});
|
||||||
|
};
|
||||||
@@ -7,6 +7,8 @@ import { Config } from '../tools/types';
|
|||||||
import { useConfig } from '../tools/state';
|
import { useConfig } from '../tools/state';
|
||||||
import { migrateToIdConfig } from '../tools/migrate';
|
import { migrateToIdConfig } from '../tools/migrate';
|
||||||
import { getConfig } from '../tools/getConfig';
|
import { getConfig } from '../tools/getConfig';
|
||||||
|
import { useColorTheme } from '../tools/color';
|
||||||
|
import Layout from '../components/layout/Layout';
|
||||||
|
|
||||||
export async function getServerSideProps({
|
export async function getServerSideProps({
|
||||||
req,
|
req,
|
||||||
@@ -28,14 +30,17 @@ export async function getServerSideProps({
|
|||||||
export default function HomePage(props: any) {
|
export default function HomePage(props: any) {
|
||||||
const { config: initialConfig }: { config: Config } = props;
|
const { config: initialConfig }: { config: Config } = props;
|
||||||
const { setConfig } = useConfig();
|
const { setConfig } = useConfig();
|
||||||
|
const { setPrimaryColor, setSecondaryColor } = useColorTheme();
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const migratedConfig = migrateToIdConfig(initialConfig);
|
const migratedConfig = migrateToIdConfig(initialConfig);
|
||||||
|
setPrimaryColor(migratedConfig.settings.primaryColor || 'red');
|
||||||
|
setSecondaryColor(migratedConfig.settings.secondaryColor || 'orange');
|
||||||
setConfig(migratedConfig);
|
setConfig(migratedConfig);
|
||||||
}, [initialConfig]);
|
}, [initialConfig]);
|
||||||
return (
|
return (
|
||||||
<>
|
<Layout>
|
||||||
<AppShelf />
|
<AppShelf />
|
||||||
<LoadConfigComponent />
|
<LoadConfigComponent />
|
||||||
</>
|
</Layout>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
111
src/pages/login.tsx
Normal file
111
src/pages/login.tsx
Normal file
@@ -0,0 +1,111 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import { PasswordInput, Anchor, Paper, Title, Text, Container, Group, Button } from '@mantine/core';
|
||||||
|
import { setCookies } from 'cookies-next';
|
||||||
|
import { useForm } from '@mantine/hooks';
|
||||||
|
import { showNotification, updateNotification } from '@mantine/notifications';
|
||||||
|
import axios from 'axios';
|
||||||
|
import { IconCheck, IconX } from '@tabler/icons';
|
||||||
|
|
||||||
|
// TODO: Add links to the wiki articles about the login process.
|
||||||
|
export default function AuthenticationTitle() {
|
||||||
|
const form = useForm({
|
||||||
|
initialValues: {
|
||||||
|
password: '',
|
||||||
|
},
|
||||||
|
});
|
||||||
|
return (
|
||||||
|
<Container
|
||||||
|
size={420}
|
||||||
|
style={{
|
||||||
|
height: '100vh',
|
||||||
|
display: 'flex',
|
||||||
|
width: 420,
|
||||||
|
flexDirection: 'column',
|
||||||
|
alignItems: 'center',
|
||||||
|
justifyContent: 'center',
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Title
|
||||||
|
align="center"
|
||||||
|
sx={(theme) => ({ fontFamily: `Greycliff CF, ${theme.fontFamily}`, fontWeight: 900 })}
|
||||||
|
>
|
||||||
|
Welcome back!
|
||||||
|
</Title>
|
||||||
|
<Text color="dimmed" size="sm" align="center" mt={5}>
|
||||||
|
Please enter the{' '}
|
||||||
|
<Anchor<'a'> href="#" size="sm" onClick={(event) => event.preventDefault()}>
|
||||||
|
password
|
||||||
|
</Anchor>
|
||||||
|
</Text>
|
||||||
|
|
||||||
|
<Paper withBorder shadow="md" p={30} mt={30} radius="md" style={{ width: 420 }}>
|
||||||
|
<form
|
||||||
|
onSubmit={form.onSubmit((values) => {
|
||||||
|
setCookies('password', values.password, {
|
||||||
|
maxAge: 60 * 60 * 24 * 30,
|
||||||
|
sameSite: 'lax',
|
||||||
|
});
|
||||||
|
showNotification({
|
||||||
|
id: 'load-data',
|
||||||
|
loading: true,
|
||||||
|
title: 'Checking your password',
|
||||||
|
message: 'Your password is being checked...',
|
||||||
|
autoClose: false,
|
||||||
|
disallowClose: true,
|
||||||
|
});
|
||||||
|
axios
|
||||||
|
.post('/api/configs/tryPassword', {
|
||||||
|
tried: values.password,
|
||||||
|
})
|
||||||
|
.then((res) => {
|
||||||
|
setTimeout(() => {
|
||||||
|
if (res.data.success === true) {
|
||||||
|
updateNotification({
|
||||||
|
id: 'load-data',
|
||||||
|
color: 'teal',
|
||||||
|
title: 'Password correct',
|
||||||
|
message:
|
||||||
|
'Notification will close in 2 seconds, you can close this notification now',
|
||||||
|
icon: <IconCheck />,
|
||||||
|
autoClose: 300,
|
||||||
|
onClose: () => {
|
||||||
|
window.location.reload();
|
||||||
|
},
|
||||||
|
});
|
||||||
|
}
|
||||||
|
if (res.data.success === false) {
|
||||||
|
updateNotification({
|
||||||
|
id: 'load-data',
|
||||||
|
color: 'red',
|
||||||
|
title: 'Password is wrong, please try again.',
|
||||||
|
message:
|
||||||
|
'Notification will close in 2 seconds, you can close this notification now',
|
||||||
|
icon: <IconX />,
|
||||||
|
autoClose: 2000,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, 500);
|
||||||
|
});
|
||||||
|
})}
|
||||||
|
>
|
||||||
|
<PasswordInput
|
||||||
|
id="password"
|
||||||
|
label="Password"
|
||||||
|
placeholder="Your password"
|
||||||
|
required
|
||||||
|
mt="md"
|
||||||
|
{...form.getInputProps('password')}
|
||||||
|
/>
|
||||||
|
<Group position="apart" mt="md">
|
||||||
|
<Anchor<'a'> onClick={(event) => event.preventDefault()} href="#" size="sm">
|
||||||
|
Forgot password?
|
||||||
|
</Anchor>
|
||||||
|
</Group>
|
||||||
|
<Button fullWidth type="submit" mt="xl">
|
||||||
|
Sign in
|
||||||
|
</Button>
|
||||||
|
</form>
|
||||||
|
</Paper>
|
||||||
|
</Container>
|
||||||
|
);
|
||||||
|
}
|
||||||
55
src/tools/addToHomarr.ts
Normal file
55
src/tools/addToHomarr.ts
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
import Dockerode from 'dockerode';
|
||||||
|
import { Config, MatchingImages, ServiceType } from './types';
|
||||||
|
|
||||||
|
async function MatchIcon(name: string) {
|
||||||
|
const res = await fetch(
|
||||||
|
`https://cdn.jsdelivr.net/gh/walkxhub/dashboard-icons/png/${name
|
||||||
|
.replace(/\s+/g, '-')
|
||||||
|
.toLowerCase()}.png`
|
||||||
|
);
|
||||||
|
return res.ok ? res.url : '/favicon.svg';
|
||||||
|
}
|
||||||
|
|
||||||
|
function tryMatchType(imageName: string): ServiceType {
|
||||||
|
// Try to find imageName inside MatchingImages
|
||||||
|
|
||||||
|
const match = MatchingImages.find(({ image }) => imageName.includes(image));
|
||||||
|
if (match) {
|
||||||
|
return match.type;
|
||||||
|
}
|
||||||
|
return 'Other';
|
||||||
|
}
|
||||||
|
|
||||||
|
export function tryMatchService(container: Dockerode.ContainerInfo | undefined) {
|
||||||
|
if (container === undefined) return {};
|
||||||
|
const name = container.Names[0].substring(1);
|
||||||
|
return {
|
||||||
|
name,
|
||||||
|
id: container.Id,
|
||||||
|
type: tryMatchType(container.Image),
|
||||||
|
url: `${container.Ports.at(0)?.IP}:${container.Ports.at(0)?.PublicPort}`,
|
||||||
|
icon: `https://cdn.jsdelivr.net/gh/walkxhub/dashboard-icons/png/${name
|
||||||
|
.replace(/\s+/g, '-')
|
||||||
|
.toLowerCase()}.png`,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export default async function addToHomarr(
|
||||||
|
container: Dockerode.ContainerInfo,
|
||||||
|
config: Config,
|
||||||
|
setConfig: (newconfig: Config) => void
|
||||||
|
) {
|
||||||
|
setConfig({
|
||||||
|
...config,
|
||||||
|
services: [
|
||||||
|
...config.services,
|
||||||
|
{
|
||||||
|
name: container.Names[0].substring(1),
|
||||||
|
id: container.Id,
|
||||||
|
type: tryMatchType(container.Image),
|
||||||
|
url: `localhost:${container.Ports.at(0)?.PublicPort}`,
|
||||||
|
icon: await MatchIcon(container.Names[0].substring(1)),
|
||||||
|
},
|
||||||
|
],
|
||||||
|
});
|
||||||
|
}
|
||||||
28
src/tools/color.ts
Normal file
28
src/tools/color.ts
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
import { createContext, useContext } from 'react';
|
||||||
|
import { MantineTheme } from '@mantine/core';
|
||||||
|
|
||||||
|
type colorThemeContextType = {
|
||||||
|
primaryColor: MantineTheme['primaryColor'];
|
||||||
|
secondaryColor: MantineTheme['primaryColor'];
|
||||||
|
primaryShade: MantineTheme['primaryShade'];
|
||||||
|
setPrimaryColor: (color: MantineTheme['primaryColor']) => void;
|
||||||
|
setSecondaryColor: (color: MantineTheme['primaryColor']) => void;
|
||||||
|
setPrimaryShade: (shade: MantineTheme['primaryShade']) => void;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const ColorTheme = createContext<colorThemeContextType>({
|
||||||
|
primaryColor: 'red',
|
||||||
|
secondaryColor: 'orange',
|
||||||
|
primaryShade: 6,
|
||||||
|
setPrimaryColor: () => {},
|
||||||
|
setSecondaryColor: () => {},
|
||||||
|
setPrimaryShade: () => {},
|
||||||
|
});
|
||||||
|
|
||||||
|
export function useColorTheme() {
|
||||||
|
const context = useContext(ColorTheme);
|
||||||
|
if (context === undefined) {
|
||||||
|
throw new Error('useColorTheme must be used within a ColorTheme.Provider');
|
||||||
|
}
|
||||||
|
return context;
|
||||||
|
}
|
||||||
22
src/tools/hooks/useSetSafeInterval.tsx
Normal file
22
src/tools/hooks/useSetSafeInterval.tsx
Normal file
@@ -0,0 +1,22 @@
|
|||||||
|
import { useEffect, useRef } from 'react';
|
||||||
|
|
||||||
|
export function useSetSafeInterval() {
|
||||||
|
const timers = useRef<NodeJS.Timer[]>([]);
|
||||||
|
|
||||||
|
function setSafeInterval(callback: () => void, delay: number) {
|
||||||
|
const newInterval = setInterval(callback, delay);
|
||||||
|
timers.current.push(newInterval);
|
||||||
|
return newInterval;
|
||||||
|
}
|
||||||
|
|
||||||
|
useEffect(
|
||||||
|
() => () => {
|
||||||
|
timers.current.forEach((t) => {
|
||||||
|
clearInterval(t);
|
||||||
|
});
|
||||||
|
},
|
||||||
|
[]
|
||||||
|
);
|
||||||
|
|
||||||
|
return setSafeInterval;
|
||||||
|
}
|
||||||
31
src/tools/humanFileSize.ts
Normal file
31
src/tools/humanFileSize.ts
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
/**
|
||||||
|
* Format bytes as human-readable text.
|
||||||
|
*
|
||||||
|
* @param bytes Number of bytes.
|
||||||
|
* @param si True to use metric (SI) units, aka powers of 1000. False to use
|
||||||
|
* binary (IEC), aka powers of 1024.
|
||||||
|
* @param dp Number of decimal places to display.
|
||||||
|
*
|
||||||
|
* @return Formatted string.
|
||||||
|
*/
|
||||||
|
export function humanFileSize(initialBytes: number, si = true, dp = 1) {
|
||||||
|
const thresh = si ? 1000 : 1024;
|
||||||
|
let bytes = initialBytes;
|
||||||
|
|
||||||
|
if (Math.abs(bytes) < thresh) {
|
||||||
|
return `${bytes} B`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const units = si
|
||||||
|
? ['kb', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB']
|
||||||
|
: ['KiB', 'MiB', 'GiB', 'TiB', 'PiB', 'EiB', 'ZiB', 'YiB'];
|
||||||
|
let u = -1;
|
||||||
|
const r = 10 ** dp;
|
||||||
|
|
||||||
|
do {
|
||||||
|
bytes /= thresh;
|
||||||
|
u += 1;
|
||||||
|
} while (Math.round(Math.abs(bytes) * r) / r >= thresh && u < units.length - 1);
|
||||||
|
|
||||||
|
return `${bytes.toFixed(dp)} ${units[u]}`;
|
||||||
|
}
|
||||||
@@ -1,6 +1,3 @@
|
|||||||
import { MantineProviderProps } from '@mantine/core';
|
import { MantineProviderProps } from '@mantine/core';
|
||||||
|
|
||||||
export const theme: MantineProviderProps['theme'] = {
|
export const theme: MantineProviderProps['theme'] = {};
|
||||||
primaryColor: 'red',
|
|
||||||
primaryShade: 6,
|
|
||||||
};
|
|
||||||
|
|||||||
@@ -1,7 +1,18 @@
|
|||||||
|
import { MantineTheme } from '@mantine/core';
|
||||||
import { OptionValues } from '../components/modules/modules';
|
import { OptionValues } from '../components/modules/modules';
|
||||||
|
|
||||||
export interface Settings {
|
export interface Settings {
|
||||||
searchUrl: string;
|
searchUrl: string;
|
||||||
|
title?: string;
|
||||||
|
logo?: string;
|
||||||
|
favicon?: string;
|
||||||
|
primaryColor?: MantineTheme['primaryColor'];
|
||||||
|
secondaryColor?: MantineTheme['primaryColor'];
|
||||||
|
primaryShade?: MantineTheme['primaryShade'];
|
||||||
|
background?: string;
|
||||||
|
appOpacity?: number;
|
||||||
|
widgetPosition?: string;
|
||||||
|
appCardWidth?: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Config {
|
export interface Config {
|
||||||
@@ -21,9 +32,36 @@ interface ConfigModule {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const StatusCodes = [
|
||||||
|
{ value: '200', label: '200 - OK', group: 'Sucessful responses' },
|
||||||
|
{ value: '204', label: '204 - No Content', group: 'Sucessful responses' },
|
||||||
|
{ value: '301', label: '301 - Moved Permanently', group: 'Redirection responses' },
|
||||||
|
{ value: '302', label: '302 - Found / Moved Temporarily', group: 'Redirection responses' },
|
||||||
|
{ value: '304', label: '304 - Not Modified', group: 'Redirection responses' },
|
||||||
|
{ value: '307', label: '307 - Temporary Redirect', group: 'Redirection responses' },
|
||||||
|
{ value: '308', label: '308 - Permanent Redirect', group: 'Redirection responses' },
|
||||||
|
{ value: '400', label: '400 - Bad Request', group: 'Client error responses' },
|
||||||
|
{ value: '401', label: '401 - Unauthorized', group: 'Client error responses' },
|
||||||
|
{ value: '403', label: '403 - Forbidden', group: 'Client error responses' },
|
||||||
|
{ value: '404', label: '404 - Not Found', group: 'Client error responses' },
|
||||||
|
{ value: '408', label: '408 - Request Timeout', group: 'Client error responses' },
|
||||||
|
{ value: '410', label: '410 - Gone', group: 'Client error responses' },
|
||||||
|
{ value: '429', label: '429 - Too Many Requests', group: 'Client error responses' },
|
||||||
|
{ value: '500', label: '500 - Internal Server Error', group: 'Server error responses' },
|
||||||
|
{ value: '502', label: '502 - Bad Gateway', group: 'Server error responses' },
|
||||||
|
{ value: '503', label: '503 - Service Unavailable', group: 'Server error responses' },
|
||||||
|
{ value: '054', label: '504 - Gateway Timeout Error', group: 'Server error responses' },
|
||||||
|
];
|
||||||
|
|
||||||
|
export const Targets = [
|
||||||
|
{ value: '_blank', label: 'New Tab' },
|
||||||
|
{ value: '_top', label: 'Same Window' },
|
||||||
|
];
|
||||||
|
|
||||||
export const ServiceTypeList = [
|
export const ServiceTypeList = [
|
||||||
'Other',
|
'Other',
|
||||||
'Emby',
|
'Emby',
|
||||||
|
'Dash.',
|
||||||
'Deluge',
|
'Deluge',
|
||||||
'Lidarr',
|
'Lidarr',
|
||||||
'Plex',
|
'Plex',
|
||||||
@@ -31,17 +69,26 @@ export const ServiceTypeList = [
|
|||||||
'Readarr',
|
'Readarr',
|
||||||
'Sonarr',
|
'Sonarr',
|
||||||
'qBittorrent',
|
'qBittorrent',
|
||||||
|
'Transmission',
|
||||||
];
|
];
|
||||||
export type ServiceType =
|
export type ServiceType =
|
||||||
| 'Other'
|
| 'Other'
|
||||||
| 'Emby'
|
| 'Emby'
|
||||||
|
| 'Dash.'
|
||||||
| 'Deluge'
|
| 'Deluge'
|
||||||
| 'Lidarr'
|
| 'Lidarr'
|
||||||
| 'Plex'
|
| 'Plex'
|
||||||
| 'Radarr'
|
| 'Radarr'
|
||||||
| 'Readarr'
|
| 'Readarr'
|
||||||
| 'Sonarr'
|
| 'Sonarr'
|
||||||
| 'qBittorrent';
|
| 'qBittorrent'
|
||||||
|
| 'Transmission';
|
||||||
|
|
||||||
|
export const MatchingImages: { image: string; type: ServiceType }[] = [
|
||||||
|
{ image: 'lscr.io/linuxserver/radarr', type: 'Radarr' },
|
||||||
|
{ image: 'lscr.io/linuxserver/sonarr', type: 'Sonarr' },
|
||||||
|
{ image: 'lscr.io/linuxserver/qbittorrent', type: 'qBittorrent' },
|
||||||
|
];
|
||||||
|
|
||||||
export interface serviceItem {
|
export interface serviceItem {
|
||||||
id: string;
|
id: string;
|
||||||
@@ -53,4 +100,7 @@ export interface serviceItem {
|
|||||||
apiKey?: string;
|
apiKey?: string;
|
||||||
password?: string;
|
password?: string;
|
||||||
username?: string;
|
username?: string;
|
||||||
|
openedUrl?: string;
|
||||||
|
newTab?: boolean;
|
||||||
|
status?: string[];
|
||||||
}
|
}
|
||||||
|
|||||||
559
yarn.lock
559
yarn.lock
@@ -1583,17 +1583,17 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@ctrl/deluge@npm:^4.0.0":
|
"@ctrl/deluge@npm:^4.1.0":
|
||||||
version: 4.0.0
|
version: 4.1.0
|
||||||
resolution: "@ctrl/deluge@npm:4.0.0"
|
resolution: "@ctrl/deluge@npm:4.1.0"
|
||||||
dependencies:
|
dependencies:
|
||||||
"@ctrl/magnet-link": ^3.1.0
|
"@ctrl/magnet-link": ^3.1.1
|
||||||
"@ctrl/shared-torrent": ^4.1.0
|
"@ctrl/shared-torrent": ^4.1.1
|
||||||
"@ctrl/url-join": ^2.0.0
|
"@ctrl/url-join": ^2.0.0
|
||||||
formdata-node: ^4.3.2
|
formdata-node: ^4.3.2
|
||||||
got: ^12.0.1
|
got: ^12.1.0
|
||||||
tough-cookie: ^4.0.0
|
tough-cookie: ^4.0.0
|
||||||
checksum: d4b828fb580a3e4c589169044b78e74d2d1c6ea3ff24f24c9aba59a5fc88320c494eebe814aa0f048e772d698ddd5979f8cd92d4144b0550227bc502342c82ed
|
checksum: a17f974e1b98a9086e1036604a86d3e14b5cf9c8d0fd997357dd4522dc296f0ef92e2697231f97f7211c0224e35256af966f722b6b316a363533328908cd8d5e
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
@@ -1606,6 +1606,15 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"@ctrl/magnet-link@npm:^3.1.1":
|
||||||
|
version: 3.1.1
|
||||||
|
resolution: "@ctrl/magnet-link@npm:3.1.1"
|
||||||
|
dependencies:
|
||||||
|
"@ctrl/ts-base32": ^2.1.1
|
||||||
|
checksum: 82533b50e2a60b2cfbad19879b0b16dbdbf2cfb633cda519d9cac7ab4039d52f98bc10185a5f6ffd29cfe415d709b8748ebe7cf763e522e0c4dcee8dde6506fe
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"@ctrl/qbittorrent@npm:^4.0.0":
|
"@ctrl/qbittorrent@npm:^4.0.0":
|
||||||
version: 4.0.0
|
version: 4.0.0
|
||||||
resolution: "@ctrl/qbittorrent@npm:4.0.0"
|
resolution: "@ctrl/qbittorrent@npm:4.0.0"
|
||||||
@@ -1630,6 +1639,15 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"@ctrl/shared-torrent@npm:^4.1.1":
|
||||||
|
version: 4.1.1
|
||||||
|
resolution: "@ctrl/shared-torrent@npm:4.1.1"
|
||||||
|
dependencies:
|
||||||
|
got: ^12.1.0
|
||||||
|
checksum: 1273c9088a920eed5afca945b11e83a6b64d4268ad0b09e916e7e2214ea8092b998ab16525885f8f24af2c75893e3fd7d4542e7e9d6dfe4688da57e47c31b165
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"@ctrl/torrent-file@npm:^2.0.1":
|
"@ctrl/torrent-file@npm:^2.0.1":
|
||||||
version: 2.0.1
|
version: 2.0.1
|
||||||
resolution: "@ctrl/torrent-file@npm:2.0.1"
|
resolution: "@ctrl/torrent-file@npm:2.0.1"
|
||||||
@@ -1639,6 +1657,18 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"@ctrl/transmission@npm:^4.1.1":
|
||||||
|
version: 4.1.1
|
||||||
|
resolution: "@ctrl/transmission@npm:4.1.1"
|
||||||
|
dependencies:
|
||||||
|
"@ctrl/magnet-link": ^3.1.0
|
||||||
|
"@ctrl/shared-torrent": ^4.1.1
|
||||||
|
"@ctrl/url-join": ^2.0.0
|
||||||
|
got: ^12.1.0
|
||||||
|
checksum: 218ed4c00f70c46c90cd2a5e90f8390beee06a2cf7d76c2445ad2bcfb89ad1e6ea9cf237a7b3aa990fdf81fc9b9d4aa9900fa21e041457e8bb177dbd0b319b0a
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"@ctrl/ts-base32@npm:^2.1.1":
|
"@ctrl/ts-base32@npm:^2.1.1":
|
||||||
version: 2.1.1
|
version: 2.1.1
|
||||||
resolution: "@ctrl/ts-base32@npm:2.1.1"
|
resolution: "@ctrl/ts-base32@npm:2.1.1"
|
||||||
@@ -2197,130 +2227,130 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@mantine/core@npm:^4.2.6":
|
"@mantine/core@npm:^4.2.8":
|
||||||
version: 4.2.7
|
version: 4.2.8
|
||||||
resolution: "@mantine/core@npm:4.2.7"
|
resolution: "@mantine/core@npm:4.2.8"
|
||||||
dependencies:
|
dependencies:
|
||||||
"@mantine/styles": 4.2.7
|
"@mantine/styles": 4.2.8
|
||||||
"@popperjs/core": ^2.9.3
|
"@popperjs/core": ^2.9.3
|
||||||
"@radix-ui/react-scroll-area": ^0.1.1
|
"@radix-ui/react-scroll-area": ^0.1.1
|
||||||
react-popper: ^2.2.5
|
react-popper: ^2.2.5
|
||||||
react-textarea-autosize: ^8.3.2
|
react-textarea-autosize: ^8.3.2
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
"@mantine/hooks": 4.2.7
|
"@mantine/hooks": 4.2.8
|
||||||
react: ">=16.8.0"
|
react: ">=16.8.0"
|
||||||
react-dom: ">=16.8.0"
|
react-dom: ">=16.8.0"
|
||||||
checksum: f86d17fe8793bf37ef40eba9bf369db268e64923fe907ebcd8d977685bf1efda8c2b6a0f490cc3de87212273e930107e7d9e7135e4babe087a3e40d0f85b44af
|
checksum: a7434d542657e5b196dc795503f667a4eff0cc4eed3870c3bd3ae1f645e01bc9c9e3dd32387907700cb96a41a70b836c0003756f5f488e7db7f61dee175386e6
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@mantine/dates@npm:^4.2.6":
|
"@mantine/dates@npm:^4.2.8":
|
||||||
version: 4.2.7
|
version: 4.2.8
|
||||||
resolution: "@mantine/dates@npm:4.2.7"
|
resolution: "@mantine/dates@npm:4.2.8"
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
"@mantine/core": 4.2.7
|
"@mantine/core": 4.2.8
|
||||||
"@mantine/hooks": 4.2.7
|
"@mantine/hooks": 4.2.8
|
||||||
dayjs: ^1.10.5
|
dayjs: ^1.10.5
|
||||||
react: ">=16.8.0"
|
react: ">=16.8.0"
|
||||||
checksum: f343252c768928be72a35aed6522d5e73b10c2934b76cbc4761695087870bafd34f4491dbc55bd47ee5e399f995041044a41f3a8a2aa3b67d233d68c59ca7931
|
checksum: 8aa69e30da0269e259b129827cf1c4496cd9f1aef22fd709fb9ae76840be3377541d289ec0e630004aeb7647fdb08a1a84651d72cb539f3491d887f626dff298
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@mantine/dropzone@npm:^4.2.6":
|
"@mantine/dropzone@npm:^4.2.8":
|
||||||
version: 4.2.7
|
version: 4.2.8
|
||||||
resolution: "@mantine/dropzone@npm:4.2.7"
|
resolution: "@mantine/dropzone@npm:4.2.8"
|
||||||
dependencies:
|
dependencies:
|
||||||
react-dropzone: ^11.4.2
|
react-dropzone: ^11.4.2
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
"@mantine/core": 4.2.7
|
"@mantine/core": 4.2.8
|
||||||
"@mantine/hooks": 4.2.7
|
"@mantine/hooks": 4.2.8
|
||||||
react: ">=16.8.0"
|
react: ">=16.8.0"
|
||||||
react-dom: ">=16.8.0"
|
react-dom: ">=16.8.0"
|
||||||
checksum: 2b8d45a36f3d5275a1d03a0f595683f881c295137b46c2e08f73fa18a180b2ec125426e522c8fb822823666c72ae8c782f8855ea24586104fbf2c8ca9122652e
|
checksum: 219e5fcc576a8d734c509b9da1b8e7e52a3c1a4aff7b2dc018a191be333e03c08139dc9695edd9911709c1e3454fff3b70b42ab2ef0e9587d1c9ff3f4f5865a4
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@mantine/form@npm:^4.2.6":
|
"@mantine/form@npm:^4.2.8":
|
||||||
version: 4.2.7
|
version: 4.2.8
|
||||||
resolution: "@mantine/form@npm:4.2.7"
|
resolution: "@mantine/form@npm:4.2.8"
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
react: ">=16.8.0"
|
react: ">=16.8.0"
|
||||||
checksum: d60cfd48ab4ef149df4dc68c3024428c51c5e1267af451882b68ef00162df2b7a07bd8fc5d734f8495957fa49769d6e444459ccf6e19e297c6737481ca85b4e1
|
checksum: 0b17d214b9e4aab58a41a7c44fa5618091b24fe95d9741c3c7aaea86cbc52f93668d35b363460f1fb278eda0482b7922c308e06e354ae2d9d49b45d9ddafaf67
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@mantine/hooks@npm:^4.2.6":
|
"@mantine/hooks@npm:^4.2.8":
|
||||||
version: 4.2.7
|
version: 4.2.8
|
||||||
resolution: "@mantine/hooks@npm:4.2.7"
|
resolution: "@mantine/hooks@npm:4.2.8"
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
react: ">=16.8.0"
|
react: ">=16.8.0"
|
||||||
checksum: 66dc8887b7913334ed1ce6f4be0353f4273142b10d1e820d4395edbad5fc7dc7f8483e07abe5956d7463fc77365340765c80843d8f825dc00c447310eb58831d
|
checksum: 371bc3fa19130838d1a53454291b84c41390f9e8d4d89166c3ba36b60e5e671502b221a98834a42be3de0c6ab878eb0a950a58f8770e44ad6d9cba1468ef0aae
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@mantine/next@npm:^4.2.6":
|
"@mantine/next@npm:^4.2.8":
|
||||||
version: 4.2.7
|
version: 4.2.8
|
||||||
resolution: "@mantine/next@npm:4.2.7"
|
resolution: "@mantine/next@npm:4.2.8"
|
||||||
dependencies:
|
dependencies:
|
||||||
"@mantine/ssr": 4.2.7
|
"@mantine/ssr": 4.2.8
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
next: "*"
|
next: "*"
|
||||||
react: ">=16.8.0"
|
react: ">=16.8.0"
|
||||||
react-dom: ">=16.8.0"
|
react-dom: ">=16.8.0"
|
||||||
checksum: 4aa9384559ca7882aa2719629a79e2eb8b60c8491a6a22ff12c535585701d9b7f2d577bc8c043c6ea7669e890440606c253fd7e4656b05b4f5d815db5c121b27
|
checksum: 48d658a6c1954a30906c34602a37da4b00ca3712819ba1cc1719045a95d412d1f3c6d847116b85f37c34dfdbac84929c525fdb20b4b93734f6016e1988924bfa
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@mantine/notifications@npm:^4.2.6":
|
"@mantine/notifications@npm:^4.2.8":
|
||||||
version: 4.2.7
|
version: 4.2.8
|
||||||
resolution: "@mantine/notifications@npm:4.2.7"
|
resolution: "@mantine/notifications@npm:4.2.8"
|
||||||
dependencies:
|
dependencies:
|
||||||
react-transition-group: ^4.4.2
|
react-transition-group: ^4.4.2
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
"@mantine/core": 4.2.7
|
"@mantine/core": 4.2.8
|
||||||
"@mantine/hooks": 4.2.7
|
"@mantine/hooks": 4.2.8
|
||||||
react: ">=16.8.0"
|
react: ">=16.8.0"
|
||||||
react-dom: ">=16.8.0"
|
react-dom: ">=16.8.0"
|
||||||
checksum: 576395cb60cd5cd0251f4c9542c4eb4c711fb542f0160c339f139ffb8eaeaf6ff440e73f9e0b9304a1f7ac2e8813db5817d2698ad1805f4593499f4816c0dcde
|
checksum: dc13bb2091526e7f2ca7eb06d82ee5b5305208b41cc3ec769fa2aac09908faf8bba3d36bd10c8098d7c1a9f0487b5da92ab443dd238a576903633153ccfc6605
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@mantine/prism@npm:^4.2.6":
|
"@mantine/prism@npm:^4.2.8":
|
||||||
version: 4.2.7
|
version: 4.2.8
|
||||||
resolution: "@mantine/prism@npm:4.2.7"
|
resolution: "@mantine/prism@npm:4.2.8"
|
||||||
dependencies:
|
dependencies:
|
||||||
prism-react-renderer: ^1.2.1
|
prism-react-renderer: ^1.2.1
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
"@mantine/core": 4.2.7
|
"@mantine/core": 4.2.8
|
||||||
"@mantine/hooks": 4.2.7
|
"@mantine/hooks": 4.2.8
|
||||||
react: ">=16.8.0"
|
react: ">=16.8.0"
|
||||||
react-dom: ">=16.8.0"
|
react-dom: ">=16.8.0"
|
||||||
checksum: 489b16b3ab775e8494b13b43eb588e79de858793b3a0fe0ce06194a163c66918a364a13822bc633f0b14091318085c1adac140650b50ab4acc2efcabb2226975
|
checksum: 0e4405993e772249633b1585db1266ea857e7b8ad21ef89a4cf78ce8e811f2be0461218ef97686a51bafd83ff41fb646f903ca587cebaae22629a8f5936c0ae2
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@mantine/ssr@npm:4.2.7":
|
"@mantine/ssr@npm:4.2.8":
|
||||||
version: 4.2.7
|
version: 4.2.8
|
||||||
resolution: "@mantine/ssr@npm:4.2.7"
|
resolution: "@mantine/ssr@npm:4.2.8"
|
||||||
dependencies:
|
dependencies:
|
||||||
"@emotion/cache": 11.7.1
|
"@emotion/cache": 11.7.1
|
||||||
"@emotion/react": 11.7.1
|
"@emotion/react": 11.7.1
|
||||||
"@emotion/serialize": 1.0.2
|
"@emotion/serialize": 1.0.2
|
||||||
"@emotion/server": 11.4.0
|
"@emotion/server": 11.4.0
|
||||||
"@emotion/utils": 1.0.0
|
"@emotion/utils": 1.0.0
|
||||||
"@mantine/styles": 4.2.7
|
"@mantine/styles": 4.2.8
|
||||||
csstype: 3.0.9
|
csstype: 3.0.9
|
||||||
html-react-parser: 1.3.0
|
html-react-parser: 1.3.0
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
react: ">=16.8.0"
|
react: ">=16.8.0"
|
||||||
react-dom: ">=16.8.0"
|
react-dom: ">=16.8.0"
|
||||||
checksum: dcbbe3c4992f16c147ebb66fe156561cc1e6f6c0166ac33cc0d422c3250864c3d80773f32bd755e6c300f7035fe3341cffcd3a0ccb919775e782e227bd3876d1
|
checksum: f2588004ffa65890e4e88ff23aae54124ccc96edda8fcdf4fee9ec93219e156a9862ce6c8473c8096b4944858c56cd8268cd8451118ce55c8358f8c569699a54
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@mantine/styles@npm:4.2.7":
|
"@mantine/styles@npm:4.2.8":
|
||||||
version: 4.2.7
|
version: 4.2.8
|
||||||
resolution: "@mantine/styles@npm:4.2.7"
|
resolution: "@mantine/styles@npm:4.2.8"
|
||||||
dependencies:
|
dependencies:
|
||||||
"@emotion/cache": 11.7.1
|
"@emotion/cache": 11.7.1
|
||||||
"@emotion/react": 11.7.1
|
"@emotion/react": 11.7.1
|
||||||
@@ -2331,7 +2361,7 @@ __metadata:
|
|||||||
peerDependencies:
|
peerDependencies:
|
||||||
react: ">=16.8.0"
|
react: ">=16.8.0"
|
||||||
react-dom: ">=16.8.0"
|
react-dom: ">=16.8.0"
|
||||||
checksum: f8a1dc2ca9be269e249671b2dc8a5849e859775c133a607723313aa2978cd7b3669217749d9ce6a8abbecfdaab567e4549af9683383030236883dde777a8628c
|
checksum: 03bbddecb1837bca42e2667cb548d821adfb758c66d71c7719390b3921483d3d4997a03b1aaceccc4e557160522000e68978aa3c8b38f2ae3e4a9d85927e519d
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
@@ -2379,111 +2409,118 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/bundle-analyzer@npm:^12.1.4":
|
"@next/bundle-analyzer@npm:^12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/bundle-analyzer@npm:12.1.6"
|
resolution: "@next/bundle-analyzer@npm:12.2.0"
|
||||||
dependencies:
|
dependencies:
|
||||||
webpack-bundle-analyzer: 4.3.0
|
webpack-bundle-analyzer: 4.3.0
|
||||||
checksum: cf37be49d45d706aea95df489656341bec64783e567067d15036b25330d7a69204987b2c402277f201b9bf943de588323b120fd8096bb3d6846a054bbb2cdc7e
|
checksum: e08770ed2f7bfa4fb38c29d58d1e3ad198fa7e9a8c061ea5e15950dd10576bed0b5b8c19266e18503af1d211a0d8d450b5fed4926f6863135b38e585d6fd1980
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/env@npm:12.1.6":
|
"@next/env@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/env@npm:12.1.6"
|
resolution: "@next/env@npm:12.2.0"
|
||||||
checksum: e6a4f189f0d653d13dc7ad510f6c9d2cf690bfd9e07c554bd501b840f8dabc3da5adcab874b0bc01aab86c3647cff4fb84692e3c3b28125af26f0b05cd4c7fcf
|
checksum: 5fb317bdb5eb2d5df12ff55e335368792dba21874c5ece3cabf8cd312cec911a1d54ecf368e69dc08640b0244669b8a98c86cd035c7874b17640602e67c1b9d9
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/eslint-plugin-next@npm:^12.1.4":
|
"@next/eslint-plugin-next@npm:^12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/eslint-plugin-next@npm:12.1.6"
|
resolution: "@next/eslint-plugin-next@npm:12.2.0"
|
||||||
dependencies:
|
dependencies:
|
||||||
glob: 7.1.7
|
glob: 7.1.7
|
||||||
checksum: 33dcaf71f299d3c8a0744cad512369f92d7a355f3c0d57f2496e888e4242080c49226ec2c59ba2efac04b3a1df51c36019b853b4177df082ca4621a1713a2229
|
checksum: 2e33b9af79af680fd873d74e91bed397930a91802c1d7a293db757227ebc431d3d856de69477dc178dec8b531635ea69d79b188293024f1371afe6c348dbe647
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/swc-android-arm-eabi@npm:12.1.6":
|
"@next/swc-android-arm-eabi@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/swc-android-arm-eabi@npm:12.1.6"
|
resolution: "@next/swc-android-arm-eabi@npm:12.2.0"
|
||||||
conditions: os=android & cpu=arm
|
conditions: os=android & cpu=arm
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/swc-android-arm64@npm:12.1.6":
|
"@next/swc-android-arm64@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/swc-android-arm64@npm:12.1.6"
|
resolution: "@next/swc-android-arm64@npm:12.2.0"
|
||||||
conditions: os=android & cpu=arm64
|
conditions: os=android & cpu=arm64
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/swc-darwin-arm64@npm:12.1.6":
|
"@next/swc-darwin-arm64@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/swc-darwin-arm64@npm:12.1.6"
|
resolution: "@next/swc-darwin-arm64@npm:12.2.0"
|
||||||
conditions: os=darwin & cpu=arm64
|
conditions: os=darwin & cpu=arm64
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/swc-darwin-x64@npm:12.1.6":
|
"@next/swc-darwin-x64@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/swc-darwin-x64@npm:12.1.6"
|
resolution: "@next/swc-darwin-x64@npm:12.2.0"
|
||||||
conditions: os=darwin & cpu=x64
|
conditions: os=darwin & cpu=x64
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/swc-linux-arm-gnueabihf@npm:12.1.6":
|
"@next/swc-freebsd-x64@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/swc-linux-arm-gnueabihf@npm:12.1.6"
|
resolution: "@next/swc-freebsd-x64@npm:12.2.0"
|
||||||
|
conditions: os=freebsd & cpu=x64
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"@next/swc-linux-arm-gnueabihf@npm:12.2.0":
|
||||||
|
version: 12.2.0
|
||||||
|
resolution: "@next/swc-linux-arm-gnueabihf@npm:12.2.0"
|
||||||
conditions: os=linux & cpu=arm
|
conditions: os=linux & cpu=arm
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/swc-linux-arm64-gnu@npm:12.1.6":
|
"@next/swc-linux-arm64-gnu@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/swc-linux-arm64-gnu@npm:12.1.6"
|
resolution: "@next/swc-linux-arm64-gnu@npm:12.2.0"
|
||||||
conditions: os=linux & cpu=arm64 & libc=glibc
|
conditions: os=linux & cpu=arm64 & libc=glibc
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/swc-linux-arm64-musl@npm:12.1.6":
|
"@next/swc-linux-arm64-musl@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/swc-linux-arm64-musl@npm:12.1.6"
|
resolution: "@next/swc-linux-arm64-musl@npm:12.2.0"
|
||||||
conditions: os=linux & cpu=arm64 & libc=musl
|
conditions: os=linux & cpu=arm64 & libc=musl
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/swc-linux-x64-gnu@npm:12.1.6":
|
"@next/swc-linux-x64-gnu@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/swc-linux-x64-gnu@npm:12.1.6"
|
resolution: "@next/swc-linux-x64-gnu@npm:12.2.0"
|
||||||
conditions: os=linux & cpu=x64 & libc=glibc
|
conditions: os=linux & cpu=x64 & libc=glibc
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/swc-linux-x64-musl@npm:12.1.6":
|
"@next/swc-linux-x64-musl@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/swc-linux-x64-musl@npm:12.1.6"
|
resolution: "@next/swc-linux-x64-musl@npm:12.2.0"
|
||||||
conditions: os=linux & cpu=x64 & libc=musl
|
conditions: os=linux & cpu=x64 & libc=musl
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/swc-win32-arm64-msvc@npm:12.1.6":
|
"@next/swc-win32-arm64-msvc@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/swc-win32-arm64-msvc@npm:12.1.6"
|
resolution: "@next/swc-win32-arm64-msvc@npm:12.2.0"
|
||||||
conditions: os=win32 & cpu=arm64
|
conditions: os=win32 & cpu=arm64
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/swc-win32-ia32-msvc@npm:12.1.6":
|
"@next/swc-win32-ia32-msvc@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/swc-win32-ia32-msvc@npm:12.1.6"
|
resolution: "@next/swc-win32-ia32-msvc@npm:12.2.0"
|
||||||
conditions: os=win32 & cpu=ia32
|
conditions: os=win32 & cpu=ia32
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"@next/swc-win32-x64-msvc@npm:12.1.6":
|
"@next/swc-win32-x64-msvc@npm:12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "@next/swc-win32-x64-msvc@npm:12.1.6"
|
resolution: "@next/swc-win32-x64-msvc@npm:12.2.0"
|
||||||
conditions: os=win32 & cpu=x64
|
conditions: os=win32 & cpu=x64
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
@@ -3753,6 +3790,15 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"@swc/helpers@npm:0.4.2":
|
||||||
|
version: 0.4.2
|
||||||
|
resolution: "@swc/helpers@npm:0.4.2"
|
||||||
|
dependencies:
|
||||||
|
tslib: ^2.4.0
|
||||||
|
checksum: 0b8c86ad03b17b8fe57dc4498e25dc294ea6bc42558a6b92d8fcd789351dac80199409bef38a2e3ac06aae0fedddfc0ab9c34409acbf74e55d1bbbd74f68b6b7
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"@szmarczak/http-timer@npm:^5.0.1":
|
"@szmarczak/http-timer@npm:^5.0.1":
|
||||||
version: 5.0.1
|
version: 5.0.1
|
||||||
resolution: "@szmarczak/http-timer@npm:5.0.1"
|
resolution: "@szmarczak/http-timer@npm:5.0.1"
|
||||||
@@ -3844,6 +3890,26 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"@types/docker-modem@npm:*":
|
||||||
|
version: 3.0.2
|
||||||
|
resolution: "@types/docker-modem@npm:3.0.2"
|
||||||
|
dependencies:
|
||||||
|
"@types/node": "*"
|
||||||
|
"@types/ssh2": "*"
|
||||||
|
checksum: 1f23db30e6e9bdd4c6d6e43572fb7ac7251d106a1906a9f3faabac393897712a5a9cd5a471baedc0ac8055dab3f48eda331f41a1e2c7c6bbe3c7f433e039151c
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"@types/dockerode@npm:^3.3.9":
|
||||||
|
version: 3.3.9
|
||||||
|
resolution: "@types/dockerode@npm:3.3.9"
|
||||||
|
dependencies:
|
||||||
|
"@types/docker-modem": "*"
|
||||||
|
"@types/node": "*"
|
||||||
|
checksum: 3d03c68addb37c50e9557fff17171d26423aa18e544cb24e4caa81ebcec39ccc1cafed7adbfb8f4220d8ed23028d231717826bb77a786d425885c4f4cc37536d
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"@types/eslint-scope@npm:^3.7.3":
|
"@types/eslint-scope@npm:^3.7.3":
|
||||||
version: 3.7.3
|
version: 3.7.3
|
||||||
resolution: "@types/eslint-scope@npm:3.7.3"
|
resolution: "@types/eslint-scope@npm:3.7.3"
|
||||||
@@ -4121,6 +4187,25 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"@types/ssh2-streams@npm:*":
|
||||||
|
version: 0.1.9
|
||||||
|
resolution: "@types/ssh2-streams@npm:0.1.9"
|
||||||
|
dependencies:
|
||||||
|
"@types/node": "*"
|
||||||
|
checksum: 190f3c235bf19787cd255f366d3ac9233875750095f3c73d15e72a1e67a826aed7e7c389603c5e89cb6420b87ff6dffc566f9174e546ddb7ff8c8dc2e8b00def
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"@types/ssh2@npm:*":
|
||||||
|
version: 0.5.52
|
||||||
|
resolution: "@types/ssh2@npm:0.5.52"
|
||||||
|
dependencies:
|
||||||
|
"@types/node": "*"
|
||||||
|
"@types/ssh2-streams": "*"
|
||||||
|
checksum: bc1c76ac727ad73ddd59ba849cf0ea3ed2e930439e7a363aff24f04f29b74f9b1976369b869dc9a018223c9fb8ad041c09a0f07aea8cf46a8c920049188cddae
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"@types/stack-utils@npm:^2.0.0":
|
"@types/stack-utils@npm:^2.0.0":
|
||||||
version: 2.0.1
|
version: 2.0.1
|
||||||
resolution: "@types/stack-utils@npm:2.0.1"
|
resolution: "@types/stack-utils@npm:2.0.1"
|
||||||
@@ -5166,6 +5251,15 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"asn1@npm:^0.2.4":
|
||||||
|
version: 0.2.6
|
||||||
|
resolution: "asn1@npm:0.2.6"
|
||||||
|
dependencies:
|
||||||
|
safer-buffer: ~2.1.0
|
||||||
|
checksum: 39f2ae343b03c15ad4f238ba561e626602a3de8d94ae536c46a4a93e69578826305366dc09fbb9b56aec39b4982a463682f259c38e59f6fa380cd72cd61e493d
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"assert@npm:^1.1.1":
|
"assert@npm:^1.1.1":
|
||||||
version: 1.5.0
|
version: 1.5.0
|
||||||
resolution: "assert@npm:1.5.0"
|
resolution: "assert@npm:1.5.0"
|
||||||
@@ -5489,7 +5583,7 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"base64-js@npm:^1.0.2":
|
"base64-js@npm:^1.0.2, base64-js@npm:^1.3.1":
|
||||||
version: 1.5.1
|
version: 1.5.1
|
||||||
resolution: "base64-js@npm:1.5.1"
|
resolution: "base64-js@npm:1.5.1"
|
||||||
checksum: 669632eb3745404c2f822a18fc3a0122d2f9a7a13f7fb8b5823ee19d1d2ff9ee5b52c53367176ea4ad093c332fd5ab4bd0ebae5a8e27917a4105a4cfc86b1005
|
checksum: 669632eb3745404c2f822a18fc3a0122d2f9a7a13f7fb8b5823ee19d1d2ff9ee5b52c53367176ea4ad093c332fd5ab4bd0ebae5a8e27917a4105a4cfc86b1005
|
||||||
@@ -5511,6 +5605,15 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"bcrypt-pbkdf@npm:^1.0.2":
|
||||||
|
version: 1.0.2
|
||||||
|
resolution: "bcrypt-pbkdf@npm:1.0.2"
|
||||||
|
dependencies:
|
||||||
|
tweetnacl: ^0.14.3
|
||||||
|
checksum: 4edfc9fe7d07019609ccf797a2af28351736e9d012c8402a07120c4453a3b789a15f2ee1530dc49eee8f7eb9379331a8dd4b3766042b9e502f74a68e7f662291
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"better-opn@npm:^2.1.1":
|
"better-opn@npm:^2.1.1":
|
||||||
version: 2.1.1
|
version: 2.1.1
|
||||||
resolution: "better-opn@npm:2.1.1"
|
resolution: "better-opn@npm:2.1.1"
|
||||||
@@ -5557,6 +5660,17 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"bl@npm:^4.0.3":
|
||||||
|
version: 4.1.0
|
||||||
|
resolution: "bl@npm:4.1.0"
|
||||||
|
dependencies:
|
||||||
|
buffer: ^5.5.0
|
||||||
|
inherits: ^2.0.4
|
||||||
|
readable-stream: ^3.4.0
|
||||||
|
checksum: 9e8521fa7e83aa9427c6f8ccdcba6e8167ef30cc9a22df26effcc5ab682ef91d2cbc23a239f945d099289e4bbcfae7a192e9c28c84c6202e710a0dfec3722662
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"bluebird@npm:^3.5.5":
|
"bluebird@npm:^3.5.5":
|
||||||
version: 3.7.2
|
version: 3.7.2
|
||||||
resolution: "bluebird@npm:3.7.2"
|
resolution: "bluebird@npm:3.7.2"
|
||||||
@@ -5812,6 +5926,23 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"buffer@npm:^5.5.0":
|
||||||
|
version: 5.7.1
|
||||||
|
resolution: "buffer@npm:5.7.1"
|
||||||
|
dependencies:
|
||||||
|
base64-js: ^1.3.1
|
||||||
|
ieee754: ^1.1.13
|
||||||
|
checksum: e2cf8429e1c4c7b8cbd30834ac09bd61da46ce35f5c22a78e6c2f04497d6d25541b16881e30a019c6fd3154150650ccee27a308eff3e26229d788bbdeb08ab84
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"buildcheck@npm:0.0.3":
|
||||||
|
version: 0.0.3
|
||||||
|
resolution: "buildcheck@npm:0.0.3"
|
||||||
|
checksum: baf30605c56e80c2ca0502e40e18f2ebc7075bb4a861c941c0b36cd468b27957ed11a62248003ce99b9e5f91a7dfa859b30aad4fa50f0090c77a6f596ba20e6d
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"builtin-status-codes@npm:^3.0.0":
|
"builtin-status-codes@npm:^3.0.0":
|
||||||
version: 3.0.0
|
version: 3.0.0
|
||||||
resolution: "builtin-status-codes@npm:3.0.0"
|
resolution: "builtin-status-codes@npm:3.0.0"
|
||||||
@@ -6548,14 +6679,14 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"cookies-next@npm:^2.0.4":
|
"cookies-next@npm:^2.1.1":
|
||||||
version: 2.0.4
|
version: 2.1.1
|
||||||
resolution: "cookies-next@npm:2.0.4"
|
resolution: "cookies-next@npm:2.1.1"
|
||||||
dependencies:
|
dependencies:
|
||||||
"@types/cookie": ^0.4.1
|
"@types/cookie": ^0.4.1
|
||||||
"@types/node": ^16.10.2
|
"@types/node": ^16.10.2
|
||||||
cookie: ^0.4.0
|
cookie: ^0.4.0
|
||||||
checksum: fc25b4215f2d7092d72f8591c9bc8b30f3ea866fca76e536e31825899c3f05eefb97cdb4152c565429cab38d20f2f937d08aea76a43d3cdd3ca36e24a347fe00
|
checksum: c5fc2c72cf2d46d6fa804e5690b5038bab3d5c7e741a8472079bfbd6920010802962f7512d999ea430ebcbfc7c89c38e16f423479e4df7cb0bb782cc1a7f9004
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
@@ -6649,6 +6780,17 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"cpu-features@npm:~0.0.4":
|
||||||
|
version: 0.0.4
|
||||||
|
resolution: "cpu-features@npm:0.0.4"
|
||||||
|
dependencies:
|
||||||
|
buildcheck: 0.0.3
|
||||||
|
nan: ^2.15.0
|
||||||
|
node-gyp: latest
|
||||||
|
checksum: a20d58e41e63182b34753dfe23bd1d967944ec13d84b70849b5d334fb4a558b7e71e7f955ed86c8e75dd65b5c5b882f1c494174d342cb6d8a062d77f79d39596
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"cpy@npm:^8.1.2":
|
"cpy@npm:^8.1.2":
|
||||||
version: 8.1.2
|
version: 8.1.2
|
||||||
resolution: "cpy@npm:8.1.2"
|
resolution: "cpy@npm:8.1.2"
|
||||||
@@ -6941,10 +7083,10 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"dayjs@npm:^1.11.2":
|
"dayjs@npm:^1.11.3":
|
||||||
version: 1.11.2
|
version: 1.11.3
|
||||||
resolution: "dayjs@npm:1.11.2"
|
resolution: "dayjs@npm:1.11.3"
|
||||||
checksum: 78f8bd04a9e5f5554aa06eacda65a7d59e162d39f621a46fd34fb3b51367c3662426d86b4e2f4ac535f81e0c4d5af3e8a83b37e672412eb556267d726c61f8f3
|
checksum: c87e06b562a51ae6568cc5b840c7579d82a0f8af7163128c858fe512d3d71d07bd8e8e464b8cc41b8698a9e26b80ab2c082d14a1cd4c33df5692d77ccdfc5a43
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
@@ -7213,6 +7355,28 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"docker-modem@npm:^3.0.0":
|
||||||
|
version: 3.0.5
|
||||||
|
resolution: "docker-modem@npm:3.0.5"
|
||||||
|
dependencies:
|
||||||
|
debug: ^4.1.1
|
||||||
|
readable-stream: ^3.5.0
|
||||||
|
split-ca: ^1.0.1
|
||||||
|
ssh2: ^1.4.0
|
||||||
|
checksum: 79027f8e719a77031790af628f9aa1d72607ec3769149de3a4b683930f2e4d113ae0e3a7345b32ff3b2289f886879f4fcf216afb17908178ba00f9661c4e0dd6
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"dockerode@npm:^3.3.2":
|
||||||
|
version: 3.3.2
|
||||||
|
resolution: "dockerode@npm:3.3.2"
|
||||||
|
dependencies:
|
||||||
|
docker-modem: ^3.0.0
|
||||||
|
tar-fs: ~2.0.1
|
||||||
|
checksum: 69b60547ed2e6156e6ec1df16fccea9150c935ee0b0517723b4d05a5d840a01d4cd945341390d24b7fa301383be64145d563d9319be56d487a5bcbf9f872ee59
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"doctrine@npm:^2.1.0":
|
"doctrine@npm:^2.1.0":
|
||||||
version: 2.1.0
|
version: 2.1.0
|
||||||
resolution: "doctrine@npm:2.1.0"
|
resolution: "doctrine@npm:2.1.0"
|
||||||
@@ -7436,7 +7600,7 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"end-of-stream@npm:^1.0.0, end-of-stream@npm:^1.1.0":
|
"end-of-stream@npm:^1.0.0, end-of-stream@npm:^1.1.0, end-of-stream@npm:^1.4.1":
|
||||||
version: 1.4.4
|
version: 1.4.4
|
||||||
resolution: "end-of-stream@npm:1.4.4"
|
resolution: "end-of-stream@npm:1.4.4"
|
||||||
dependencies:
|
dependencies:
|
||||||
@@ -8686,6 +8850,13 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"fs-constants@npm:^1.0.0":
|
||||||
|
version: 1.0.0
|
||||||
|
resolution: "fs-constants@npm:1.0.0"
|
||||||
|
checksum: 18f5b718371816155849475ac36c7d0b24d39a11d91348cfcb308b4494824413e03572c403c86d3a260e049465518c4f0d5bd00f0371cdfcad6d4f30a85b350d
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"fs-extra@npm:^10.1.0":
|
"fs-extra@npm:^10.1.0":
|
||||||
version: 10.1.0
|
version: 10.1.0
|
||||||
resolution: "fs-extra@npm:10.1.0"
|
resolution: "fs-extra@npm:10.1.0"
|
||||||
@@ -9081,7 +9252,7 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"got@npm:^12.0.1, got@npm:^12.0.4":
|
"got@npm:^12.0.1, got@npm:^12.0.4, got@npm:^12.1.0":
|
||||||
version: 12.1.0
|
version: 12.1.0
|
||||||
resolution: "got@npm:12.1.0"
|
resolution: "got@npm:12.1.0"
|
||||||
dependencies:
|
dependencies:
|
||||||
@@ -9395,34 +9566,37 @@ __metadata:
|
|||||||
resolution: "homarr@workspace:."
|
resolution: "homarr@workspace:."
|
||||||
dependencies:
|
dependencies:
|
||||||
"@babel/core": ^7.17.8
|
"@babel/core": ^7.17.8
|
||||||
"@ctrl/deluge": ^4.0.0
|
"@ctrl/deluge": ^4.1.0
|
||||||
"@ctrl/qbittorrent": ^4.0.0
|
"@ctrl/qbittorrent": ^4.0.0
|
||||||
"@ctrl/shared-torrent": ^4.1.0
|
"@ctrl/shared-torrent": ^4.1.0
|
||||||
|
"@ctrl/transmission": ^4.1.1
|
||||||
"@dnd-kit/core": ^6.0.1
|
"@dnd-kit/core": ^6.0.1
|
||||||
"@dnd-kit/sortable": ^7.0.0
|
"@dnd-kit/sortable": ^7.0.0
|
||||||
"@dnd-kit/utilities": ^3.2.0
|
"@dnd-kit/utilities": ^3.2.0
|
||||||
"@mantine/core": ^4.2.6
|
"@mantine/core": ^4.2.8
|
||||||
"@mantine/dates": ^4.2.6
|
"@mantine/dates": ^4.2.8
|
||||||
"@mantine/dropzone": ^4.2.6
|
"@mantine/dropzone": ^4.2.8
|
||||||
"@mantine/form": ^4.2.6
|
"@mantine/form": ^4.2.8
|
||||||
"@mantine/hooks": ^4.2.6
|
"@mantine/hooks": ^4.2.8
|
||||||
"@mantine/next": ^4.2.6
|
"@mantine/next": ^4.2.8
|
||||||
"@mantine/notifications": ^4.2.6
|
"@mantine/notifications": ^4.2.8
|
||||||
"@mantine/prism": ^4.2.6
|
"@mantine/prism": ^4.2.8
|
||||||
"@next/bundle-analyzer": ^12.1.4
|
"@next/bundle-analyzer": ^12.2.0
|
||||||
"@next/eslint-plugin-next": ^12.1.4
|
"@next/eslint-plugin-next": ^12.2.0
|
||||||
"@nivo/core": ^0.79.0
|
"@nivo/core": ^0.79.0
|
||||||
"@nivo/line": ^0.79.1
|
"@nivo/line": ^0.79.1
|
||||||
"@storybook/react": ^6.5.4
|
"@storybook/react": ^6.5.4
|
||||||
"@tabler/icons": ^1.68.0
|
"@tabler/icons": ^1.68.0
|
||||||
|
"@types/dockerode": ^3.3.9
|
||||||
"@types/node": ^17.0.23
|
"@types/node": ^17.0.23
|
||||||
"@types/react": 17.0.43
|
"@types/react": 17.0.43
|
||||||
"@types/uuid": ^8.3.4
|
"@types/uuid": ^8.3.4
|
||||||
"@typescript-eslint/eslint-plugin": ^5.16.0
|
"@typescript-eslint/eslint-plugin": ^5.16.0
|
||||||
"@typescript-eslint/parser": ^5.16.0
|
"@typescript-eslint/parser": ^5.16.0
|
||||||
axios: ^0.27.2
|
axios: ^0.27.2
|
||||||
cookies-next: ^2.0.4
|
cookies-next: ^2.1.1
|
||||||
dayjs: ^1.11.2
|
dayjs: ^1.11.3
|
||||||
|
dockerode: ^3.3.2
|
||||||
eslint: ^8.11.0
|
eslint: ^8.11.0
|
||||||
eslint-config-airbnb: ^19.0.4
|
eslint-config-airbnb: ^19.0.4
|
||||||
eslint-config-airbnb-typescript: ^16.1.0
|
eslint-config-airbnb-typescript: ^16.1.0
|
||||||
@@ -9438,7 +9612,7 @@ __metadata:
|
|||||||
framer-motion: ^6.3.1
|
framer-motion: ^6.3.1
|
||||||
jest: ^28.1.0
|
jest: ^28.1.0
|
||||||
js-file-download: ^0.4.12
|
js-file-download: ^0.4.12
|
||||||
next: 12.1.6
|
next: ^12.2.0
|
||||||
prettier: ^2.6.2
|
prettier: ^2.6.2
|
||||||
prism-react-renderer: ^1.3.1
|
prism-react-renderer: ^1.3.1
|
||||||
react: ^17.0.1
|
react: ^17.0.1
|
||||||
@@ -9673,7 +9847,7 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"ieee754@npm:^1.1.4":
|
"ieee754@npm:^1.1.13, ieee754@npm:^1.1.4":
|
||||||
version: 1.2.1
|
version: 1.2.1
|
||||||
resolution: "ieee754@npm:1.2.1"
|
resolution: "ieee754@npm:1.2.1"
|
||||||
checksum: 5144c0c9815e54ada181d80a0b810221a253562422e7c6c3a60b1901154184f49326ec239d618c416c1c5945a2e197107aee8d986a3dd836b53dffefd99b5e7e
|
checksum: 5144c0c9815e54ada181d80a0b810221a253562422e7c6c3a60b1901154184f49326ec239d618c416c1c5945a2e197107aee8d986a3dd836b53dffefd99b5e7e
|
||||||
@@ -11810,6 +11984,13 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"mkdirp-classic@npm:^0.5.2":
|
||||||
|
version: 0.5.3
|
||||||
|
resolution: "mkdirp-classic@npm:0.5.3"
|
||||||
|
checksum: 3f4e088208270bbcc148d53b73e9a5bd9eef05ad2cbf3b3d0ff8795278d50dd1d11a8ef1875ff5aea3fa888931f95bfcb2ad5b7c1061cfefd6284d199e6776ac
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"mkdirp@npm:^0.5.1, mkdirp@npm:^0.5.3":
|
"mkdirp@npm:^0.5.1, mkdirp@npm:^0.5.3":
|
||||||
version: 0.5.6
|
version: 0.5.6
|
||||||
resolution: "mkdirp@npm:0.5.6"
|
resolution: "mkdirp@npm:0.5.6"
|
||||||
@@ -11889,7 +12070,7 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"nan@npm:^2.12.1":
|
"nan@npm:^2.12.1, nan@npm:^2.15.0, nan@npm:^2.16.0":
|
||||||
version: 2.16.0
|
version: 2.16.0
|
||||||
resolution: "nan@npm:2.16.0"
|
resolution: "nan@npm:2.16.0"
|
||||||
dependencies:
|
dependencies:
|
||||||
@@ -11954,26 +12135,29 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"next@npm:12.1.6":
|
"next@npm:^12.2.0":
|
||||||
version: 12.1.6
|
version: 12.2.0
|
||||||
resolution: "next@npm:12.1.6"
|
resolution: "next@npm:12.2.0"
|
||||||
dependencies:
|
dependencies:
|
||||||
"@next/env": 12.1.6
|
"@next/env": 12.2.0
|
||||||
"@next/swc-android-arm-eabi": 12.1.6
|
"@next/swc-android-arm-eabi": 12.2.0
|
||||||
"@next/swc-android-arm64": 12.1.6
|
"@next/swc-android-arm64": 12.2.0
|
||||||
"@next/swc-darwin-arm64": 12.1.6
|
"@next/swc-darwin-arm64": 12.2.0
|
||||||
"@next/swc-darwin-x64": 12.1.6
|
"@next/swc-darwin-x64": 12.2.0
|
||||||
"@next/swc-linux-arm-gnueabihf": 12.1.6
|
"@next/swc-freebsd-x64": 12.2.0
|
||||||
"@next/swc-linux-arm64-gnu": 12.1.6
|
"@next/swc-linux-arm-gnueabihf": 12.2.0
|
||||||
"@next/swc-linux-arm64-musl": 12.1.6
|
"@next/swc-linux-arm64-gnu": 12.2.0
|
||||||
"@next/swc-linux-x64-gnu": 12.1.6
|
"@next/swc-linux-arm64-musl": 12.2.0
|
||||||
"@next/swc-linux-x64-musl": 12.1.6
|
"@next/swc-linux-x64-gnu": 12.2.0
|
||||||
"@next/swc-win32-arm64-msvc": 12.1.6
|
"@next/swc-linux-x64-musl": 12.2.0
|
||||||
"@next/swc-win32-ia32-msvc": 12.1.6
|
"@next/swc-win32-arm64-msvc": 12.2.0
|
||||||
"@next/swc-win32-x64-msvc": 12.1.6
|
"@next/swc-win32-ia32-msvc": 12.2.0
|
||||||
|
"@next/swc-win32-x64-msvc": 12.2.0
|
||||||
|
"@swc/helpers": 0.4.2
|
||||||
caniuse-lite: ^1.0.30001332
|
caniuse-lite: ^1.0.30001332
|
||||||
postcss: 8.4.5
|
postcss: 8.4.5
|
||||||
styled-jsx: 5.0.2
|
styled-jsx: 5.0.2
|
||||||
|
use-sync-external-store: 1.1.0
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
fibers: ">= 3.1.0"
|
fibers: ">= 3.1.0"
|
||||||
node-sass: ^6.0.0 || ^7.0.0
|
node-sass: ^6.0.0 || ^7.0.0
|
||||||
@@ -11989,6 +12173,8 @@ __metadata:
|
|||||||
optional: true
|
optional: true
|
||||||
"@next/swc-darwin-x64":
|
"@next/swc-darwin-x64":
|
||||||
optional: true
|
optional: true
|
||||||
|
"@next/swc-freebsd-x64":
|
||||||
|
optional: true
|
||||||
"@next/swc-linux-arm-gnueabihf":
|
"@next/swc-linux-arm-gnueabihf":
|
||||||
optional: true
|
optional: true
|
||||||
"@next/swc-linux-arm64-gnu":
|
"@next/swc-linux-arm64-gnu":
|
||||||
@@ -12014,7 +12200,7 @@ __metadata:
|
|||||||
optional: true
|
optional: true
|
||||||
bin:
|
bin:
|
||||||
next: dist/bin/next
|
next: dist/bin/next
|
||||||
checksum: 670d544fd47670c29681d10824e6da625e9d4a048e564c8d9cb80d37f33c9ff9b5ca0a53e6d84d8d618b1fe7c9bb4e6b45040cb7e57a5c46b232a8f914425dc1
|
checksum: 38456c33935122ac1581367e4982034be23269039a8470a66443d710334336f8f3fb587f25d172d138d84cf18c01d3a76627fb610c2e2e57bd1692277c23fa2b
|
||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
@@ -13664,7 +13850,7 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"readable-stream@npm:^3.6.0":
|
"readable-stream@npm:^3.1.1, readable-stream@npm:^3.4.0, readable-stream@npm:^3.5.0, readable-stream@npm:^3.6.0":
|
||||||
version: 3.6.0
|
version: 3.6.0
|
||||||
resolution: "readable-stream@npm:3.6.0"
|
resolution: "readable-stream@npm:3.6.0"
|
||||||
dependencies:
|
dependencies:
|
||||||
@@ -14147,7 +14333,7 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"safer-buffer@npm:>= 2.1.2 < 3, safer-buffer@npm:>= 2.1.2 < 3.0.0, safer-buffer@npm:^2.1.0":
|
"safer-buffer@npm:>= 2.1.2 < 3, safer-buffer@npm:>= 2.1.2 < 3.0.0, safer-buffer@npm:^2.1.0, safer-buffer@npm:~2.1.0":
|
||||||
version: 2.1.2
|
version: 2.1.2
|
||||||
resolution: "safer-buffer@npm:2.1.2"
|
resolution: "safer-buffer@npm:2.1.2"
|
||||||
checksum: cab8f25ae6f1434abee8d80023d7e72b598cf1327164ddab31003c51215526801e40b66c5e65d658a0af1e9d6478cadcb4c745f4bd6751f97d8644786c0978b0
|
checksum: cab8f25ae6f1434abee8d80023d7e72b598cf1327164ddab31003c51215526801e40b66c5e65d658a0af1e9d6478cadcb4c745f4bd6751f97d8644786c0978b0
|
||||||
@@ -14628,6 +14814,13 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"split-ca@npm:^1.0.1":
|
||||||
|
version: 1.0.1
|
||||||
|
resolution: "split-ca@npm:1.0.1"
|
||||||
|
checksum: 1e7409938a95ee843fe2593156a5735e6ee63772748ee448ea8477a5a3e3abde193c3325b3696e56a5aff07c7dcf6b1f6a2f2a036895b4f3afe96abb366d893f
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"split-string@npm:^3.0.1, split-string@npm:^3.0.2":
|
"split-string@npm:^3.0.1, split-string@npm:^3.0.2":
|
||||||
version: 3.1.0
|
version: 3.1.0
|
||||||
resolution: "split-string@npm:3.1.0"
|
resolution: "split-string@npm:3.1.0"
|
||||||
@@ -14644,6 +14837,23 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"ssh2@npm:^1.4.0":
|
||||||
|
version: 1.11.0
|
||||||
|
resolution: "ssh2@npm:1.11.0"
|
||||||
|
dependencies:
|
||||||
|
asn1: ^0.2.4
|
||||||
|
bcrypt-pbkdf: ^1.0.2
|
||||||
|
cpu-features: ~0.0.4
|
||||||
|
nan: ^2.16.0
|
||||||
|
dependenciesMeta:
|
||||||
|
cpu-features:
|
||||||
|
optional: true
|
||||||
|
nan:
|
||||||
|
optional: true
|
||||||
|
checksum: e40cb9f171741a807c170dc555078aa8c49dc93dd36fc9c8be026fce1cfd31f0d37078d9b60a0f2cfb11d0e007ed5407376b72f8a0ef9f2cb89574632bbfb824
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"ssri@npm:^6.0.1":
|
"ssri@npm:^6.0.1":
|
||||||
version: 6.0.2
|
version: 6.0.2
|
||||||
resolution: "ssri@npm:6.0.2"
|
resolution: "ssri@npm:6.0.2"
|
||||||
@@ -15094,6 +15304,31 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"tar-fs@npm:~2.0.1":
|
||||||
|
version: 2.0.1
|
||||||
|
resolution: "tar-fs@npm:2.0.1"
|
||||||
|
dependencies:
|
||||||
|
chownr: ^1.1.1
|
||||||
|
mkdirp-classic: ^0.5.2
|
||||||
|
pump: ^3.0.0
|
||||||
|
tar-stream: ^2.0.0
|
||||||
|
checksum: 26cd297ed2421bc8038ce1a4ca442296b53739f409847d495d46086e5713d8db27f2c03ba2f461d0f5ddbc790045628188a8544f8ae32cbb6238b279b68d0247
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
|
"tar-stream@npm:^2.0.0":
|
||||||
|
version: 2.2.0
|
||||||
|
resolution: "tar-stream@npm:2.2.0"
|
||||||
|
dependencies:
|
||||||
|
bl: ^4.0.3
|
||||||
|
end-of-stream: ^1.4.1
|
||||||
|
fs-constants: ^1.0.0
|
||||||
|
inherits: ^2.0.3
|
||||||
|
readable-stream: ^3.1.1
|
||||||
|
checksum: 699831a8b97666ef50021c767f84924cfee21c142c2eb0e79c63254e140e6408d6d55a065a2992548e72b06de39237ef2b802b99e3ece93ca3904a37622a66f3
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"tar@npm:^6.0.2, tar@npm:^6.1.11, tar@npm:^6.1.2":
|
"tar@npm:^6.0.2, tar@npm:^6.1.11, tar@npm:^6.1.2":
|
||||||
version: 6.1.11
|
version: 6.1.11
|
||||||
resolution: "tar@npm:6.1.11"
|
resolution: "tar@npm:6.1.11"
|
||||||
@@ -15448,7 +15683,7 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
"tslib@npm:^2.0.0, tslib@npm:^2.0.1, tslib@npm:^2.0.3, tslib@npm:^2.1.0":
|
"tslib@npm:^2.0.0, tslib@npm:^2.0.1, tslib@npm:^2.0.3, tslib@npm:^2.1.0, tslib@npm:^2.4.0":
|
||||||
version: 2.4.0
|
version: 2.4.0
|
||||||
resolution: "tslib@npm:2.4.0"
|
resolution: "tslib@npm:2.4.0"
|
||||||
checksum: 8c4aa6a3c5a754bf76aefc38026134180c053b7bd2f81338cb5e5ebf96fefa0f417bff221592bf801077f5bf990562f6264fecbc42cd3309b33872cb6fc3b113
|
checksum: 8c4aa6a3c5a754bf76aefc38026134180c053b7bd2f81338cb5e5ebf96fefa0f417bff221592bf801077f5bf990562f6264fecbc42cd3309b33872cb6fc3b113
|
||||||
@@ -15473,6 +15708,13 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"tweetnacl@npm:^0.14.3":
|
||||||
|
version: 0.14.5
|
||||||
|
resolution: "tweetnacl@npm:0.14.5"
|
||||||
|
checksum: 6061daba1724f59473d99a7bb82e13f211cdf6e31315510ae9656fefd4779851cb927adad90f3b488c8ed77c106adc0421ea8055f6f976ff21b27c5c4e918487
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"type-check@npm:^0.4.0, type-check@npm:~0.4.0":
|
"type-check@npm:^0.4.0, type-check@npm:~0.4.0":
|
||||||
version: 0.4.0
|
version: 0.4.0
|
||||||
resolution: "type-check@npm:0.4.0"
|
resolution: "type-check@npm:0.4.0"
|
||||||
@@ -15877,6 +16119,15 @@ __metadata:
|
|||||||
languageName: node
|
languageName: node
|
||||||
linkType: hard
|
linkType: hard
|
||||||
|
|
||||||
|
"use-sync-external-store@npm:1.1.0":
|
||||||
|
version: 1.1.0
|
||||||
|
resolution: "use-sync-external-store@npm:1.1.0"
|
||||||
|
peerDependencies:
|
||||||
|
react: ^16.8.0 || ^17.0.0 || ^18.0.0
|
||||||
|
checksum: 8993a0b642f91d7fcdbb02b7b3ac984bd3af4769686f38291fe7fcfe73dfb73d6c64d20dfb7e5e7fbf5a6da8f5392d6f8e5b00c243a04975595946e82c02b883
|
||||||
|
languageName: node
|
||||||
|
linkType: hard
|
||||||
|
|
||||||
"use@npm:^3.1.0":
|
"use@npm:^3.1.0":
|
||||||
version: 3.1.1
|
version: 3.1.1
|
||||||
resolution: "use@npm:3.1.1"
|
resolution: "use@npm:3.1.1"
|
||||||
|
|||||||
Reference in New Issue
Block a user