diff --git a/.dockerignore b/.dockerignore index 40d5f5bbf6..74fa836a6c 100644 --- a/.dockerignore +++ b/.dockerignore @@ -4,3 +4,4 @@ Dockerfile .gitignore .DS_Store node_modules +.idea \ No newline at end of file diff --git a/.eslintrc.json b/.eslintrc.json index 6ea8791edd..691ae90c7f 100644 --- a/.eslintrc.json +++ b/.eslintrc.json @@ -2,14 +2,10 @@ "env": { "browser": true, "es2020": true, - "node": true + "node": true, + "jest": true }, - "extends": [ - "eslint:recommended", - "plugin:prettier/recommended", - "plugin:import/recommended", - "next" - ], + "parser": "@typescript-eslint/parser", "parserOptions": { "ecmaFeatures": { "jsx": true @@ -19,30 +15,37 @@ }, "settings": { "import/resolver": { - "alias": { - "map": [ - ["assets", "./assets"], - ["components", "./components"], - ["db", "./db"], - ["hooks", "./hooks"], - ["lang", "./lang"], - ["lib", "./lib"], - ["public", "./public"], - ["queries", "./queries"], - ["store", "./store"], - ["styles", "./styles"] - ], - "extensions": [".ts", ".js", ".jsx", ".json"] + "node": { + "moduleDirectory": ["node_modules", "src/"] } } }, + "extends": [ + "plugin:@typescript-eslint/recommended", + "eslint:recommended", + "plugin:prettier/recommended", + "plugin:import/errors", + "plugin:import/typescript", + "plugin:css-modules/recommended", + "plugin:cypress/recommended", + "prettier", + "next" + ], + "plugins": ["@typescript-eslint", "prettier", "promise", "css-modules", "cypress"], "rules": { "no-console": "error", "react/display-name": "off", + "react-hooks/exhaustive-deps": "off", "react/react-in-jsx-scope": "off", "react/prop-types": "off", "import/no-anonymous-default-export": "off", - "@next/next/no-img-element": "off" + "import/no-named-as-default": "off", + "@next/next/no-img-element": "off", + "@typescript-eslint/no-empty-function": "off", + "@typescript-eslint/no-explicit-any": "off", + "@typescript-eslint/no-var-requires": "off", + "@typescript-eslint/no-empty-interface": "off", + "@typescript-eslint/no-unused-vars": ["error", { "ignoreRestSiblings": true }] }, "globals": { "React": "writable" diff --git a/.github/ISSUE_TEMPLATE/1.bug_report.yml b/.github/ISSUE_TEMPLATE/1.bug_report.yml new file mode 100644 index 0000000000..711468f259 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/1.bug_report.yml @@ -0,0 +1,36 @@ +name: '🐛 Bug Report' +description: Create a bug report for Umami. +body: + - type: textarea + attributes: + label: Describe the Bug + description: A clear and concise description of what the bug is. + validations: + required: true + - type: dropdown + attributes: + label: Database + description: What database are you using? + options: + - PostgreSQL + - MySQL + - Umami Cloud + validations: + required: true + - type: textarea + attributes: + label: Relevant log output + description: Please copy and paste any relevant log output. This will be automatically formatted into code, so no need for backticks. + render: shell + - type: input + attributes: + label: Which Umami version are you using? (if relevant) + description: 'For example: Chrome, Edge, Firefox, etc' + - type: input + attributes: + label: Which browser are you using? (if relevant) + description: 'For example: Chrome, Edge, Firefox, etc' + - type: input + attributes: + label: How are you deploying your application? (if relevant) + description: 'For example: Vercel, Railway, Docker, etc' diff --git a/.github/ISSUE_TEMPLATE/2.feature_request.yml b/.github/ISSUE_TEMPLATE/2.feature_request.yml new file mode 100644 index 0000000000..529a6c7328 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/2.feature_request.yml @@ -0,0 +1,9 @@ +name: '✨ Feature Request' +description: Create a feature or enhancement request for Umami. +body: + - type: textarea + attributes: + label: Describe the feature or enhancement + description: A clear and concise description of what the feature or enhancement is. + validations: + required: true diff --git a/.github/ISSUE_TEMPLATE/config.yml b/.github/ISSUE_TEMPLATE/config.yml new file mode 100644 index 0000000000..92132929a6 --- /dev/null +++ b/.github/ISSUE_TEMPLATE/config.yml @@ -0,0 +1,6 @@ + +blank_issues_enabled: false +contact_links: + - name: "🤔 Ask a question" + url: https://github.com/umami-software/umami/discussions + about: Ask questions and discuss with other community members. \ No newline at end of file diff --git a/.github/stale.yml b/.github/stale.yml deleted file mode 100644 index 2dc5b6750d..0000000000 --- a/.github/stale.yml +++ /dev/null @@ -1,19 +0,0 @@ -# Number of days of inactivity before an issue becomes stale -daysUntilStale: 60 -# Number of days of inactivity before a stale issue is closed -daysUntilClose: 7 -# Issues with these labels will never be considered stale -exemptLabels: - - pinned - - security - - enhancement - - bug -# Label to use when marking an issue as stale -staleLabel: wontfix -# Comment to post when marking an issue as stale. Set to `false` to disable -markComment: > - This issue has been automatically marked as stale because it has not had - recent activity. It will be closed if no further activity occurs. Thank you - for your contributions. -# Comment to post when closing a stale issue. Set to `false` to disable -closeComment: false diff --git a/.github/workflows/cd-manual.yml b/.github/workflows/cd-manual.yml index 6b04dd847a..ac701fcc50 100644 --- a/.github/workflows/cd-manual.yml +++ b/.github/workflows/cd-manual.yml @@ -18,14 +18,26 @@ jobs: db-type: [postgresql, mysql] steps: - - uses: actions/checkout@v2 + - uses: actions/checkout@v3 - - uses: mr-smithers-excellent/docker-build-push@v5 - name: Build & push Docker image for ${{ matrix.db-type }} + - uses: mr-smithers-excellent/docker-build-push@v6 + name: Build & push Docker image to ghcr.io for ${{ matrix.db-type }} with: image: umami tags: ${{ matrix.db-type }}-${{ inputs.version }}, ${{ matrix.db-type }}-latest buildArgs: DATABASE_TYPE=${{ matrix.db-type }} registry: ghcr.io + multiPlatform: true + platform: linux/amd64,linux/arm64 username: ${{ github.actor }} password: ${{ secrets.GITHUB_TOKEN }} + + - uses: mr-smithers-excellent/docker-build-push@v6 + name: Build & push Docker image to docker.io for ${{ matrix.db-type }} + with: + image: umamisoftware/umami + tags: ${{ matrix.db-type }}-${{ inputs.version }}, ${{ matrix.db-type }}-latest + buildArgs: DATABASE_TYPE=${{ matrix.db-type }} + registry: docker.io + username: ${{ secrets.DOCKER_USERNAME }} + password: ${{ secrets.DOCKER_PASSWORD }} \ No newline at end of file diff --git a/.github/workflows/cd.yml b/.github/workflows/cd.yml index 6d8a7f7309..b4de549922 100644 --- a/.github/workflows/cd.yml +++ b/.github/workflows/cd.yml @@ -3,7 +3,6 @@ name: Create docker images on: [create] jobs: - build: name: Build, push, and deploy if: ${{ startsWith(github.ref, 'refs/tags/v') }} @@ -14,17 +13,32 @@ jobs: db-type: [postgresql, mysql] steps: - - uses: actions/checkout@v2 + - uses: actions/checkout@v3 - name: Set env - run: echo "RELEASE_VERSION=${GITHUB_REF#refs/*/}" >> $GITHUB_ENV + run: | + echo "RELEASE_VERSION=${GITHUB_REF#refs/*/}" >> $GITHUB_ENV + echo "NOW=$(date -u +'%Y-%m-%dT%H:%M:%SZ')" >> $GITHUB_ENV - - uses: mr-smithers-excellent/docker-build-push@v5 - name: Build & push Docker image for ${{ matrix.db-type }} + - uses: mr-smithers-excellent/docker-build-push@v6 + name: Build & push Docker image to ghcr.io for ${{ matrix.db-type }} with: image: umami tags: ${{ matrix.db-type }}-${{ env.RELEASE_VERSION }}, ${{ matrix.db-type }}-latest buildArgs: DATABASE_TYPE=${{ matrix.db-type }} registry: ghcr.io + multiPlatform: true + platform: linux/amd64,linux/arm64 username: ${{ github.actor }} password: ${{ secrets.GITHUB_TOKEN }} + + + - uses: mr-smithers-excellent/docker-build-push@v6 + name: Build & push Docker image to docker.io for ${{ matrix.db-type }} + with: + image: umamisoftware/umami + tags: ${{ matrix.db-type }}-${{ env.RELEASE_VERSION }}, ${{ matrix.db-type }}-latest + buildArgs: DATABASE_TYPE=${{ matrix.db-type }} + registry: docker.io + username: ${{ secrets.DOCKER_USERNAME }} + password: ${{ secrets.DOCKER_PASSWORD }} \ No newline at end of file diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index c24c2e6d01..22a083c08a 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -16,24 +16,21 @@ jobs: strategy: matrix: include: - - node-version: 14.x + - node-version: 18.17 db-type: postgresql - - node-version: 14.x - db-type: mysql - - node-version: 16.x - db-type: postgresql - - node-version: 16.x + - node-version: 18.17 db-type: mysql steps: - - uses: actions/checkout@v2 + - uses: actions/checkout@v4 - name: Use Node.js ${{ matrix.node-version }} - uses: actions/setup-node@v2 + uses: actions/setup-node@v4 with: node-version: ${{ matrix.node-version }} - cache: 'npm' + cache: 'yarn' env: DATABASE_TYPE: ${{ matrix.db-type }} - run: npm install --global yarn - - run: yarn install --frozen-lockfile + - run: yarn install + - run: yarn test - run: yarn build diff --git a/.github/workflows/stale-issues.yml b/.github/workflows/stale-issues.yml new file mode 100644 index 0000000000..f160401497 --- /dev/null +++ b/.github/workflows/stale-issues.yml @@ -0,0 +1,25 @@ +name: Close stale issues +on: + schedule: + - cron: '30 1 * * *' + +jobs: + stale: + runs-on: ubuntu-latest + permissions: + issues: write + pull-requests: write + steps: + - uses: actions/stale@v8 + with: + days-before-issue-stale: 60 + days-before-issue-close: 7 + stale-issue-label: 'stale' + stale-issue-message: 'This issue is stale because it has been open for 60 days with no activity.' + close-issue-message: 'This issue was closed because it has been inactive for 7 days since being marked as stale.' + days-before-pr-stale: -1 + days-before-pr-close: -1 + operations-per-run: 200 + ascending: true + repo-token: ${{ secrets.GITHUB_TOKEN }} + exempt-issue-labels: bug,enhancement diff --git a/.gitignore b/.gitignore index 5441032437..b11f450938 100644 --- a/.gitignore +++ b/.gitignore @@ -1,8 +1,8 @@ # See https://help.github.com/articles/ignoring-files/ for more about ignoring files. # dependencies -/node_modules -/.pnp +node_modules +.pnp .pnp.js # testing @@ -15,15 +15,18 @@ # production /build -/public/umami.js -/public/geo +/public/script.js +/geo +/dist # misc .DS_Store .idea +.yarn *.iml *.log -/.vscode/ +.vscode +.tool-versions # debug npm-debug.log* @@ -32,9 +35,8 @@ yarn-error.log* # local env files .env -.env.development.local -.env.test.local -.env.production.local +.env.* +*.env.* *.dev.yml diff --git a/.prettierignore b/.prettierignore index 15ce475e76..08076d4054 100644 --- a/.prettierignore +++ b/.prettierignore @@ -1 +1 @@ -/public/ \ No newline at end of file +/public/script.js \ No newline at end of file diff --git a/.stylelintrc.json b/.stylelintrc.json index 9a05af1442..b940b63e41 100644 --- a/.stylelintrc.json +++ b/.stylelintrc.json @@ -5,13 +5,7 @@ "stylelint-config-prettier" ], "rules": { - "no-descending-specificity": null, - "selector-pseudo-class-no-unknown": [ - true, - { - "ignorePseudoClasses": ["global", "horizontal", "vertical"] - } - ] + "no-descending-specificity": null }, "ignoreFiles": ["**/*.js", "**/*.md"] } diff --git a/Dockerfile b/Dockerfile index eb7ebf1a53..801b2bc20a 100644 --- a/Dockerfile +++ b/Dockerfile @@ -1,16 +1,19 @@ # Install dependencies only when needed -FROM node:16-alpine AS deps +FROM node:18-alpine AS deps # Check https://github.com/nodejs/docker-node/tree/b4117f9333da4138b03a546ec926ef50a31506c3#nodealpine to understand why libc6-compat might be needed. RUN apk add --no-cache libc6-compat WORKDIR /app COPY package.json yarn.lock ./ +# Add yarn timeout to handle slow CPU when Github Actions +RUN yarn config set network-timeout 300000 RUN yarn install --frozen-lockfile # Rebuild the source code only when needed -FROM node:16-alpine AS builder +FROM node:18-alpine AS builder WORKDIR /app COPY --from=deps /app/node_modules ./node_modules COPY . . +COPY docker/middleware.js ./src ARG DATABASE_TYPE ARG BASE_PATH @@ -23,7 +26,7 @@ ENV NEXT_TELEMETRY_DISABLED 1 RUN yarn build-docker # Production image, copy all the files and run next -FROM node:16-alpine AS runner +FROM node:18-alpine AS runner WORKDIR /app ENV NODE_ENV production @@ -32,7 +35,9 @@ ENV NEXT_TELEMETRY_DISABLED 1 RUN addgroup --system --gid 1001 nodejs RUN adduser --system --uid 1001 nextjs -RUN yarn add npm-run-all dotenv prisma +RUN set -x \ + && apk add --no-cache curl \ + && yarn add npm-run-all dotenv prisma semver # You only need to copy next.config.js if you are NOT using the default configuration COPY --from=builder /app/next.config.js . @@ -50,6 +55,7 @@ USER nextjs EXPOSE 3000 +ENV HOSTNAME 0.0.0.0 ENV PORT 3000 CMD ["yarn", "start-docker"] diff --git a/LICENSE b/LICENSE index 512776558e..eff4136934 100644 --- a/LICENSE +++ b/LICENSE @@ -1,6 +1,6 @@ MIT License -Copyright (c) 2020 Mike Cao +Copyright (c) 2022 Umami Software, Inc. Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal diff --git a/README.md b/README.md index 3f457631ac..6d9755d4de 100644 --- a/README.md +++ b/README.md @@ -10,8 +10,8 @@ A detailed getting started guide can be found at [https://umami.is/docs/](https: ### Requirements -- A server with Node.js version 12 or newer -- A database. Umami supports [MySQL](https://www.mysql.com/) and [Postgresql](https://www.postgresql.org/) databases. +- A server with Node.js version 16.13 or newer +- A database. Umami supports [MySQL](https://www.mysql.com/) (minimum v8.0) and [Postgresql](https://www.postgresql.org/) (minimum v12.14) databases. ### Install Yarn @@ -36,6 +36,7 @@ DATABASE_URL=connection-url ``` The connection url is in the following format: + ``` postgresql://username:mypassword@localhost:5432/mydb @@ -48,7 +49,7 @@ mysql://username:mypassword@localhost:3306/mydb yarn build ``` -The build step will also create tables in your database if you ae installing for the first time. It will also create a login account with username **admin** and password **umami**. +The build step will also create tables in your database if you are installing for the first time. It will also create a login user with username **admin** and password **umami**. ### Start the application @@ -65,17 +66,19 @@ or change the [port](https://nextjs.org/docs/api-reference/cli#production) to se To build the umami container and start up a Postgres database, run: ```bash -docker compose up +docker compose up -d ``` Alternatively, to pull just the Umami Docker image with PostgreSQL support: + ```bash -docker pull docker.umami.dev/umami-software/umami:postgresql-latest +docker pull docker.umami.is/umami-software/umami:postgresql-latest ``` Or with MySQL support: + ```bash -docker pull docker.umami.dev/umami-software/umami:mysql-latest +docker pull docker.umami.is/umami-software/umami:mysql-latest ``` ## Getting updates @@ -98,3 +101,4 @@ docker compose up --force-recreate ## License MIT + diff --git a/app.json b/app.json index ba8b8c457d..cec5fda109 100644 --- a/app.json +++ b/app.json @@ -6,7 +6,7 @@ "repository": "https://github.com/umami-software/umami", "addons": ["heroku-postgresql"], "env": { - "HASH_SALT": { + "APP_SECRET": { "description": "Used to generate unique values for your installation", "required": true, "generator": "secret" diff --git a/assets/arrow-right.svg b/assets/arrow-right.svg deleted file mode 100644 index 6fc9390999..0000000000 --- a/assets/arrow-right.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/arrow-up-right-from-square.svg b/assets/arrow-up-right-from-square.svg deleted file mode 100644 index 90ad457f1f..0000000000 --- a/assets/arrow-up-right-from-square.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/bars.svg b/assets/bars.svg deleted file mode 100644 index fdb2d6e48a..0000000000 --- a/assets/bars.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/chart-bar.svg b/assets/chart-bar.svg deleted file mode 100644 index d1d72fdc24..0000000000 --- a/assets/chart-bar.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/check.svg b/assets/check.svg deleted file mode 100644 index 1a7abdce95..0000000000 --- a/assets/check.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/chevron-down.svg b/assets/chevron-down.svg deleted file mode 100644 index cb9d8fe135..0000000000 --- a/assets/chevron-down.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/code.svg b/assets/code.svg deleted file mode 100644 index cd29765e02..0000000000 --- a/assets/code.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/ellipsis-h.svg b/assets/ellipsis-h.svg deleted file mode 100644 index 5bb083591e..0000000000 --- a/assets/ellipsis-h.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/exclamation-triangle.svg b/assets/exclamation-triangle.svg deleted file mode 100644 index 46bef5bcba..0000000000 --- a/assets/exclamation-triangle.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/external-link.svg b/assets/external-link.svg deleted file mode 100644 index ed09306fc7..0000000000 --- a/assets/external-link.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/gear.svg b/assets/gear.svg deleted file mode 100644 index ab97a69308..0000000000 --- a/assets/gear.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/list-ul.svg b/assets/list-ul.svg deleted file mode 100644 index 5e63212638..0000000000 --- a/assets/list-ul.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/logo.svg b/assets/logo.svg deleted file mode 100644 index f0e522614a..0000000000 --- a/assets/logo.svg +++ /dev/null @@ -1,2 +0,0 @@ - - diff --git a/assets/moon.svg b/assets/moon.svg deleted file mode 100644 index 6c8955aed1..0000000000 --- a/assets/moon.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/pen.svg b/assets/pen.svg deleted file mode 100644 index 426c520c28..0000000000 --- a/assets/pen.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/plus.svg b/assets/plus.svg deleted file mode 100644 index e4774d8722..0000000000 --- a/assets/plus.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/sun.svg b/assets/sun.svg deleted file mode 100644 index ebc20eb27d..0000000000 --- a/assets/sun.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/times.svg b/assets/times.svg deleted file mode 100644 index c528bcdd30..0000000000 --- a/assets/times.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/trash.svg b/assets/trash.svg deleted file mode 100644 index 2f525c8f50..0000000000 --- a/assets/trash.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/user.svg b/assets/user.svg deleted file mode 100644 index c0094666d5..0000000000 --- a/assets/user.svg +++ /dev/null @@ -1 +0,0 @@ -Asset 1 \ No newline at end of file diff --git a/assets/visitor.svg b/assets/visitor.svg deleted file mode 100644 index 591873a5b7..0000000000 --- a/assets/visitor.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/assets/xmark.svg b/assets/xmark.svg deleted file mode 100644 index 340f479ea0..0000000000 --- a/assets/xmark.svg +++ /dev/null @@ -1 +0,0 @@ - \ No newline at end of file diff --git a/components/common/Button.js b/components/common/Button.js deleted file mode 100644 index d3b7d2adc4..0000000000 --- a/components/common/Button.js +++ /dev/null @@ -1,63 +0,0 @@ -import React from 'react'; -import PropTypes from 'prop-types'; -import ReactTooltip from 'react-tooltip'; -import classNames from 'classnames'; -import Icon from './Icon'; -import styles from './Button.module.css'; - -function Button({ - type = 'button', - icon, - size, - variant, - children, - className, - tooltip, - tooltipId, - disabled, - iconRight, - onClick = () => {}, - ...props -}) { - return ( - - ); -} - -Button.propTypes = { - type: PropTypes.oneOf(['button', 'submit', 'reset']), - icon: PropTypes.node, - size: PropTypes.oneOf(['xlarge', 'large', 'medium', 'small', 'xsmall']), - variant: PropTypes.oneOf(['action', 'danger', 'light']), - children: PropTypes.node, - className: PropTypes.string, - tooltip: PropTypes.node, - tooltipId: PropTypes.string, - disabled: PropTypes.bool, - iconRight: PropTypes.bool, - onClick: PropTypes.func, -}; - -export default Button; diff --git a/components/common/Button.module.css b/components/common/Button.module.css deleted file mode 100644 index b911095f51..0000000000 --- a/components/common/Button.module.css +++ /dev/null @@ -1,102 +0,0 @@ -.button { - display: flex; - justify-content: center; - align-items: center; - font-size: var(--font-size-normal); - color: var(--gray900); - background: var(--gray100); - padding: 8px 16px; - border-radius: 4px; - border: 0; - outline: none; - cursor: pointer; - position: relative; -} - -.button:hover { - background: var(--gray200); -} - -.button:active { - color: var(--gray900); -} - -.label { - white-space: nowrap; - text-overflow: ellipsis; - overflow: hidden; - max-width: 300px; -} - -.large { - font-size: var(--font-size-large); -} - -.small { - font-size: var(--font-size-small); -} - -.xsmall { - font-size: var(--font-size-xsmall); -} - -.action, -.action:active { - color: var(--gray50); - background: var(--gray900); -} - -.action:hover { - background: var(--gray800); -} - -.danger, -.danger:active { - color: var(--gray50); - background: var(--red500); -} - -.danger:hover { - background: var(--red400); -} - -.light, -.light:active { - color: var(--gray900); - background: transparent; -} - -.light:hover { - background: inherit; -} - -.button .icon + * { - margin-left: 10px; -} - -.button.iconRight .icon { - order: 1; - margin-left: 10px; -} - -.button.iconRight .icon + * { - margin: 0; -} - -.button:disabled { - cursor: default; - color: var(--gray500); - background: var(--gray75); -} - -.button:disabled:active { - color: var(--gray500); -} - -.button:disabled:hover { - background: var(--gray75); -} - -.button.light:disabled { - background: var(--gray50); -} diff --git a/components/common/ButtonGroup.js b/components/common/ButtonGroup.js deleted file mode 100644 index 353ce690c3..0000000000 --- a/components/common/ButtonGroup.js +++ /dev/null @@ -1,42 +0,0 @@ -import React from 'react'; -import PropTypes from 'prop-types'; -import classNames from 'classnames'; -import Button from './Button'; -import styles from './ButtonGroup.module.css'; - -function ButtonGroup({ items = [], selectedItem, className, size, icon, onClick = () => {} }) { - return ( -
- {items.map(item => { - const { label, value } = item; - return ( - - ); - })} -
- ); -} - -ButtonGroup.propTypes = { - items: PropTypes.arrayOf( - PropTypes.shape({ - label: PropTypes.node, - value: PropTypes.any.isRequired, - }), - ), - selectedItem: PropTypes.any, - className: PropTypes.string, - size: PropTypes.oneOf(['xlarge', 'large', 'medium', 'small', 'xsmall']), - icon: PropTypes.node, - onClick: PropTypes.func, -}; - -export default ButtonGroup; diff --git a/components/common/ButtonGroup.module.css b/components/common/ButtonGroup.module.css deleted file mode 100644 index bc60f8d363..0000000000 --- a/components/common/ButtonGroup.module.css +++ /dev/null @@ -1,31 +0,0 @@ -.group { - display: inline-flex; - border-radius: 4px; - overflow: hidden; - border: 1px solid var(--gray500); -} - -.group .button { - border-radius: 0; - color: var(--gray800); - background: var(--gray50); - border-left: 1px solid var(--gray500); - padding: 4px 8px; -} - -.group .button:first-child { - border: 0; -} - -.group .button:hover { - background: var(--gray100); -} - -.group .button + .button { - margin: 0; -} - -.group .button.selected { - color: var(--gray900); - font-weight: 600; -} diff --git a/components/common/Calendar.js b/components/common/Calendar.js deleted file mode 100644 index b6c5cd0b17..0000000000 --- a/components/common/Calendar.js +++ /dev/null @@ -1,273 +0,0 @@ -import React, { useState } from 'react'; -import classNames from 'classnames'; -import { - startOfWeek, - startOfMonth, - startOfYear, - endOfMonth, - addDays, - subDays, - addYears, - subYears, - addMonths, - setMonth, - setYear, - isSameDay, - isBefore, - isAfter, -} from 'date-fns'; -import Button from './Button'; -import useLocale from 'hooks/useLocale'; -import { dateFormat } from 'lib/date'; -import { chunk } from 'lib/array'; -import { getDateLocale } from 'lib/lang'; -import Chevron from 'assets/chevron-down.svg'; -import Cross from 'assets/times.svg'; -import styles from './Calendar.module.css'; -import Icon from './Icon'; - -export default function Calendar({ date, minDate, maxDate, onChange }) { - const { locale } = useLocale(); - const [selectMonth, setSelectMonth] = useState(false); - const [selectYear, setSelectYear] = useState(false); - - const month = dateFormat(date, 'MMMM', locale); - const year = date.getFullYear(); - - function toggleMonthSelect() { - setSelectYear(false); - setSelectMonth(state => !state); - } - - function toggleYearSelect() { - setSelectMonth(false); - setSelectYear(state => !state); - } - - function handleChange(value) { - setSelectMonth(false); - setSelectYear(false); - if (value) { - onChange(value); - } - } - - return ( -
-
-
{date.getDate()}
-
- {month} - : } size="small" /> -
-
- {year} - : } size="small" /> -
-
-
- {!selectMonth && !selectYear && ( - - )} - {selectMonth && ( - - )} - {selectYear && ( - - )} -
-
- ); -} - -const DaySelector = ({ date, minDate, maxDate, locale, onSelect }) => { - const dateLocale = getDateLocale(locale); - const weekStartsOn = dateLocale?.options?.weekStartsOn || 0; - const startWeek = startOfWeek(date, { - locale: dateLocale, - weekStartsOn, - }); - const startMonth = startOfMonth(date); - const startDay = subDays(startMonth, startMonth.getDay() - weekStartsOn); - const month = date.getMonth(); - const year = date.getFullYear(); - - const daysOfWeek = []; - for (let i = 0; i < 7; i++) { - daysOfWeek.push(addDays(startWeek, i)); - } - - const days = []; - for (let i = 0; i < 35; i++) { - days.push(addDays(startDay, i)); - } - - return ( - - - - {daysOfWeek.map((day, i) => ( - - ))} - - - - {chunk(days, 7).map((week, i) => ( - - {week.map((day, j) => { - const disabled = isBefore(day, minDate) || isAfter(day, maxDate); - return ( - - ); - })} - - ))} - -
- {dateFormat(day, 'EEE', locale)} -
onSelect(day) : null} - > - {day.getDate()} -
- ); -}; - -const MonthSelector = ({ date, minDate, maxDate, locale, onSelect }) => { - const start = startOfYear(date); - const months = []; - for (let i = 0; i < 12; i++) { - months.push(addMonths(start, i)); - } - - function handleSelect(value) { - onSelect(setMonth(date, value)); - } - - return ( - - - {chunk(months, 3).map((row, i) => ( - - {row.map((month, j) => { - const disabled = - isBefore(endOfMonth(month), minDate) || isAfter(startOfMonth(month), maxDate); - return ( - - ); - })} - - ))} - -
handleSelect(month.getMonth()) : null} - > - {dateFormat(month, 'MMMM', locale)} -
- ); -}; - -const YearSelector = ({ date, minDate, maxDate, onSelect }) => { - const [currentDate, setCurrentDate] = useState(date); - const year = date.getFullYear(); - const currentYear = currentDate.getFullYear(); - const minYear = minDate.getFullYear(); - const maxYear = maxDate.getFullYear(); - const years = []; - for (let i = 0; i < 15; i++) { - years.push(currentYear - 7 + i); - } - - function handleSelect(value) { - onSelect(setYear(date, value)); - } - - function handlePrevClick() { - setCurrentDate(state => subYears(state, 15)); - } - - function handleNextClick() { - setCurrentDate(state => addYears(state, 15)); - } - - return ( -
-
-
-
- - - {chunk(years, 5).map((row, i) => ( - - {row.map((n, j) => ( - - ))} - - ))} - -
maxYear, - })} - onClick={() => (n < minYear || n > maxYear ? null : handleSelect(n))} - > - {n} -
-
-
-
-
- ); -}; diff --git a/components/common/Calendar.module.css b/components/common/Calendar.module.css deleted file mode 100644 index 9751cf25b8..0000000000 --- a/components/common/Calendar.module.css +++ /dev/null @@ -1,111 +0,0 @@ -.calendar { - display: flex; - flex-direction: column; - font-size: var(--font-size-small); - flex: 1; - min-height: 306px; -} - -.calendar table { - width: 100%; - border-spacing: 5px; -} - -.calendar td { - color: var(--gray800); - cursor: pointer; - text-align: center; - vertical-align: center; - height: 40px; - width: 40px; - border-radius: 5px; - border: 1px solid transparent; -} - -.calendar td:hover { - border: 1px solid var(--gray300); - background: var(--gray75); -} - -.calendar td.faded { - color: var(--gray500); -} - -.calendar td.selected { - font-weight: 600; - border: 1px solid var(--gray600); -} - -.calendar td.selected:hover { - background: transparent; -} - -.calendar td.disabled { - color: var(--gray400); - background: var(--gray75); -} - -.calendar td.disabled:hover { - cursor: default; - background: var(--gray75); - border-color: transparent; -} - -.calendar td.faded.disabled { - background: var(--gray100); -} - -.header { - display: flex; - justify-content: space-evenly; - align-items: center; - font-weight: 700; - line-height: 40px; - font-size: var(--font-size-normal); -} - -.body { - display: flex; -} - -.selector { - cursor: pointer; -} - -.pager { - display: flex; - flex: 1; -} - -.pager button { - align-self: center; -} - -.middle { - flex: 1; -} - -.left, -.right { - display: flex; - justify-content: center; - align-items: center; -} - -.left svg { - transform: rotate(90deg); -} - -.right svg { - transform: rotate(-90deg); -} - -.icon { - margin-left: 10px; -} - -@media only screen and (max-width: 992px) { - .calendar table { - max-width: calc(100vw - 30px); - } -} diff --git a/components/common/Checkbox.js b/components/common/Checkbox.js deleted file mode 100644 index 0cd0dcad9a..0000000000 --- a/components/common/Checkbox.js +++ /dev/null @@ -1,39 +0,0 @@ -import React, { useRef } from 'react'; -import PropTypes from 'prop-types'; -import Icon from 'components/common/Icon'; -import Check from 'assets/check.svg'; -import styles from './Checkbox.module.css'; - -function Checkbox({ name, value, label, onChange }) { - const ref = useRef(); - - const onClick = () => ref.current.click(); - - return ( -
-
- {value && } size="small" />} -
- - -
- ); -} - -Checkbox.propTypes = { - name: PropTypes.string, - value: PropTypes.any, - label: PropTypes.node, - onChange: PropTypes.func, -}; - -export default Checkbox; diff --git a/components/common/Checkbox.module.css b/components/common/Checkbox.module.css deleted file mode 100644 index c9a01eacb7..0000000000 --- a/components/common/Checkbox.module.css +++ /dev/null @@ -1,30 +0,0 @@ -.container { - display: flex; - align-items: center; - position: relative; - overflow: hidden; -} - -.checkbox { - display: flex; - justify-content: center; - align-items: center; - width: 20px; - height: 20px; - border: 1px solid var(--gray500); - border-radius: 4px; -} - -.label { - margin-left: 10px; - user-select: none; /* disable text selection when clicking to toggle the checkbox */ -} - -.input { - position: absolute; - visibility: hidden; - height: 0; - width: 0; - bottom: 100%; - right: 100%; -} diff --git a/components/common/CopyButton.js b/components/common/CopyButton.js deleted file mode 100644 index b300ef31f1..0000000000 --- a/components/common/CopyButton.js +++ /dev/null @@ -1,37 +0,0 @@ -import React, { useState } from 'react'; -import PropTypes from 'prop-types'; -import Button from './Button'; -import { FormattedMessage } from 'react-intl'; - -const defaultText = ( - -); - -function CopyButton({ element, ...props }) { - const [text, setText] = useState(defaultText); - - function handleClick() { - if (element?.current) { - element.current.select(); - document.execCommand('copy'); - setText(); - window.getSelection().removeAllRanges(); - } - } - - return ( - - ); -} - -CopyButton.propTypes = { - element: PropTypes.shape({ - current: PropTypes.shape({ - select: PropTypes.func.isRequired, - }), - }), -}; - -export default CopyButton; diff --git a/components/common/DateFilter.js b/components/common/DateFilter.js deleted file mode 100644 index d568a889e8..0000000000 --- a/components/common/DateFilter.js +++ /dev/null @@ -1,138 +0,0 @@ -import React, { useState } from 'react'; -import PropTypes from 'prop-types'; -import { FormattedMessage } from 'react-intl'; -import { endOfYear, isSameDay } from 'date-fns'; -import Modal from './Modal'; -import DropDown from './DropDown'; -import DatePickerForm from 'components/forms/DatePickerForm'; -import useLocale from 'hooks/useLocale'; -import { dateFormat } from 'lib/date'; -import Calendar from 'assets/calendar-alt.svg'; -import Icon from './Icon'; - -export const filterOptions = [ - { label: , value: '1day' }, - { - label: ( - - ), - value: '24hour', - }, - { - label: , - value: '-1day', - }, - { - label: , - value: '1week', - divider: true, - }, - { - label: ( - - ), - value: '7day', - }, - { - label: , - value: '1month', - divider: true, - }, - { - label: ( - - ), - value: '30day', - }, - { - label: ( - - ), - value: '90day', - }, - { label: , value: '1year' }, - { - label: , - value: 'all', - divider: true, - }, - { - label: , - value: 'custom', - divider: true, - }, -]; - -function DateFilter({ value, startDate, endDate, onChange, className, options }) { - const [showPicker, setShowPicker] = useState(false); - const displayValue = - value === 'custom' ? ( - handleChange('custom')} /> - ) : ( - value - ); - - async function handleChange(value) { - if (value === 'custom') { - setShowPicker(true); - return; - } - onChange(value); - } - - function handlePickerChange(value) { - setShowPicker(false); - onChange(value); - } - - return ( - <> - - {showPicker && ( - - setShowPicker(false)} - /> - - )} - - ); -} - -const CustomRange = ({ startDate, endDate, onClick }) => { - const { locale } = useLocale(); - - function handleClick(e) { - e.stopPropagation(); - - onClick(); - } - - return ( - <> - } className="mr-2" onClick={handleClick} /> - {dateFormat(startDate, 'd LLL y', locale)} - {!isSameDay(startDate, endDate) && ` — ${dateFormat(endDate, 'd LLL y', locale)}`} - - ); -}; - -DateFilter.propTypes = { - value: PropTypes.string, - startDate: PropTypes.instanceOf(Date), - endDate: PropTypes.instanceOf(Date), - onChange: PropTypes.func, - className: PropTypes.string, -}; - -export default DateFilter; diff --git a/components/common/Dot.js b/components/common/Dot.js deleted file mode 100644 index 81454c487e..0000000000 --- a/components/common/Dot.js +++ /dev/null @@ -1,26 +0,0 @@ -import React from 'react'; -import PropTypes from 'prop-types'; -import classNames from 'classnames'; -import styles from './Dot.module.css'; - -function Dot({ color, size, className }) { - return ( -
-
-
- ); -} - -Dot.propTypes = { - color: PropTypes.string, - size: PropTypes.oneOf(['small', 'large']), - className: PropTypes.string, -}; - -export default Dot; diff --git a/components/common/Dot.module.css b/components/common/Dot.module.css deleted file mode 100644 index 258d6e8788..0000000000 --- a/components/common/Dot.module.css +++ /dev/null @@ -1,22 +0,0 @@ -.wrapper { - background: var(--gray50); - margin-right: 10px; - border-radius: 100%; -} - -.dot { - background: var(--green400); - width: 10px; - height: 10px; - border-radius: 100%; -} - -.dot.small { - width: 8px; - height: 8px; -} - -.dot.large { - width: 16px; - height: 16px; -} diff --git a/components/common/DropDown.js b/components/common/DropDown.js deleted file mode 100644 index 00d20e34d3..0000000000 --- a/components/common/DropDown.js +++ /dev/null @@ -1,64 +0,0 @@ -import React, { useState, useRef } from 'react'; -import PropTypes from 'prop-types'; -import classNames from 'classnames'; -import Menu from './Menu'; -import useDocumentClick from 'hooks/useDocumentClick'; -import Chevron from 'assets/chevron-down.svg'; -import styles from './Dropdown.module.css'; -import Icon from './Icon'; - -function DropDown({ value, className, menuClassName, options = [], onChange = () => {} }) { - const [showMenu, setShowMenu] = useState(false); - const ref = useRef(); - const selectedOption = options.find(e => e.value === value); - - function handleShowMenu() { - setShowMenu(state => !state); - } - - function handleSelect(selected, e) { - e.stopPropagation(); - setShowMenu(false); - - onChange(selected); - } - - useDocumentClick(e => { - if (!ref.current?.contains(e.target)) { - setShowMenu(false); - } - }); - - return ( -
-
-
{options.find(e => e.value === value)?.label || value}
- } className={styles.icon} size="small" /> -
- {showMenu && ( - - )} -
- ); -} - -DropDown.propTypes = { - value: PropTypes.any, - className: PropTypes.string, - menuClassName: PropTypes.string, - options: PropTypes.arrayOf( - PropTypes.shape({ - value: PropTypes.any.isRequired, - label: PropTypes.node, - }), - ), - onChange: PropTypes.func, -}; - -export default DropDown; diff --git a/components/common/Dropdown.module.css b/components/common/Dropdown.module.css deleted file mode 100644 index 9738b00770..0000000000 --- a/components/common/Dropdown.module.css +++ /dev/null @@ -1,28 +0,0 @@ -.dropdown { - position: relative; - display: flex; - justify-content: space-between; - align-items: center; - border: 1px solid var(--gray500); - border-radius: 4px; - cursor: pointer; -} - -.value { - flex: 1; - display: flex; - justify-content: space-between; - font-size: var(--font-size-small); - flex-wrap: nowrap; - white-space: nowrap; - padding: 4px 16px; - min-width: 160px; -} - -.text { - flex: 1; -} - -.icon { - padding-left: 20px; -} diff --git a/components/common/EmptyPlaceholder.js b/components/common/EmptyPlaceholder.js deleted file mode 100644 index b5394e85e5..0000000000 --- a/components/common/EmptyPlaceholder.js +++ /dev/null @@ -1,22 +0,0 @@ -import React from 'react'; -import PropTypes from 'prop-types'; -import Icon from 'components/common/Icon'; -import Logo from 'assets/logo.svg'; -import styles from './EmptyPlaceholder.module.css'; - -function EmptyPlaceholder({ msg, children }) { - return ( -
- } size="xlarge" /> -

{msg}

- {children} -
- ); -} - -EmptyPlaceholder.propTypes = { - msg: PropTypes.node, - children: PropTypes.node, -}; - -export default EmptyPlaceholder; diff --git a/components/common/EmptyPlaceholder.module.css b/components/common/EmptyPlaceholder.module.css deleted file mode 100644 index a9231836b7..0000000000 --- a/components/common/EmptyPlaceholder.module.css +++ /dev/null @@ -1,15 +0,0 @@ -.placeholder { - display: flex; - flex-direction: column; - justify-content: center; - align-items: center; - min-height: 600px; -} - -.icon { - margin-bottom: 30px; -} - -.msg { - margin-bottom: 15px; -} diff --git a/components/common/ErrorMessage.js b/components/common/ErrorMessage.js deleted file mode 100644 index 5747f2269f..0000000000 --- a/components/common/ErrorMessage.js +++ /dev/null @@ -1,14 +0,0 @@ -import React from 'react'; -import { FormattedMessage } from 'react-intl'; -import Icon from './Icon'; -import Exclamation from 'assets/exclamation-triangle.svg'; -import styles from './ErrorMessage.module.css'; - -export default function ErrorMessage() { - return ( -
- } className={styles.icon} size="large" /> - -
- ); -} diff --git a/components/common/ErrorMessage.module.css b/components/common/ErrorMessage.module.css deleted file mode 100644 index 88769cf5ce..0000000000 --- a/components/common/ErrorMessage.module.css +++ /dev/null @@ -1,15 +0,0 @@ -.error { - position: absolute; - top: 50%; - left: 50%; - transform: translate(-50%, -50%); - margin: auto; - display: flex; - z-index: 1; - background-color: var(--gray50); - padding: 10px; -} - -.icon { - margin-right: 10px; -} diff --git a/components/common/EventDataButton.js b/components/common/EventDataButton.js deleted file mode 100644 index 2b89584029..0000000000 --- a/components/common/EventDataButton.js +++ /dev/null @@ -1,48 +0,0 @@ -import List from 'assets/list-ul.svg'; -import Modal from 'components/common/Modal'; -import PropTypes from 'prop-types'; -import { useState } from 'react'; -import { FormattedMessage } from 'react-intl'; -import Button from './Button'; -import EventDataForm from 'components/forms/EventDataForm'; -import styles from './EventDataButton.module.css'; - -function EventDataButton({ websiteId }) { - const [showEventData, setShowEventData] = useState(false); - - function handleClick() { - if (!showEventData) { - setShowEventData(true); - } - } - - function handleClose() { - setShowEventData(false); - } - - return ( - <> - - {showEventData && ( - }> - - - )} - - ); -} - -EventDataButton.propTypes = { - websiteId: PropTypes.oneOfType([PropTypes.string, PropTypes.number]), -}; - -export default EventDataButton; diff --git a/components/common/EventDataButton.module.css b/components/common/EventDataButton.module.css deleted file mode 100644 index cd2a2ed6ee..0000000000 --- a/components/common/EventDataButton.module.css +++ /dev/null @@ -1,3 +0,0 @@ -.button { - width: fit-content; -} diff --git a/components/common/Favicon.js b/components/common/Favicon.js deleted file mode 100644 index d72cd3c7a0..0000000000 --- a/components/common/Favicon.js +++ /dev/null @@ -1,28 +0,0 @@ -import React from 'react'; -import PropTypes from 'prop-types'; -import styles from './Favicon.module.css'; - -function getHostName(url) { - const match = url.match(/^(?:https?:\/\/)?(?:[^@\n]+@)?(?:www\.)?([^:/\n?=]+)/im); - return match && match.length > 1 ? match[1] : null; -} - -function Favicon({ domain, ...props }) { - const hostName = domain ? getHostName(domain) : null; - - return hostName ? ( - - ) : null; -} - -Favicon.propTypes = { - domain: PropTypes.string, -}; - -export default Favicon; diff --git a/components/common/Favicon.module.css b/components/common/Favicon.module.css deleted file mode 100644 index 82c85c4269..0000000000 --- a/components/common/Favicon.module.css +++ /dev/null @@ -1,3 +0,0 @@ -.favicon { - margin-right: 8px; -} diff --git a/components/common/FilterButtons.js b/components/common/FilterButtons.js deleted file mode 100644 index ea811216e6..0000000000 --- a/components/common/FilterButtons.js +++ /dev/null @@ -1,25 +0,0 @@ -import React from 'react'; -import PropTypes from 'prop-types'; -import ButtonLayout from 'components/layout/ButtonLayout'; -import ButtonGroup from './ButtonGroup'; - -function FilterButtons({ buttons, selected, onClick }) { - return ( - - - - ); -} - -FilterButtons.propTypes = { - buttons: PropTypes.arrayOf( - PropTypes.shape({ - label: PropTypes.node, - value: PropTypes.any.isRequired, - }), - ), - selected: PropTypes.any, - onClick: PropTypes.func, -}; - -export default FilterButtons; diff --git a/components/common/FilterLink.js b/components/common/FilterLink.js deleted file mode 100644 index f16258f1a6..0000000000 --- a/components/common/FilterLink.js +++ /dev/null @@ -1,34 +0,0 @@ -import React from 'react'; -import classNames from 'classnames'; -import Link from 'next/link'; -import { safeDecodeURI } from 'next-basics'; -import usePageQuery from 'hooks/usePageQuery'; -import External from 'assets/arrow-up-right-from-square.svg'; -import Icon from './Icon'; -import styles from './FilterLink.module.css'; - -export default function FilterLink({ id, value, label, externalUrl }) { - const { resolve, query } = usePageQuery(); - const active = query[id] !== undefined; - const selected = query[id] === value; - - return ( - - ); -} diff --git a/components/common/FilterLink.module.css b/components/common/FilterLink.module.css deleted file mode 100644 index 45b049dac7..0000000000 --- a/components/common/FilterLink.module.css +++ /dev/null @@ -1,32 +0,0 @@ -.row { - display: flex; - align-items: center; -} - -.row .inactive { - color: var(--gray500); -} - -.row .active { - color: var(--gray900); - font-weight: 600; -} - -.row .link { - display: none; - margin-left: 20px; -} - -.row .label { - white-space: nowrap; - overflow: hidden; - text-overflow: ellipsis; -} - -.row:hover .link { - display: block; -} - -.icon { - cursor: pointer; -} diff --git a/components/common/HamburgerButton.js b/components/common/HamburgerButton.js deleted file mode 100644 index 501b8c95c3..0000000000 --- a/components/common/HamburgerButton.js +++ /dev/null @@ -1,44 +0,0 @@ -import Button from 'components/common/Button'; -import XMark from 'assets/xmark.svg'; -import Bars from 'assets/bars.svg'; -import { useState } from 'react'; -import styles from './HamburgerButton.module.css'; -import MobileMenu from './MobileMenu'; -import { FormattedMessage } from 'react-intl'; - -const menuItems = [ - { - label: , - value: '/dashboard', - }, - { label: , value: '/realtime' }, - { label: , value: '/settings' }, - { - label: , - value: '/settings/profile', - }, - { label: , value: '/logout' }, -]; - -export default function HamburgerButton() { - const [active, setActive] = useState(false); - - function handleClick() { - setActive(state => !state); - } - - function handleClose() { - setActive(false); - } - - return ( - <> - - {showMenu && ( - - )} -
- ); -} - -MenuButton.propTypes = { - icon: PropTypes.node, - value: PropTypes.any, - options: PropTypes.arrayOf( - PropTypes.shape({ - label: PropTypes.node, - value: PropTypes.any, - className: PropTypes.string, - render: PropTypes.func, - divider: PropTypes.bool, - }), - ), - buttonClassName: PropTypes.string, - menuClassName: PropTypes.string, - menuPosition: PropTypes.oneOf(['top', 'bottom']), - menuAlign: PropTypes.oneOf(['left', 'right']), - onSelect: PropTypes.func, - renderValue: PropTypes.func, -}; - -export default MenuButton; diff --git a/components/common/MenuButton.module.css b/components/common/MenuButton.module.css deleted file mode 100644 index 7e9dd7e142..0000000000 --- a/components/common/MenuButton.module.css +++ /dev/null @@ -1,20 +0,0 @@ -.container { - display: flex; - position: relative; - cursor: pointer; -} - -.button { - border: 1px solid transparent; - border-radius: 4px; -} - -.text { - font-size: var(--font-size-small); -} - -.open, -.open:hover { - background: var(--gray50); - border: 1px solid var(--gray500); -} diff --git a/components/common/MobileMenu.js b/components/common/MobileMenu.js deleted file mode 100644 index 1241b1e6b0..0000000000 --- a/components/common/MobileMenu.js +++ /dev/null @@ -1,22 +0,0 @@ -import classNames from 'classnames'; -import Link from './Link'; -import Button from './Button'; -import XMark from 'assets/xmark.svg'; -import styles from './MobileMenu.module.css'; - -export default function MobileMenu({ items = [], onClose }) { - return ( -
-
-
-
- {items.map(({ label, value }) => ( - - {label} - - ))} -
-
- ); -} diff --git a/components/common/MobileMenu.module.css b/components/common/MobileMenu.module.css deleted file mode 100644 index 78a440d91e..0000000000 --- a/components/common/MobileMenu.module.css +++ /dev/null @@ -1,41 +0,0 @@ -.menu { - position: fixed; - top: 0; - left: 0; - right: 0; - bottom: 0; - margin: auto; - z-index: 100; - display: flex; - flex-direction: column; - background-color: var(--gray50); - overflow: auto; -} - -.items { - flex: 1; - display: flex; - flex-direction: column; - justify-content: center; - align-items: center; -} - -.item { - font-size: var(--font-size-large); -} - -.item + .item { - margin-top: 20px; -} - -.item:last-child { - margin-top: 60px; -} - -.header { - display: flex; - justify-content: flex-end; - align-items: center; - height: 100px; - padding: 0 30px; -} diff --git a/components/common/Modal.js b/components/common/Modal.js deleted file mode 100644 index 694fba6064..0000000000 --- a/components/common/Modal.js +++ /dev/null @@ -1,26 +0,0 @@ -import React from 'react'; -import PropTypes from 'prop-types'; -import ReactDOM from 'react-dom'; -import { useSpring, animated } from 'react-spring'; -import styles from './Modal.module.css'; - -function Modal({ title, children }) { - const props = useSpring({ opacity: 1, from: { opacity: 0 } }); - - return ReactDOM.createPortal( - -
- {title &&
{title}
} -
{children}
-
-
, - document.getElementById('__modals'), - ); -} - -Modal.propTypes = { - title: PropTypes.node, - children: PropTypes.node, -}; - -export default Modal; diff --git a/components/common/Modal.module.css b/components/common/Modal.module.css deleted file mode 100644 index bf2491c71a..0000000000 --- a/components/common/Modal.module.css +++ /dev/null @@ -1,46 +0,0 @@ -.modal { - position: fixed; - top: 0; - left: 0; - right: 0; - bottom: 0; - margin: auto; - z-index: 2; -} - -.modal:before { - content: ''; - position: absolute; - top: 0; - left: 0; - right: 0; - bottom: 0; - margin: auto; - background: #000; - opacity: 0.5; -} - -.content { - position: absolute; - top: 50%; - left: 50%; - transform: translate(-50%, -50%); - background: var(--gray50); - min-width: 400px; - min-height: 100px; - max-width: 100vw; - z-index: 1; - border: 1px solid var(--gray300); - padding: 30px; - border-radius: 4px; -} - -.header { - font-weight: 600; - margin-bottom: 20px; -} - -.body { - display: flex; - flex-direction: column; -} diff --git a/components/common/NavMenu.js b/components/common/NavMenu.js deleted file mode 100644 index 82d97fffed..0000000000 --- a/components/common/NavMenu.js +++ /dev/null @@ -1,47 +0,0 @@ -import React from 'react'; -import PropTypes from 'prop-types'; -import { useRouter } from 'next/router'; -import classNames from 'classnames'; -import styles from './NavMenu.module.css'; - -function NavMenu({ options = [], className, onSelect = () => {} }) { - const router = useRouter(); - - return ( -
- {options - .filter(({ hidden }) => !hidden) - .map(option => { - const { label, value, className: customClassName, render } = option; - - return render ? ( - render(option) - ) : ( -
onSelect(value, e)} - > - {label} -
- ); - })} -
- ); -} - -NavMenu.propTypes = { - options: PropTypes.arrayOf( - PropTypes.shape({ - label: PropTypes.node, - value: PropTypes.any, - className: PropTypes.string, - render: PropTypes.func, - }), - ), - className: PropTypes.string, - onSelect: PropTypes.func, -}; -export default NavMenu; diff --git a/components/common/NavMenu.module.css b/components/common/NavMenu.module.css deleted file mode 100644 index 7be7397376..0000000000 --- a/components/common/NavMenu.module.css +++ /dev/null @@ -1,22 +0,0 @@ -.menu { - color: var(--gray800); - border: 1px solid var(--gray500); - border-radius: 4px; - overflow: hidden; - z-index: 2; -} - -.option { - padding: 8px 16px; - cursor: pointer; - border-radius: 4px; -} - -.option:hover { - background: var(--gray75); -} - -.selected { - color: var(--gray900); - font-weight: 600; -} diff --git a/components/common/NoData.js b/components/common/NoData.js deleted file mode 100644 index 9d523437ba..0000000000 --- a/components/common/NoData.js +++ /dev/null @@ -1,19 +0,0 @@ -import React from 'react'; -import PropTypes from 'prop-types'; -import classNames from 'classnames'; -import { FormattedMessage } from 'react-intl'; -import styles from './NoData.module.css'; - -function NoData({ className }) { - return ( -
- -
- ); -} - -NoData.propTypes = { - className: PropTypes.string, -}; - -export default NoData; diff --git a/components/common/NoData.module.css b/components/common/NoData.module.css deleted file mode 100644 index 518fa488e6..0000000000 --- a/components/common/NoData.module.css +++ /dev/null @@ -1,11 +0,0 @@ -.container { - color: var(--gray500); - font-size: var(--font-size-normal); - position: relative; - display: flex; - align-items: center; - justify-content: center; - text-align: center; - width: 100%; - height: 100%; -} diff --git a/components/common/OverflowText.js b/components/common/OverflowText.js deleted file mode 100644 index d67c492304..0000000000 --- a/components/common/OverflowText.js +++ /dev/null @@ -1,66 +0,0 @@ -import PropTypes from 'prop-types'; -import React, { useCallback, useEffect, useRef, useState } from 'react'; -import ReactTooltip from 'react-tooltip'; - -import styles from './OverflowText.module.css'; - -const OverflowText = ({ children, tooltipId }) => { - const measureEl = useRef(); - const [isOverflown, setIsOverflown] = useState(false); - - const measure = useCallback( - el => { - if (!el) return; - setIsOverflown(el.scrollWidth > el.clientWidth); - }, - [setIsOverflown], - ); - - // Do one measure on mount - useEffect(() => { - measure(measureEl.current); - }, [measure]); - - // Set up resize listener for subsequent measures - useEffect(() => { - if (!measureEl.current) return; - - // Destructure ref in case it changes out from under us - const el = measureEl.current; - - if ('ResizeObserver' in global) { - // Ideally, we have access to ResizeObservers - const observer = new ResizeObserver(() => { - measure(el); - }); - observer.observe(el); - return () => observer.unobserve(el); - } else { - // Otherwise, fall back to measuring on window resizes - const handler = () => measure(el); - - window.addEventListener('resize', handler, { passive: true }); - return () => window.removeEventListener('resize', handler, { passive: true }); - } - }); - - return ( - - {children} - {isOverflown && {children}} - - ); -}; - -OverflowText.propTypes = { - children: PropTypes.oneOfType([PropTypes.number, PropTypes.string]).isRequired, - tooltipId: PropTypes.string.isRequired, -}; - -export default OverflowText; diff --git a/components/common/OverflowText.module.css b/components/common/OverflowText.module.css deleted file mode 100644 index c206663142..0000000000 --- a/components/common/OverflowText.module.css +++ /dev/null @@ -1,6 +0,0 @@ -.root { - max-width: 100%; - overflow: hidden; - text-overflow: ellipsis; - white-space: nowrap; -} diff --git a/components/common/RefreshButton.js b/components/common/RefreshButton.js deleted file mode 100644 index be1a8a1d83..0000000000 --- a/components/common/RefreshButton.js +++ /dev/null @@ -1,47 +0,0 @@ -import React, { useState, useEffect, useCallback } from 'react'; -import PropTypes from 'prop-types'; -import { FormattedMessage } from 'react-intl'; -import useStore from 'store/queries'; -import { setDateRange } from 'store/websites'; -import Button from './Button'; -import Refresh from 'assets/redo.svg'; -import Dots from 'assets/ellipsis-h.svg'; -import useDateRange from 'hooks/useDateRange'; - -function RefreshButton({ websiteId }) { - const [dateRange] = useDateRange(websiteId); - const [loading, setLoading] = useState(false); - const selector = useCallback(state => state[`/websites/${websiteId}/stats`], [websiteId]); - const completed = useStore(selector); - - function handleClick() { - if (!loading && dateRange) { - setLoading(true); - if (/^[\d]+/.test(dateRange.value)) { - setDateRange(websiteId, dateRange.value); - } else { - setDateRange(websiteId, dateRange); - } - } - } - - useEffect(() => { - setLoading(false); - }, [completed]); - - return ( - - - - - ); -} diff --git a/components/common/UpdateNotice.module.css b/components/common/UpdateNotice.module.css deleted file mode 100644 index 52bac611c8..0000000000 --- a/components/common/UpdateNotice.module.css +++ /dev/null @@ -1,18 +0,0 @@ -.notice { - display: flex; - justify-content: center; - align-items: center; - padding-top: 20px; -} - -.message { - font-size: var(--font-size-small); - font-weight: 600; - flex: 1; - text-align: center; - margin-right: 20px; -} - -.buttons { - flex: 0; -} diff --git a/components/common/WorldMap.js b/components/common/WorldMap.js deleted file mode 100644 index 0f9a1d6af0..0000000000 --- a/components/common/WorldMap.js +++ /dev/null @@ -1,103 +0,0 @@ -import React, { useState, useMemo } from 'react'; -import { useRouter } from 'next/router'; -import PropTypes from 'prop-types'; -import ReactTooltip from 'react-tooltip'; -import { ComposableMap, Geographies, Geography, ZoomableGroup } from 'react-simple-maps'; -import classNames from 'classnames'; -import { colord } from 'colord'; -import useTheme from 'hooks/useTheme'; -import { ISO_COUNTRIES, THEME_COLORS, MAP_FILE } from 'lib/constants'; -import styles from './WorldMap.module.css'; -import useCountryNames from 'hooks/useCountryNames'; -import useLocale from 'hooks/useLocale'; - -function WorldMap({ data, className }) { - const { basePath } = useRouter(); - const [tooltip, setTooltip] = useState(); - const [theme] = useTheme(); - const colors = useMemo( - () => ({ - baseColor: THEME_COLORS[theme].primary, - fillColor: THEME_COLORS[theme].gray100, - strokeColor: THEME_COLORS[theme].primary, - hoverColor: THEME_COLORS[theme].primary, - }), - [theme], - ); - const { locale } = useLocale(); - const countryNames = useCountryNames(locale); - - function getFillColor(code) { - if (code === 'AQ') return; - const country = data?.find(({ x }) => x === code); - - if (!country) { - return colors.fillColor; - } - - return colord(colors.baseColor) - [theme === 'light' ? 'lighten' : 'darken'](0.4 * (1.0 - country.z / 100)) - .toHex(); - } - - function getOpacity(code) { - return code === 'AQ' ? 0 : 1; - } - - function handleHover(code) { - if (code === 'AQ') return; - const country = data?.find(({ x }) => x === code); - setTooltip(`${countryNames[code]}: ${country?.y || 0} visitors`); - } - - return ( -
- - - - {({ geographies }) => { - return geographies.map(geo => { - const code = ISO_COUNTRIES[geo.id]; - - return ( - handleHover(code)} - onMouseOut={() => setTooltip(null)} - /> - ); - }); - }} - - - - {tooltip} -
- ); -} - -WorldMap.propTypes = { - data: PropTypes.arrayOf( - PropTypes.shape({ - x: PropTypes.string, - y: PropTypes.number, - z: PropTypes.number, - }), - ), - className: PropTypes.string, -}; - -export default WorldMap; diff --git a/components/forms/AccountEditForm.js b/components/forms/AccountEditForm.js deleted file mode 100644 index 70125656d8..0000000000 --- a/components/forms/AccountEditForm.js +++ /dev/null @@ -1,89 +0,0 @@ -import React, { useState } from 'react'; -import { FormattedMessage } from 'react-intl'; -import { Formik, Form, Field } from 'formik'; -import Button from 'components/common/Button'; -import FormLayout, { - FormButtons, - FormError, - FormMessage, - FormRow, -} from 'components/layout/FormLayout'; -import useApi from 'hooks/useApi'; - -const initialValues = { - username: '', - password: '', -}; - -const validate = ({ id, username, password }) => { - const errors = {}; - - if (!username) { - errors.username = ; - } - if (!id && !password) { - errors.password = ; - } - - return errors; -}; - -export default function AccountEditForm({ values, onSave, onClose }) { - const { post } = useApi(); - const [message, setMessage] = useState(); - - const handleSubmit = async values => { - const { id } = values; - const { ok, data } = await post(id ? `/accounts/${id}` : '/accounts', values); - - if (ok) { - onSave(); - } else { - setMessage( - data || , - ); - } - }; - - return ( - - - {() => ( -
- - -
- - -
-
- - -
- - -
-
- - - - - {message} -
- )} -
-
- ); -} diff --git a/components/forms/ChangePasswordForm.js b/components/forms/ChangePasswordForm.js deleted file mode 100644 index dcad6f1786..0000000000 --- a/components/forms/ChangePasswordForm.js +++ /dev/null @@ -1,107 +0,0 @@ -import React, { useState } from 'react'; -import { FormattedMessage } from 'react-intl'; -import { Formik, Form, Field } from 'formik'; -import Button from 'components/common/Button'; -import FormLayout, { - FormButtons, - FormError, - FormMessage, - FormRow, -} from 'components/layout/FormLayout'; -import useApi from 'hooks/useApi'; -import useUser from 'hooks/useUser'; - -const initialValues = { - current_password: '', - new_password: '', - confirm_password: '', -}; - -const validate = ({ current_password, new_password, confirm_password }) => { - const errors = {}; - - if (!current_password) { - errors.current_password = ; - } - if (!new_password) { - errors.new_password = ; - } - if (!confirm_password) { - errors.confirm_password = ; - } else if (new_password !== confirm_password) { - errors.confirm_password = ( - - ); - } - - return errors; -}; - -export default function ChangePasswordForm({ values, onSave, onClose }) { - const { post } = useApi(); - const [message, setMessage] = useState(); - const { user } = useUser(); - - const handleSubmit = async values => { - const { ok, error } = await post(`/accounts/${user.accountUuid}/password`, values); - - if (ok) { - onSave(); - } else { - setMessage( - error || , - ); - } - }; - - return ( - - - {() => ( -
- - -
- - -
-
- - -
- - -
-
- - -
- - -
-
- - - - - {message} -
- )} -
-
- ); -} diff --git a/components/forms/DatePickerForm.js b/components/forms/DatePickerForm.js deleted file mode 100644 index 9669f7418c..0000000000 --- a/components/forms/DatePickerForm.js +++ /dev/null @@ -1,83 +0,0 @@ -import React, { useState } from 'react'; -import { FormattedMessage } from 'react-intl'; -import { isAfter, isBefore, isSameDay } from 'date-fns'; -import Calendar from 'components/common/Calendar'; -import Button from 'components/common/Button'; -import { FormButtons } from 'components/layout/FormLayout'; -import { getDateRangeValues } from 'lib/date'; -import styles from './DatePickerForm.module.css'; -import ButtonGroup from 'components/common/ButtonGroup'; - -const FILTER_DAY = 0; -const FILTER_RANGE = 1; - -export default function DatePickerForm({ - startDate: defaultStartDate, - endDate: defaultEndDate, - minDate, - maxDate, - onChange, - onClose, -}) { - const [selected, setSelected] = useState( - isSameDay(defaultStartDate, defaultEndDate) ? FILTER_DAY : FILTER_RANGE, - ); - const [date, setDate] = useState(defaultStartDate); - const [startDate, setStartDate] = useState(defaultStartDate); - const [endDate, setEndDate] = useState(defaultEndDate); - - const disabled = - selected === FILTER_DAY - ? isAfter(minDate, date) && isBefore(maxDate, date) - : isAfter(startDate, endDate); - - const buttons = [ - { - label: , - value: FILTER_DAY, - }, - { - label: , - value: FILTER_RANGE, - }, - ]; - - function handleSave() { - if (selected === FILTER_DAY) { - onChange({ ...getDateRangeValues(date, date), value: 'custom' }); - } else { - onChange({ ...getDateRangeValues(startDate, endDate), value: 'custom' }); - } - } - - return ( -
-
- -
-
- {selected === FILTER_DAY ? ( - - ) : ( - <> - - - - )} -
- - - - -
- ); -} diff --git a/components/forms/DatePickerForm.module.css b/components/forms/DatePickerForm.module.css deleted file mode 100644 index 96e2d2ecc4..0000000000 --- a/components/forms/DatePickerForm.module.css +++ /dev/null @@ -1,40 +0,0 @@ -.container { - display: flex; - flex-direction: column; - max-width: 100vw; -} - -.calendars { - display: flex; - justify-content: center; -} - -.calendars > div { - width: 380px; -} - -.calendars > div + div { - margin-left: 20px; - padding-left: 20px; - border-left: 1px solid var(--gray300); -} - -.filter { - display: flex; - justify-content: center; - align-items: center; - margin-bottom: 20px; -} - -@media only screen and (max-width: 768px) { - .calendars { - flex-direction: column; - } - - .calendars > div + div { - padding: 0; - margin-left: 0; - margin-top: 20px; - border: 0; - } -} diff --git a/components/forms/DeleteForm.js b/components/forms/DeleteForm.js deleted file mode 100644 index 09da94d627..0000000000 --- a/components/forms/DeleteForm.js +++ /dev/null @@ -1,105 +0,0 @@ -import React, { useState } from 'react'; -import { FormattedMessage } from 'react-intl'; -import { Formik, Form, Field } from 'formik'; -import Button from 'components/common/Button'; -import FormLayout, { - FormButtons, - FormError, - FormMessage, - FormRow, -} from 'components/layout/FormLayout'; -import Loading from 'components/common/Loading'; -import useApi from 'hooks/useApi'; - -const CONFIRMATION_WORD = 'DELETE'; - -const validate = ({ confirmation }) => { - const errors = {}; - - if (confirmation !== CONFIRMATION_WORD) { - errors.confirmation = !confirmation ? ( - - ) : ( - - ); - } - - return errors; -}; - -export default function DeleteForm({ values, onSave, onClose }) { - const { del } = useApi(); - const [message, setMessage] = useState(); - const [deleting, setDeleting] = useState(false); - - const handleSubmit = async ({ type, id }) => { - setDeleting(true); - - const { ok, data } = await del(`/${type}/${id}`); - - if (ok) { - onSave(); - } else { - setMessage( - data || , - ); - - setDeleting(false); - } - }; - - return ( - - {deleting && } - - {props => ( -
-
- {values.name} }} - /> -
-
- -
-

- {CONFIRMATION_WORD} }} - /> -

- -
- - -
-
- - - - - {message} -
- )} -
-
- ); -} diff --git a/components/forms/EventDataForm.js b/components/forms/EventDataForm.js deleted file mode 100644 index e236aa3d30..0000000000 --- a/components/forms/EventDataForm.js +++ /dev/null @@ -1,262 +0,0 @@ -import classNames from 'classnames'; -import Button from 'components/common/Button'; -import DateFilter from 'components/common/DateFilter'; -import DropDown from 'components/common/DropDown'; -import FormLayout, { - FormButtons, - FormError, - FormMessage, - FormRow, -} from 'components/layout/FormLayout'; -import DataTable from 'components/metrics/DataTable'; -import FilterTags from 'components/metrics/FilterTags'; -import { Field, Form, Formik } from 'formik'; -import useApi from 'hooks/useApi'; -import useDateRange from 'hooks/useDateRange'; -import { useState, useEffect } from 'react'; -import { FormattedMessage } from 'react-intl'; -import styles from './EventDataForm.module.css'; -import useTimezone from 'hooks/useTimezone'; - -export const filterOptions = [ - { label: 'Count', value: 'count' }, - { label: 'Average', value: 'avg' }, - { label: 'Minimum', value: 'min' }, - { label: 'Maximum', value: 'max' }, - { label: 'Sum', value: 'sum' }, -]; - -export const dateOptions = [ - { label: , value: '1day' }, - { - label: ( - - ), - value: '24hour', - }, - { - label: , - value: '-1day', - }, - { - label: , - value: '1week', - divider: true, - }, - { - label: ( - - ), - value: '7day', - }, - { - label: , - value: '1month', - divider: true, - }, - { - label: ( - - ), - value: '30day', - }, - { - label: ( - - ), - value: '90day', - }, - { label: , value: '1year' }, - { - label: , - value: 'custom', - divider: true, - }, -]; - -export default function EventDataForm({ websiteId, onClose, className }) { - const { post } = useApi(); - const [message, setMessage] = useState(); - const [columns, setColumns] = useState({}); - const [filters, setFilters] = useState({}); - const [data, setData] = useState([]); - const [dateRange, setDateRange] = useDateRange('report'); - const { startDate, endDate, value } = dateRange; - const [timezone] = useTimezone(); - const [isValid, setIsValid] = useState(false); - - useEffect(() => { - if (Object.keys(columns).length > 0) { - setIsValid(true); - } else { - setIsValid(false); - } - }, [columns]); - - const handleAddTag = (value, list, setState, resetForm) => { - setState({ ...list, [`${value.field}`]: value.value }); - resetForm(); - }; - - const handleRemoveTag = (value, list, setState) => { - const newList = { ...list }; - - delete newList[`${value}`]; - - setState(newList); - }; - - const handleSubmit = async () => { - const params = { - website_id: websiteId, - start_at: +startDate, - end_at: +endDate, - timezone, - columns, - filters, - }; - - const { ok, data } = await post(`/websites/${websiteId}/eventdata`, params); - - if (!ok) { - setMessage(); - setData([]); - } else { - setData(data); - setMessage(null); - } - }; - - return ( - <> - {message} -
-
- -
- - - - -
-
- - handleAddTag(value, columns, setColumns, resetForm) - } - > - {({ values, setFieldValue }) => ( -
- - -
- - -
-
- - -
- setFieldValue('value', value)} - className={styles.dropdown} - name="value" - options={filterOptions} - /> - -
-
- - - -
- )} -
- handleRemoveTag(value, columns, setColumns)} - /> -
-
- - handleAddTag(value, filters, setFilters, resetForm) - } - > - {({ values }) => ( -
- - -
- - -
-
- - -
- - -
-
- - - -
- )} -
- handleRemoveTag(value, filters, setFilters)} - /> -
-
-
-
- -
-
- - - - - - ); -} diff --git a/components/forms/EventDataForm.module.css b/components/forms/EventDataForm.module.css deleted file mode 100644 index 19d76f7756..0000000000 --- a/components/forms/EventDataForm.module.css +++ /dev/null @@ -1,38 +0,0 @@ -.container { - display: flex; -} - -.form { - border-right: 1px solid var(--gray300); - width: 420px; -} - -.filters { - padding: 10px 5px; -} - -.filters + .filters { - border-top: 1px solid var(--gray300); - min-height: 250px; -} - -.table { - padding: 10px; - min-height: 430px; - min-width: 400px; -} - -.formButtons { - justify-content: flex-start; - margin-left: 20px; -} - -.dropdown { - min-height: 39px; - min-width: 240px; -} - -.filterTag { - flex-wrap: wrap; - margin: 10px 5px 5px 5px; -} diff --git a/components/forms/LoginForm.js b/components/forms/LoginForm.js deleted file mode 100644 index 8c8aa09e47..0000000000 --- a/components/forms/LoginForm.js +++ /dev/null @@ -1,111 +0,0 @@ -import React, { useState } from 'react'; -import { FormattedMessage } from 'react-intl'; -import { Formik, Form, Field } from 'formik'; -import { setItem } from 'next-basics'; -import { useRouter } from 'next/router'; -import Button from 'components/common/Button'; -import FormLayout, { - FormButtons, - FormError, - FormMessage, - FormRow, -} from 'components/layout/FormLayout'; -import Icon from 'components/common/Icon'; -import useApi from 'hooks/useApi'; -import { AUTH_TOKEN } from 'lib/constants'; -import { setUser } from 'store/app'; -import Logo from 'assets/logo.svg'; -import styles from './LoginForm.module.css'; - -const validate = ({ username, password }) => { - const errors = {}; - - if (!username) { - errors.username = ; - } - if (!password) { - errors.password = ; - } - - return errors; -}; - -export default function LoginForm() { - const { post } = useApi(); - const router = useRouter(); - const [message, setMessage] = useState(); - - const handleSubmit = async ({ username, password }) => { - const { ok, status, data } = await post('/auth/login', { - username, - password, - }); - - if (ok) { - setItem(AUTH_TOKEN, data.token); - - setUser(data.user); - - await router.push('/'); - - return null; - } else { - setMessage( - status === 401 ? ( - - ) : ( - data - ), - ); - } - }; - - return ( - - - {() => ( -
-
- } size="xlarge" className={styles.icon} /> -

umami

-
- - -
- - -
-
- - -
- - -
-
- - - - {message} -
- )} -
-
- ); -} diff --git a/components/forms/LoginForm.module.css b/components/forms/LoginForm.module.css deleted file mode 100644 index dfd5456a7b..0000000000 --- a/components/forms/LoginForm.module.css +++ /dev/null @@ -1,23 +0,0 @@ -.login { - display: flex; - flex-direction: column; - margin-top: 80px; -} - -.login form { - margin: 0 auto; -} - -.icon { - display: flex; - justify-content: center; - margin: 0 auto; -} - -.header { - margin-bottom: 30px; -} - -.header h1 { - margin: 12px 0; -} diff --git a/components/forms/ResetForm.js b/components/forms/ResetForm.js deleted file mode 100644 index 924aa7b1f1..0000000000 --- a/components/forms/ResetForm.js +++ /dev/null @@ -1,98 +0,0 @@ -import React, { useState } from 'react'; -import { FormattedMessage } from 'react-intl'; -import { Formik, Form, Field } from 'formik'; -import Button from 'components/common/Button'; -import FormLayout, { - FormButtons, - FormError, - FormMessage, - FormRow, -} from 'components/layout/FormLayout'; -import useApi from 'hooks/useApi'; - -const CONFIRMATION_WORD = 'RESET'; - -const validate = ({ confirmation }) => { - const errors = {}; - - if (confirmation !== CONFIRMATION_WORD) { - errors.confirmation = !confirmation ? ( - - ) : ( - - ); - } - - return errors; -}; - -export default function ResetForm({ values, onSave, onClose }) { - const { post } = useApi(); - const [message, setMessage] = useState(); - - const handleSubmit = async ({ type, id }) => { - const { ok, data } = await post(`/${type}/${id}/reset`); - - if (ok) { - onSave(); - } else { - setMessage( - data || , - ); - } - }; - - return ( - - - {props => ( -
-
- {values.name} }} - /> -
-
- -
-

- {CONFIRMATION_WORD} }} - /> -

- -
- - -
-
- - - - - {message} -
- )} -
-
- ); -} diff --git a/components/forms/ShareUrlForm.js b/components/forms/ShareUrlForm.js deleted file mode 100644 index 9800e54b92..0000000000 --- a/components/forms/ShareUrlForm.js +++ /dev/null @@ -1,42 +0,0 @@ -import React, { useRef } from 'react'; -import { FormattedMessage } from 'react-intl'; -import { useRouter } from 'next/router'; -import Button from 'components/common/Button'; -import FormLayout, { FormButtons, FormRow } from 'components/layout/FormLayout'; -import CopyButton from 'components/common/CopyButton'; - -export default function TrackingCodeForm({ values, onClose }) { - const ref = useRef(); - const { basePath } = useRouter(); - const { name, shareId } = values; - - return ( - -

- {values.name} }} - /> -

- -