Compare commits
17 Commits
Author | SHA1 | Date | |
---|---|---|---|
debfcf7226
|
|||
15bbc27238
|
|||
16bd44c599
|
|||
d13b16c032
|
|||
d144f7385b
|
|||
0283c293ef
|
|||
8d75849c55 | |||
58b44dddad | |||
c9de531389 | |||
d48cfe12f2 | |||
3a9cc20f86 | |||
54a34ef5ee | |||
f097d7761d | |||
03717113f4 | |||
284a4c5520 | |||
f03faabbee | |||
353fb3899e |
0
.dockerignore
Normal file → Executable file
0
.dockerignore
Normal file → Executable file
0
.editorconfig
Normal file → Executable file
0
.editorconfig
Normal file → Executable file
@ -1,42 +0,0 @@
|
||||
# These environment variables will be used by default if you do not create any
|
||||
# yourself in .env. This file should be safe to check into your version control
|
||||
# system. Any custom values should go in .env and .env should *not* be checked
|
||||
# into version control.
|
||||
|
||||
# location of the test database for api service scenarios (defaults to ./.redwood/test.db if not set)
|
||||
# TEST_DATABASE_URL=file:./.redwood/test.db
|
||||
|
||||
# disables Prisma CLI update notifier
|
||||
PRISMA_HIDE_UPDATE_MESSAGE=true
|
||||
|
||||
# Option to override the current environment's default api-side log level
|
||||
# See: https://redwoodjs.com/docs/logger for level options, defaults to "trace" otherwise.
|
||||
# Most applications want "debug" or "info" during dev, "trace" when you have issues and "warn" in production.
|
||||
# Ordered by how verbose they are: trace | debug | info | warn | error | silent
|
||||
# LOG_LEVEL=debug
|
||||
|
||||
FIRST_NAME=firstname
|
||||
LAST_NAME=lastname
|
||||
|
||||
COUNTRY=US
|
||||
|
||||
SMTP_HOST=smtp.example.com
|
||||
SMTP_PORT=465
|
||||
SMTP_SECURE=true
|
||||
SMTP_USER=noreply@example.com
|
||||
EMAIL_FROM=noreply@example.com
|
||||
EMAIL_TO=email@example.com
|
||||
SMTP_PASSWORD=password
|
||||
|
||||
DOMAIN=example.com
|
||||
API_DOMAIN=api.example.com
|
||||
|
||||
# Must not end with "/"
|
||||
ADDRESS_PROD=https://portfolio.example.com
|
||||
ADDRESS_DEV=http://localhost:8910
|
||||
API_ADDRESS_PROD=https://api-portfolio.example.com
|
||||
API_ADDRESS_DEV=http://localhost:8911
|
||||
|
||||
MAX_HTTP_CONNECTIONS_PER_MINUTE=60
|
||||
|
||||
DATABASE_URL=postgresql://user:password@localhost:5432/rw_portfolio
|
4
.env.example
Normal file → Executable file
4
.env.example
Normal file → Executable file
@ -6,7 +6,11 @@
|
||||
FIRST_NAME=firstname
|
||||
LAST_NAME=lastname
|
||||
|
||||
DEFAULT_THEME=light
|
||||
|
||||
COUNTRY=US
|
||||
STATE=New York
|
||||
CITY=Manhattan
|
||||
|
||||
SMTP_HOST=smtp.example.com
|
||||
SMTP_PORT=465
|
||||
|
30
.gitea/workflows/ci-dev.yml
Executable file
30
.gitea/workflows/ci-dev.yml
Executable file
@ -0,0 +1,30 @@
|
||||
version: "1"
|
||||
name: Publish Development Docker Image
|
||||
on:
|
||||
push:
|
||||
tags:
|
||||
- "v[0-9]+\\.[0-9]+\\.[0-9]+-dev"
|
||||
|
||||
jobs:
|
||||
build:
|
||||
name: Publish Development Docker Image
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v2
|
||||
|
||||
- name: Set up Docker Buildx
|
||||
uses: docker/setup-buildx-action@v1
|
||||
|
||||
- name: Login to Registry
|
||||
run: echo "${{ secrets.ACCESS_TOKEN }}" | docker login git.altaiar.dev -u "${{ secrets.USERNAME }}" --password-stdin
|
||||
|
||||
- name: Build & Tag Image
|
||||
run: |
|
||||
docker build --build-arg APP_VERSION=${{ gitea.ref_name }} -t git.altaiar.dev/${{ gitea.repository }}:${{ gitea.ref_name }} .
|
||||
docker tag git.altaiar.dev/${{ gitea.repository }}:${{ gitea.ref_name }} git.altaiar.dev/${{ gitea.repository }}:dev
|
||||
|
||||
- name: Push Images
|
||||
run: |
|
||||
docker push git.altaiar.dev/${{ gitea.repository }}:${{ gitea.ref_name }}
|
||||
docker push git.altaiar.dev/${{ gitea.repository }}:dev
|
5
.gitea/workflows/ci.yml
Normal file → Executable file
5
.gitea/workflows/ci.yml
Normal file → Executable file
@ -1,10 +1,9 @@
|
||||
version: "1"
|
||||
name: Publish Docker Image
|
||||
|
||||
on:
|
||||
push:
|
||||
tags:
|
||||
- "*"
|
||||
- "v[0-9]+\\.[0-9]+\\.[0-9]+"
|
||||
|
||||
jobs:
|
||||
build:
|
||||
@ -22,7 +21,7 @@ jobs:
|
||||
|
||||
- name: Build & Tag Image
|
||||
run: |
|
||||
docker build -t git.altaiar.dev/${{ gitea.repository }}:${{ gitea.ref_name }} .
|
||||
docker build --build-arg APP_VERSION=${{ gitea.ref_name }} -t git.altaiar.dev/${{ gitea.repository }}:${{ gitea.ref_name }} .
|
||||
docker tag git.altaiar.dev/${{ gitea.repository }}:${{ gitea.ref_name }} git.altaiar.dev/${{ gitea.repository }}:latest
|
||||
|
||||
- name: Push Images
|
||||
|
0
.gitignore
vendored
Normal file → Executable file
0
.gitignore
vendored
Normal file → Executable file
0
.redwood/README.md
Normal file → Executable file
0
.redwood/README.md
Normal file → Executable file
0
.vscode/extensions.json
vendored
Normal file → Executable file
0
.vscode/extensions.json
vendored
Normal file → Executable file
0
.vscode/launch.json
vendored
Normal file → Executable file
0
.vscode/launch.json
vendored
Normal file → Executable file
0
.vscode/settings.json
vendored
Normal file → Executable file
0
.vscode/settings.json
vendored
Normal file → Executable file
0
.vscode/tasks.json
vendored
Normal file → Executable file
0
.vscode/tasks.json
vendored
Normal file → Executable file
0
.yarnrc.yml
Normal file → Executable file
0
.yarnrc.yml
Normal file → Executable file
38
Dockerfile
Normal file → Executable file
38
Dockerfile
Normal file → Executable file
@ -1,6 +1,6 @@
|
||||
# base
|
||||
# ----
|
||||
FROM node:20-bookworm-slim as base
|
||||
FROM node:20-bookworm-slim AS base
|
||||
|
||||
RUN corepack enable
|
||||
|
||||
@ -29,11 +29,10 @@ RUN --mount=type=cache,target=/home/node/.yarn/berry/cache,uid=1000 \
|
||||
|
||||
COPY --chown=node:node redwood.toml .
|
||||
COPY --chown=node:node graphql.config.js .
|
||||
COPY --chown=node:node .env.defaults .env.defaults
|
||||
|
||||
# api build
|
||||
# ---------
|
||||
FROM base as api_build
|
||||
FROM base AS api_build
|
||||
|
||||
# If your api side build relies on build-time environment variables,
|
||||
# specify them here as ARGs. (But don't put secrets in your Dockerfile!)
|
||||
@ -52,37 +51,52 @@ ARG EMAIL_FROM
|
||||
ARG EMAIL_TO
|
||||
ARG FIRST_NAME
|
||||
ARG LAST_NAME
|
||||
ARG APP_VERSION
|
||||
|
||||
COPY --chown=node:node api api
|
||||
RUN yarn rw build api
|
||||
|
||||
# web prerender build
|
||||
# -------------------
|
||||
FROM api_build as web_build_with_prerender
|
||||
FROM api_build AS web_build_with_prerender
|
||||
|
||||
ARG FIRST_NAME
|
||||
ARG LAST_NAME
|
||||
ARG COUNTRY
|
||||
ARG STATE
|
||||
ARG CITY
|
||||
ARG DEFAULT_THEME
|
||||
ARG API_ADDRESS_PROD
|
||||
ARG API_ADDRESS_DEV
|
||||
ARG APP_VERSION
|
||||
|
||||
ENV APP_VERSION=$APP_VERSION
|
||||
|
||||
COPY --chown=node:node web web
|
||||
RUN yarn rw build web
|
||||
|
||||
# web build
|
||||
# ---------
|
||||
FROM base as web_build
|
||||
FROM base AS web_build
|
||||
|
||||
ARG FIRST_NAME
|
||||
ARG LAST_NAME
|
||||
ARG COUNTRY
|
||||
ARG STATE
|
||||
ARG CITY
|
||||
ARG DEFAULT_THEME
|
||||
ARG API_ADDRESS_PROD
|
||||
ARG API_ADDRESS_DEV
|
||||
ARG APP_VERSION
|
||||
|
||||
ENV APP_VERSION=$APP_VERSION
|
||||
|
||||
COPY --chown=node:node web web
|
||||
RUN yarn rw build web --no-prerender
|
||||
|
||||
# api serve
|
||||
# ---------
|
||||
FROM node:20-bookworm-slim as api_serve
|
||||
FROM node:20-bookworm-slim AS api_serve
|
||||
|
||||
RUN corepack enable
|
||||
|
||||
@ -107,13 +121,15 @@ RUN --mount=type=cache,target=/home/node/.yarn/berry/cache,uid=1000 \
|
||||
|
||||
COPY --chown=node:node redwood.toml .
|
||||
COPY --chown=node:node graphql.config.js .
|
||||
COPY --chown=node:node .env.defaults .env.defaults
|
||||
|
||||
COPY --chown=node:node --from=api_build /home/node/app/api/dist /home/node/app/api/dist
|
||||
COPY --chown=node:node --from=api_build /home/node/app/api/db /home/node/app/api/db
|
||||
COPY --chown=node:node --from=api_build /home/node/app/node_modules/.prisma /home/node/app/node_modules/.prisma
|
||||
|
||||
ARG APP_VERSION
|
||||
|
||||
ENV NODE_ENV=production
|
||||
ENV APP_VERSION=$APP_VERSION
|
||||
|
||||
# default api serve command
|
||||
# ---------
|
||||
@ -126,7 +142,7 @@ CMD [ "./api/dist/server.js" ]
|
||||
|
||||
# web serve
|
||||
# ---------
|
||||
FROM node:20-bookworm-slim as web_serve
|
||||
FROM node:20-bookworm-slim AS web_serve
|
||||
|
||||
RUN corepack enable
|
||||
|
||||
@ -147,10 +163,12 @@ RUN --mount=type=cache,target=/home/node/.yarn/berry/cache,uid=1000 \
|
||||
|
||||
COPY --chown=node:node redwood.toml .
|
||||
COPY --chown=node:node graphql.config.js .
|
||||
COPY --chown=node:node .env.defaults .env.defaults
|
||||
|
||||
COPY --chown=node:node --from=web_build /home/node/app/web/dist /home/node/app/web/dist
|
||||
|
||||
ARG APP_VERSION
|
||||
ENV APP_VERSION=$APP_VERSION
|
||||
|
||||
ENV NODE_ENV=production \
|
||||
API_PROXY_TARGET=http://api:8911
|
||||
|
||||
@ -159,7 +177,7 @@ CMD "node_modules/.bin/rw-web-server" "--api-proxy-target" "$API_PROXY_TARGET"
|
||||
|
||||
# console
|
||||
# -------
|
||||
FROM base as console
|
||||
FROM base AS console
|
||||
|
||||
# To add more packages:
|
||||
#
|
||||
|
16
README.md
Normal file → Executable file
16
README.md
Normal file → Executable file
@ -16,6 +16,7 @@ services:
|
||||
portfolio:
|
||||
container_name: portfolio
|
||||
image: git.altaiar.dev/ahmed/portfolio:latest
|
||||
restart: unless-stopped
|
||||
environment:
|
||||
- NODE_ENV=production
|
||||
- API_PROXY_TARGET=http://localhost:8911
|
||||
@ -25,6 +26,9 @@ services:
|
||||
- FIRST_NAME=first name # Your first name
|
||||
- LAST_NAME=lastname # Your last name
|
||||
- COUNTRY=US # ISO-3166-1 alpha-2 country code, https://en.wikipedia.org/wiki/ISO_3166-1#Codes
|
||||
- STATE=New York # Optional, state or province
|
||||
- CITY=Manhattan # Optional
|
||||
- DEFAULT_THEME=light # 'light' or 'dark'
|
||||
- SMTP_HOST=smtp.example.com
|
||||
- SMTP_PORT=465
|
||||
- SMTP_SECURE=true
|
||||
@ -41,7 +45,8 @@ services:
|
||||
- 8910:8910 # Web
|
||||
- 8911:8911 # API
|
||||
depends_on:
|
||||
- db
|
||||
db:
|
||||
condition: service_healthy
|
||||
volumes:
|
||||
- files:/home/node/app/api/files_prod
|
||||
command: >
|
||||
@ -51,14 +56,19 @@ services:
|
||||
yarn rw prisma db seed &&
|
||||
yarn rw serve"
|
||||
|
||||
|
||||
db:
|
||||
container_name: portfolio-db
|
||||
image: postgres:16-bookworm
|
||||
environment:
|
||||
- POSTGRES_USER=redwood
|
||||
- POSTGRES_PASSWORD=changeme
|
||||
- POSTGRES_PASSWORD=changeme # Change to a more secure password
|
||||
- POSTGRES_DB=portfolio
|
||||
restart: unless-stopped
|
||||
healthcheck:
|
||||
test: ["CMD-SHELL", "pg_isready -d DATABASE_URL"] # Replace DATABASE_URL with the database URL from the portfolio container
|
||||
interval: 10s
|
||||
timeout: 5s
|
||||
retries: 5
|
||||
volumes:
|
||||
- postgres:/var/lib/postgresql/data
|
||||
|
||||
|
0
api/db/migrations/20240810184713_user/migration.sql
Normal file → Executable file
0
api/db/migrations/20240810184713_user/migration.sql
Normal file → Executable file
0
api/db/migrations/20240819213158_social/migration.sql
Normal file → Executable file
0
api/db/migrations/20240819213158_social/migration.sql
Normal file → Executable file
0
api/db/migrations/20240821020900_portrait/migration.sql
Normal file → Executable file
0
api/db/migrations/20240821020900_portrait/migration.sql
Normal file → Executable file
0
api/db/migrations/20240824001030_project/migration.sql
Normal file → Executable file
0
api/db/migrations/20240824001030_project/migration.sql
Normal file → Executable file
0
api/db/migrations/20240921164727_project_date_is_mandatory_default_today_12_am/migration.sql
Normal file → Executable file
0
api/db/migrations/20240921164727_project_date_is_mandatory_default_today_12_am/migration.sql
Normal file → Executable file
0
api/db/migrations/20240921181721_more_socials/migration.sql
Normal file → Executable file
0
api/db/migrations/20240921181721_more_socials/migration.sql
Normal file → Executable file
0
api/db/migrations/20240921183021_even_more_socials/migration.sql
Normal file → Executable file
0
api/db/migrations/20240921183021_even_more_socials/migration.sql
Normal file → Executable file
0
api/db/migrations/20240927031102_/migration.sql
Normal file → Executable file
0
api/db/migrations/20240927031102_/migration.sql
Normal file → Executable file
0
api/db/migrations/20240929164343_/migration.sql
Normal file → Executable file
0
api/db/migrations/20240929164343_/migration.sql
Normal file → Executable file
0
api/db/migrations/20241001005227_title_and_resume/migration.sql
Normal file → Executable file
0
api/db/migrations/20241001005227_title_and_resume/migration.sql
Normal file → Executable file
0
api/db/migrations/20241005014130_/migration.sql
Normal file → Executable file
0
api/db/migrations/20241005014130_/migration.sql
Normal file → Executable file
0
api/db/migrations/20241015183037_matrix/migration.sql
Normal file → Executable file
0
api/db/migrations/20241015183037_matrix/migration.sql
Normal file → Executable file
0
api/db/migrations/migration_lock.toml
Normal file → Executable file
0
api/db/migrations/migration_lock.toml
Normal file → Executable file
0
api/db/schema.prisma
Normal file → Executable file
0
api/db/schema.prisma
Normal file → Executable file
0
api/jest.config.js
Normal file → Executable file
0
api/jest.config.js
Normal file → Executable file
6
api/package.json
Normal file → Executable file
6
api/package.json
Normal file → Executable file
@ -9,9 +9,9 @@
|
||||
"@redwoodjs/api-server": "8.4.0",
|
||||
"@redwoodjs/auth-dbauth-api": "8.4.0",
|
||||
"@redwoodjs/graphql-server": "8.4.0",
|
||||
"@tus/file-store": "^1.4.0",
|
||||
"@tus/server": "^1.7.0",
|
||||
"country-flag-icons": "^1.5.13",
|
||||
"@tus/file-store": "^2.0.0",
|
||||
"@tus/server": "^2.0.0",
|
||||
"countries-list": "^3.1.1",
|
||||
"graphql-scalars": "^1.23.0",
|
||||
"nodemailer": "^6.9.14"
|
||||
},
|
||||
|
0
api/quick-lint-js.config
Normal file → Executable file
0
api/quick-lint-js.config
Normal file → Executable file
0
api/src/directives/requireAuth/requireAuth.ts
Normal file → Executable file
0
api/src/directives/requireAuth/requireAuth.ts
Normal file → Executable file
0
api/src/directives/skipAuth/skipAuth.ts
Normal file → Executable file
0
api/src/directives/skipAuth/skipAuth.ts
Normal file → Executable file
0
api/src/functions/auth.ts
Normal file → Executable file
0
api/src/functions/auth.ts
Normal file → Executable file
0
api/src/functions/graphql.ts
Normal file → Executable file
0
api/src/functions/graphql.ts
Normal file → Executable file
0
api/src/graphql/.keep
Normal file → Executable file
0
api/src/graphql/.keep
Normal file → Executable file
0
api/src/graphql/portrait.sdl.ts
Normal file → Executable file
0
api/src/graphql/portrait.sdl.ts
Normal file → Executable file
0
api/src/graphql/projects.sdl.ts
Normal file → Executable file
0
api/src/graphql/projects.sdl.ts
Normal file → Executable file
0
api/src/graphql/resume.sdl.ts
Normal file → Executable file
0
api/src/graphql/resume.sdl.ts
Normal file → Executable file
0
api/src/graphql/scalars.sdl.ts
Normal file → Executable file
0
api/src/graphql/scalars.sdl.ts
Normal file → Executable file
0
api/src/graphql/socials.sdl.ts
Normal file → Executable file
0
api/src/graphql/socials.sdl.ts
Normal file → Executable file
0
api/src/graphql/tags.sdl.ts
Normal file → Executable file
0
api/src/graphql/tags.sdl.ts
Normal file → Executable file
0
api/src/graphql/title.sdl.ts
Normal file → Executable file
0
api/src/graphql/title.sdl.ts
Normal file → Executable file
0
api/src/lib/auth.ts
Normal file → Executable file
0
api/src/lib/auth.ts
Normal file → Executable file
0
api/src/lib/cors.ts
Normal file → Executable file
0
api/src/lib/cors.ts
Normal file → Executable file
0
api/src/lib/db.ts
Normal file → Executable file
0
api/src/lib/db.ts
Normal file → Executable file
0
api/src/lib/email.ts
Normal file → Executable file
0
api/src/lib/email.ts
Normal file → Executable file
0
api/src/lib/logger.ts
Normal file → Executable file
0
api/src/lib/logger.ts
Normal file → Executable file
6
api/src/lib/tus.ts
Normal file → Executable file
6
api/src/lib/tus.ts
Normal file → Executable file
@ -19,7 +19,7 @@ interface User {
|
||||
resetTokenExpiresAt: Date | null
|
||||
}
|
||||
|
||||
export const handleTusUpload = (
|
||||
export const handleTusUpload = async (
|
||||
req: FastifyRequest,
|
||||
res: FastifyReply,
|
||||
tusHandler: Server,
|
||||
@ -28,7 +28,7 @@ export const handleTusUpload = (
|
||||
if (isProduction) {
|
||||
if (req.method === 'OPTIONS') handleOptionsRequest(res)
|
||||
else if (isPublicEndpoint && req.method === 'GET')
|
||||
tusHandler.handle(req.raw, res.raw)
|
||||
await tusHandler.handle(req.raw, res.raw)
|
||||
else if (['GET', 'POST', 'HEAD', 'PATCH'].includes(req.method)) {
|
||||
if (req.headers.cookie) handleAuthenticatedRequest(req, res, tusHandler)
|
||||
else {
|
||||
@ -41,7 +41,7 @@ export const handleTusUpload = (
|
||||
}
|
||||
} else {
|
||||
setCorsHeaders(res)
|
||||
tusHandler.handle(req.raw, res.raw)
|
||||
await tusHandler.handle(req.raw, res.raw)
|
||||
}
|
||||
}
|
||||
|
||||
|
28
api/src/server.ts
Normal file → Executable file
28
api/src/server.ts
Normal file → Executable file
@ -1,22 +1,35 @@
|
||||
/* eslint-disable @typescript-eslint/no-explicit-any */
|
||||
import Cors from '@fastify/cors'
|
||||
import RateLimit from '@fastify/rate-limit'
|
||||
import { FileStore } from '@tus/file-store'
|
||||
import { Server } from '@tus/server'
|
||||
|
||||
import { isProduction } from '@redwoodjs/api/logger'
|
||||
import { createServer } from '@redwoodjs/api-server'
|
||||
|
||||
import { logger } from 'src/lib/logger'
|
||||
import { handleTusUpload } from 'src/lib/tus'
|
||||
;(async () => {
|
||||
const { hasFlag } = await import('country-flag-icons')
|
||||
|
||||
if (!hasFlag(process.env.COUNTRY))
|
||||
enum Theme {
|
||||
light = 'light',
|
||||
dark = 'dark',
|
||||
}
|
||||
|
||||
;(async () => {
|
||||
const { countries } = await import('countries-list')
|
||||
const { FileStore } = await import('@tus/file-store')
|
||||
const { Server } = await import('@tus/server')
|
||||
|
||||
if (!Object.keys(countries).includes(process.env.COUNTRY))
|
||||
throw new Error(
|
||||
'Invalid COUNTRY environment variable, please select a valid ISO-3166-1 alpha-2 country code\n See https://en.wikipedia.org/wiki/ISO_3166-1#Codes'
|
||||
)
|
||||
|
||||
if (!(process.env.DEFAULT_THEME.toLowerCase() in Theme))
|
||||
throw new Error(
|
||||
'Invalid DEFAULT_THEME environment variable, please select either light or dark'
|
||||
)
|
||||
|
||||
logger.info(`Portfolio ${process.env.APP_VERSION}`)
|
||||
|
||||
const server = await createServer({
|
||||
logger,
|
||||
configureApiServer: async (server) => {
|
||||
@ -41,7 +54,10 @@ import { handleTusUpload } from 'src/lib/tus'
|
||||
datastore: new FileStore({
|
||||
directory: `./files_${isProduction ? 'prod' : 'dev'}`,
|
||||
}),
|
||||
onResponseError: (_req, res, _err) => logger.error(res),
|
||||
onResponseError(_, err) {
|
||||
logger.error(err)
|
||||
return { status_code: 500, body: 'Internal Server Error' }
|
||||
},
|
||||
})
|
||||
|
||||
server.addContentTypeParser(
|
||||
|
0
api/src/services/.keep
Normal file → Executable file
0
api/src/services/.keep
Normal file → Executable file
0
api/src/services/portrait/portrait.ts
Normal file → Executable file
0
api/src/services/portrait/portrait.ts
Normal file → Executable file
0
api/src/services/projects/projects.ts
Normal file → Executable file
0
api/src/services/projects/projects.ts
Normal file → Executable file
0
api/src/services/resume/resume.ts
Normal file → Executable file
0
api/src/services/resume/resume.ts
Normal file → Executable file
0
api/src/services/socials/socials.ts
Normal file → Executable file
0
api/src/services/socials/socials.ts
Normal file → Executable file
0
api/src/services/tags/tags.ts
Normal file → Executable file
0
api/src/services/tags/tags.ts
Normal file → Executable file
0
api/src/services/title/title.ts
Normal file → Executable file
0
api/src/services/title/title.ts
Normal file → Executable file
4
api/tsconfig.json
Normal file → Executable file
4
api/tsconfig.json
Normal file → Executable file
@ -4,8 +4,8 @@
|
||||
"allowJs": true,
|
||||
"esModuleInterop": true,
|
||||
"target": "ES2023",
|
||||
"module": "Node16",
|
||||
"moduleResolution": "Node16",
|
||||
"module": "NodeNext",
|
||||
"moduleResolution": "nodenext",
|
||||
"skipLibCheck": false,
|
||||
"rootDirs": [
|
||||
"./src",
|
||||
|
16
docker-compose.yml
Normal file → Executable file
16
docker-compose.yml
Normal file → Executable file
@ -4,6 +4,7 @@ services:
|
||||
portfolio:
|
||||
container_name: portfolio
|
||||
image: git.altaiar.dev/ahmed/portfolio:latest
|
||||
restart: unless-stopped
|
||||
environment:
|
||||
- NODE_ENV=production
|
||||
- API_PROXY_TARGET=http://localhost:8911
|
||||
@ -13,6 +14,9 @@ services:
|
||||
- FIRST_NAME=first name # Your first name
|
||||
- LAST_NAME=lastname # Your last name
|
||||
- COUNTRY=US # ISO-3166-1 alpha-2 country code, https://en.wikipedia.org/wiki/ISO_3166-1#Codes
|
||||
- STATE=New York # Optional, state or province
|
||||
- CITY=Manhattan # Optional
|
||||
- DEFAULT_THEME=light # 'light' or 'dark'
|
||||
- SMTP_HOST=smtp.example.com
|
||||
- SMTP_PORT=465
|
||||
- SMTP_SECURE=true
|
||||
@ -29,7 +33,8 @@ services:
|
||||
- 8910:8910 # Web
|
||||
- 8911:8911 # API
|
||||
depends_on:
|
||||
- db
|
||||
db:
|
||||
condition: service_healthy
|
||||
volumes:
|
||||
- files:/home/node/app/api/files_prod
|
||||
command: >
|
||||
@ -39,14 +44,19 @@ services:
|
||||
yarn rw prisma db seed &&
|
||||
yarn rw serve"
|
||||
|
||||
|
||||
db:
|
||||
container_name: portfolio-db
|
||||
image: postgres:16-bookworm
|
||||
environment:
|
||||
- POSTGRES_USER=redwood
|
||||
- POSTGRES_PASSWORD=changeme
|
||||
- POSTGRES_PASSWORD=changeme # Change to a more secure password
|
||||
- POSTGRES_DB=portfolio
|
||||
restart: unless-stopped
|
||||
healthcheck:
|
||||
test: ["CMD-SHELL", "pg_isready -d DATABASE_URL"] # Replace DATABASE_URL with the database URL from the portfolio container
|
||||
interval: 10s
|
||||
timeout: 5s
|
||||
retries: 5
|
||||
volumes:
|
||||
- postgres:/var/lib/postgresql/data
|
||||
|
||||
|
0
graphql.config.js
Normal file → Executable file
0
graphql.config.js
Normal file → Executable file
0
jest.config.js
Normal file → Executable file
0
jest.config.js
Normal file → Executable file
0
package.json
Normal file → Executable file
0
package.json
Normal file → Executable file
0
prettier.config.mjs
Normal file → Executable file
0
prettier.config.mjs
Normal file → Executable file
2
redwood.toml
Normal file → Executable file
2
redwood.toml
Normal file → Executable file
@ -9,7 +9,7 @@
|
||||
title = "${FIRST_NAME} ${LAST_NAME}"
|
||||
port = 8910
|
||||
apiUrl = "/api"
|
||||
includeEnvironmentVariables = ["FIRST_NAME", "LAST_NAME", "COUNTRY", "API_ADDRESS_PROD", "API_ADDRESS_DEV"]
|
||||
includeEnvironmentVariables = ["FIRST_NAME", "LAST_NAME", "COUNTRY", "STATE", "CITY", "DEFAULT_THEME", "API_ADDRESS_PROD", "API_ADDRESS_DEV", "APP_VERSION"]
|
||||
[generate]
|
||||
tests = false
|
||||
stories = false
|
||||
|
0
scripts/.keep
Normal file → Executable file
0
scripts/.keep
Normal file → Executable file
6
scripts/seed.ts
Normal file → Executable file
6
scripts/seed.ts
Normal file → Executable file
@ -3,8 +3,6 @@ import { db } from 'api/src/lib/db'
|
||||
|
||||
import { hashPassword } from '@redwoodjs/auth-dbauth-api'
|
||||
|
||||
const MAX_TITLES = 5
|
||||
|
||||
export default async () => {
|
||||
try {
|
||||
const admin = {
|
||||
@ -44,9 +42,7 @@ export default async () => {
|
||||
if (!titles)
|
||||
await db.titles.create({
|
||||
data: {
|
||||
titles: Array.from({ length: MAX_TITLES }).map(
|
||||
(_, i) => `a title ${i + 1}`
|
||||
),
|
||||
titles: Array.from({ length: 3 }).map((_, i) => `title ${i + 1}`),
|
||||
},
|
||||
})
|
||||
} catch (error) {
|
||||
|
0
scripts/tsconfig.json
Normal file → Executable file
0
scripts/tsconfig.json
Normal file → Executable file
0
web/config/postcss.config.js
Normal file → Executable file
0
web/config/postcss.config.js
Normal file → Executable file
0
web/config/tailwind.config.js
Normal file → Executable file
0
web/config/tailwind.config.js
Normal file → Executable file
0
web/jest.config.js
Normal file → Executable file
0
web/jest.config.js
Normal file → Executable file
5
web/package.json
Normal file → Executable file
5
web/package.json
Normal file → Executable file
@ -35,14 +35,13 @@
|
||||
"@uppy/react": "^4.0.1",
|
||||
"@uppy/tus": "^4.0.0",
|
||||
"@uppy/webcam": "^4.0.1",
|
||||
"country-flag-icons": "^1.5.13",
|
||||
"countries-list": "^3.1.1",
|
||||
"date-fns": "^4.1.0",
|
||||
"humanize-string": "2.1.0",
|
||||
"react": "18.3.1",
|
||||
"react-colorful": "^5.6.1",
|
||||
"react-dom": "18.3.1",
|
||||
"react-html-parser": "^2.0.2",
|
||||
"react-typed": "^2.0.12"
|
||||
"react-html-parser": "^2.0.2"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@redwoodjs/vite": "8.4.0",
|
||||
|
0
web/public/README.md
Normal file → Executable file
0
web/public/README.md
Normal file → Executable file
0
web/public/favicon.png
Normal file → Executable file
0
web/public/favicon.png
Normal file → Executable file
Before Width: | Height: | Size: 757 B After Width: | Height: | Size: 757 B |
0
web/public/no_portrait.webp
Normal file → Executable file
0
web/public/no_portrait.webp
Normal file → Executable file
Before Width: | Height: | Size: 7.8 KiB After Width: | Height: | Size: 7.8 KiB |
0
web/public/no_resume.pdf
Normal file → Executable file
0
web/public/no_resume.pdf
Normal file → Executable file
0
web/public/robots.txt
Normal file → Executable file
0
web/public/robots.txt
Normal file → Executable file
0
web/quick-lint-js.config
Normal file → Executable file
0
web/quick-lint-js.config
Normal file → Executable file
0
web/src/App.tsx
Normal file → Executable file
0
web/src/App.tsx
Normal file → Executable file
4
web/src/Routes.tsx
Normal file → Executable file
4
web/src/Routes.tsx
Normal file → Executable file
@ -31,11 +31,11 @@ const Routes = () => {
|
||||
<Route path="/admin/titles" page={TitleTitlesPage} name="titles" />
|
||||
</Set>
|
||||
|
||||
<Set wrap={ScaffoldLayout} title="Resume" titleTo="adminResume">
|
||||
<Set wrap={ScaffoldLayout} title="Résumé" titleTo="adminResume">
|
||||
<Route path="/admin/resume" page={ResumeAdminResumePage} name="adminResume" />
|
||||
</Set>
|
||||
|
||||
<Set wrap={ScaffoldLayout} title="Projects" titleTo="projects" buttonLabel="New Project" buttonTo="newProject">
|
||||
<Set wrap={ScaffoldLayout} title="Projects" titleTo="adminProjects" buttonLabel="New Project" buttonTo="newProject">
|
||||
<Route path="/admin/projects/new" page={ProjectNewProjectPage} name="newProject" />
|
||||
<Route path="/admin/projects/{id:Int}/edit" page={ProjectEditProjectPage} name="editProject" />
|
||||
<Route path="/admin/projects/{id:Int}" page={ProjectAdminProjectPage} name="adminProject" />
|
||||
|
0
web/src/auth.ts
Normal file → Executable file
0
web/src/auth.ts
Normal file → Executable file
0
web/src/components/.keep
Normal file → Executable file
0
web/src/components/.keep
Normal file → Executable file
@ -1,56 +0,0 @@
|
||||
import { useState, useRef, useCallback, useEffect } from 'react'
|
||||
|
||||
const SCROLL_INTERVAL_SECONDS = 3
|
||||
|
||||
interface AutoCarouselProps {
|
||||
images: string[]
|
||||
}
|
||||
|
||||
const AutoCarousel = ({ images }: AutoCarouselProps) => {
|
||||
const [activeItem, setActiveItem] = useState<number>(0)
|
||||
const ref = useRef<HTMLDivElement>(null)
|
||||
|
||||
const scroll = useCallback(() => {
|
||||
setActiveItem((prev) => {
|
||||
if (images.length - 1 > prev) return prev + 1
|
||||
else return 0
|
||||
})
|
||||
}, [images.length])
|
||||
|
||||
const autoScroll = useCallback(
|
||||
() => setInterval(scroll, SCROLL_INTERVAL_SECONDS * 1000),
|
||||
[scroll]
|
||||
)
|
||||
|
||||
useEffect(() => {
|
||||
const play = autoScroll()
|
||||
return () => clearInterval(play)
|
||||
}, [autoScroll])
|
||||
|
||||
useEffect(() => {
|
||||
const width = ref.current?.getBoundingClientRect().width
|
||||
ref.current?.scroll({ left: activeItem * (width || 0) })
|
||||
}, [activeItem])
|
||||
|
||||
return (
|
||||
<div
|
||||
ref={ref}
|
||||
className="carousel carousel-center p-2 space-x-2 rounded-box"
|
||||
>
|
||||
{images.map((image, i) => (
|
||||
<div
|
||||
key={i}
|
||||
className="carousel-item w-full h-fit my-auto justify-center"
|
||||
>
|
||||
<img
|
||||
src={image}
|
||||
alt={`${i}`}
|
||||
className="object-contain rounded-xl size-fit"
|
||||
/>
|
||||
</div>
|
||||
))}
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
export default AutoCarousel
|
0
web/src/components/Cell/CellEmpty/CellEmpty.tsx
Normal file → Executable file
0
web/src/components/Cell/CellEmpty/CellEmpty.tsx
Normal file → Executable file
0
web/src/components/Cell/CellFailure/CellFailure.tsx
Normal file → Executable file
0
web/src/components/Cell/CellFailure/CellFailure.tsx
Normal file → Executable file
0
web/src/components/Cell/CellLoading/CellLoading.tsx
Normal file → Executable file
0
web/src/components/Cell/CellLoading/CellLoading.tsx
Normal file → Executable file
90
web/src/components/ColorPicker/ColorPicker.tsx
Normal file → Executable file
90
web/src/components/ColorPicker/ColorPicker.tsx
Normal file → Executable file
@ -9,54 +9,50 @@ interface ColorPickerProps {
|
||||
setColor: React.Dispatch<React.SetStateAction<string>>
|
||||
}
|
||||
|
||||
const ColorPicker = ({ color, setColor }: ColorPickerProps) => {
|
||||
return (
|
||||
<div className="flex flex-col space-y-2 bg-base-100 p-2 rounded-xl w-min">
|
||||
<section className="w-52">
|
||||
<HexColorPicker color={color} onChange={setColor} />
|
||||
</section>
|
||||
<div className="flex space-x-2 w-52">
|
||||
{/* eslint-disable-next-line jsx-a11y/label-has-associated-control */}
|
||||
<label className="input input-bordered flex items-center gap-2 input-sm w-full grow">
|
||||
<Icon path={mdiPound} className="size-4 opacity-70" />
|
||||
<HexColorInput color={color} className="w-16" />
|
||||
</label>
|
||||
<button
|
||||
type="button"
|
||||
className="btn btn-square btn-sm"
|
||||
onClick={async () => {
|
||||
try {
|
||||
await navigator.clipboard.writeText(color)
|
||||
toast.success('Copied color to clipboard')
|
||||
} catch {
|
||||
toast.error(`Failed to copy, please try again`)
|
||||
}
|
||||
}}
|
||||
>
|
||||
<Icon path={mdiContentCopy} className="size-4" />
|
||||
</button>
|
||||
<button
|
||||
type="button"
|
||||
className="btn btn-square btn-sm "
|
||||
onClick={async () => {
|
||||
try {
|
||||
const clipboardText = await navigator.clipboard.readText()
|
||||
const hexColorRegex =
|
||||
/^#(?:(?:[\da-f]{3}){1,2}|(?:[\da-f]{4}){1,2})$/i
|
||||
const ColorPicker = ({ color, setColor }: ColorPickerProps) => (
|
||||
<div className="flex flex-col space-y-2 bg-base-100 p-2 rounded-xl">
|
||||
<HexColorPicker color={color} onChange={setColor} />
|
||||
<div className="flex space-x-2">
|
||||
{/* eslint-disable-next-line jsx-a11y/label-has-associated-control */}
|
||||
<label className="input input-bordered flex items-center gap-2 input-sm grow">
|
||||
<Icon path={mdiPound} className="size-4 opacity-70" />
|
||||
<HexColorInput color={color} className="w-16" />
|
||||
</label>
|
||||
<button
|
||||
type="button"
|
||||
className="btn btn-square btn-sm"
|
||||
onClick={async () => {
|
||||
try {
|
||||
await navigator.clipboard.writeText(color)
|
||||
toast.success('Copied color to clipboard')
|
||||
} catch {
|
||||
toast.error(`Failed to copy, please try again`)
|
||||
}
|
||||
}}
|
||||
>
|
||||
<Icon path={mdiContentCopy} className="size-4" />
|
||||
</button>
|
||||
<button
|
||||
type="button"
|
||||
className="btn btn-square btn-sm"
|
||||
onClick={async () => {
|
||||
try {
|
||||
const clipboardText = await navigator.clipboard.readText()
|
||||
const hexColorRegex =
|
||||
/^#(?:(?:[\da-f]{3}){1,2}|(?:[\da-f]{4}){1,2})$/i
|
||||
|
||||
if (!hexColorRegex.test(clipboardText))
|
||||
toast.error(`Text is not a valid hex color`)
|
||||
else setColor(clipboardText)
|
||||
} catch {
|
||||
toast.error(`Failed to paste, please try again`)
|
||||
}
|
||||
}}
|
||||
>
|
||||
<Icon path={mdiContentPaste} className="size-4" />
|
||||
</button>
|
||||
</div>
|
||||
if (!hexColorRegex.test(clipboardText))
|
||||
toast.error(`Text is not a valid hex color`)
|
||||
else setColor(clipboardText)
|
||||
} catch {
|
||||
toast.error(`Failed to paste, please try again`)
|
||||
}
|
||||
}}
|
||||
>
|
||||
<Icon path={mdiContentPaste} className="size-4" />
|
||||
</button>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
</div>
|
||||
)
|
||||
|
||||
export default ColorPicker
|
||||
|
30
web/src/components/ContactCard/ContactCard/ContactCard.tsx
Normal file → Executable file
30
web/src/components/ContactCard/ContactCard/ContactCard.tsx
Normal file → Executable file
@ -10,8 +10,8 @@ interface ContactCardProps {
|
||||
}
|
||||
|
||||
const ContactCard = ({ portraitUrl, socials }: ContactCardProps) => {
|
||||
const [width, setWidth] = useState()
|
||||
const [height, setHeight] = useState()
|
||||
const [width, setWidth] = useState<number>(0)
|
||||
const [height, setHeight] = useState<number>(0)
|
||||
|
||||
const observedDiv = useRef(null)
|
||||
|
||||
@ -53,24 +53,18 @@ const ContactCard = ({ portraitUrl, socials }: ContactCardProps) => {
|
||||
}}
|
||||
/>
|
||||
|
||||
<div className="flex min-h-[calc(100vh-6rem)] items-center justify-center">
|
||||
<div className="card bg-base-100 shadow-xl md:card-side">
|
||||
<figure>
|
||||
<img
|
||||
className="contact-me-image aspect-portrait object-cover"
|
||||
src={portraitUrl}
|
||||
alt={`${process.env.FIRST_NAME} ${process.env.LAST_NAME}`}
|
||||
/>
|
||||
</figure>
|
||||
<div
|
||||
className="card-body mx-auto h-fit w-fit md:mx-0"
|
||||
ref={observedDiv}
|
||||
>
|
||||
<h2 className="card-title justify-center text-3xl md:justify-start">
|
||||
<div className="flex items-center justify-center">
|
||||
<div className="card card-compact bg-base-100 shadow-xl md:card-side">
|
||||
<img
|
||||
className="contact-me-image rounded-box aspect-portrait p-2 object-cover"
|
||||
src={portraitUrl}
|
||||
alt={`${process.env.FIRST_NAME} ${process.env.LAST_NAME}`}
|
||||
/>
|
||||
<div className="card-body mx-auto w-fit md:mx-0" ref={observedDiv}>
|
||||
<h2 className="card-title justify-center text-3xl pb-2 md:justify-start">
|
||||
Contact Me
|
||||
</h2>
|
||||
<p className="p-2"></p>
|
||||
<div className="card-actions">
|
||||
<div className="card-actions rounded-btn">
|
||||
<SocialLinks socials={socials} />
|
||||
</div>
|
||||
</div>
|
||||
|
0
web/src/components/ContactCard/ContactCardCell/ContactCardCell.tsx
Normal file → Executable file
0
web/src/components/ContactCard/ContactCardCell/ContactCardCell.tsx
Normal file → Executable file
0
web/src/components/DatePicker/DatePicker.tsx
Normal file → Executable file
0
web/src/components/DatePicker/DatePicker.tsx
Normal file → Executable file
0
web/src/components/FormTextList/FormTextList.tsx
Normal file → Executable file
0
web/src/components/FormTextList/FormTextList.tsx
Normal file → Executable file
43
web/src/components/PDF/PDF.tsx
Normal file → Executable file
43
web/src/components/PDF/PDF.tsx
Normal file → Executable file
@ -1,19 +1,38 @@
|
||||
import { useState } from 'react'
|
||||
|
||||
import { mdiAlertOutline } from '@mdi/js'
|
||||
import Icon from '@mdi/react'
|
||||
|
||||
interface PDFProps {
|
||||
url: string
|
||||
form?: boolean
|
||||
}
|
||||
|
||||
const PDF = ({ url, form = false }: PDFProps) => (
|
||||
<embed
|
||||
src={url}
|
||||
title="PDF"
|
||||
type="application/pdf"
|
||||
style={{
|
||||
width: 'calc(100vw - 1rem)',
|
||||
height: `calc(100vh - ${form ? '8.5rem' : '6rem'})`,
|
||||
}}
|
||||
className="rounded-xl"
|
||||
/>
|
||||
)
|
||||
const PDF = ({ url, form = false }: PDFProps) => {
|
||||
const [error, setError] = useState<boolean>(false)
|
||||
|
||||
return error ? (
|
||||
<div role="alert" className="alert alert-warning">
|
||||
<Icon path={mdiAlertOutline} className="size-7" />
|
||||
<span>
|
||||
Could not load PDF, this is common in in-app browsers, try opening this
|
||||
page in a regular browser
|
||||
</span>
|
||||
</div>
|
||||
) : (
|
||||
<iframe
|
||||
src={url}
|
||||
title="PDF"
|
||||
content="application/pdf"
|
||||
style={{
|
||||
width: 'calc(100vw - 1rem)',
|
||||
height: `calc(100vh - ${form ? '8.5rem' : '6rem'})`,
|
||||
}}
|
||||
className="rounded-xl"
|
||||
onError={() => setError(true)}
|
||||
onLoad={() => setError(false)}
|
||||
/>
|
||||
)
|
||||
}
|
||||
|
||||
export default PDF
|
||||
|
0
web/src/components/Portrait/PortraitCell/PortraitCell.tsx
Normal file → Executable file
0
web/src/components/Portrait/PortraitCell/PortraitCell.tsx
Normal file → Executable file
7
web/src/components/Portrait/PortraitForm/PortraitForm.tsx
Normal file → Executable file
7
web/src/components/Portrait/PortraitForm/PortraitForm.tsx
Normal file → Executable file
@ -127,14 +127,13 @@ const PortraitForm = ({ portrait }: PortraitFormProps) => {
|
||||
)
|
||||
else
|
||||
return (
|
||||
<div className="mx-auto w-fit space-y-2">
|
||||
<div className="mx-auto max-w-prose space-y-2">
|
||||
{!fileId ? (
|
||||
<>
|
||||
<Uploader
|
||||
onComplete={onUploadComplete}
|
||||
width="22rem"
|
||||
height="34.5rem"
|
||||
className="flex justify-center"
|
||||
width="auto"
|
||||
height="30rem"
|
||||
/>
|
||||
<p className="text-center">
|
||||
High quality, 4:5 aspect ratio image recommended
|
||||
|
50
web/src/components/Project/AdminProject/AdminProject.tsx
Normal file → Executable file
50
web/src/components/Project/AdminProject/AdminProject.tsx
Normal file → Executable file
@ -46,29 +46,28 @@ const AdminProject = ({ project }: Props) => {
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="flex w-full justify-center">
|
||||
<div className="flex justify-center">
|
||||
<div>
|
||||
<div className="overflow-hidden overflow-x-auto rounded-xl bg-base-100">
|
||||
<table className="table">
|
||||
<thead className="bg-base-200 font-syne">
|
||||
<tr>
|
||||
<th className="w-0">
|
||||
<th colSpan={2}>
|
||||
Project {project.id}: {project.title}
|
||||
</th>
|
||||
<th> </th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
<tr>
|
||||
<th>ID</th>
|
||||
<th className="text-right">ID</th>
|
||||
<td>{project.id}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Title</th>
|
||||
<th className="text-right">Title</th>
|
||||
<td>{project.title}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Description</th>
|
||||
<th className="text-right">Description</th>
|
||||
<td>
|
||||
<article className="prose">
|
||||
{parseHtml(project.description)}
|
||||
@ -76,11 +75,11 @@ const AdminProject = ({ project }: Props) => {
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Date</th>
|
||||
<th className="text-right">Date</th>
|
||||
<td>{timeTag(project.date)}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Images</th>
|
||||
<th className="text-right">Images</th>
|
||||
<td>
|
||||
<div className="flex flex-wrap gap-2">
|
||||
{project.images.map((image, i) => (
|
||||
@ -88,7 +87,7 @@ const AdminProject = ({ project }: Props) => {
|
||||
key={i}
|
||||
href={image}
|
||||
target="_blank"
|
||||
className="btn btn-sm btn-square"
|
||||
className={`btn btn-sm btn-square ${i === 0 && 'btn-primary'}`}
|
||||
rel="noreferrer"
|
||||
>
|
||||
{i + 1}
|
||||
@ -98,7 +97,7 @@ const AdminProject = ({ project }: Props) => {
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Tags</th>
|
||||
<th className="text-right">Tags</th>
|
||||
<td>
|
||||
<div className="flex flex-wrap gap-2">
|
||||
{project.tags.map((tag, i) => (
|
||||
@ -120,19 +119,30 @@ const AdminProject = ({ project }: Props) => {
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th>Links</th>
|
||||
<th className="text-right">Links</th>
|
||||
<td>
|
||||
<div className="flex flex-wrap gap-2">
|
||||
{project.links.map((link, i) => (
|
||||
<a
|
||||
href={link}
|
||||
target="_blank"
|
||||
className="badge badge-ghost text-nowrap"
|
||||
key={i}
|
||||
rel="noreferrer"
|
||||
>
|
||||
{link}
|
||||
</a>
|
||||
<>
|
||||
<a
|
||||
href={link}
|
||||
target="_blank"
|
||||
className="hidden sm:flex badge badge-ghost text-nowrap"
|
||||
key={i}
|
||||
rel="noreferrer"
|
||||
>
|
||||
{link}
|
||||
</a>
|
||||
<a
|
||||
href={link}
|
||||
target="_blank"
|
||||
className="btn btn-sm btn-square sm:hidden"
|
||||
key={i}
|
||||
rel="noreferrer"
|
||||
>
|
||||
{i + 1}
|
||||
</a>
|
||||
</>
|
||||
))}
|
||||
</div>
|
||||
</td>
|
||||
|
0
web/src/components/Project/AdminProjectCell/AdminProjectCell.tsx
Normal file → Executable file
0
web/src/components/Project/AdminProjectCell/AdminProjectCell.tsx
Normal file → Executable file
8
web/src/components/Project/EditProjectCell/EditProjectCell.tsx
Normal file → Executable file
8
web/src/components/Project/EditProjectCell/EditProjectCell.tsx
Normal file → Executable file
@ -73,12 +73,12 @@ export const Success = ({ project }: CellSuccessProps<EditProjectById>) => {
|
||||
) => updateProject({ variables: { id, input } })
|
||||
|
||||
return (
|
||||
<div className="flex w-full justify-center">
|
||||
<div className="overflow-hidden overflow-x-auto rounded-xl bg-base-100">
|
||||
<table className="table w-80">
|
||||
<div className="flex mx-auto max-w-prose justify-center">
|
||||
<div className="overflow-hidden w-full overflow-x-auto rounded-xl bg-base-100">
|
||||
<table className="table">
|
||||
<thead className="bg-base-200 font-syne">
|
||||
<tr>
|
||||
<th className="w-0">Edit Project {project.id}</th>
|
||||
<th>Edit Project {project.id}</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
|
6
web/src/components/Project/NewProject/NewProject.tsx
Normal file → Executable file
6
web/src/components/Project/NewProject/NewProject.tsx
Normal file → Executable file
@ -38,9 +38,9 @@ const NewProject = () => {
|
||||
createProject({ variables: { input } })
|
||||
|
||||
return (
|
||||
<div className="flex w-full justify-center">
|
||||
<div className="overflow-hidden overflow-x-auto rounded-xl bg-base-100">
|
||||
<table className="table w-80">
|
||||
<div className="flex mx-auto max-w-prose justify-center">
|
||||
<div className="overflow-hidden w-full overflow-x-auto rounded-xl bg-base-100">
|
||||
<table className="table">
|
||||
<thead className="bg-base-200 font-syne">
|
||||
<tr>
|
||||
<th>New Project</th>
|
||||
|
4
web/src/components/Project/Project/Project.tsx
Normal file → Executable file
4
web/src/components/Project/Project/Project.tsx
Normal file → Executable file
@ -73,7 +73,7 @@ const Project = ({ project }: Props) => {
|
||||
<h2 className="sm:hidden font-bold text-3xl text-center">Images</h2>
|
||||
)}
|
||||
</div>
|
||||
<div className="flex flex-wrap gap-4 pt-8 justify-center h-fit">
|
||||
<div className="flex flex-wrap gap-4 pt-8 justify-center h-fit sm:p-8">
|
||||
{project.images.length > 0 &&
|
||||
project.images.map((image, i) => (
|
||||
<a
|
||||
@ -81,7 +81,7 @@ const Project = ({ project }: Props) => {
|
||||
target="_blank"
|
||||
rel="noreferrer"
|
||||
key={i}
|
||||
className="rounded-xl"
|
||||
className="rounded-box"
|
||||
>
|
||||
<img src={image} alt="" className="rounded-xl" />
|
||||
</a>
|
||||
|
0
web/src/components/Project/ProjectCell/ProjectCell.tsx
Normal file → Executable file
0
web/src/components/Project/ProjectCell/ProjectCell.tsx
Normal file → Executable file
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user