1 Commits

Author SHA1 Message Date
Starystars67 d80dc4dea0 Update index.html 2025-12-20 02:22:46 +00:00
43 changed files with 1044 additions and 2374 deletions
+13 -16
View File
@@ -1,34 +1,31 @@
name: Build Docker image and push to release
on:
release:
types: [published]
push:
# Sequence of patterns matched against refs/tags
tags:
- 'v*' # Push events to matching v*, i.e. v1.0, v20.15.10
jobs:
docker:
if: "!github.event.release.prerelease"
runs-on: ubuntu-latest
steps:
- name: Connect to Tailscale
uses: tailscale/github-action@v4
with:
oauth-client-id: ${{ secrets.TS_OAUTH_CLIENT_ID }}
oauth-secret: ${{ secrets.TS_OAUTH_SECRET }}
tags: tag:ci
- name: Set up QEMU
-
name: Set up QEMU
uses: docker/setup-qemu-action@v3
- name: Set up Docker Buildx
-
name: Set up Docker Buildx
uses: docker/setup-buildx-action@v3
with:
driver: docker
- name: Login to Docker Registry
-
name: Login to Docker Registry
uses: docker/login-action@v3
with:
registry: ${{ secrets.REGISTRY_URL }}
username: ${{ secrets.REGISTRY_USERNAME }}
password: ${{ secrets.REGISTRY_PASSWORD }}
- name: Build and push
-
name: Build and push
uses: docker/build-push-action@v5
with:
push: true
tags: ${{ secrets.REGISTRY_URL }}/beammp/website:${{ github.REF_NAME }}, ${{ secrets.REGISTRY_URL }}/beammp/website:latest, ${{ secrets.REGISTRY_URL }}/beammp/website:production
tags: registry.beammp.com/beammp/website:${{ github.ref_name }}
+13 -17
View File
@@ -1,34 +1,30 @@
name: Build Docker image and push to prerelease
name: Build Docker image and push to staging
on:
release:
types: [published]
push:
branches:
- "dev"
jobs:
docker:
if: github.event.release.prerelease
runs-on: ubuntu-latest
steps:
- name: Connect to Tailscale
uses: tailscale/github-action@v4
with:
oauth-client-id: ${{ secrets.TS_OAUTH_CLIENT_ID }}
oauth-secret: ${{ secrets.TS_OAUTH_SECRET }}
tags: tag:ci
- name: Set up QEMU
-
name: Set up QEMU
uses: docker/setup-qemu-action@v3
- name: Set up Docker Buildx
-
name: Set up Docker Buildx
uses: docker/setup-buildx-action@v3
with:
driver: docker
- name: Login to Docker Registry
-
name: Login to Docker Registry
uses: docker/login-action@v3
with:
registry: ${{ secrets.REGISTRY_URL }}
username: ${{ secrets.REGISTRY_USERNAME }}
password: ${{ secrets.REGISTRY_PASSWORD }}
- name: Build and push
-
name: Build and push
uses: docker/build-push-action@v5
with:
push: true
tags: ${{ secrets.REGISTRY_URL }}/beammp/website:${{ github.REF_NAME }}, ${{ secrets.REGISTRY_URL }}/beammp/website:latest
tags: registry.beammp.com/beammp/website:staging
+1 -1
View File
@@ -102,6 +102,7 @@ dist
# TernJS port file
.tern-port
*.exe
*.zip
# Logs
@@ -128,4 +129,3 @@ frontend/*.ntvs*
frontend/*.njsproj
frontend/*.sln
frontend/*.sw?
.DS_Store
-27
View File
@@ -1,27 +0,0 @@
{
"cSpell.words": [
"beammp",
"beammpservers",
"beamng",
"Deutsch",
"Español",
"Français",
"freeroam",
"gridlines",
"Italiano",
"Lucide",
"maxplayers",
"modlist",
"modstotal",
"modstotalsize",
"Offroad",
"playerslist",
"reka",
"rels",
"roleplay",
"sdesc",
"sname",
"vueuse",
"Русский"
]
}
+4 -11
View File
@@ -1,29 +1,22 @@
# Step 1: Build stage
FROM node:lts-alpine AS build
FROM node:22-alpine3.21 AS build
WORKDIR /app
COPY package*.json ./
RUN npm ci
COPY . .
ENV NODE_ENV=production
ENV NODE_ENV=development
RUN npm run build
# Step 2: Serve stage
FROM nginx:stable
# Remove default nginx static assets
RUN rm -rf /usr/share/nginx/html/*
FROM nginx:alpine
# Copy built files from the previous stage
COPY --from=build /app/dist /usr/share/nginx/html
# Copy secure nginx configs
COPY nginx.main.conf /etc/nginx/nginx.conf
# Add a custom Nginx configuration
COPY nginx.conf /etc/nginx/conf.d/default.conf
# Use non-root user for security
USER nginx
# Expose port 80
EXPOSE 80
+33 -16
View File
@@ -1,6 +1,6 @@
# BeamMP Website
# BeamMP Website (Frontend)
This repository is home for the BeamMP website. The website features support for translations too now!
This repository is home for the BeamMP website, this is a rebuild of the website from the ground up using Vue+Vite. We are also making use of Tailwindcss v4.
**Tech Stack**
- **Framework:** Vue 3 (`vue`, `vue-router`)
@@ -59,6 +59,27 @@ Vite will print the local dev URL (usually `http://localhost:5173`).
- `vite.config.js` Vite configuration
- `components.json` Shadcn-Vue component registry
## Styling and UI
- **Tailwind CSS 4** is set up via `tailwind.config.js` and imported in `src/style.css`.
- **reka-ui** and shadcn-vue-style patterns are used for accessible, composable UI.
- **lucide-vue-next** provides icons.
## Adding New UI Components (shadcn-vue)
Were standardizing on shadcn-vue-compatible components for consistency. Use the CLI to add new components:
```powershell
npx shadcn-vue@latest add <component>
# examples
npx shadcn-vue@latest add button
npx shadcn-vue@latest add navigation-menu
```
The CLI reads `components.json` and will scaffold files under `src/components/ui/`.
## Routing
Use `vue-router` for navigation. Keep routes co-located with views and prefer lazy-loaded routes for large pages.
## Contributing
We welcome contributions! Heres how to get started.
@@ -94,19 +115,15 @@ We welcome contributions! Heres how to get started.
- Open a PR describing the problem, solution, screenshots if UI changes, and any follow-ups.
- Link related issues. Keep PRs small; big changes should be split.
## Environment & Configuration
- Tailwind and Vite are preconfigured. If you need globals, add them in `vite.config.js`.
- For icons, use `lucide-vue-next` and keep icon size consistent via props/classes.
- If adding new pages, prefer code-splitting with dynamic imports.
# Translations
BeamMP makes an effort to be maintained for multiple languages.
The current progress of this sits at:
[![gitlocalized ](https://gitlocalize.com/repo/10617/whole_project/badge.svg)](https://gitlocalize.com/repo/10617?utm_source=badge)
We use [GitLocalize](https://gitlocalize.com/) for managing this. You can contribute if you wish here: https://gitlocalize.com/repo/10617.
## FAQ
- “Why Vite?” Fast dev server, optimized builds, and great Vue tooling.
- “Can I use Yarn or pnpm?” Yes—adjust commands accordingly.
- “Design system?” We favor shadcn-vue patterns + Tailwind + reka-ui primitives for consistent UI.
The individual language progress is as follows:
| Language | Badge |
|-----------------------|---------------------------------------------------------------------------------------------------------------------------------|
| French | [![gitlocalized ](https://gitlocalize.com/repo/10617/fr/badge.svg)](https://gitlocalize.com/repo/10617/fr?utm_source=badge) |
| German | [![gitlocalized ](https://gitlocalize.com/repo/10617/de/badge.svg)](https://gitlocalize.com/repo/10617/de?utm_source=badge) |
| Italian | [![gitlocalized ](https://gitlocalize.com/repo/10617/it/badge.svg)](https://gitlocalize.com/repo/10617/it?utm_source=badge) |
| Russian | [![gitlocalized ](https://gitlocalize.com/repo/10617/ru/badge.svg)](https://gitlocalize.com/repo/10617/ru?utm_source=badge) |
| Spanish | [![gitlocalized ](https://gitlocalize.com/repo/10617/es/badge.svg)](https://gitlocalize.com/repo/10617/es?utm_source=badge) |
## License
Unless otherwise noted in the root repository, this project follows the BeamMP websites standard license. If clarifications are needed, open an issue and we will update this section.
-8
View File
@@ -1,8 +0,0 @@
services:
website:
build:
dockerfile: Dockerfile
container_name: website
restart: unless-stopped
ports:
- "127.0.0.1:3000:80"
+422 -597
View File
File diff suppressed because it is too large Load Diff
+10 -35
View File
@@ -2,46 +2,21 @@ server {
listen 80;
server_name _;
server_tokens off;
root /usr/share/nginx/html;
index index.html;
# Baseline security hardening for a static SPA.
# NOTE: add_header directives are NOT inherited by child location blocks that
# define their own add_header. To avoid silently dropping security headers,
# use the `expires` directive (not add_header Cache-Control) in location blocks.
add_header X-Frame-Options "DENY" always;
add_header X-Content-Type-Options "nosniff" always;
add_header Referrer-Policy "strict-origin-when-cross-origin" always;
add_header Permissions-Policy "camera=(), microphone=(), geolocation=()" always;
add_header Content-Security-Policy "default-src 'self'; img-src 'self' data: https:; script-src 'self'; style-src 'self' 'unsafe-inline'; font-src 'self' data:; connect-src 'self' https://backend.beammp.com; object-src 'none'; base-uri 'self'; frame-ancestors 'none'" always;
# Installer directory: serve files directly; return 404 for anything missing.
location ^~ /installer/ {
try_files $uri =404;
}
# Vite-hashed assets (JS, CSS, fonts, images): serve directly with a long cache.
# These files have content hashes in their names so stale cache is never an issue.
# Return 404 if a file is genuinely missing rather than silently serving index.html.
location ~* \.(js|css|woff2?|ttf|eot|svg|webp|avif|png|jpg|jpeg|gif|ico|map)$ {
try_files $uri =404;
expires 1y;
access_log off;
}
# Data / download files: serve directly, no client-side cache, 404 if missing.
location ~* \.(json|txt|exe|zip)$ {
try_files $uri =404;
expires -1;
access_log off;
}
# SPA fallback: all other paths go to index.html so Vue Router can render the
# correct view (including the NotFound page for unknown routes).
location / {
try_files $uri /index.html;
# SPA fallback: serve index.html for non-file routes so Vue Router can render NotFound
try_files $uri $uri/ /index.html;
}
# Let real 404s for assets return 404s; Vue handles route-level 404 via the SPA fallback above.
location /static/ {
# Serve static files directly
expires max;
access_log off;
}
}
-27
View File
@@ -1,27 +0,0 @@
worker_processes auto;
pid /tmp/nginx.pid;
error_log /tmp/error.log warn;
events {
worker_connections 1024;
}
http {
include /etc/nginx/mime.types;
default_type application/octet-stream;
access_log /tmp/access.log;
sendfile on;
keepalive_timeout 65;
server_tokens off;
# Redefine temp paths to writable tmpfs locations
client_body_temp_path /tmp/client_temp;
proxy_temp_path /tmp/proxy_temp;
fastcgi_temp_path /tmp/fastcgi_temp;
uwsgi_temp_path /tmp/uwsgi_temp;
scgi_temp_path /tmp/scgi_temp;
include /etc/nginx/conf.d/*.conf;
}
+227 -326
View File
File diff suppressed because it is too large Load Diff
+2 -2
View File
@@ -1,7 +1,7 @@
{
"name": "beammp-website",
"private": true,
"version": "2.4.20",
"version": "2.1.0",
"type": "module",
"scripts": {
"dev": "vite",
@@ -16,7 +16,7 @@
"class-variance-authority": "^0.7.1",
"clsx": "^2.1.1",
"lucide-vue-next": "^0.555.0",
"reka-ui": "^2.7.0",
"reka-ui": "^2.6.0",
"tailwind-merge": "^3.4.0",
"tailwindcss": "^4.1.17",
"tailwindcss-animate": "^1.0.7",
Binary file not shown.

Before

Width:  |  Height:  |  Size: 5.1 MiB

Binary file not shown.
-86
View File
@@ -1,86 +0,0 @@
[
{
"name": "Horizon Hosting",
"website": "https://hrzn.link/beammp",
"from": "$3.34/mo",
"logo": "https://hrznhosting.com/assets/logo.svg"
},
{
"name": "RackGenius",
"website": "https://rackgeni.us/beammp-plans",
"from": "$0.5/mo",
"logo": "https://rackgenius.com/rackgenius-logo.png"
},
{
"name": "Connect Hosting",
"website": "https://connecthosting.net/beammp",
"from": "$1.49/mo",
"logo": "https://connecthosting.net/img/logo.webp"
},
{
"name": "Assetto Hosting",
"website": "https://assettohosting.com/games/beamng",
"from": "$2.30/mo",
"logo": "https://assettohosting.com/_next/image?url=https%3A%2F%2Fstrapi.assettohosting.com%2Fuploads%2Flogo_2228c8bbfb.png&w=640&q=100"
},
{
"name": "Zap-Hosting",
"website": "https://zap-hosting.com/itsbeammp",
"from": "$6.46/mo",
"logo": "https://wormhole.ifyouwantmorepower.com/online/assets/beammp/zap-logo.png"
},
{
"name": "HostHavoc",
"website": "https://hosthavoc.com/game-servers/beammp",
"from": "$3.75/mo",
"logo": "https://hosthavoc.com/images/logo.svg"
},
{
"name": "PedalHost",
"website": "https://pedal.host",
"from": "$1.31/mo",
"logo": "https://pedal.host/pedalhost_horizontal_light.svg"
},
{
"name": "Vyper Hosting",
"website": "https://vyperhosting.com/r/beammp",
"from": "$2.68/mo",
"logo": "https://vyperhosting.com/assets/logo.png"
},
{
"name": "BisecHosting",
"website": "https://bisecthosting.com/beammp-server-hosting",
"from": "$5.99/mo",
"logo": "https://www.bisecthosting.com/_ipx/q_100&s_140x46/images/logo-dark-theme.svg"
},
{
"name": "Four Season Hosting",
"website": "https://fourseasonshosting.com",
"from": "$3.00/mo",
"logo": "https://fourseasonshosting.com/_next/image?url=%2Flogo.png&w=256&q=75"
},
{
"name": "Vertuo Hosting",
"website": "https://vertuohosting.com",
"from": "$1.99/mo",
"logo": "https://vertuohosting.com/assets/img/logo.png"
},
{
"name": "Winheberg",
"website": "https://winheberg.fr/offres/gaming/beammp",
"from": "$2.99/mo",
"logo": "https://winheberg.fr/img/logo-white.png"
},
{
"name": "Wabbanode",
"website": "https://wabbanode.com/partner/beammp",
"from": "$5.96/mo",
"logo": "https://wabbanode.com/_nuxt/wabba_logo.UPNIVeXa.webp"
},
{
"name": "Iceline Hosting",
"website": "https://iceline-hosting.com/games/beammp",
"from": "$3.64/mo",
"logo": "https://iceline-hosting.com/_astro/logo.DTnWV5zL_157TDr.webp"
}
]
Binary file not shown.

Before

Width:  |  Height:  |  Size: 157 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 16 MiB

+42 -50
View File
@@ -1,80 +1,72 @@
<script setup>
import { Github, Facebook, Instagram, createLucideIcon } from 'lucide-vue-next'
import { getLocalizedPath } from '@/utils/locale'
import { RouterLink, useRoute } from 'vue-router'
import { RouterLink } from 'vue-router'
const route = useRoute()
// Generate localized route
function localRoute(path) {
return getLocalizedPath(path, route.params.locale)
}
const XIcon = createLucideIcon('X', [
const XIcon = createLucideIcon("X", [
[
'path',
"path",
{
d: 'M18.901 1.153h3.68l-8.04 9.19L24 22.846h-7.406l-5.8-7.584-6.638 7.584H.474l8.6-9.83L0 1.154h7.594l5.243 6.932ZM17.61 20.644h2.039L6.486 3.24H4.298Z',
stroke: 'none',
fill: 'currentColor',
d: "M18.901 1.153h3.68l-8.04 9.19L24 22.846h-7.406l-5.8-7.584-6.638 7.584H.474l8.6-9.83L0 1.154h7.594l5.243 6.932ZM17.61 20.644h2.039L6.486 3.24H4.298Z",
stroke: "none",
fill: "currentColor",
},
],
])
]);
const BlueskyIcon = createLucideIcon('Bluesky', [
const BlueskyIcon = createLucideIcon("Bluesky", [
[
'path',
"path",
{
d: 'M5.202 2.857C7.954 4.922 10.913 9.11 12 11.358c1.087-2.247 4.046-6.436 6.798-8.501C20.783 1.366 24 .213 24 3.883c0 .732-.42 6.156-.667 7.037-.856 3.061-3.978 3.842-6.755 3.37 4.854.826 6.089 3.562 3.422 6.299-5.065 5.196-7.28-1.304-7.847-2.97-.104-.305-.152-.448-.153-.327 0-.121-.05.022-.153.327-.568 1.666-2.782 8.166-7.847 2.97-2.667-2.737-1.432-5.473 3.422-6.3-2.777.473-5.899-.308-6.755-3.369C.42 10.04 0 4.615 0 3.883c0-3.67 3.217-2.517 5.202-1.026',
stroke: 'none',
fill: 'currentColor',
d: "M5.202 2.857C7.954 4.922 10.913 9.11 12 11.358c1.087-2.247 4.046-6.436 6.798-8.501C20.783 1.366 24 .213 24 3.883c0 .732-.42 6.156-.667 7.037-.856 3.061-3.978 3.842-6.755 3.37 4.854.826 6.089 3.562 3.422 6.299-5.065 5.196-7.28-1.304-7.847-2.97-.104-.305-.152-.448-.153-.327 0-.121-.05.022-.153.327-.568 1.666-2.782 8.166-7.847 2.97-2.667-2.737-1.432-5.473 3.422-6.3-2.777.473-5.899-.308-6.755-3.369C.42 10.04 0 4.615 0 3.883c0-3.67 3.217-2.517 5.202-1.026",
stroke: "none",
fill: "currentColor",
},
],
])
]);
const RedditIcon = createLucideIcon('X', [
const RedditIcon = createLucideIcon("X", [
[
'path',
"path",
{
d: 'M12 0A12 12 0 0 0 0 12a12 12 0 0 0 12 12 12 12 0 0 0 12-12A12 12 0 0 0 12 0zm5.01 4.744c.688 0 1.25.561 1.25 1.249a1.25 1.25 0 0 1-2.498.056l-2.597-.547-.8 3.747c1.824.07 3.48.632 4.674 1.488.308-.309.73-.491 1.207-.491.968 0 1.754.786 1.754 1.754 0 .716-.435 1.333-1.01 1.614a3.111 3.111 0 0 1 .042.52c0 2.694-3.13 4.87-7.004 4.87-3.874 0-7.004-2.176-7.004-4.87 0-.183.015-.366.043-.534A1.748 1.748 0 0 1 4.028 12c0-.968.786-1.754 1.754-1.754.463 0 .898.196 1.207.49 1.207-.883 2.878-1.43 4.744-1.487l.885-4.182a.342.342 0 0 1 .14-.197.35.35 0 0 1 .238-.042l2.906.617a1.214 1.214 0 0 1 1.108-.701zM9.25 12C8.561 12 8 12.562 8 13.25c0 .687.561 1.248 1.25 1.248.687 0 1.248-.561 1.248-1.249 0-.688-.561-1.249-1.249-1.249zm5.5 0c-.687 0-1.248.561-1.248 1.25 0 .687.561 1.248 1.249 1.248.688 0 1.249-.561 1.249-1.249 0-.687-.562-1.249-1.25-1.249zm-5.466 3.99a.327.327 0 0 0-.231.094.33.33 0 0 0 0 .463c.842.842 2.484.913 2.961.913.477 0 2.105-.056 2.961-.913a.361.361 0 0 0 .029-.463.33.33 0 0 0-.464 0c-.547.533-1.684.73-2.512.73-.828 0-1.979-.196-2.512-.73a.326.326 0 0 0-.232-.095z',
stroke: 'none',
fill: 'currentColor',
d: "M12 0A12 12 0 0 0 0 12a12 12 0 0 0 12 12 12 12 0 0 0 12-12A12 12 0 0 0 12 0zm5.01 4.744c.688 0 1.25.561 1.25 1.249a1.25 1.25 0 0 1-2.498.056l-2.597-.547-.8 3.747c1.824.07 3.48.632 4.674 1.488.308-.309.73-.491 1.207-.491.968 0 1.754.786 1.754 1.754 0 .716-.435 1.333-1.01 1.614a3.111 3.111 0 0 1 .042.52c0 2.694-3.13 4.87-7.004 4.87-3.874 0-7.004-2.176-7.004-4.87 0-.183.015-.366.043-.534A1.748 1.748 0 0 1 4.028 12c0-.968.786-1.754 1.754-1.754.463 0 .898.196 1.207.49 1.207-.883 2.878-1.43 4.744-1.487l.885-4.182a.342.342 0 0 1 .14-.197.35.35 0 0 1 .238-.042l2.906.617a1.214 1.214 0 0 1 1.108-.701zM9.25 12C8.561 12 8 12.562 8 13.25c0 .687.561 1.248 1.25 1.248.687 0 1.248-.561 1.248-1.249 0-.688-.561-1.249-1.249-1.249zm5.5 0c-.687 0-1.248.561-1.248 1.25 0 .687.561 1.248 1.249 1.248.688 0 1.249-.561 1.249-1.249 0-.687-.562-1.249-1.25-1.249zm-5.466 3.99a.327.327 0 0 0-.231.094.33.33 0 0 0 0 .463c.842.842 2.484.913 2.961.913.477 0 2.105-.056 2.961-.913a.361.361 0 0 0 .029-.463.33.33 0 0 0-.464 0c-.547.533-1.684.73-2.512.73-.828 0-1.979-.196-2.512-.73a.326.326 0 0 0-.232-.095z",
stroke: "none",
fill: "currentColor",
},
],
])
]);
const DiscordIcon = createLucideIcon('Discord', [
const DiscordIcon = createLucideIcon("Discord", [
[
'path',
"path",
{
d: 'M20.317 4.37a19.791 19.791 0 0 0-4.885-1.515.074.074 0 0 0-.079.037c-.21.375-.444.864-.608 1.25a18.27 18.27 0 0 0-5.487 0 12.64 12.64 0 0 0-.617-1.25.077.077 0 0 0-.079-.037A19.736 19.736 0 0 0 3.677 4.37a.07.07 0 0 0-.032.027C.533 9.046-.32 13.58.099 18.057a.082.082 0 0 0 .031.057 19.9 19.9 0 0 0 5.993 3.03.078.078 0 0 0 .084-.028 14.09 14.09 0 0 0 1.226-1.994.076.076 0 0 0-.041-.106 13.107 13.107 0 0 1-1.872-.892.077.077 0 0 1-.008-.128 10.2 10.2 0 0 0 .372-.292.074.074 0 0 1 .077-.01c3.928 1.793 8.18 1.793 12.062 0a.074.074 0 0 1 .078.01c.12.098.246.198.373.292a.077.077 0 0 1-.006.127 12.299 12.299 0 0 1-1.873.892.077.077 0 0 0-.041.107c.36.698.772 1.362 1.225 1.993a.076.076 0 0 0 .084.028 19.839 19.839 0 0 0 6.002-3.03.077.077 0 0 0 .032-.054c.5-5.177-.838-9.674-3.549-13.66a.061.061 0 0 0-.031-.03zM8.02 15.33c-1.183 0-2.157-1.085-2.157-2.419 0-1.333.956-2.419 2.157-2.419 1.21 0 2.176 1.096 2.157 2.42 0 1.333-.956 2.418-2.157 2.418zm7.975 0c-1.183 0-2.157-1.085-2.157-2.419 0-1.333.955-2.419 2.157-2.419 1.21 0 2.176 1.096 2.157 2.42 0 1.333-.946 2.418-2.157 2.418z',
stroke: 'none',
fill: 'currentColor',
d: "M20.317 4.37a19.791 19.791 0 0 0-4.885-1.515.074.074 0 0 0-.079.037c-.21.375-.444.864-.608 1.25a18.27 18.27 0 0 0-5.487 0 12.64 12.64 0 0 0-.617-1.25.077.077 0 0 0-.079-.037A19.736 19.736 0 0 0 3.677 4.37a.07.07 0 0 0-.032.027C.533 9.046-.32 13.58.099 18.057a.082.082 0 0 0 .031.057 19.9 19.9 0 0 0 5.993 3.03.078.078 0 0 0 .084-.028 14.09 14.09 0 0 0 1.226-1.994.076.076 0 0 0-.041-.106 13.107 13.107 0 0 1-1.872-.892.077.077 0 0 1-.008-.128 10.2 10.2 0 0 0 .372-.292.074.074 0 0 1 .077-.01c3.928 1.793 8.18 1.793 12.062 0a.074.074 0 0 1 .078.01c.12.098.246.198.373.292a.077.077 0 0 1-.006.127 12.299 12.299 0 0 1-1.873.892.077.077 0 0 0-.041.107c.36.698.772 1.362 1.225 1.993a.076.076 0 0 0 .084.028 19.839 19.839 0 0 0 6.002-3.03.077.077 0 0 0 .032-.054c.5-5.177-.838-9.674-3.549-13.66a.061.061 0 0 0-.031-.03zM8.02 15.33c-1.183 0-2.157-1.085-2.157-2.419 0-1.333.956-2.419 2.157-2.419 1.21 0 2.176 1.096 2.157 2.42 0 1.333-.956 2.418-2.157 2.418zm7.975 0c-1.183 0-2.157-1.085-2.157-2.419 0-1.333.955-2.419 2.157-2.419 1.21 0 2.176 1.096 2.157 2.42 0 1.333-.946 2.418-2.157 2.418z",
stroke: "none",
fill: "currentColor",
},
],
])
]);
const YouTubeIcon = createLucideIcon('YouTube', [
const YouTubeIcon = createLucideIcon("YouTube", [
[
'path',
"path",
{
d: 'M23.498 6.186a3.016 3.016 0 0 0-2.122-2.136C19.505 3.545 12 3.545 12 3.545s-7.505 0-9.377.505A3.017 3.017 0 0 0 .502 6.186C0 8.07 0 12 0 12s0 3.93.502 5.814a3.016 3.016 0 0 0 2.122 2.136c1.871.505 9.376.505 9.376.505s7.505 0 9.377-.505a3.015 3.015 0 0 0 2.122-2.136C24 15.93 24 12 24 12s0-3.93-.502-5.814zM9.545 15.568V8.432L15.818 12l-6.273 3.568z',
stroke: 'none',
fill: 'currentColor',
d: "M23.498 6.186a3.016 3.016 0 0 0-2.122-2.136C19.505 3.545 12 3.545 12 3.545s-7.505 0-9.377.505A3.017 3.017 0 0 0 .502 6.186C0 8.07 0 12 0 12s0 3.93.502 5.814a3.016 3.016 0 0 0 2.122 2.136c1.871.505 9.376.505 9.376.505s7.505 0 9.377-.505a3.015 3.015 0 0 0 2.122-2.136C24 15.93 24 12 24 12s0-3.93-.502-5.814zM9.545 15.568V8.432L15.818 12l-6.273 3.568z",
stroke: "none",
fill: "currentColor",
},
],
])
]);
const TikTokIcon = createLucideIcon('TikTok', [
const TikTokIcon = createLucideIcon("TikTok", [
[
'path',
"path",
{
d: 'M12.525.02c1.31-.02 2.61-.01 3.91-.02.08 1.53.63 3.09 1.75 4.17 1.12 1.11 2.7 1.62 4.24 1.79v4.03c-1.44-.05-2.89-.35-4.2-.97-.57-.26-1.1-.59-1.62-.93-.01 2.92.01 5.84-.02 8.75-.08 1.4-.54 2.79-1.35 3.94-1.31 1.92-3.58 3.17-5.91 3.21-1.43.08-2.86-.31-4.08-1.03-2.02-1.19-3.44-3.37-3.65-5.71-.02-.5-.03-1-.01-1.49.18-1.9 1.12-3.72 2.58-4.96 1.66-1.44 3.98-2.13 6.15-1.72.02 1.48-.04 2.96-.04 4.44-.99-.32-2.15-.23-3.02.37-.63.41-1.11 1.04-1.36 1.75-.21.51-.15 1.07-.14 1.61.24 1.64 1.82 3.02 3.5 2.87 1.12-.01 2.19-.66 2.77-1.61.19-.33.4-.67.41-1.06.1-1.79.06-3.57.07-5.36.01-4.03-.01-8.05.02-12.07z',
stroke: 'none',
fill: 'currentColor',
d: "M12.525.02c1.31-.02 2.61-.01 3.91-.02.08 1.53.63 3.09 1.75 4.17 1.12 1.11 2.7 1.62 4.24 1.79v4.03c-1.44-.05-2.89-.35-4.2-.97-.57-.26-1.1-.59-1.62-.93-.01 2.92.01 5.84-.02 8.75-.08 1.4-.54 2.79-1.35 3.94-1.31 1.92-3.58 3.17-5.91 3.21-1.43.08-2.86-.31-4.08-1.03-2.02-1.19-3.44-3.37-3.65-5.71-.02-.5-.03-1-.01-1.49.18-1.9 1.12-3.72 2.58-4.96 1.66-1.44 3.98-2.13 6.15-1.72.02 1.48-.04 2.96-.04 4.44-.99-.32-2.15-.23-3.02.37-.63.41-1.11 1.04-1.36 1.75-.21.51-.15 1.07-.14 1.61.24 1.64 1.82 3.02 3.5 2.87 1.12-.01 2.19-.66 2.77-1.61.19-.33.4-.67.41-1.06.1-1.79.06-3.57.07-5.36.01-4.03-.01-8.05.02-12.07z",
stroke: "none",
fill: "currentColor",
},
],
])
]);
</script>
<template>
@@ -180,7 +172,7 @@ const TikTokIcon = createLucideIcon('TikTok', [
rel="noopener noreferrer"
class="text-xs text-orange-600 hover:text-orange-700 underline-offset-2 hover:underline dark:text-orange-400 dark:hover:text-orange-300"
>
{{ $t('message.footer.support_on_patreon') }}
{{ $t("message.footer.support_on_patreon") }}
</a>
</div>
@@ -191,10 +183,10 @@ const TikTokIcon = createLucideIcon('TikTok', [
<p>&copy; 2019 - {{ new Date().getFullYear() }} | BeamMP Mod Team All Rights Reserved</p>
<div class="flex gap-3">
<RouterLink
:to="localRoute('about')"
to="/about"
class="text-neutral-700 hover:text-beammp-blue transition-colors dark:text-neutral-500 dark:hover:text-blue-400"
>
{{ $t('message.footer.about') }}
{{ $t("message.footer.about") }}
</RouterLink>
<span>&middot;</span>
<a
@@ -203,7 +195,7 @@ const TikTokIcon = createLucideIcon('TikTok', [
rel="noopener noreferrer"
class="text-neutral-700 hover:text-beammp-blue transition-colors dark:text-neutral-500 dark:hover:text-blue-400"
>
{{ $t('message.footer.privacy_policy') }}
{{ $t("message.footer.privacy_policy") }}
</a>
<span>&middot;</span>
<a
@@ -212,7 +204,7 @@ const TikTokIcon = createLucideIcon('TikTok', [
rel="noopener noreferrer"
class="text-neutral-700 hover:text-beammp-blue transition-colors dark:text-neutral-500 dark:hover:text-blue-400"
>
{{ $t('message.footer.terms_conditions') }}
{{ $t("message.footer.terms_conditions") }}
</a>
</div>
</div>
+61 -153
View File
@@ -1,6 +1,6 @@
<script setup>
import { ref } from 'vue'
import { RouterLink, useRoute } from 'vue-router'
import { RouterLink } from 'vue-router'
import {
NavigationMenu,
NavigationMenuItem,
@@ -12,10 +12,8 @@ import { cn } from '@/lib/utils'
import ThemeToggle from '@/components/ThemeToggle.vue'
import LanguageSelector from '@/components/LanguageSelector.vue'
import { Menu, X } from 'lucide-vue-next'
import { getLocalizedPath } from '@/utils/locale'
const mobileMenuOpen = ref(false)
const route = useRoute()
function toggleMobileMenu() {
mobileMenuOpen.value = !mobileMenuOpen.value
@@ -24,23 +22,12 @@ function toggleMobileMenu() {
function closeMobileMenu() {
mobileMenuOpen.value = false
}
// Generate localized route
function localRoute(path) {
return getLocalizedPath(path, route.params.locale)
}
</script>
<template>
<header
class="border-b border-neutral-200 dark:border-neutral-800 bg-white/80 dark:bg-neutral-900/70 backdrop-blur sticky top-0 z-20"
>
<header class="border-b border-neutral-200 dark:border-neutral-800 bg-white/80 dark:bg-neutral-900/70 backdrop-blur sticky top-0 z-20">
<nav class="max-w-6xl mx-auto px-4 h-16 flex items-center justify-between">
<RouterLink
:to="localRoute('')"
class="flex items-center gap-2 shrink-0 mr-4"
@click="closeMobileMenu"
>
<RouterLink to="/" class="flex items-center gap-2 shrink-0" @click="closeMobileMenu">
<!-- Light mode logo (black) -->
<img
src="/src/assets/BeamMP_blk.png"
@@ -57,100 +44,28 @@ function localRoute(path) {
<!-- Desktop Navigation -->
<!-- Switch to mobile earlier (avoid logo crowding) -->
<div class="hidden xl:flex items-center gap-4">
<div class="hidden lg:flex items-center gap-4">
<NavigationMenu>
<NavigationMenuList>
<!--<NavigationMenuItem>
<NavigationMenuLink as-child>
<RouterLink
:to="localRoute('communities')"
:class="
cn(
navigationMenuTriggerStyle(),
'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white'
)
"
>
{{ $t('message.nav.communities') }}
</RouterLink>
</NavigationMenuLink>
</NavigationMenuItem>-->
<NavigationMenuItem>
<NavigationMenuLink as-child>
<RouterLink
:to="localRoute('partners')"
:class="
cn(
navigationMenuTriggerStyle(),
'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white'
)
"
>
{{ $t('message.nav.partners') }}
</RouterLink>
</NavigationMenuLink>
</NavigationMenuItem>
<!--<NavigationMenuItem>
<NavigationMenuLink as-child>
<RouterLink
:to="localRoute('servers')"
:class="
cn(
navigationMenuTriggerStyle(),
'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white'
)
"
>
{{ $t('message.nav.servers') }}
</RouterLink>
</NavigationMenuLink>
</NavigationMenuItem>-->
<!--<NavigationMenuItem>
<NavigationMenuLink as-child>
<RouterLink
:to="localRoute('stats')"
:class="
cn(
navigationMenuTriggerStyle(),
'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white'
)
"
>
{{ $t('message.nav.statistics') }}
</RouterLink>
</NavigationMenuLink>
</NavigationMenuItem>-->
<!--<NavigationMenuItem>
<NavigationMenuLink as-child>
<a
href="https://forum.beammp.com"
target="_blank"
rel="noopener noreferrer"
aria-label="External link"
:class="
cn(
navigationMenuTriggerStyle(),
'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white'
)
"
:class="cn(navigationMenuTriggerStyle(), 'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white')"
>
{{ $t('message.nav.forums') }}
</a>
</NavigationMenuLink>
</NavigationMenuItem>-->
</NavigationMenuItem>
<NavigationMenuItem>
<NavigationMenuLink as-child>
<a
href="https://docs.beammp.com"
target="_blank"
rel="noopener noreferrer"
aria-label="External link"
:class="
cn(
navigationMenuTriggerStyle(),
'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white'
)
"
:class="cn(navigationMenuTriggerStyle(), 'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white')"
>
{{ $t('message.nav.docs') }}
</a>
@@ -158,20 +73,32 @@ function localRoute(path) {
</NavigationMenuItem>
<NavigationMenuItem>
<NavigationMenuLink as-child>
<a
href="https://store.beammp.com"
target="_blank"
rel="noopener noreferrer"
aria-label="External link"
:class="
cn(
navigationMenuTriggerStyle(),
'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white'
)
"
<RouterLink
to="/communities"
:class="cn(navigationMenuTriggerStyle(), 'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white')"
>
{{ $t('message.nav.store') }}
</a>
{{ $t('message.nav.communities') }}
</RouterLink>
</NavigationMenuLink>
</NavigationMenuItem>
<NavigationMenuItem>
<NavigationMenuLink as-child>
<RouterLink
to="/servers"
:class="cn(navigationMenuTriggerStyle(), 'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white')"
>
{{ $t('message.nav.servers') }}
</RouterLink>
</NavigationMenuLink>
</NavigationMenuItem>
<NavigationMenuItem>
<NavigationMenuLink as-child>
<RouterLink
to="/stats"
:class="cn(navigationMenuTriggerStyle(), 'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white')"
>
{{ $t('message.nav.statistics') }}
</RouterLink>
</NavigationMenuLink>
</NavigationMenuItem>
<NavigationMenuItem>
@@ -180,13 +107,7 @@ function localRoute(path) {
href="https://github.com/BeamMP/BeamMP"
target="_blank"
rel="noopener noreferrer"
aria-label="External link"
:class="
cn(
navigationMenuTriggerStyle(),
'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white'
)
"
:class="cn(navigationMenuTriggerStyle(), 'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white')"
>
{{ $t('message.nav.github') }}
</a>
@@ -198,13 +119,7 @@ function localRoute(path) {
href="https://www.patreon.com/BeamMP"
target="_blank"
rel="noopener noreferrer"
aria-label="External link"
:class="
cn(
navigationMenuTriggerStyle(),
'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white'
)
"
:class="cn(navigationMenuTriggerStyle(), 'bg-transparent text-neutral-900 hover:bg-neutral-100 hover:text-neutral-900 dark:bg-transparent dark:text-white dark:hover:bg-neutral-800 dark:hover:text-white')"
>
{{ $t('message.nav.patreon') }}
</a>
@@ -212,20 +127,20 @@ function localRoute(path) {
</NavigationMenuItem>
</NavigationMenuList>
</NavigationMenu>
<div class="flex items-center gap-1 sm:gap-2 flex-shrink-0">
<div class="flex items-center gap-2">
<LanguageSelector />
<ThemeToggle />
</div>
</div>
<!-- Mobile Menu Button and Theme Toggle -->
<div class="flex xl:hidden items-center gap-2">
<div class="flex lg:hidden items-center gap-2">
<LanguageSelector />
<ThemeToggle />
<button
@click="toggleMobileMenu"
class="p-2 text-neutral-900 dark:text-white hover:bg-neutral-100 dark:hover:bg-neutral-800 rounded-md transition-colors"
aria-label="Toggle menu"
@click="toggleMobileMenu"
>
<Menu v-if="!mobileMenuOpen" class="w-6 h-6" />
<X v-else class="w-6 h-6" />
@@ -244,38 +159,10 @@ function localRoute(path) {
>
<div
v-if="mobileMenuOpen"
class="xl:hidden border-t border-neutral-200 dark:border-neutral-800 bg-white dark:bg-neutral-900"
class="md:hidden border-t border-neutral-200 dark:border-neutral-800 bg-white dark:bg-neutral-900"
>
<div class="px-4 py-3 space-y-1">
<!--<RouterLink
:to="localRoute('communities')"
class="block px-4 py-3 text-neutral-900 dark:text-white hover:bg-neutral-100 dark:hover:bg-neutral-800 rounded-md transition-colors"
@click="closeMobileMenu"
>
{{ $t('message.nav.communities') }}
</RouterLink>-->
<RouterLink
:to="localRoute('partners')"
class="block px-4 py-3 text-neutral-900 dark:text-white hover:bg-neutral-100 dark:hover:bg-neutral-800 rounded-md transition-colors"
@click="closeMobileMenu"
>
{{ $t('message.nav.partners') }}
</RouterLink>
<RouterLink
:to="localRoute('servers')"
class="block px-4 py-3 text-neutral-900 dark:text-white hover:bg-neutral-100 dark:hover:bg-neutral-800 rounded-md transition-colors"
@click="closeMobileMenu"
>
{{ $t('message.nav.servers') }}
</RouterLink>
<!--<RouterLink
:to="localRoute('stats')"
class="block px-4 py-3 text-neutral-900 dark:text-white hover:bg-neutral-100 dark:hover:bg-neutral-800 rounded-md transition-colors"
@click="closeMobileMenu"
>
{{ $t('message.nav.statistics') }}
</RouterLink>-->
<!--<a
<a
href="https://forum.beammp.com"
target="_blank"
rel="noopener noreferrer"
@@ -283,7 +170,7 @@ function localRoute(path) {
@click="closeMobileMenu"
>
{{ $t('message.nav.forums') }}
</a>-->
</a>
<a
href="https://docs.beammp.com"
target="_blank"
@@ -293,6 +180,27 @@ function localRoute(path) {
>
{{ $t('message.nav.docs') }}
</a>
<RouterLink
to="/communities"
class="block px-4 py-3 text-neutral-900 dark:text-white hover:bg-neutral-100 dark:hover:bg-neutral-800 rounded-md transition-colors"
@click="closeMobileMenu"
>
{{ $t('message.nav.communities') }}
</RouterLink>
<RouterLink
to="/servers"
class="block px-4 py-3 text-neutral-900 dark:text-white hover:bg-neutral-100 dark:hover:bg-neutral-800 rounded-md transition-colors"
@click="closeMobileMenu"
>
{{ $t('message.nav.servers') }}
</RouterLink>
<RouterLink
to="/stats"
class="block px-4 py-3 text-neutral-900 dark:text-white hover:bg-neutral-100 dark:hover:bg-neutral-800 rounded-md transition-colors"
@click="closeMobileMenu"
>
{{ $t('message.nav.statistics') }}
</RouterLink>
<a
href="https://github.com/BeamMP/BeamMP"
target="_blank"
+4 -18
View File
@@ -1,14 +1,10 @@
<script setup>
import { ref, onMounted } from 'vue'
import { useI18n } from 'vue-i18n'
import { useRouter, useRoute } from 'vue-router'
import { LANGUAGES, loadLocaleMessages } from '@/i18n'
import { switchLocale, getCurrentLocale } from '@/utils/locale'
import { ChevronDown } from 'lucide-vue-next'
const { locale } = useI18n()
const router = useRouter()
const route = useRoute()
const isOpen = ref(false)
const currentLanguage = () => {
@@ -26,24 +22,14 @@ const selectLanguage = async (langCode) => {
locale.value = langCode
localStorage.setItem('lang', langCode)
isOpen.value = false
// Navigate to the new locale route
const currentPath = route.fullPath
const newPath = switchLocale(langCode, currentPath)
router.push(newPath)
}
const toggleDropdown = () => {
isOpen.value = !isOpen.value
}
const getFlagUrl = (flagName) => {
//return new URL(`../assets/flags/${flagName}.png`, import.meta.url).href
return `/flags/${flagName}.png`
}
onMounted(async () => {
const saved = localStorage.getItem('lang') || getCurrentLocale()
const saved = localStorage.getItem('lang')
if (saved && saved !== locale.value) {
try {
await loadLocaleMessages(window.i18n, saved)
@@ -59,13 +45,13 @@ onMounted(async () => {
<div class="relative">
<button
class="flex items-center gap-2 px-3 py-2 rounded-lg bg-neutral-200 dark:bg-neutral-800/50 hover:bg-neutral-300 dark:hover:bg-neutral-800 transition-colors text-neutral-900 dark:text-white text-sm font-medium"
style="height: 40px"
style="height: 40px;"
:aria-expanded="isOpen"
:aria-label="$t('message.nav.language')"
@click="toggleDropdown"
>
<img
:src="getFlagUrl(currentLanguage().flag)"
:src="`/flags/${currentLanguage().flag}.png`"
:alt="currentLanguage().name"
class="w-5 h-4 rounded-sm"
/>
@@ -90,7 +76,7 @@ onMounted(async () => {
]"
@click="selectLanguage(code)"
>
<img :src="getFlagUrl(lang.flag)" :alt="lang.name" class="w-5 h-4 rounded-sm" />
<img :src="`/flags/${lang.flag}.png`" :alt="lang.name" class="w-5 h-4 rounded-sm" />
<span>{{ lang.name }}</span>
<span v-if="locale === code" class="ml-auto text-neutral-500"></span>
</button>
+2 -4
View File
@@ -1,6 +1,6 @@
<script setup>
import { ref, onMounted } from 'vue'
import { Sun, Moon, Monitor, Smartphone, Tablet } from 'lucide-vue-next'
import { Sun, Moon, Monitor } from 'lucide-vue-next'
const theme = ref('system')
@@ -60,9 +60,7 @@ onMounted(() => {
:title="$t('message.theme.system')"
@click="setTheme('system')"
>
<Smartphone class="w-4 h-4 md:hidden" />
<Tablet class="w-4 h-4 hidden md:block lg:hidden" />
<Monitor class="w-4 h-4 hidden lg:block" />
<Monitor class="w-4 h-4" />
</button>
<button
:class="[
-42
View File
@@ -1,42 +0,0 @@
<script setup>
import { reactiveOmit } from "@vueuse/core";
import { Check } from "lucide-vue-next";
import { CheckboxIndicator, CheckboxRoot, useForwardPropsEmits } from "reka-ui";
import { cn } from "@/lib/utils";
const props = defineProps({
defaultValue: { type: [Boolean, String], required: false },
modelValue: { type: [Boolean, String, null], required: false },
disabled: { type: Boolean, required: false },
value: { type: null, required: false },
id: { type: String, required: false },
asChild: { type: Boolean, required: false },
as: { type: null, required: false },
name: { type: String, required: false },
required: { type: Boolean, required: false },
class: { type: null, required: false },
});
const emits = defineEmits(["update:modelValue"]);
const delegatedProps = reactiveOmit(props, "class");
const forwarded = useForwardPropsEmits(delegatedProps, emits);
</script>
<template>
<CheckboxRoot
v-bind="forwarded"
:class="
cn(
'grid place-content-center peer h-4 w-4 shrink-0 rounded-sm border border-primary shadow focus-visible:outline-none focus-visible:ring-1 focus-visible:ring-ring disabled:cursor-not-allowed disabled:opacity-50 data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground',
props.class,
)
"
>
<CheckboxIndicator class="grid place-content-center text-current">
<slot>
<Check class="h-4 w-4" />
</slot>
</CheckboxIndicator>
</CheckboxRoot>
</template>
-1
View File
@@ -1 +0,0 @@
export { default as Checkbox } from "./Checkbox.vue";
+1 -2
View File
@@ -1,7 +1,7 @@
import { nextTick } from 'vue'
import { createI18n } from 'vue-i18n'
export const SUPPORT_LOCALES = ['en', 'es', 'fr', 'de', 'it', 'ru', 'zh']
export const SUPPORT_LOCALES = ['en', 'es', 'fr', 'de', 'it', 'ru']
export const LANGUAGES = {
en: { name: 'English', code: 'en', flag: 'gb' },
@@ -10,7 +10,6 @@ export const LANGUAGES = {
de: { name: 'Deutsch', code: 'de', flag: 'de' },
it: { name: 'Italiano', code: 'it', flag: 'it' },
ru: { name: 'Русский', code: 'ru', flag: 'ru' },
zh: { name: '中文', code: 'zh', flag: 'cn' },
}
export function setupI18n(options = { locale: 'en' }) {
+3 -18
View File
@@ -10,7 +10,6 @@
"forums": "Forum",
"docs": "Docs",
"communities": "Communities",
"partners": "Partners",
"servers": "Server",
"statistics": "Statistiken",
"github": "GitHub",
@@ -46,7 +45,7 @@
"public_servers": "Öffentliche Server",
"all_servers": "Alle Server"
},
"why_beammp": "Warum BeamMP?",
"why_choose_beammp": "Warum BeamMP?",
"features": {
"stable_servers": {
"title": "Stabile Server",
@@ -62,7 +61,7 @@
},
"sync": {
"title": "Synchronisationsqualität",
"description": "BeamMP aktualisiert die Position deines Fahrzeugs ~50 Mal pro Sekunde für ein flüssiges Erlebnis."
"description": "BeamMP aktualisiert die Position deines Fahrzeugs ~100 Mal pro Sekunde für ein flüssiges Erlebnis."
}
},
"communities": {
@@ -118,14 +117,7 @@
"title": "Bereit, deinen eigenen Server zu hosten?",
"description": "Lade die Serverdateien herunter und erschaffe deine eigene BeamMP-Erfahrung",
"windows": "Windows-Server",
"linux": "Linux Builds",
"partners": {
"description": "Starte schnell mit unseren vertrauenswürdigen Hosting-Partnern - problemlose Einrichtung, 24/7 Support und optimierte Leistung.",
"view_partners": "Hosting-Partner ansehen"
},
"self_host": {
"title": "Oder selbst hosten"
}
"linux": "Linux-Builds"
}
},
"faq": {
@@ -171,13 +163,6 @@
"join_discord": "Discord beitreten"
}
},
"partners": {
"title": "Hosting-Partner",
"description": "Wähle aus unseren vertrauenswürdigen Hosting-Partnern. Einfache Einrichtung, optimierte Leistung und Support, damit dein BeamMP-Server reibungslos läuft.",
"from_price": "Ab {price}",
"visit_website": "Website besuchen",
"error_loading": "Partner konnten nicht geladen werden"
},
"servers": {
"title": "Server",
"description": "Durchsuche und trete BeamMP-Servern weltweit bei. Finde deine Lieblingsspielmodi, Communities und Erlebnisse an einem Ort.",
+9 -25
View File
@@ -10,10 +10,8 @@
"forums": "Forums",
"docs": "Docs",
"communities": "Communities",
"partners": "Hosting Partners",
"servers": "Servers",
"statistics": "Statistics",
"store": "Merch Store",
"github": "GitHub",
"patreon": "Patreon",
"language": "Select Language",
@@ -47,11 +45,11 @@
"public_servers": "Public Servers",
"all_servers": "All Servers"
},
"why_beammp": "Why BeamMP?",
"why_choose_beammp": "Why Choose BeamMP?",
"features": {
"stable_servers": {
"title": "Stable Servers",
"description": "BeamMP allows for a stable environment, with a variety of servers located across the globe."
"description": "BeamMP allows for stable servers, with a variety of servers located across the globe."
},
"beamng": {
"title": "BeamNG.drive",
@@ -63,12 +61,12 @@
},
"sync": {
"title": "Sync Quality",
"description": "BeamMP updates your vehicle position ~50 times per second, allowing for a smooth overall experience."
"description": "BeamMP updates your vehicle position ~100 times per second, allowing for a smooth overall experience."
}
},
"communities": {
"join": "Join a Thriving Community",
"description": "Discover diverse gameplay experiences across thousands of unique servers",
"description": "Discover diverse gameplay experiences across hundreds of unique servers",
"racing": {
"name": "Racing Communities",
"description": "Competitive racing leagues and time trials with players worldwide"
@@ -82,8 +80,8 @@
"description": "Demolition derbies and destruction-focused gameplay modes"
},
"freeroam": {
"name": "Freeroam & Custom Servers",
"description": "Explore open-world servers or build your own with custom scripts, maps, and mods."
"name": "Freeroam",
"description": "Casual multiplayer sessions exploring maps with friends"
}
},
"find": {
@@ -119,14 +117,7 @@
"title": "Ready to Host Your Own Server?",
"description": "Download the server files and create your own unique BeamMP experience",
"windows": "Windows Server",
"linux": "Linux Builds",
"partners": {
"description": "Get started quickly with our trusted hosting partners - hassle-free setup, 24/7 support, and optimized performance.",
"view_partners": "View Hosting Partners"
},
"self_host": {
"title": "Or Host It Yourself"
}
"linux": "Linux Builds"
}
},
"faq": {
@@ -154,12 +145,12 @@
"4": {
"question": "How do I host a server?",
"answer":
"The server files required for hosting your own server can be found at the top of this page below the client download. You will also require an authentication key which can be found from keymaster. Further information around the setup can be found on our docs."
"The server files required for hosting your own server can be found at the top of this page below the client download. You will also require an authentication key which can be found from keymaster. Further information around the setup can be found on our wiki."
},
"5": {
"question": "Can I use mods?",
"answer":
"Mods are supported. These are installed on the server. See our docs for more information."
"Mods are supported. These are installed on the server. See our wiki for more information."
}
},
"stats": {
@@ -178,13 +169,6 @@
"join_discord": "Join Our Discord"
}
},
"partners": {
"title": "Hosting Partners",
"description": "Choose from our trusted hosting partners. Simple setup, optimized performance, and support to keep your BeamMP server running smoothly.",
"from_price": "From {price}",
"visit_website": "Visit Website",
"error_loading": "Unable to fetch partners"
},
"servers": {
"title": "Servers",
"description": "Browse and join BeamMP servers from around the world. Find your favorite game modes, communities, and experiences all in one place.",
+3 -18
View File
@@ -10,7 +10,6 @@
"forums": "Foros",
"docs": "Docs",
"communities": "Comunidades",
"partners": "Socios",
"servers": "Servidores",
"statistics": "Estadísticas",
"github": "GitHub",
@@ -46,7 +45,7 @@
"public_servers": "Servidores públicos",
"all_servers": "Todos los servidores"
},
"why_beammp": "¿Por qué BeamMP?",
"why_choose_beammp": "¿Por qué elegir BeamMP?",
"features": {
"stable_servers": {
"title": "Servidores estables",
@@ -62,7 +61,7 @@
},
"sync": {
"title": "Calidad de sincronización",
"description": "BeamMP actualiza la posición de tu vehículo ~50 veces por segundo, permitiendo una experiencia fluida."
"description": "BeamMP actualiza la posición de tu vehículo ~100 veces por segundo, permitiendo una experiencia fluida."
}
},
"communities": {
@@ -118,14 +117,7 @@
"title": "¿Listo para alojar tu propio servidor?",
"description": "Descarga los archivos del servidor y crea tu experiencia única en BeamMP",
"windows": "Servidor Windows",
"linux": "Compilaciones Linux",
"partners": {
"description": "Comienza rápidamente con nuestros socios de hosting de confianza - configuración sin complicaciones, soporte 24/7 y rendimiento optimizado.",
"view_partners": "Ver socios de hosting"
},
"self_host": {
"title": "O alójalo tú mismo"
}
"linux": "Compilaciones Linux"
}
},
"faq": {
@@ -171,13 +163,6 @@
"join_discord": "Unirte a Discord"
}
},
"partners": {
"title": "Socios de alojamiento",
"description": "Elige entre nuestros socios de alojamiento de confianza. Configuración sencilla, rendimiento optimizado y soporte para mantener tu servidor BeamMP funcionando sin problemas.",
"from_price": "Desde {price}",
"visit_website": "Visitar sitio web",
"error_loading": "No se pudieron cargar los socios"
},
"servers": {
"title": "Servidores",
"description": "Explora y únete a servidores de BeamMP en todo el mundo. Encuentra tus modos de juego, comunidades y experiencias favoritas en un solo lugar.",
+18 -33
View File
@@ -10,7 +10,6 @@
"forums": "Forums",
"docs": "Docs",
"communities": "Communautés",
"partners": "Partenaires",
"servers": "Serveurs",
"statistics": "Statistiques",
"github": "GitHub",
@@ -23,7 +22,7 @@
}
},
"footer": {
"support_on_patreon": "Soutenez-nous sur notre Patreon",
"support_on_patreon": "Soutenez-nous sur Patreon",
"about": "À propos",
"privacy_policy": "Politique de confidentialité",
"terms_conditions": "Conditions générales"
@@ -36,7 +35,7 @@
"home": {
"hero": {
"title": "Multijoueur pour BeamNG.drive",
"subtitle": "Amusez-vous avec vos amis comme jamais. Courses, roleplay ou simples balades : tout est possible avec BeamMP !",
"subtitle": "Profitez de la physique soft-body avec vos amis. Faites la course, du roleplay ou une balade ensemble.",
"download_now": "Télécharger",
"browse_servers": "Parcourir les serveurs"
},
@@ -46,23 +45,23 @@
"public_servers": "Serveurs publics",
"all_servers": "Tous les serveurs"
},
"why_beammp": "Pourquoi BeamMP ?",
"why_choose_beammp": "Pourquoi choisir BeamMP ?",
"features": {
"stable_servers": {
"title": "Serveurs stables",
"description": "BeamMP offre des serveurs stables et une grande variété de mondes à explorer partout dans le monde."
"description": "BeamMP permet des serveurs stables, avec une grande variété de serveurs à travers le monde."
},
"beamng": {
"title": "BeamNG.drive",
"description": "Les mêmes cartes, véhicules et mods que vous connaissez — BeamMP rend la transition simple et rapide !"
"description": "BeamMP utilise les mêmes cartes, véhicules et mods — pas besoin d'apprendre quoi que ce soit de nouveau !"
},
"standalone": {
"title": "Indépendant",
"description": "BeamMP préserve votre jeu original et vous permet de jouer en solo ou avec des amis."
"description": "BeamMP ne modifie pas votre installation originale, vous pouvez jouer en solo ou en multijoueur."
},
"sync": {
"title": "Qualité de synchronisation",
"description": "BeamMP met à jour la position de votre véhicule ~50 fois par seconde pour une expérience fluide."
"description": "BeamMP met à jour la position de votre véhicule ~100 fois par seconde pour une expérience fluide."
}
},
"communities": {
@@ -74,7 +73,7 @@
},
"roleplay": {
"name": "Serveurs roleplay",
"description": "Des expériences immersives variées, des interventions de police aux services de livraison."
"description": "Expériences immersives, des poursuites policières aux services de livraison"
},
"crash": {
"name": "Crash & derby",
@@ -94,7 +93,7 @@
"moderation": "Modération active",
"moderation_desc": "Un environnement de jeu sûr et convivial",
"global": "Réseau mondial",
"global_desc": "Serveurs mondiaux, latence minimale"
"global_desc": "Serveurs dans le monde entier pour une faible latence"
},
"browse_all_servers": "Parcourir tous les serveurs"
},
@@ -103,7 +102,7 @@
"description": "Créez des modes de jeu, hébergez votre serveur et contribuez au projet",
"lua": {
"title": "API Lua",
"description": "Puissant scrip côté serveur en Lua pour des fonctionnalités personnalisées"
"description": "Puissant scripting côté serveur en Lua pour des fonctionnalités personnalisées"
},
"docs": {
"title": "Documentation",
@@ -111,32 +110,25 @@
},
"openSource": {
"title": "Open source",
"description": "Développement communautaire avec le code open source sur GitHub"
"description": "Développement communautaire avec code disponible sur GitHub"
},
"learn_more": "En savoir plus",
"host": {
"title": "Prêt à héberger votre propre serveur ?",
"description": "Téléchargez les fichiers serveur et créez votre expérience BeamMP unique",
"windows": "Serveur Windows",
"linux": "Versions Linux",
"partners": {
"description": "Démarrez rapidement avec nos partenaires d'hébergement de confiance - configuration sans tracas, support 24/7 et performances optimisées.",
"view_partners": "Voir les partenaires d'hébergement"
},
"self_host": {
"title": "Ou hébergez-le vous-même"
}
"linux": "Builds Linux"
}
},
"faq": {
"title": "Foire aux questions",
"0": {
"question": "La liste des serveurs n'apparaît pas !",
"answer": "Essayez de redémarrer BeamMP. Si le problème persiste, créez un post sur notre forum ou ouvrez un ticket dassistance sur notre serveur Discord."
"answer": "Essayez de redémarrer BeamMP. Si cela ne fonctionne pas, créez un sujet sur le forum ou rendez-vous sur le canal support de Discord."
},
"1": {
"question": "Comment ouvrir un ticket si quelque chose ne fonctionne pas ?",
"answer": "Consultez le canal #how-to-use sur Discord ainsi que le forum. Décrivez précisément les actions effectuées afin que léquipe support puisse vous aider rapidement et efficacement."
"answer": "Consultez le canal #how-to-use sur Discord et le forum. Décrivez précisément ce que vous avez fait pour que l'équipe support puisse vous aider rapidement et efficacement."
},
"2": {
"question": "À l'aide ! Je reçois des codes d'erreur",
@@ -144,7 +136,7 @@
},
"3": {
"question": "Est-ce compatible avec des versions piratées de BeamNG.drive ?",
"answer": "Les versions piratées de BeamNG.drive ne sont pas prises en charge et ne bénéficieront daucun support."
"answer": "Nous ne savons pas si BeamNG.drive piraté fonctionne, et nous n'offrirons aucun support pour des copies non légitimes."
},
"4": {
"question": "Comment héberger un serveur ?",
@@ -168,16 +160,9 @@
"title": "Envie de créer votre communauté ?",
"description": "Hébergez votre serveur BeamMP et bâtissez une communauté autour de vos modes favoris",
"setup_guide": "Guide de configuration du serveur",
"join_discord": "Rejoindre notre Discord"
"join_discord": "Rejoindre Discord"
}
},
"partners": {
"title": "Partenaires dhébergement",
"description": "Choisissez parmi nos partenaires dhébergement de confiance. Configuration simple, performances optimisées et support pour que votre serveur BeamMP fonctionne sans interruption.",
"from_price": "À partir de {price}",
"visit_website": "Visiter le site",
"error_loading": "Impossible de charger les partenaires"
},
"servers": {
"title": "Serveurs",
"description": "Parcourez et rejoignez des serveurs BeamMP dans le monde entier. Retrouvez vos modes de jeu, communautés et expériences préférés au même endroit.",
@@ -245,7 +230,7 @@
},
"get_involved": {
"title": "Participer",
"description": "Participez de multiples façons — discussions, signalement, code, ou soutien financier via notre",
"description": "Participez de multiples façons — discussions, signalement, code, ou soutien financier via",
"patreon": "Patreon",
"forum": "Forums",
"docs": "Docs",
@@ -254,7 +239,7 @@
},
"funding": {
"title": "Financement et durabilité",
"description": "BeamMP repose sur le soutien de la communauté. Les dons couvrent l'infrastructure, la bande passante, les outils et le temps de développement. Si vous appréciez le projet, pensez à nous soutenir via notre Patreon ou en visitant notre Merch !",
"description": "BeamMP repose sur le soutien de la communauté. Les dons couvrent l'infrastructure, la bande passante, les outils et le temps de développement. Si vous appréciez le projet, pensez à nous soutenir.",
"patreon": "Soutenez-nous sur Patreon",
"learn": "En savoir plus sur GitHub"
},
+3 -18
View File
@@ -10,7 +10,6 @@
"forums": "Forum",
"docs": "Docs",
"communities": "Community",
"partners": "Partner",
"servers": "Server",
"statistics": "Statistiche",
"github": "GitHub",
@@ -46,7 +45,7 @@
"public_servers": "Server pubblici",
"all_servers": "Tutti i server"
},
"why_beammp": "Perché BeamMP?",
"why_choose_beammp": "Perché scegliere BeamMP?",
"features": {
"stable_servers": {
"title": "Server stabili",
@@ -62,7 +61,7 @@
},
"sync": {
"title": "Qualità di sincronizzazione",
"description": "BeamMP aggiorna la posizione del veicolo ~50 volte al secondo, offrendo un'esperienza fluida."
"description": "BeamMP aggiorna la posizione del veicolo ~100 volte al secondo, offrendo un'esperienza fluida."
}
},
"communities": {
@@ -118,14 +117,7 @@
"title": "Pronto a ospitare il tuo server?",
"description": "Scarica i file del server e crea la tua esperienza unica su BeamMP",
"windows": "Server Windows",
"linux": "Build Linux",
"partners": {
"description": "Inizia rapidamente con i nostri partner di hosting affidabili - configurazione semplice, supporto 24/7 e prestazioni ottimizzate.",
"view_partners": "Visualizza partner di hosting"
},
"self_host": {
"title": "O ospitalo tu stesso"
}
"linux": "Build Linux"
}
},
"faq": {
@@ -171,13 +163,6 @@
"join_discord": "Unisciti a Discord"
}
},
"partners": {
"title": "Partner di hosting",
"description": "Scegli tra i nostri partner di hosting di fiducia. Configurazione semplice, prestazioni ottimizzate e supporto per mantenere il tuo server BeamMP sempre operativo.",
"from_price": "Da {price}",
"visit_website": "Visita il sito",
"error_loading": "Impossibile caricare i partner"
},
"servers": {
"title": "Server",
"description": "Sfoglia e unisciti ai server di BeamMP in tutto il mondo. Trova modalità, community ed esperienze preferite in un unico posto.",
+3 -18
View File
@@ -10,7 +10,6 @@
"forums": "Форум",
"docs": "Документация",
"communities": "Сообщества",
"partners": "Партнёры",
"servers": "Серверы",
"statistics": "Статистика",
"github": "GitHub",
@@ -46,7 +45,7 @@
"public_servers": "Публичные серверы",
"all_servers": "Все серверы"
},
"why_beammp": "Почему BeamMP?",
"why_choose_beammp": "Почему BeamMP?",
"features": {
"stable_servers": {
"title": "Стабильные серверы",
@@ -62,7 +61,7 @@
},
"sync": {
"title": "Качество синхронизации",
"description": "BeamMP обновляет позицию вашего автомобиля ~50 раз в секунду, обеспечивая плавный игровой процесс."
"description": "BeamMP обновляет позицию вашего автомобиля ~100 раз в секунду, обеспечивая плавный игровой процесс."
}
},
"communities": {
@@ -118,14 +117,7 @@
"title": "Готовы разместить свой сервер?",
"description": "Скачайте серверные файлы и создайте уникальный опыт в BeamMP",
"windows": "Windows-сервер",
"linux": "Linux-сборки",
"partners": {
"description": "Быстро начните работу с нашими надежными хостинг-партнерами - простая настройка, поддержка 24/7 и оптимизированная производительность.",
"view_partners": "Посмотреть хостинг-партнеров"
},
"self_host": {
"title": "Или разместите самостоятельно"
}
"linux": "Linux-сборки"
}
},
"faq": {
@@ -171,13 +163,6 @@
"join_discord": "Вступить в Discord"
}
},
"partners": {
"title": "Партнёры по хостингу",
"description": "Выберите из наших надёжных хостинг‑партнёров. Простая настройка, оптимизированная производительность и поддержка, чтобы ваш сервер BeamMP работал стабильно.",
"from_price": "От {price}",
"visit_website": "Перейти на сайт",
"error_loading": "Не удалось загрузить партнёров"
},
"servers": {
"title": "Серверы",
"description": "Просматривайте и присоединяйтесь к серверам BeamMP по всему миру. Найдите любимые режимы, сообщества и впечатления в одном месте.",
-278
View File
@@ -1,278 +0,0 @@
{
"message": {
"theme": {
"light": "明亮模式",
"dark": "黑暗模式",
"system": "跟随系统"
},
"nav": {
"home": "主页",
"forums": "论坛",
"docs": "文档",
"communities": "社区",
"partners": "托管合作伙伴",
"servers": "服务器",
"statistics": "统计",
"store": "周边商店",
"github": "GitHub",
"patreon": "在 Patreon 上为我们捐献",
"language": "选择语言",
"theme": {
"light": "明亮主题",
"dark": "黑暗主题",
"system": "跟随系统"
}
},
"footer": {
"support_on_patreon": "Patreon",
"about": "关于我们",
"privacy_policy": "隐私政策",
"terms_conditions": "条款与细则"
},
"404": {
"title": "找不到页面",
"description": "此页面不存在,\n此页面可能不存在或已经被删除",
"return_home": "返回首页"
},
"home": {
"hero": {
"title": "为BeamNG.Drive 提供多人游戏",
"subtitle": "联机体验软体物理引擎\n和朋友一起赛车 角色扮演或闲逛",
"download_now": "现在下载",
"browse_servers": "查看服务器"
},
"metrics": {
"active_players": "活跃玩家",
"players_online": "在线玩家",
"public_servers": "公开服务器",
"all_servers": "所有服务器"
},
"why_beammp": "为什么选择BeamMP",
"features": {
"stable_servers": {
"title": "稳定",
"description": "BeamMP在全球开设服务器 提供稳定体验"
},
"beamng": {
"title": "易用",
"description": "BeamMP与BeamNG使用相同的地图 车辆和模组 无需学习新东西"
},
"standalone": {
"title": "独立",
"description": "BeamMP不会修改你的BeamNG 你可以无缝使用单机和联机"
},
"sync": {
"title": "流畅",
"description": "BeamMP 每秒更新~50次同步 带来流畅的联机体验"
}
},
"communities": {
"join": "加入这个充满活力的社区",
"description": "探索数千个独特服务器上的多样化游戏体验",
"racing": {
"name": "赛车",
"description": "为全球玩家提供联机赛车与计时赛"
},
"roleplay": {
"name": "角色扮演",
"description": "从警匪追逐到货运模拟都能获得沉浸的体验"
},
"crash": {
"name": "破坏",
"description": "以德比和破坏为玩法的主题玩法"
},
"freeroam": {
"name": "自由漫游与自定义服务器",
"description": "探索开放世界服务器,或使用自定义脚本、地图和模组构建你自己的服务器。"
}
},
"find": {
"title": "寻找你的完美服务器",
"description": "查看数百个活跃服务器 拥有不同的游戏模式 模组和社区 从竞速到休闲 每个人都能在这里找到适合自己的服务器",
"points": {
"custom": "独特体验",
"custom_desc": "每个玩法都由社区创造",
"moderation": "环境友好",
"moderation_desc": "安全友好的游戏环境",
"global": "极低延迟",
"global_desc": "全球服务器实现低延迟"
},
"browse_all_servers": "浏览全部服务器"
},
"devFeatures": {
"title": "为开发者而生",
"description": "打造独特游戏规则 开设自己的服务器 为项目做出贡献",
"lua": {
"title": "Lua API",
"description": "强大的服务器端Lua脚本支持,用于实现自定义游戏模式与功能。"
},
"docs": {
"title": "文档",
"description": "面向服务器开发的详尽指南与API参考文档"
},
"openSource": {
"title": "开源",
"description": "采用社区驱动开发模式,并在GitHub上开源"
},
"learn_more": "了解更多",
"host": {
"title": "准备搭建您的专属服务器了吗?",
"description": "下载服务器文件,即可打造您独特的BeamMP游戏体验。",
"windows": "Windows 服务器端",
"linux": "Linux 服务器端",
"partners": {
"description": "通过我们可信赖的托管合作伙伴快速上手,享受无忧设置、24/7技术支持和优化性能。",
"view_partners": "查看托管合作伙伴"
},
"self_host": {
"title": "或者自托管"
}
}
},
"faq": {
"title": "常见问题回答",
"0": {
"question": "服务器列表没有显示!",
"answer": "如果遇到此问题,可尝试重启BeamMP客户端。若仍未解决,请在官方论坛发帖或加入Discord支持频道寻求帮助。"
},
"1": {
"question": "若遇到问题或有疑问,如何提交工单?",
"answer": "请查看Discord及官方论坛中的#how-to-use频道。详细描述您已尝试的操作步骤,以便支持团队能够快速有效地为您提供帮助。"
},
"2": {
"question": "求助!我遇到错误代码了",
"answer": "请前往我们的论坛查看是否有人遇到过相同问题。很可能已有其他用户遇到过类似情况,回复中通常也会附有解决方案。或者可以访问我们的Discord频道,那里是目前更活跃的社区,大家都很热心帮助。"
},
"3": {
"question": "我可以在盗版的BeamNG.Drive上玩吗?",
"answer": "我们尚不确定其是否兼容BeamNG.drive的盗版版本,但对于非正版游戏副本,我们将不提供任何技术支持,也不会兼容它"
},
"4": {
"question": "如何托管服务器?",
"answer": "托管服务器所需的文件可在本页顶部客户端下载下方找到。同时您需要在Keymaster获取认证密钥。更多设置相关信息请查阅我们的官方文档。"
},
"5": {
"question": "我可以使用模组吗?",
"answer": "模组是支持的,这些安装在服务器上。请查看我们的文档以获取更多信息。"
}
},
"stats": {
"active_players": "活跃玩家",
"global_servers": "全球服务器",
"total_downloads": "总下载量"
}
},
"communities": {
"title": "社区",
"description": "发现围绕BeamMP建立的繁荣社区。从竞技赛车联盟到休闲自由漫游会话,找到您完美的玩伴群组。",
"starting": {
"title": "想要创建您自己的社区吗?",
"description": "托管您自己的BeamMP服务器,并围绕您钟爱的游戏模式建立社区。",
"setup_guide": "服务器设置指南",
"join_discord": "加入我们的Discord"
}
},
"partners": {
"title": "托管合作伙伴",
"description": "选择我们值得信赖的托管合作伙伴。简单的设置、优化的性能和支持,以确保您的BeamMP服务器运行顺畅。",
"from_price": "从 {price}",
"visit_website": "访问网站",
"error_loading": "无法获取合作伙伴信息"
},
"servers": {
"title": "服务器",
"description": "探索并加入来自全球的BeamMP服务器。在此一站式发现您钟爱的游戏模式、活跃社群与丰富玩法。",
"show_only": "仅显示:",
"loading_servers": "加载服务器中...",
"server_count": "0 个发现的服务器 | 1 个发现的服务器 | {count} 发现的服务器",
"players_found": "0 个玩家 | 1 个玩家 | {count} 个玩家",
"filters": {
"hide_empty": "隐藏空服务器",
"hide_full": "隐藏爆满服务器",
"search": "搜索服务器中",
"hide_password": "隐藏受密码保护的服务器",
"show_official": "官方服务器",
"show_partner": "合作伙伴服务器",
"show_featured": "推荐服务器"
},
"table_headers": {
"location": "位置",
"name": "服务器名称",
"map": "地图",
"players": "玩家",
"mods": "模组",
"status": "状态"
}
},
"statistics": {
"title": "统计数据",
"description": "提供活跃服务器的实时快照指标与玩家数量时间线。支持在图表上标注版本发布日期。",
"loading": "加载中...",
"peak": "峰值: {count}",
"current_players": "当前玩家",
"current_servers": "当前服务器",
"avg_players_per_server": "服务器平均玩家数量",
"official_servers": "官方服务器",
"partner_servers": "合作伙伴服务器",
"player_volume": "在线人数时间轴",
"hint": "版本发布标记显示为橙色垂直虚线(例如 v3.0.0、v4.0.0)。悬停鼠标至数据点可查看详细信息。",
"server_volume": "服务器数量时间轴",
"server_hint": "服务器数量呈现相似趋势。发布标记指示主要版本发布。"
},
"about": {
"title": "关于BeamMP",
"description": "BeamMP可为BeamNG.drive带来多人游戏体验。它由社区共建、为社区服务,注重稳定性与性能,致力于让您与朋友共享真实的驾驶体验。",
"note": {
"title": "开发者的话",
"content": "BeamMP 始于一个简单的想法:我想和我的兄弟们一起玩BeamNG.drive。这个最初作为实验的项目,迅速成长为一个社区驱动的倡议,专注于实现多人游戏体验与乐趣。我们深切关注于提供尽可能最佳的用户体验,并让玩家能够轻松加入,享受共同驾驶的快乐。该项目向所有技能水平的贡献者开放——无论您是编写代码、管理服务器、设计场景,还是帮助他人完成设置,您的努力都是 BeamMP 蓬勃发展的关键部分。感谢您的参与,帮助我们共同构建这个非凡的项目。"
},
"provides": {
"title": "BeamMP的功能与服务",
"points": {
"0": "提供BeamNG.drive的多人联机会话,支持服务器浏览器与筛选功能。",
"1": "服务器端管理工具与配置选项",
"2": "模组支持与付费内容保护机制",
"3": "活跃的社区(论坛、Discord),可以提供帮助与协作支持。"
}
},
"project": {
"title": "项目价值观",
"points": {
"0": "社区优先:决策由真实玩家/服务器需求引导",
"1": "可靠性:我们始终把游戏稳定性、数据同步精度和性能表现放在首位,能优化到的地方都全力以赴。",
"2": "开放:欢迎贡献、反馈与透明",
"3": "安全:促进公平游戏和尊重互动"
}
},
"get_involved": {
"title": "参与进来",
"description": "参与的方式有很多种——比如加入讨论、反馈问题、贡献代码,或者通过捐赠等途径给项目提供财务支持。",
"patreon": "Patreon",
"forum": "论坛",
"docs": "文档",
"github": "GitHub",
"discord": "Discord"
},
"funding": {
"title": "资金与可持续发展",
"description": "BeamMP的发展离不开社区的支持。捐款将用于支付服务器等基础设施、带宽、工具及开发时间的成本。如果您认可本项目并愿意支持我们持续成长,请考虑为我们提供捐助。",
"patreon": "在Patreon上支持我们",
"learn": "了解更多内容请访问GitHub"
},
"credits": {
"title": "致谢与贡献者",
"description": "BeamMP项目由模组团队及一众优秀的社区贡献者共同维护。我们同时感谢所有服务器所有者、模组创作者、测试人员以及积极反馈问题的用户——你们的时间与热情是项目持续前进的动力。",
"desc_2": "衷心感谢BeamNG.drive的开发团队,他们打造的卓越平台为多人游戏功能的实现奠定了坚实基础。",
"desc_3": "同时衷心感谢以下现任与曾经的社区成员,感谢他们多年来对BeamMP项目作出的重大贡献:",
"mentions": {
"0": "Jojos38(联合创始人)—感谢他早期在设计各方面的工作,帮助将想法变为现实。",
"1": "Jettajetta.cat)—负责标志的设计与制作。",
"2": "Anonymous275 与 Lionkor — 在项目重构期间负责使用 C++ 重写代码的工作。",
"3": "Tixx — 感谢他对整个项目代码库作出的重大贡献。",
"4": "以及更多未列名的贡献者—衷心感谢每一位以各种方式支持过本项目的朋友!"
},
"thank_you": "我们正在共同打造一个独特的作品—真诚感谢您成为BeamMP社区的一员!"
}
}
}
}
+4 -2
View File
@@ -7,7 +7,6 @@ import './style.css'
//const i18n = createI18n({
const initialLocale = localStorage.getItem('lang') || 'en'
const i18n = setupI18n({
legacy: false,
locale: initialLocale,
fallbackLocale: 'en',
messages: {
@@ -22,4 +21,7 @@ window.locale = i18n.global.locale
import router from './routes'
createApp(App).use(i18n).use(router).mount('#app')
createApp(App)
.use(i18n)
.use(router)
.mount('#app')
+21 -50
View File
@@ -1,12 +1,10 @@
import { loadLocaleMessages, setI18nLanguage, SUPPORT_LOCALES } from '@/i18n'
import { createRouter, createWebHistory } from 'vue-router'
import NotFound from '@/views/NotFound.vue'
import { RouterView } from 'vue-router'
// Base routes without locale prefix
const baseRoutes = [
const routes = [
{
path: '',
path: '/',
name: 'Home',
component: () => import('@/views/Home.vue'),
meta: {
@@ -16,7 +14,7 @@ const baseRoutes = [
},
},
{
path: 'about',
path: '/about',
name: 'About',
component: () => import('@/views/About.vue'),
meta: {
@@ -25,8 +23,18 @@ const baseRoutes = [
requiresAuth: false,
},
},
/*{
path: '/contact',
name: 'Contact',
component: Contact,
meta: {
title: 'Contact - BeamMP',
description: 'Get in touch with us',
requiresAuth: false
}
},*/
{
path: 'communities',
path: '/communities',
name: 'Communities',
component: () => import('@/views/Communities.vue'),
meta: {
@@ -36,7 +44,7 @@ const baseRoutes = [
},
},
{
path: 'servers',
path: '/servers',
name: 'Servers',
component: () => import('@/views/Servers.vue'),
meta: {
@@ -46,7 +54,7 @@ const baseRoutes = [
},
},
{
path: 'stats',
path: '/stats',
name: 'Statistics',
component: () => import('@/views/Statistics.vue'),
meta: {
@@ -55,38 +63,6 @@ const baseRoutes = [
requiresAuth: false,
},
},
{
path: 'partners',
name: 'Partners',
component: () => import('@/views/Partners.vue'),
meta: {
title: 'Partners - BeamMP',
description: 'Explore trusted hosting partners for BeamMP servers',
requiresAuth: false,
},
},
]
const routes = [
{
path: `/:locale(${SUPPORT_LOCALES.join('|')})`,
component: RouterView,
beforeEnter: (to) => {
console.log('Entering locale route:', to.params.locale)
// Validate locale
if (!SUPPORT_LOCALES.includes(to.params.locale)) {
return false
}
},
children: baseRoutes,
},
{
path: '/',
redirect: () => {
const locale = localStorage.getItem('lang') || 'en'
return `/${locale}`
},
},
{
path: '/:pathMatch(.*)*',
name: 'NotFound',
@@ -104,16 +80,14 @@ const router = createRouter({
routes,
})
// Global navigation guard for meta data and locale
// Global navigation guard for meta data
router.beforeEach(async (to, from, next) => {
const paramsLocale = to.params.locale || 'en'
const i18n = window.i18n
// Ensure i18n is available
if (!i18n) {
next()
return
}
// use locale if paramsLocale is not in SUPPORT_LOCALES
/*if (!SUPPORT_LOCALES.includes(paramsLocale)) {
return next(`/${locale}`)
}*/
// load locale messages
if (!i18n.global.availableLocales.includes(paramsLocale)) {
@@ -123,9 +97,6 @@ router.beforeEach(async (to, from, next) => {
// set i18n language
setI18nLanguage(i18n, paramsLocale)
// Store current locale in localStorage
localStorage.setItem('lang', paramsLocale)
// Set page title
document.title = to.meta.title || 'BeamMP'
-9
View File
@@ -5,15 +5,6 @@
@custom-variant dark (&:is(.dark *));
@theme {
--color-beammp-orange: #F36D24;
--color-beammp-white: #FFFFFF;
--color-beammp-black: #000000;
--color-beammp-gray: #333333;
--color-beammp-green: #1D9749;
--color-beammp-blue: #4470B6;
}
@theme inline {
--radius-sm: calc(var(--radius) - 4px);
--radius-md: calc(var(--radius) - 2px);
-42
View File
@@ -1,42 +0,0 @@
/**
* Get the current locale from localStorage or window.i18n
* @returns {string} The current locale code
*/
export function getCurrentLocale() {
if (window.i18n) {
const locale = window.i18n.global.locale
return locale.value || locale
}
return localStorage.getItem('lang') || 'en'
}
/**
* Generate a localized route path
* @param {string} route - The route name or path (without leading slash)
* @param {string} locale - Optional locale override
* @returns {string} The full path including locale
*/
export function getLocalizedPath(route, locale = null) {
const currentLocale = locale || getCurrentLocale()
const path = route.startsWith('/') ? route.slice(1) : route
return `/${currentLocale}/${path}`.replace(/\/+/g, '/')
}
/**
* Create a redirect to the current route in a different locale
* @param {string} newLocale - The new locale to switch to
* @param {string} currentPath - The current route path (can include locale and query params)
* @returns {string} The new path with the different locale
*/
export function switchLocale(newLocale, currentPath) {
// Split path from query string
const [pathOnly] = currentPath.split('?')
// Remove locale from the beginning: /en/servers -> /servers, /en/ -> /
const pathWithoutLocale = pathOnly.replace(/^\/[a-z]{2}(?:\/|$)/, '/')
// Remove trailing slashes and ensure we have the right format
const cleanPath = pathWithoutLocale.replace(/\/+/g, '/').replace(/\/$/, '') || ''
return `/${newLocale}${cleanPath}`.replace(/\/+/g, '/')
}
+1 -3
View File
@@ -54,9 +54,7 @@
<!-- How to Get Involved -->
<div class="rounded-lg border border-neutral-200 dark:border-neutral-800 p-6 mb-10">
<h3 class="text-xl font-semibold text-neutral-900 dark:text-white mb-3">
{{ $t('message.about.get_involved.title') }}
</h3>
<h3 class="text-xl font-semibold text-neutral-900 dark:text-white mb-3">{{ $t('message.about.get_involved.title') }}</h3>
<p class="text-neutral-700 dark:text-neutral-300 mb-4">
{{ $t('message.about.get_involved.description') }}
<a
+3 -4
View File
@@ -1,8 +1,7 @@
<script setup>
import { computed } from 'vue'
import { ExternalLink, Users, Trophy, Gamepad2, Shield, MapPin } from 'lucide-vue-next'
const communities = computed(() => [
const communities = [
{
name: 'BeamMP Racing League',
description:
@@ -122,7 +121,7 @@ const communities = computed(() => [
},
color: 'from-amber-600 to-yellow-600',
},
])
]
</script>
<template>
@@ -251,7 +250,7 @@ const communities = computed(() => [
<!-- CTA Section -->
<section class="text-center py-16 bg-neutral-50 dark:bg-neutral-900/30 -mx-4 px-4 rounded-xl">
<div class="max-w-2xl mx-auto space-y-6">
<h2 class="text-3xl md:text-4xl font-bold">{{ $t('message.communities.starting.title') }}</h2>
<h2 class="text-3xl md:text-4xl font-bold">{{ $t('message.communities.starting') }}</h2>
<p class="text-lg text-neutral-600 dark:text-neutral-400">
{{ $t('message.communities.starting.description') }}
</p>
+71 -148
View File
File diff suppressed because one or more lines are too long
-104
View File
@@ -1,104 +0,0 @@
<script setup>
import { ref, onMounted } from 'vue'
const partners = ref([])
const loading = ref(true)
const error = ref(null)
async function loadPartners() {
try {
const res = await fetch('/partners.json', { cache: 'no-store' })
if (!res.ok) throw new Error(`Failed to load partners (${res.status})`)
const data = await res.json()
const partnersArray = Array.isArray(data) ? data : []
// Create a seed based on the current date (YYYY-MM-DD)
const today = new Date().toISOString().split('T')[0]
const seed = today.split('-').reduce((acc, val) => acc + parseInt(val), 0)
// Simple seeded shuffle using the date as seed
const shuffled = [...partnersArray]
for (let i = shuffled.length - 1; i > 0; i--) {
const j = Math.floor(((seed * (i + 1) * 9301 + 49297) % 233280) / 233280 * (i + 1))
;[shuffled[i], shuffled[j]] = [shuffled[j], shuffled[i]]
}
partners.value = shuffled
} catch (e) {
error.value = e.message || 'Unable to fetch partners'
} finally {
loading.value = false
}
}
onMounted(loadPartners)
</script>
<template>
<section class="py-16 px-4">
<div class="max-w-6xl mx-auto">
<div class="text-center mb-12">
<h1 class="text-4xl font-bold">{{ $t('message.partners.title') }}</h1>
<p class="text-neutral-600 dark:text-neutral-400 max-w-2xl mx-auto">
{{ $t('message.partners.description') }}
</p>
</div>
<!-- Loading State -->
<div v-if="loading" class="grid sm:grid-cols-2 lg:grid-cols-3 gap-6">
<div
v-for="n in 6"
:key="n"
class="rounded-xl border border-neutral-200 dark:border-neutral-800 p-6 bg-white/70 dark:bg-neutral-900/50 animate-pulse"
>
<div class="h-16 bg-neutral-200 dark:bg-neutral-800 rounded mb-4" />
<div class="h-5 bg-neutral-200 dark:bg-neutral-800 rounded w-2/3 mb-2" />
<div class="h-4 bg-neutral-200 dark:bg-neutral-800 rounded w-1/3" />
<div class="h-9 bg-neutral-200 dark:bg-neutral-800 rounded w-full mt-6" />
</div>
</div>
<!-- Error State -->
<div v-else-if="error" class="text-center">
<p class="text-red-600 dark:text-red-400">{{ $t('message.partners.error_loading') }}</p>
</div>
<!-- Partners Grid -->
<div v-else class="grid sm:grid-cols-2 lg:grid-cols-3 gap-6">
<div
v-for="(p, idx) in partners"
:key="idx"
class="shadow-md hover:-translate-y-1 duration-150 group rounded-xl border border-neutral-200 dark:border-neutral-800 dark:bg-neutral-900/50 hover:border-beammp-orange dark:hover:border-beammp-orange hover:shadow-xl transition-all"
>
<div class="flex items-center justify-center min-h-16 mb-4 p-6 bg-neutral-900/80 border-b border-beammp-orange border-b-2 rounded-t-xl" style="height: 114px;">
<img :src="p.logo" :alt="p.name" class="max-h-16 object-contain" />
</div>
<div class="p-6">
<div class="space-y-1">
<h3 class="text-xl text-neutral-900 dark:text-white font-semibold">{{ p.name }}</h3>
<p class="text-sm text-neutral-600 dark:text-neutral-400">
{{ $t('message.partners.from_price', { price: p.from }) }}
</p>
</div>
<a
:href="p.website"
target="_blank"
rel="noopener noreferrer"
class="mt-6 inline-flex items-center justify-center gap-2 w-full bg-beammp-orange hover:bg-beammp-orange/90 text-white px-4 py-2 rounded-lg font-semibold transition-colors dark:bg-beammp-orange dark:hover:bg-beammp-orange/90"
>
{{ $t('message.partners.visit_website') }}
<svg class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
<path
stroke-linecap="round"
stroke-linejoin="round"
stroke-width="2"
d="M9 5l7 7-7 7"
/>
</svg>
</a>
</div>
</div>
</div>
</div>
</section>
</template>
+39 -109
View File
@@ -1,8 +1,8 @@
<template>
<div class="servers-container">
<div class="servers-header">
<h1 class="text-4xl font-bold">{{ $t('message.servers.title') }}</h1>
<p class="text-base servers-subtitle text-neutral-600 dark:text-neutral-400">{{ $t('message.servers.description') }}</p>
<h1 class="servers-title">{{ $t('message.servers.title') }}</h1>
<p class="servers-subtitle">{{ $t('message.servers.description') }}</p>
</div>
<div class="servers-filters">
@@ -16,11 +16,11 @@
</div>
<div class="filter-group">
<label class="filter-label text-neutral-600 dark:text-neutral-400">
<label class="filter-label">
<input v-model="filters.hideEmpty" type="checkbox" />
{{ $t('message.servers.filters.hide_empty') }}
</label>
<label class="filter-label text-neutral-600 dark:text-neutral-400">
<label class="filter-label">
<input v-model="filters.hideFull" type="checkbox" />
{{ $t('message.servers.filters.hide_full') }}
</label>
@@ -32,15 +32,15 @@
<div class="filter-group">
<span class="filter-heading">{{ $t('message.servers.show_only') }}</span>
<label class="filter-label text-neutral-600 dark:text-neutral-400">
<label class="filter-label">
<input v-model="filters.official" type="checkbox" />
{{ $t('message.servers.filters.show_official') }}
</label>
<label class="filter-label text-neutral-600 dark:text-neutral-400">
<label class="filter-label">
<input v-model="filters.partner" type="checkbox" />
{{ $t('message.servers.filters.show_partner') }}
</label>
<label class="filter-label text-neutral-600 dark:text-neutral-400">
<label class="filter-label">
<input v-model="filters.featured" type="checkbox" />
{{ $t('message.servers.filters.show_featured') }}
</label>
@@ -244,8 +244,8 @@ const regionDisplay = computed(() => {
onMounted(async () => {
try {
//const response = await fetch('/servers.json')
const response = await fetch('https://backend.beammp.com/servers-info')
const response = await fetch('/servers.json')
//const response = await fetch('https://backend.beammp.com/servers-info')
console.log(response)
if (!response.ok) throw new Error('Failed to fetch servers')
const data = await response.json()
@@ -449,6 +449,13 @@ function joinServer(server) {
margin-bottom: 40px;
}
.servers-title {
font-size: 2.5rem;
font-weight: 700;
margin: 0 0 10px 0;
color: var(--text-color, #1a1a1a);
}
.servers-subtitle {
font-size: 1.1rem;
color: var(--text-muted, #666);
@@ -456,17 +463,15 @@ function joinServer(server) {
}
.servers-filters {
background: var(--filters-bg);
border: 1px solid var(--filters-border);
border-radius: 12px;
padding: 24px;
background: var(--card-bg, #f9f9f9);
border: 1px solid var(--border-color, #e0e0e0);
border-radius: 8px;
padding: 20px;
margin-bottom: 30px;
box-shadow: var(--filters-shadow);
backdrop-filter: blur(10px);
}
.filter-group {
margin-bottom: 20px;
margin-bottom: 15px;
}
.filter-group:last-child {
@@ -477,86 +482,30 @@ function joinServer(server) {
width: 100%;
padding: 12px 16px;
font-size: 1rem;
border: 2px solid var(--border-color, #e0e0e0);
border-radius: 8px;
border: 1px solid var(--border-color, #d0d0d0);
border-radius: 6px;
background: var(--input-bg, #fff);
color: var(--text-color, #1a1a1a);
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
font-weight: 500;
transition: border-color 0.2s;
}
.search-input:focus {
outline: none;
border-color: #ff6a00;
box-shadow: 0 0 0 3px rgba(255, 106, 0, 0.1);
}
.search-input::placeholder {
color: var(--placeholder-color, var(--text-muted, #999));
}
.filter-heading {
display: block;
margin-bottom: 12px;
font-size: 0.9rem;
font-weight: 600;
color: var(--text-color, #1a1a1a);
text-transform: uppercase;
letter-spacing: 0.5px;
opacity: 0.7;
border-color: var(--primary-color, #5d9cec);
}
.filter-label {
display: inline-flex;
align-items: center;
margin-right: 24px;
margin-bottom: 12px;
margin-right: 20px;
font-size: 0.95rem;
cursor: pointer;
font-weight: 500;
transition: all 0.2s ease;
}
.filter-label:hover {
color: #ff6a00;
color: var(--text-color, #1a1a1a);
}
.filter-label input[type='checkbox'] {
appearance: none;
-webkit-appearance: none;
-moz-appearance: none;
width: 20px;
height: 20px;
margin-right: 10px;
margin-right: 6px;
cursor: pointer;
border: 2px solid var(--checkbox-border, #d0d0d0);
border-radius: 6px;
background: var(--checkbox-bg, #fff);
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
flex-shrink: 0;
position: relative;
}
.filter-label input[type='checkbox']:hover {
border-color: #ff6a00;
box-shadow: 0 0 0 3px var(--checkbox-hover-shadow, rgba(255, 106, 0, 0.1));
}
.filter-label input[type='checkbox']:checked {
background: linear-gradient(135deg, #ff6a00 0%, #ff8c26 100%);
border-color: #ff6a00;
box-shadow: 0 2px 8px rgba(255, 106, 0, 0.3);
}
.filter-label input[type='checkbox']:checked::after {
content: '✓';
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
color: white;
font-weight: bold;
font-size: 12px;
}
.server-count {
@@ -945,27 +894,20 @@ function joinServer(server) {
color: var(--text-muted, #666);
}
:global(.dark) .servers-container,
/* Dark mode support */
@media (prefers-color-scheme: dark) {
.servers-container {
--text-color: #e0e0e0;
--text-muted: #999999;
--text-muted: #999;
--card-bg: #2a2a2a;
--border-color: #404040;
--input-bg: #1a1a1a;
--tag-bg: #404040;
--primary-color: #5d9cec;
--header-bg: #1f1f1f;
--hover-bg: #333333;
--hover-bg: #333;
--active-bg: #2a3f5f;
--details-bg: #252525;
--filters-bg: linear-gradient(135deg, rgba(30, 30, 30, 0.95) 0%, rgba(20, 20, 20, 0.98) 100%);
--filters-border: rgba(255, 106, 0, 0.2);
--filters-shadow: 0 2px 8px rgba(0, 0, 0, 0.3);
--checkbox-bg: rgba(255, 255, 255, 0.05);
--checkbox-border: rgba(255, 255, 255, 0.15);
--checkbox-hover-shadow: rgba(255, 106, 0, 0.15);
--placeholder-color: #666666;
}
}
@@ -975,6 +917,14 @@ function joinServer(server) {
padding: 15px;
}
.servers-title {
font-size: 2rem;
}
.servers-subtitle {
font-size: 1rem;
}
.servers-filters {
padding: 15px;
}
@@ -1013,23 +963,3 @@ function joinServer(server) {
}
}
</style>
/* Light theme defaults */
--text-color: #1a1a1a;
--text-muted: #666666;
--card-bg: #ffffff;
--border-color: #e0e0e0;
--input-bg: #ffffff;
--tag-bg: #e8e8e8;
--primary-color: #5d9cec;
--header-bg: #f5f5f5;
--hover-bg: #f9f9f9;
--active-bg: #f0f7ff;
--details-bg: #fafafa;
--filters-bg: linear-gradient(135deg, rgba(255, 255, 255, 0.9) 0%, rgba(249, 249, 249, 0.95) 100%);
--filters-border: var(--border-color, #e0e0e0);
--filters-shadow: 0 2px 8px rgba(0, 0, 0, 0.05);
--checkbox-bg: #ffffff;
--checkbox-border: #d0d0d0;
--checkbox-hover-shadow: rgba(255, 106, 0, 0.1);
--placeholder-color: var(--text-muted, #999999);
+6 -46
View File
@@ -7,8 +7,6 @@ const servers = ref([])
// Backend endpoints (use null to fall back to synthetic data)
const STATS_ENDPOINT = null // e.g., 'https://backend.beammp.com/stats-timeseries'
const MOD_RELEASES_ENDPOINT = 'https://api.github.com/repos/BeamMP/BeamMP/releases' // e.g., 'https://backend.beammp.com/releases'
const SERVER_RELEASES_ENDPOINT = 'https://api.github.com/repos/BeamMP/BeamMP-Server/releases' // e.g., 'https://backend.beammp.com/server-releases'
// All-time high tracking
const peakPlayers = ref(0)
@@ -35,52 +33,14 @@ const selectedRangeKey = ref('7d')
const selectedRange = computed(() => ranges.find((r) => r.key === selectedRangeKey.value))
// Release markers (fill with real dates/labels as desired)
const mod_releases = ref([
//{ date: '2025-11-24', label: 'v4.0.0' },
//{ date: '2025-11-27', label: 'v4.1.0' },
//{ date: '2025-11-29', label: 'v4.2.0' },
])
const server_releases = ref([
//{ date: '2025-11-25', label: 'v4.0.0' },
//{ date: '2025-11-28', label: 'v4.1.0' },
//{ date: '2025-11-30', label: 'v4.2.0' },
const releases = ref([
{ date: '2025-11-24', label: 'v4.0.0' },
{ date: '2025-11-27', label: 'v4.1.0' },
{ date: '2025-11-29', label: 'v4.2.0' },
])
// Load time-series from backend or generate synthetic
async function loadTimeSeries(points, stepHours) {
if (MOD_RELEASES_ENDPOINT) {
try {
const res = await fetch(MOD_RELEASES_ENDPOINT)
if (res.ok) {
const r = await res.json()
r.forEach((rel) => {
const existing = mod_releases.value.find((e) => e.label === rel.tag_name)
if (!existing) {
mod_releases.value.push({ date: rel.published_at, label: rel.tag_name })
}
})
}
} catch (e) {
console.error('Failed to load releases:', e)
}
}
if (SERVER_RELEASES_ENDPOINT) {
try {
const res = await fetch(SERVER_RELEASES_ENDPOINT)
if (res.ok) {
const r = await res.json()
r.forEach((rel) => {
const existing = server_releases.value.find((e) => e.label === rel.tag_name)
if (!existing) {
server_releases.value.push({ date: rel.published_at, label: rel.tag_name })
}
})
}
} catch (e) {
console.error('Failed to load releases:', e)
}
}
if (STATS_ENDPOINT) {
try {
const res = await fetch(STATS_ENDPOINT)
@@ -296,7 +256,7 @@ function drawPlayers() {
ctx.strokeStyle = 'rgba(255,106,0,0.8)'
ctx.fillStyle = 'rgba(255,106,0,0.9)'
ctx.lineWidth = 1
const rels = (mod_releases.value || [])
const rels = (releases.value || [])
.map((r) => ({ ...r, time: new Date(r.date).getTime() }))
.filter((r) => !isNaN(r.time) && r.time >= minT && r.time <= maxT)
releaseMarkersPlayers = []
@@ -420,7 +380,7 @@ function drawServers() {
ctx.strokeStyle = 'rgba(255,106,0,0.8)'
ctx.fillStyle = 'rgba(255,106,0,0.9)'
ctx.lineWidth = 1
const rels = (server_releases.value || [])
const rels = (releases.value || [])
.map((r) => ({ ...r, time: new Date(r.date).getTime() }))
.filter((r) => !isNaN(r.time) && r.time >= minT && r.time <= maxT)
releaseMarkersServers = []
+18 -6
View File
@@ -5,12 +5,24 @@ export default {
theme: {
extend: {
colors: {
'beammp-orange': '#F36D24',
'beammp-white': '#FFFFFF',
'beammp-black': '#000000',
'beammp-gray': '#333333',
'beammp-green': '#1D9749',
'beammp-blue': '#4470B6',
'beammp-orange': {
DEFAULT: '#F36D24', // Base color
},
'beammp-white': {
DEFAULT: '#FFFFFF', // Base color
},
'beammp-black': {
DEFAULT: '#000000', // Base color
},
'beammp-gray': {
DEFAULT: '#333333', // Base color
},
'beammp-green': {
DEFAULT: '#1D9749', // Base color
},
'beammp-blue': {
DEFAULT: '#4470B6', // Base color
},
},
},
},
+3
View File
@@ -11,4 +11,7 @@ export default defineConfig({
'@': path.resolve(__dirname, './src'),
},
},
server: {
port: 3000,
},
})