diff --git a/.devcontainer/devcontainer.json b/.devcontainer/devcontainer.json new file mode 100644 index 0000000..1c8114e --- /dev/null +++ b/.devcontainer/devcontainer.json @@ -0,0 +1,19 @@ +{ + "name": "Astro", + "image": "ghcr.io/acdh-oeaw/devcontainer-frontend:20", + "customizations": { + "vscode": { + "extensions": [ + "astro-build.astro-vscode", + "bradlc.vscode-tailwindcss", + "dbaeumer.vscode-eslint", + "editorconfig.editorconfig", + "esbenp.prettier-vscode", + "mikestead.dotenv", + "ms-playwright.playwright", + "stylelint.vscode-stylelint", + "unifiedjs.vscode-mdx" + ] + } + } +} diff --git a/.dockerignore b/.dockerignore new file mode 100644 index 0000000..793e18a --- /dev/null +++ b/.dockerignore @@ -0,0 +1,63 @@ +## .gitignore ## + +# dependencies +node_modules/ +.pnpm-store/ + +# logs +*.log + +# non-public environment variables +.env.local +.env.*.local + +# caches +.eslintcache +.stylelintcache +*.tsbuildinfo + +# vercel +.vercel + +# misc +.DS_Store +.idea/ + +# astro +dist/ +.astro/ + +# test +/coverage/ + +# playwright +/blob-report/ +/playwright/.cache/ +/playwright-report/ +/test-results/ + + +## .dockerignore ## + +# git +.git/ +.gitattributes +.gitignore + +# github +.github/ + +# vscode settings +.vscode/ + +# environment variables +.env +.env.* + +# tests +playwright.config.ts +/e2e/ +/test/ + +# misc +.editorconfig diff --git a/.editorconfig b/.editorconfig new file mode 100644 index 0000000..7d73f1e --- /dev/null +++ b/.editorconfig @@ -0,0 +1,13 @@ +root = true + +[*] +charset = utf-8 +end_of_line = lf +indent_size = 2 +indent_style = tab +insert_final_newline = true +max_line_length = 100 +trim_trailing_whitespace = true + +[*.{yaml,yml}] +indent_style = space diff --git a/.env.local.example b/.env.local.example new file mode 100644 index 0000000..b3be690 --- /dev/null +++ b/.env.local.example @@ -0,0 +1,46 @@ +# ------------------------------------------------------------------------------------------------- +# environment variables +# ------------------------------------------------------------------------------------------------- +# - public environment variables must be prefixed with `PUBLIC_`. +# - when adding new environment variables, don't forget to also update the +# validation schema in `./config/env.config.ts`. + +# ------------------------------------------------------------------------------------------------- +# app +# ------------------------------------------------------------------------------------------------- +PUBLIC_APP_BASE_URL="http://localhost:3000" +# PUBLIC_APP_BASE_PATH= +# imprint service +PUBLIC_REDMINE_ID="22508" +# web crawlers +PUBLIC_BOTS="disabled" +# validate environment variables +ENV_VALIDATION="enabled" + +# ------------------------------------------------------------------------------------------------- +# analytics +# ------------------------------------------------------------------------------------------------- +# PUBLIC_GOOGLE_SITE_VERIFICATION= +PUBLIC_MATOMO_BASE_URL="https://matomo.acdh.oeaw.ac.at" +# PUBLIC_MATOMO_ID= + +# ------------------------------------------------------------------------------------------------- +# keystatic cms +# ------------------------------------------------------------------------------------------------- +# KEYSTATIC_GITHUB_CLIENT_ID= +# KEYSTATIC_GITHUB_CLIENT_SECRET= +# KEYSTATIC_SECRET= +# PUBLIC_KEYSTATIC_GITHUB_APP_SLUG= +# PUBLIC_KEYSTATIC_GITHUB_REPO_NAME= +# PUBLIC_KEYSTATIC_GITHUB_REPO_OWNER= +# PUBLIC_KEYSTATIC_MODE="github" + +# ------------------------------------------------------------------------------------------------- +# registration form +# ------------------------------------------------------------------------------------------------- +# EMAIL_CONTACT_ADDRESS= +# EMAIL_CONTACT_ADDRESS_BCC= +# EMAIL_SMTP_PORT="587" +# EMAIL_SMTP_SERVER="smtp.oeaw.ac.at" +# EMAIL_SMTP_USERNAME +# EMAIL_SMTP_PASSWORD diff --git a/.gitattributes b/.gitattributes new file mode 100644 index 0000000..6313b56 --- /dev/null +++ b/.gitattributes @@ -0,0 +1 @@ +* text=auto eol=lf diff --git a/.github/workflows/build-deploy.yml b/.github/workflows/build-deploy.yml new file mode 100644 index 0000000..bd2aa3e --- /dev/null +++ b/.github/workflows/build-deploy.yml @@ -0,0 +1,131 @@ +name: Build and deploy + +concurrency: + group: "${{ github.workflow }}-${{ github.ref }}-build-deploy" + cancel-in-progress: true + +on: + workflow_call: + workflow_dispatch: + +jobs: + env: + name: Generate environment variables + runs-on: ubuntu-latest + steps: + - name: Derive environment from git ref + id: environment + run: | + if [ "${{ github.ref }}" = "refs/heads/main" ]; then + ENVIRONMENT="production" + APP_NAME_SUFFIX="" + elif [ "${{ github.ref }}" = "refs/heads/develop" ]; then + ENVIRONMENT="development" + APP_NAME_SUFFIX="-development" + elif [ "${{github.event_name}}" = "pull_request"]; then + ENVIRONMENT="pr/${{ github.event.pull_request.number }}" + APP_NAME_SUFFIX="-pr-${{ github.event.pull_request.number }}" + else + exit 1 + fi + + echo "ENVIRONMENT=$ENVIRONMENT" >> $GITHUB_OUTPUT + echo "APP_NAME_SUFFIX=$APP_NAME_SUFFIX" >> $GITHUB_OUTPUT + outputs: + environment: "${{ steps.environment.outputs.ENVIRONMENT }}" + app_name: "bruckner-online-website${{ steps.environment.outputs.APP_NAME_SUFFIX }}" + registry: "ghcr.io" + image: "${{ github.repository }}" + + vars: + name: Generate public url + needs: [env] + runs-on: ubuntu-latest + environment: + name: "${{ needs.env.outputs.environment }}" + steps: + - name: Generate public URL + id: public_url + run: | + if [ -z "${{ vars.PUBLIC_URL }}" ]; then + PUBLIC_URL="https://${{ needs.env.outputs.app_name }}.${{ vars.KUBE_INGRESS_BASE_DOMAIN }}" + else + PUBLIC_URL="${{ vars.PUBLIC_URL }}" + fi + + echo "PUBLIC_URL=$PUBLIC_URL" >> $GITHUB_OUTPUT + outputs: + public_url: "${{ steps.public_url.outputs.PUBLIC_URL }}" + + build: + name: Build and push docker image + needs: [env, vars] + runs-on: ubuntu-latest + permissions: + contents: read + packages: write + + steps: + - name: Checkout repository + uses: actions/checkout@v4 + + - name: Set up Docker Buildx + uses: docker/setup-buildx-action@v3 + + - name: Log in to the Container registry + uses: docker/login-action@v3 + with: + registry: "${{ needs.env.outputs.registry }}" + username: "${{ github.actor }}" + password: "${{ secrets.GITHUB_TOKEN }}" + + - name: Extract metadata (tags, labels) for Docker + id: meta + uses: docker/metadata-action@v5 + with: + images: "${{ needs.env.outputs.registry }}/${{ needs.env.outputs.image }}" + tags: | + type=raw,value={{sha}} + type=ref,event=branch + # type=ref,event=pr + # type=semver,pattern={{version}} + # type=semver,pattern={{major}}.{{minor}} + # type=raw,value=latest + + - name: Build and push Docker image + uses: docker/build-push-action@v5 + with: + context: . + push: true + tags: "${{ steps.meta.outputs.tags }}" + labels: "${{ steps.meta.outputs.labels }}" + build-args: | + "PUBLIC_APP_BASE_URL=${{ needs.vars.outputs.public_url }}" + "PUBLIC_BOTS=${{ vars.PUBLIC_BOTS }}" + "PUBLIC_KEYSTATIC_GITHUB_APP_SLUG=${{ vars.PUBLIC_KEYSTATIC_GITHUB_APP_SLUG }}" + "PUBLIC_KEYSTATIC_GITHUB_REPO_NAME=${{ vars.PUBLIC_KEYSTATIC_GITHUB_REPO_NAME }}" + "PUBLIC_KEYSTATIC_GITHUB_REPO_OWNER=${{ vars.PUBLIC_KEYSTATIC_GITHUB_REPO_OWNER }}" + "PUBLIC_KEYSTATIC_MODE=${{ vars.PUBLIC_KEYSTATIC_MODE }}" + "PUBLIC_MATOMO_BASE_URL=${{ vars.PUBLIC_MATOMO_BASE_URL }}" + "PUBLIC_MATOMO_ID=${{ vars.PUBLIC_MATOMO_ID }}" + "PUBLIC_REDMINE_ID=${{ vars.SERVICE_ID }}" + secrets: | + "KEYSTATIC_GITHUB_CLIENT_ID=${{ secrets.K8S_SECRET_KEYSTATIC_GITHUB_CLIENT_ID }}" + "KEYSTATIC_GITHUB_CLIENT_SECRET=${{ secrets.K8S_SECRET_KEYSTATIC_GITHUB_CLIENT_SECRET }}" + "KEYSTATIC_SECRET=${{ secrets.K8S_SECRET_KEYSTATIC_SECRET }}" + cache-from: type=gha + cache-to: type=gha,mode=max + + deploy: + name: Deploy docker image + needs: [env, vars, build] + uses: acdh-oeaw/gl-autodevops-minimal-port/.github/workflows/deploy.yml@main + secrets: inherit + with: + environment: "${{ needs.env.outputs.environment }}" + DOCKER_TAG: "${{ needs.env.outputs.registry }}/${{ needs.env.outputs.image }}" + APP_NAME: "${{ needs.env.outputs.app_name }}" + APP_ROOT: "/" + SERVICE_ID: "${{ vars.SERVICE_ID }}" + PUBLIC_URL: "${{ needs.vars.outputs.public_url }}" + default_port: "3000" diff --git a/.github/workflows/validate.yml b/.github/workflows/validate.yml new file mode 100644 index 0000000..3c6f4cd --- /dev/null +++ b/.github/workflows/validate.yml @@ -0,0 +1,102 @@ +name: Validate + +concurrency: + group: "${{ github.workflow }}-${{ github.ref }}-validate" + cancel-in-progress: true + +on: + pull_request: + branches: + - main + push: + branches: + - main + +jobs: + validate: + name: Validate + runs-on: ${{ matrix.os }} + timeout-minutes: 60 + + strategy: + fail-fast: true + matrix: + node-version: [20.x] + os: [ubuntu-latest] + + steps: + - name: Checkout repository + uses: actions/checkout@v4 + + # Necessary because `actions/setup-node` does not yet support `corepack`. + # @see https://github.com/actions/setup-node/issues/531 + - name: Install pnpm + uses: pnpm/action-setup@v4 + + - name: Use Node.js ${{ matrix.node-version }} + uses: actions/setup-node@v4 + with: + node-version: ${{ matrix.node-version }} + cache: "pnpm" + + - name: Install dependencies + run: pnpm install --frozen-lockfile + + - name: Format + run: pnpm run format:check + + - name: Lint + run: pnpm run lint:check + + - name: Typecheck + run: pnpm run types:check + + - name: Test + run: pnpm run test + + - name: Get playwright version + run: | + PLAYWRIGHT_VERSION=$(pnpm ls @playwright/test --json | jq --raw-output '.[0].devDependencies["@playwright/test"].version') + echo "PLAYWRIGHT_VERSION=$PLAYWRIGHT_VERSION" >> $GITHUB_ENV + + - name: Cache playwright browsers + uses: actions/cache@v4 + id: cache-playwright-browsers + with: + path: "~/.cache/ms-playwright" + key: "${{ matrix.os }}-playwright-browsers-${{ env.PLAYWRIGHT_VERSION }}" + + - name: Install playwright browsers + if: steps.cache-playwright-browsers.outputs.cache-hit != 'true' + run: pnpm exec playwright install --with-deps + - name: Install playwright browsers (operating system dependencies) + if: steps.cache-playwright-browsers.outputs.cache-hit == 'true' + run: pnpm exec playwright install-deps + + - name: Build app + run: pnpm run build + env: + PUBLIC_APP_BASE_URL: "http://localhost:3000" + PUBLIC_REDMINE_ID: "${{ vars.SERVICE_ID }}" + + - name: Run e2e tests + run: pnpm run test:e2e + env: + PUBLIC_APP_BASE_URL: "http://localhost:3000" + + - uses: actions/upload-artifact@v4 + if: always() + with: + name: playwright-report + path: playwright-report/ + retention-days: 30 + + build-deploy: + if: ${{ github.event_name == 'push' }} + needs: [validate] + uses: ./.github/workflows/build-deploy.yml + secrets: inherit + # https://docs.github.com/en/actions/using-workflows/reusing-workflows#access-and-permissions + permissions: + contents: read + packages: write diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..622b980 --- /dev/null +++ b/.gitignore @@ -0,0 +1,35 @@ +# dependencies +node_modules/ +.pnpm-store/ + +# logs +*.log + +# non-public environment variables +.env.local +.env.*.local + +# caches +.eslintcache +.stylelintcache +*.tsbuildinfo + +# vercel +.vercel + +# misc +.DS_Store +.idea/ + +# astro +dist/ +.astro/ + +# test +/coverage/ + +# playwright +/blob-report/ +/playwright/.cache/ +/playwright-report/ +/test-results/ diff --git a/.npmrc b/.npmrc new file mode 100644 index 0000000..e284682 --- /dev/null +++ b/.npmrc @@ -0,0 +1,4 @@ +engine-strict=true +package-manager-strict=false +shell-emulator=true +use-node-version=20.14.0 diff --git a/.vscode/extensions.json b/.vscode/extensions.json new file mode 100644 index 0000000..4f00a02 --- /dev/null +++ b/.vscode/extensions.json @@ -0,0 +1,13 @@ +{ + "recommendations": [ + "astro-build.astro-vscode", + "bradlc.vscode-tailwindcss", + "dbaeumer.vscode-eslint", + "editorconfig.editorconfig", + "esbenp.prettier-vscode", + "mikestead.dotenv", + "ms-playwright.playwright", + "stylelint.vscode-stylelint", + "unifiedjs.vscode-mdx" + ] +} diff --git a/.vscode/launch.json b/.vscode/launch.json new file mode 100644 index 0000000..98a64ed --- /dev/null +++ b/.vscode/launch.json @@ -0,0 +1,24 @@ +{ + "configurations": [ + { + "name": "Apps: Web (server-side)", + "request": "launch", + "runtimeArgs": ["run", "dev"], + "runtimeExecutable": "pnpm", + "skipFiles": ["/**"], + "type": "node" + }, + { + "name": "Apps: Web (client-side)", + "type": "chrome", + "request": "launch", + "url": "http://localhost:3000" + } + ], + "compounds": [ + { + "name": "Apps: Web", + "configurations": ["Apps: Web (server-side)", "Apps: Web (client-side)"] + } + ] +} diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 0000000..9967b32 --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,54 @@ +{ + "css.validate": false, + "editor.codeActionsOnSave": { + "source.fixAll": "explicit" + }, + "editor.defaultFormatter": "esbenp.prettier-vscode", + "editor.formatOnSave": true, + "editor.inlayHints.enabled": "offUnlessPressed", + "editor.linkedEditing": true, + "editor.quickSuggestions": { + "strings": true + }, + "editor.rulers": [100], + "editor.stickyScroll.enabled": true, + "eslint.enable": true, + "eslint.options": { + "ignorePath": ".gitignore" + }, + "eslint.validate": ["astro", "javascript", "typescript", "typescriptreact"], + "files.associations": { + "*.css": "tailwindcss" + }, + "files.eol": "\n", + "html.autoCreateQuotes": false, + "less.validate": false, + "mdx.server.enable": true, + "prettier.documentSelectors": ["**/*.astro"], + "prettier.ignorePath": ".gitignore", + "scss.validate": false, + "stylelint.enable": true, + "stylelint.snippet": ["astro", "css", "postcss", "tailwindcss"], + "stylelint.validate": ["astro", "css", "postcss", "tailwindcss"], + "tailwindCSS.classAttributes": ["class", "class:list"], + "tailwindCSS.experimental.classRegex": [ + ["cn\\(([^)]*)\\)", "(?:'|\"|`)([^']*)(?:'|\"|`)"], + ["variants\\(([^)]*)\\)", "[\"'`]([^\"'`]*).*?[\"'`]"] + ], + "tailwindCSS.validate": true, + "terminal.integrated.enablePersistentSessions": false, + "typescript.enablePromptUseWorkspaceTsdk": true, + "typescript.inlayHints.parameterNames.enabled": "all", + "typescript.preferences.importModuleSpecifier": "non-relative", + "typescript.preferences.preferTypeOnlyAutoImports": true, + "typescript.tsdk": "node_modules/typescript/lib", + "workbench.editor.labelFormat": "medium", + "workbench.tree.enableStickyScroll": true, + "[markdown]": { + "editor.wordWrap": "on" + }, + "[mdx]": { + "editor.formatOnSave": false, + "editor.wordWrap": "on" + } +} diff --git a/Dockerfile b/Dockerfile new file mode 100644 index 0000000..723d0e6 --- /dev/null +++ b/Dockerfile @@ -0,0 +1,73 @@ +# syntax=docker/dockerfile:1 + +# using alpine base image to avoid `sharp` memory leaks. +# @see https://sharp.pixelplumbing.com/install#linux-memory-allocator + +# base +FROM node:20-alpine AS base + +RUN corepack enable + +RUN mkdir /app && chown -R node:node /app +WORKDIR /app + +USER node + +COPY --chown=node:node .npmrc package.json pnpm-lock.yaml ./ +RUN sed -i "s/use-node-version/# use-node-version/" .npmrc + +RUN pnpm fetch --prod +RUN pnpm install --frozen-lockfile --ignore-scripts --offline --prod + +# build +FROM base as build + +RUN pnpm fetch --dev + +COPY --chown=node:node ./ ./ +RUN sed -i "s/use-node-version/# use-node-version/" .npmrc + +ARG PUBLIC_APP_BASE_PATH +ARG PUBLIC_APP_BASE_URL +ARG PUBLIC_BOTS +ARG PUBLIC_KEYSTATIC_GITHUB_APP_SLUG +ARG PUBLIC_KEYSTATIC_GITHUB_REPO_NAME +ARG PUBLIC_KEYSTATIC_GITHUB_REPO_OWNER +ARG PUBLIC_KEYSTATIC_MODE +ARG PUBLIC_MATOMO_BASE_URL +ARG PUBLIC_MATOMO_ID +ARG PUBLIC_REDMINE_ID + +# disable validation for runtime environment variables +ENV ENV_VALIDATION=public + +RUN pnpm install --frozen-lockfile --offline + +ENV NODE_ENV=production + +RUN --mount=type=secret,id=KEYSTATIC_GITHUB_CLIENT_ID,uid=1000 \ + --mount=type=secret,id=KEYSTATIC_GITHUB_CLIENT_SECRET,uid=1000 \ + --mount=type=secret,id=KEYSTATIC_SECRET,uid=1000 \ + KEYSTATIC_GITHUB_CLIENT_ID=$(cat /run/secrets/KEYSTATIC_GITHUB_CLIENT_ID) \ + KEYSTATIC_GITHUB_CLIENT_SECRET=$(cat /run/secrets/KEYSTATIC_GITHUB_CLIENT_SECRET) \ + KEYSTATIC_SECRET=$(cat /run/secrets/KEYSTATIC_SECRET) \ + pnpm run build + +# serve +FROM node:20-alpine AS serve + +RUN mkdir /app && chown -R node:node /app +WORKDIR /app + +USER node + +COPY --from=base --chown=node:node /app/node_modules ./node_modules +COPY --from=build --chown=node:node /app/dist ./ + +ENV NODE_ENV=production +ENV HOST=0.0.0.0 +ENV PORT=3000 + +EXPOSE 3000 + +CMD [ "node", "./server/entry.mjs" ] diff --git a/Dockerfile.static b/Dockerfile.static new file mode 100644 index 0000000..1472598 --- /dev/null +++ b/Dockerfile.static @@ -0,0 +1,53 @@ +# syntax=docker/dockerfile:1 + +# using alpine base image to avoid `sharp` memory leaks. +# @see https://sharp.pixelplumbing.com/install#linux-memory-allocator + +# build +FROM node:20-alpine AS build + +RUN corepack enable + +RUN mkdir /app && chown -R node:node /app +WORKDIR /app + +USER node + +COPY --chown=node:node .npmrc package.json pnpm-lock.yaml ./ +RUN sed -i "s/use-node-version/# use-node-version/" .npmrc + +RUN pnpm fetch + +COPY --chown=node:node ./ ./ +RUN sed -i "s/use-node-version/# use-node-version/" .npmrc + +ARG PUBLIC_APP_BASE_PATH +ARG PUBLIC_APP_BASE_URL +ARG PUBLIC_BOTS +ARG PUBLIC_KEYSTATIC_GITHUB_APP_SLUG +ARG PUBLIC_KEYSTATIC_GITHUB_REPO_NAME +ARG PUBLIC_KEYSTATIC_GITHUB_REPO_OWNER +ARG PUBLIC_KEYSTATIC_MODE +ARG PUBLIC_MATOMO_BASE_URL +ARG PUBLIC_MATOMO_ID +ARG PUBLIC_REDMINE_ID + +# disable validation for runtime environment variables +ENV ENV_VALIDATION=public + +ENV NODE_ENV=production + +RUN pnpm install --frozen-lockfile --offline + +RUN pnpm run build + +# serve +FROM caddy:2-alpine AS serve + +WORKDIR /usr/share/caddy + +COPY --from=build /app/dist /usr/share/caddy + +EXPOSE 3000 + +CMD ["caddy", "file-server", "--listen", ":3000"] diff --git a/astro.config.ts b/astro.config.ts new file mode 100644 index 0000000..10f8c50 --- /dev/null +++ b/astro.config.ts @@ -0,0 +1,89 @@ +import mdx from "@astrojs/mdx"; +import node from "@astrojs/node"; +import react from "@astrojs/react"; +import sitemap from "@astrojs/sitemap"; +import keystatic from "@keystatic/astro"; +import { defineConfig } from "astro/config"; +import icon from "astro-icon"; +import { loadEnv } from "vite"; + +// import { defaultLocale } from "./src/config/i18n.config"; +// import { createConfig as createMdxConfig } from "./src/config/mdx.config"; + +const env = loadEnv(import.meta.env.MODE, process.cwd(), ""); + +export default defineConfig({ + /** + * When switching to static site generation, place an empty `index.astro` file in + * the `src/pages` folder, so `astro` will generate a redirect to the default locale + * via ``. + */ + adapter: node({ + mode: "standalone", + }), + base: env.PUBLIC_APP_BASE_PATH, + experimental: { + contentCollectionCache: true, + }, + integrations: [ + icon({ + /** @see https://www.astroicon.dev/reference/configuration/#include */ + include: { + lucide: [ + "chevron-down", + "chevron-left", + "chevron-right", + "circle-pause", + "circle-play", + "menu", + "quote", + "search", + "x", + ], + }, + svgoOptions: { + multipass: true, + plugins: [ + { + name: "preset-default", + params: { + overrides: { + removeViewBox: false, + }, + }, + }, + ], + }, + }), + keystatic(), + mdx(), + react(), + sitemap(), + ], + // // @ts-expect-error Astro types are incomplete. + // markdown: { + // ...(await createMdxConfig(defaultLocale)), + // gfm: false, + // smartypants: false, + // syntaxHighlight: false, + // }, + output: "hybrid", + prefetch: { + defaultStrategy: "hover", + prefetchAll: true, + }, + redirects: { + "/admin": { + destination: "/keystatic", + status: 307, + }, + }, + scopedStyleStrategy: "where", + security: { + checkOrigin: true, + }, + server: { + port: 3000, + }, + site: env.PUBLIC_APP_BASE_URL, +}); diff --git a/content/index-page/index.json b/content/index-page/index.json new file mode 100644 index 0000000..e7dbc74 --- /dev/null +++ b/content/index-page/index.json @@ -0,0 +1,87 @@ +{ + "hero": { + "slides": [ + { + "title": "Jubiläumsjahr 2024", + "summary": "Veranstaltungen zum 200. Geburtstag Anton Bruckners", + "alignment": "right", + "image": "/assets/content/index-page/hero/slides/0/image.jpg", + "page": "lexikon" + }, + { + "title": "Digitales Werkverzeichnis Anton Bruckner", + "summary": "Datenbank mit Informationen zu Werken Anton Bruckners und zu weiteren Quellen aus dem musikalischen Nachlass", + "alignment": "left", + "image": "/assets/content/index-page/hero/slides/1/image.jpg", + "page": "werkverzeichnis/wab" + }, + { + "title": "Anton Bruckner-Lexikon", + "summary": "Mit Informationen zu Leben und Werk des Komponisten", + "alignment": "right", + "image": "/assets/content/index-page/hero/slides/2/image.jpg", + "page": "lexikon" + }, + { + "title": "Musik-Handschriften ", + "summary": "31.000 Farbdigitalisate in hoher Qualität von über 700 Autographen und frühen Abschriften Bruckners (zugänglich über das Werkverzeichnis)", + "alignment": "left", + "image": "/assets/content/index-page/hero/slides/3/image.jpg", + "page": "werkverzeichnis/wab" + }, + { + "title": "Musik-Drucke ", + "summary": "5500 Farbdigitalisate aller zur Verfügung stehenden Erstausgaben und der Alten Gesamtausgabe", + "alignment": "right", + "image": "/assets/content/index-page/hero/slides/4/image.jpg", + "page": "drucke/erstdrucke" + }, + { + "title": "Biographie", + "summary": "Zusammenstellung der wichtigsten Stationen in Bruckners Leben und zeitliche Einordung der Werke", + "alignment": "left", + "image": "/assets/content/index-page/hero/slides/5/image.jpg", + "page": "leben/kindheit-und-jugend" + }, + { + "title": "Kitzler-Studienbuch", + "summary": "Digitale Edition und automatisierte Harmonieanalyse auf Grundlage MEI-codierter Notation.", + "alignment": "left", + "image": "/assets/content/index-page/hero/slides/6/image.jpg", + "page": "forschung/kitzler-studienbuch" + }, + { + "title": "Bibliographie", + "summary": "Datenbank mit 9749 Literaturzitaten", + "alignment": "left", + "image": "/assets/content/index-page/hero/slides/7/image.jpg", + "page": "leben/bibliographie" + }, + { + "title": "Anton Bruckner Chronologie Datenbank", + "summary": "von Franz Scheder", + "alignment": "right", + "image": "/assets/content/index-page/hero/slides/8/image.jpg", + "page": "forschung/chronologie-datenbank" + } + ] + }, + "quote": "Anton Bruckner-Jubiläumsjahr 2024 an der Akademie der Wissenschaften: Symposion (10.-12.04). Jubiläumskonzert im historischen Festsaal (10.04.). Bruckner-Fest im Arkadenhof der Akademie (07.09.). ", + "links": [ + { + "title": "Werkverzeichnis", + "summary": "Datenbank mit Informationen zu Werken Anton Bruckners und zu weiteren Quellen aus dem musikalischen Nachlass", + "page": "werkverzeichnis/wab" + }, + { + "title": "ABLO", + "summary": "Anton Bruckner-Lexikon online mit Informationen zu Leben und Werk des Komponisten", + "page": "lexikon" + }, + { + "title": "ABCD", + "summary": "Anton Bruckner Chronologie Datenbank der biographisch erfassten Informationen zu Leben und Werk des Komponisten", + "page": "forschung/chronologie-datenbank" + } + ] +} diff --git a/content/metadata.json b/content/metadata.json new file mode 100644 index 0000000..ed8e3c2 --- /dev/null +++ b/content/metadata.json @@ -0,0 +1,5 @@ +{ + "title": "Bruckner Online", + "shortTitle": "Bruckner Online", + "description": "bruckner-online.at ist ein umfangreich angelegtes Bruckner-Internet-Portal (Webarchiv), in dem neben der elektronischen Dokumentation sämtlicher hand­schriftlicher Quellen auch alle Kompositionen, relevante Personen und Orte enthalten sind. Zudem werden von allen Quellen, Erstdrucken und der Alten Gesamtausgabe vollständige Digitalisate zur Verfügung gestellt." +} diff --git a/content/navigation.json b/content/navigation.json new file mode 100644 index 0000000..3a94f6f --- /dev/null +++ b/content/navigation.json @@ -0,0 +1,304 @@ +{ + "links": [ + { + "discriminant": "menu", + "value": { + "label": "Projekt", + "links": [ + { + "discriminant": "page", + "value": { + "label": "Mitarbeiter", + "reference": "projekt/mitarbeiter" + } + }, + { + "discriminant": "page", + "value": { + "label": "Beschreibung", + "reference": "projekt/projektbeschreibung" + } + }, + { + "discriminant": "page", + "value": { + "label": "Bruckner-Forschung (ÖAW)", + "reference": "projekt/bruckner-forschung-oeaw" + } + } + ] + } + }, + { + "discriminant": "menu", + "value": { + "label": "Bruckner", + "links": [ + { + "discriminant": "page", + "value": { + "label": "Leben: Kindheit und Jugend (1824-1845)", + "reference": "leben/kindheit-und-jugend" + } + }, + { + "discriminant": "page", + "value": { + "label": "Leben: St. Florian (1845-1855)", + "reference": "leben/st-florian" + } + }, + { + "discriminant": "page", + "value": { + "label": "Leben: Linz (1856-1868)", + "reference": "leben/linz" + } + }, + { + "discriminant": "page", + "value": { + "label": "Leben: Wien I (1868-1877)", + "reference": "leben/wien-i" + } + }, + { + "discriminant": "page", + "value": { + "label": "Leben: Wien II (1878-1889)", + "reference": "leben/wien-ii" + } + }, + { + "discriminant": "page", + "value": { + "label": "Leben: Wien III (1890-1896)", + "reference": "leben/wien-iii" + } + }, + { + "discriminant": "page", + "value": { + "label": "Bibliographie", + "reference": "leben/bibliographie" + } + } + ] + } + }, + { + "discriminant": "menu", + "value": { + "label": "Werkverzeichnis", + "links": [ + { + "discriminant": "page", + "value": { + "label": "Werkverzeichnis", + "reference": "werkverzeichnis/wab" + } + }, + { + "discriminant": "page", + "value": { + "label": "Gedrucktes Werkverzeichnis", + "reference": "werkverzeichnis/gedrucktes-werkverzeichnis" + } + }, + { + "discriminant": "page", + "value": { + "label": "Werkübersicht", + "reference": "werkverzeichnis/werkuebersicht" + } + }, + { + "discriminant": "page", + "value": { + "label": "Werksystematik", + "reference": "werkverzeichnis/werksystematik" + } + }, + { + "discriminant": "page", + "value": { + "label": "WAB-Konkordanz", + "reference": "werkverzeichnis/wab-konkordanz" + } + }, + { + "discriminant": "page", + "value": { + "label": "Werkeinträge / Quellenbeschreibungen", + "reference": "werkverzeichnis/werkeintraege-quellenbeschreibungen" + } + }, + { + "discriminant": "page", + "value": { + "label": "Aufbewahrungsorte", + "reference": "werkverzeichnis/aufbewahrungsorte" + } + }, + { + "discriminant": "page", + "value": { + "label": "Abkürzungen", + "reference": "werkverzeichnis/abkuerzungen" + } + } + ] + } + }, + { + "discriminant": "menu", + "value": { + "label": "Drucke", + "links": [ + { + "discriminant": "page", + "value": { + "label": "Erstdrucke", + "reference": "drucke/erstdrucke" + } + }, + { + "discriminant": "page", + "value": { + "label": "Digitalisate der Erstdrucke", + "reference": "drucke/erstdrucke-digitalisate" + } + }, + { + "discriminant": "page", + "value": { + "label": "Anton Bruckner Sämtliche Werke (AGA)", + "reference": "drucke/alte-gesamtausgabe" + } + }, + { + "discriminant": "page", + "value": { + "label": "Anton Bruckner Gesamtausgabe (NGA)", + "reference": "drucke/gesamtausgabe" + } + }, + { + "discriminant": "link", + "value": { + "label": "Neue Anton Bruckner Gesamtausgabe", + "href": "https://www.mwv.at/content/seite/NeueAntonBrucknerGesamtausgabeab2016/" + } + }, + { + "discriminant": "link", + "value": { + "label": "Anton Bruckner Urtext Gesamtausgabe", + "href": "https://hermann.eu/home/label/BEW" + } + } + ] + } + }, + { + "discriminant": "menu", + "value": { + "label": "Forschung", + "links": [ + { + "discriminant": "page", + "value": { + "label": "Kitzler-Studienbuch", + "reference": "forschung/kitzler-studienbuch" + } + }, + { + "discriminant": "page", + "value": { + "label": "Chronologie-Datenbank", + "reference": "forschung/chronologie-datenbank" + } + }, + { + "discriminant": "page", + "value": { + "label": "Transkriptionstool Hanschrift Bruckners", + "reference": "forschung/transkriptionstool" + } + }, + { + "discriminant": "page", + "value": { + "label": "Anton Bruckner Kopisten", + "reference": "forschung/kopisten" + } + }, + { + "discriminant": "page", + "value": { + "label": "Bruckner Forschung (ÖAW)", + "reference": "projekt/bruckner-forschung-oeaw" + } + }, + { + "discriminant": "page", + "value": { + "label": "Bruckner Forschung International", + "reference": "forschung/international" + } + }, + { + "discriminant": "page", + "value": { + "label": "Bruckner Gesellschaft", + "reference": "forschung/bruckner-gesellschaft" + } + }, + { + "discriminant": "page", + "value": { + "label": "Geschichte Dom-Musikarchiv Linz", + "reference": "forschung/geschichte-dom-musikarchiv-linz" + } + }, + { + "discriminant": "page", + "value": { + "label": "Weitere Links", + "reference": "forschung/links" + } + } + ] + } + }, + { + "discriminant": "page", + "value": { + "label": "Lexikon", + "reference": "lexikon" + } + }, + { + "discriminant": "menu", + "value": { + "label": "Audio", + "links": [ + { + "discriminant": "page", + "value": { + "label": "Symphonien", + "reference": "audio/symphonien" + } + }, + { + "discriminant": "page", + "value": { + "label": "Weitere Audiobeispiele", + "reference": "audio/beispiele" + } + } + ] + } + } + ] +} diff --git a/content/pages/anton-bruckner-2024.mdx b/content/pages/anton-bruckner-2024.mdx new file mode 100644 index 0000000..979a123 --- /dev/null +++ b/content/pages/anton-bruckner-2024.mdx @@ -0,0 +1,68 @@ +--- +title: anton bruckner 2024 +--- +**Aktivitäten im Jubiläumsjahr** + +Am 4. September 2024 jährt sich zum 200. Mal der Geburtstag des bedeutenden österreichischen Komponisten Anton Bruckner (1824-1896). Dieses Jubiläum nimmt die Österreichische Akademie der Wissenschaften zum Anlass für verschiedene Veranstaltungen. + +**10.-12. April 2024** + +**Internationales Symposion** **der Akademie der Wissenschaften**\ +Bruckner-Aspekte: 200 Jahre Bruckner – 100 Jahre Bruckner-Forschung. + +**Mittwoch, 10. April 2024** – **19 Uhr** + +**Jubiläumskonzert im historischen Festsaal der Akademie der Wissenschaften** + +Zwei Fanfaren über Werke Anton Bruckners von Vinzenz Goller. Ausführende: Blechbläser:innen-Kammermusikensemble *BrassAdventure* der Universität für Musik und darstellende Kunst Wien.\ +Symphonie Nr. 7 E-Dur in einer Bearbeitung für groß besetztes Kammerensemble von Hanns Eisler, Erwin Stein und Karl Rankl.\ +Ausführende: Musiker:innen des Bruckner Orchester Linz. + + + +**Samstag, 7. September 2024** + +**09.30 Uhr**: **Festgottesdienst in der Jesuitenkirche Wien** + +Messe in e-Moll, Ausführende: „Chorvereinigung St. Augustin“ unter der Leitung von Andreas Pixner\ +Zelebranten: Seine Eminenz Kardinal Christoph Schönborn, Erzbischof von Wien und Propst Johann Holzinger von St. Florian, Generalabt der Augustiner-Chorherren. + +Präsentation des neuen, gedruckten [*Werkverzeichnis Anton Bruckner*](http://www.bruckner-online.at/?page_id=2660) + +**11-18 Uhr: Campusfestival im Arkadenhof der Akademie der Wissenschaften** + +Grätzlfest für die Wiener Bevölkerung mit Bewirtung, Musik- und Kinderprogramm + +*Vorläufiges Programm*\ +11.00-12.00 Uhr [Musikverein Kremsmünster](https://www.mvkremsmuenster.at/)\ +12.00-12.30 Uhr [Bachl Chor](https://www.bachlchor.at/)\ +12.30-13.30 Uhr [Musikverein Kremsmünster](https://www.mvkremsmuenster.at/)\ +13.30-14.00 Uhr [Andorfer Chöre](https://andorferchoere.at/)\ +14.00-15.00 Uhr [Musikverein St. Florian](https://www.mv-stflorian.at/index.html)\ +15.00-15.30 Uhr [Camerata Garstina](https://www.chvooe.at/chvooe/mitgliedschoere/5051-camerata-garstina)\ +15.30-16.00 Uhr [Musikverein St. Florian](https://www.mv-stflorian.at/index.html)\ +16.00-16.30 Uhr Chor der ÖAW\ +16.30-18.00 Uhr [Blaskapelle der Boku Wien](https://blaskapelle.boku.ac.at/) + +Die Veranstaltung findet in Kooperation mit dem **Remasuri Straßenfest** in der Wollzeile (Wien) statt. + +--- + +#### Donnerstag, 21. November 2024 – 18 Uhr + +**Leibniz Lecture im Rahmen der Akademievorlesungen** + +Prof.em. Dr. Jürgen Stolzenberg, Institut für Philosophie der Universität Halle:\ +Zur Religion in Bruckners Neunter Symphonie\ +\ +Österreichische Akademie der Wissenschaften, Seminarraum am Arkadenhof + +--- + +### **anton bruckner 2024 in Oberösterreich** + +Oberösterreich war Anton Bruckners Heimat. In ­Ansfelden wurde er 1824 geboren, im Stift des Nachbarortes St. Florian musikalisch sozialisiert, gefördert und ausgebildet. In Linz entwickelte er sich zum umjubelten Orgelimprovisator und zu einem ­Komponisten mit unverwechselbarer Tonsprache. Hier erklingt seine Musik heute im renommierten Konzerthaus, das seinen Namen trägt, hier spielt sie das nach ihm benannte Orchester und hier lehrt und erforscht sie eine Universität, deren Namensgeber er ist. + +Sein runder Geburtstag ist willkommener Anlass, gemeinsam oberösterreichische Kultur in ihrer ganzen Vielfalt zu präsentieren, was in zahlreichen Projekten rund um Bruckner seinen Ausdruck finden wird, auf Plätzen und an Orten, in Dörfern und Städten, überall in Oberösterreich. + +[www.anton-bruckner-2024.at](https://www.anton-bruckner-2024.at/) diff --git a/content/pages/audio/beispiele.mdx b/content/pages/audio/beispiele.mdx new file mode 100644 index 0000000..7a51ada --- /dev/null +++ b/content/pages/audio/beispiele.mdx @@ -0,0 +1,20 @@ +--- +title: Auswahl weiterer Audio- und Musikvideobeispiele +--- +## Fanfaren über Themen Anton Bruckners von Vinzenz Goller + +