Compare commits
26 Commits
7f48a725d8
...
master
| Author | SHA1 | Date | |
|---|---|---|---|
| 94e2b8bd47 | |||
| 030060286f | |||
| b7ad2dc406 | |||
| 2023542229 | |||
| eaac266ebb | |||
| ed782f2fc6 | |||
| f2fd36664c | |||
| 456a0490a7 | |||
| 306dd8cabc | |||
| 50b3e421df | |||
| 963295e76b | |||
| e12b48b3e1 | |||
| b0509818b2 | |||
| ee553eae82 | |||
| 8c230d3596 | |||
| da3e015631 | |||
| f532152d57 | |||
| 818ae39e34 | |||
| 836f42be4d | |||
| 3fdced84bf | |||
| 139e7d0ef5 | |||
| ffb23b08eb | |||
| e6d66529e3 | |||
| 76bb9fbb30 | |||
| b928212608 | |||
| 329ed5adb0 |
@@ -13,9 +13,32 @@ jobs:
|
||||
- name: Checkout
|
||||
uses: actions/checkout@v4
|
||||
|
||||
- name: Install dependencies
|
||||
run: |
|
||||
apt-get update -y && apt-get install -y openssh-client
|
||||
apt-get install -y git
|
||||
|
||||
- name: Setup SSH inside container
|
||||
run: |
|
||||
rm -rf ~/.ssh
|
||||
mkdir -p ~/.ssh
|
||||
echo "${{ secrets.DEPLOY_KEY }}" | tr -d '\r' > ~/.ssh/id_rsa
|
||||
chmod 600 ~/.ssh/id_rsa
|
||||
|
||||
# Add the container host to known_hosts
|
||||
ssh-keyscan -H ${{ secrets.DEPLOY_HOST }} >> ~/.ssh/known_hosts
|
||||
|
||||
- name: Log in to registry
|
||||
run: |
|
||||
echo "${{ secrets.REGISTRY_PASSWORD }}" | docker login git.aranroig.com -u "${{ secrets.REGISTRY_USER }}" --password-stdin
|
||||
|
||||
- name: Build frontend
|
||||
run: |
|
||||
docker build -t git.aranroig.com/${{ secrets.REGISTRY_USER }}/dragonroll-frontend:latest ./frontend
|
||||
docker build -t git.aranroig.com/${{ secrets.REGISTRY_USER }}/dragonroll-frontend:latest \
|
||||
--build-arg NUXT_PUBLIC_GIT_COMMIT=$(git rev-parse --short HEAD) \
|
||||
--build-arg NUXT_PUBLIC_GIT_TAG=$(git describe --tags --abbrev=0) \
|
||||
--build-arg NUXT_PUBLIC_GIT_BRANCH=$(git rev-parse --abbrev-ref HEAD) \
|
||||
./frontend
|
||||
docker push git.aranroig.com/${{ secrets.REGISTRY_USER }}/dragonroll-frontend:latest
|
||||
|
||||
- name: Build backend
|
||||
@@ -23,17 +46,17 @@ jobs:
|
||||
docker build -t git.aranroig.com/${{ secrets.REGISTRY_USER }}/dragonroll-backend:latest ./backend
|
||||
docker push git.aranroig.com/${{ secrets.REGISTRY_USER }}/dragonroll-backend:latest
|
||||
|
||||
# - name: Copy files
|
||||
# run: |
|
||||
# scp docker-compose.yml deploy@${{ secrets.DEPLOY_HOST}}:/var/www/app/
|
||||
# scp nginx.conf deploy@${{ secrets.DEPLOY_HOST }}:/var/www/app/nginx.conf
|
||||
- name: Copy files
|
||||
run: |
|
||||
scp docker-compose.yml deploy@${{ secrets.DEPLOY_HOST}}:/var/www/app/
|
||||
scp nginx.conf deploy@${{ secrets.DEPLOY_HOST }}:/var/www/app/nginx.conf
|
||||
|
||||
#- name: Deploy
|
||||
# run: |
|
||||
# ssh deploy@${{ secrets.DEPLOY_HOST }} << 'EOF'
|
||||
# echo "${{ secrets.REGISTRY_PASSWORD }}" | docker login git.aranroig.com -u "${{ secrets.REGISTRY_USER }}" --password-stdin
|
||||
# cd /var/www/app/
|
||||
# docker-compose pull
|
||||
# docker-compose up -d
|
||||
# EOF
|
||||
- name: Deploy
|
||||
run: |
|
||||
ssh deploy@${{ secrets.DEPLOY_HOST }} << 'EOF'
|
||||
echo "${{ secrets.REGISTRY_PASSWORD }}" | docker login git.aranroig.com -u "${{ secrets.REGISTRY_USER }}" --password-stdin
|
||||
cd /var/www/app/
|
||||
docker compose pull
|
||||
docker compose up -d
|
||||
EOF
|
||||
|
||||
|
||||
@@ -7,6 +7,7 @@ const path = require('path');
|
||||
const dotenv = require('dotenv');
|
||||
|
||||
if(process.env.NODE_ENV) {
|
||||
console.log(`.env.${process.env.NODE_ENV}`);
|
||||
dotenv.config({
|
||||
path: `.env.${process.env.NODE_ENV}`
|
||||
});
|
||||
@@ -20,7 +21,7 @@ const connectDB = require("./db");
|
||||
|
||||
// PUBLIC
|
||||
const uploadDir = path.join(__dirname, 'uploads');
|
||||
app.use('/public', express.static(uploadDir));
|
||||
app.use('/api/public', express.static(uploadDir));
|
||||
|
||||
// JSON LIMIT EXPRESS
|
||||
app.use(express.json({ limit: '50mb' }));
|
||||
@@ -45,13 +46,15 @@ app.use(cors({
|
||||
}));
|
||||
|
||||
// ROUTES (NO AUTH)
|
||||
app.use('/user', require('./routes/user'));
|
||||
app.use('/api/user', require('./routes/user'));
|
||||
|
||||
// AUTH
|
||||
checkAuth = passport.authenticate('jwt', { session: false });
|
||||
app.use(checkAuth);
|
||||
|
||||
// ROUTES WITH AUTH
|
||||
app.use('/api/campaign', require('./routes/campaign'));
|
||||
app.use('/api/note', require('./routes/note'));
|
||||
/*
|
||||
app.use('/campaign', require('./routes/campaign'));
|
||||
app.use('/maps', require('./routes/map'));
|
||||
|
||||
17
backend/src/models/Campaign.js
Normal file
17
backend/src/models/Campaign.js
Normal file
@@ -0,0 +1,17 @@
|
||||
const mongoose = require("mongoose");
|
||||
const Schema = mongoose.Schema;
|
||||
|
||||
const CampaignSchema = new Schema({
|
||||
name: {type: String, required: true},
|
||||
description: {type: String},
|
||||
color: {type: String},
|
||||
createdBy: {
|
||||
type: Schema.Types.ObjectId,
|
||||
ref: 'User',
|
||||
required: true
|
||||
},
|
||||
date: { type: Date, default: Date.now},
|
||||
settings: { type: Object }
|
||||
});
|
||||
|
||||
module.exports = mongoose.model('Campaign', CampaignSchema);
|
||||
15
backend/src/models/Note.js
Normal file
15
backend/src/models/Note.js
Normal file
@@ -0,0 +1,15 @@
|
||||
const mongoose = require('mongoose');
|
||||
const Schema = mongoose.Schema;
|
||||
|
||||
const NoteSchema = new Schema({
|
||||
title: { type: String },
|
||||
content: { type: String },
|
||||
campaign: {
|
||||
type: Schema.Types.ObjectId,
|
||||
ref: 'Campaign',
|
||||
required: true
|
||||
},
|
||||
date: { type: Date, default: Date.now }
|
||||
});
|
||||
|
||||
module.exports = mongoose.model('Note', NoteSchema);
|
||||
46
backend/src/routes/campaign.js
Normal file
46
backend/src/routes/campaign.js
Normal file
@@ -0,0 +1,46 @@
|
||||
const express = require('express')
|
||||
const router = express.Router();
|
||||
|
||||
const Campaign = require("../models/Campaign");
|
||||
|
||||
router.post('/create', async (req, res) => {
|
||||
try {
|
||||
const { name, description, color, settings } = req.body;
|
||||
const newCampaign = new Campaign({
|
||||
name,
|
||||
description,
|
||||
color,
|
||||
settings,
|
||||
createdBy: req.user.id
|
||||
});
|
||||
await newCampaign.save();
|
||||
res.json({ status: "ok", campaign: newCampaign });
|
||||
} catch (err) {
|
||||
console.error(err);
|
||||
res.json({ status: "error", msg: "errors.internal" });
|
||||
}
|
||||
});
|
||||
|
||||
router.get('/list', async (req, res) => {
|
||||
try {
|
||||
const campaigns = await Campaign.find({ createdBy: req.user.id });
|
||||
res.json({ status: "ok", campaigns });
|
||||
} catch (err) {
|
||||
res.json({ status: "error", msg: "errors.internal", err });
|
||||
}
|
||||
});
|
||||
|
||||
router.get('/retrieve/:id', async (req, res) => {
|
||||
try {
|
||||
if (!req.user?.id) return res.status(401).json({ status: "error", msg: "errors.unauthorized" });
|
||||
|
||||
const campaign = await Campaign.findOne({ _id: req.params.id, createdBy: req.user.id });
|
||||
if (!campaign) return res.json({ status: "error", msg: "errors.not-found" });
|
||||
|
||||
res.json({ status: "ok", campaign });
|
||||
} catch (err) {
|
||||
res.json({ status: "error", msg: "errors.internal", err });
|
||||
}
|
||||
});
|
||||
|
||||
module.exports = router;
|
||||
86
backend/src/routes/note.js
Normal file
86
backend/src/routes/note.js
Normal file
@@ -0,0 +1,86 @@
|
||||
const express = require('express');
|
||||
const router = express.Router();
|
||||
|
||||
const Campaign = require("../models/Campaign");
|
||||
const Note = require("../models/Note");
|
||||
|
||||
async function userOwnsCampaign(campaignId, userId) {
|
||||
const campaign = await Campaign.findOne({ _id: campaignId, createdBy: userId }).lean();
|
||||
return Boolean(campaign);
|
||||
}
|
||||
|
||||
router.get('/list', async (req, res) => {
|
||||
try {
|
||||
const { campaign } = req.query;
|
||||
if (!campaign) return res.json({ status: "error", msg: "errors.missing-data" });
|
||||
|
||||
const hasAccess = await userOwnsCampaign(campaign, req.user.id);
|
||||
if (!hasAccess) return res.json({ status: "error", msg: "unauthorized" });
|
||||
|
||||
const notes = await Note.find({ campaign })
|
||||
.select('_id title content date campaign')
|
||||
.sort({ date: -1 })
|
||||
.lean();
|
||||
|
||||
res.json({ status: "ok", notes });
|
||||
} catch (err) {
|
||||
console.error(err);
|
||||
res.json({ status: "error", msg: "errors.internal" });
|
||||
}
|
||||
});
|
||||
|
||||
router.post('/create', async (req, res) => {
|
||||
try {
|
||||
const { title, content, campaign } = req.body;
|
||||
const hasAccess = await userOwnsCampaign(campaign, req.user.id);
|
||||
if (!hasAccess) return res.json({ status: "error", msg: "unauthorized" });
|
||||
|
||||
const newNote = new Note({
|
||||
title,
|
||||
content,
|
||||
campaign
|
||||
});
|
||||
await newNote.save();
|
||||
res.json({ status: "ok", note: newNote });
|
||||
} catch (err) {
|
||||
console.error(err);
|
||||
res.json({ status: "error", msg: "errors.internal" });
|
||||
}
|
||||
});
|
||||
|
||||
router.post('/update', async (req, res) => {
|
||||
try {
|
||||
const { id, title, content } = req.body;
|
||||
const note = await Note.findById(id);
|
||||
if (!note) return res.json({ status: "error", msg: "errors.notfound" });
|
||||
const hasAccess = await userOwnsCampaign(note.campaign, req.user.id);
|
||||
if (!hasAccess) return res.json({ status: "error", msg: "unauthorized" });
|
||||
|
||||
if(title) note.title = title;
|
||||
note.content = content;
|
||||
note.date = Date.now();
|
||||
await note.save();
|
||||
res.json({ status: "ok", note });
|
||||
} catch (err) {
|
||||
console.error(err);
|
||||
res.json({ status: "error", msg: "errors.internal" });
|
||||
}
|
||||
});
|
||||
|
||||
router.post('/delete', async (req, res) => {
|
||||
try {
|
||||
const { id } = req.body;
|
||||
const note = await Note.findById(id);
|
||||
if (!note) return res.json({ status: "error", msg: "errors.notfound" });
|
||||
const hasAccess = await userOwnsCampaign(note.campaign, req.user.id);
|
||||
if (!hasAccess) return res.json({ status: "error", msg: "unauthorized" });
|
||||
|
||||
await note.remove();
|
||||
res.json({ status: "ok" });
|
||||
} catch (err) {
|
||||
console.error(err);
|
||||
res.json({ status: "error", msg: "errors.internal" });
|
||||
}
|
||||
});
|
||||
|
||||
module.exports = router;
|
||||
21
docker-compose.yml
Normal file
21
docker-compose.yml
Normal file
@@ -0,0 +1,21 @@
|
||||
version: "3.9"
|
||||
|
||||
services:
|
||||
nginx:
|
||||
image: nginx:latest
|
||||
ports:
|
||||
- "3000:80"
|
||||
volumes:
|
||||
- ./nginx.conf:/etc/nginx/nginx.conf:ro
|
||||
depends_on:
|
||||
- frontend
|
||||
- backend
|
||||
restart: always
|
||||
|
||||
frontend:
|
||||
image: git.aranroig.com/syndria98/dragonroll-frontend:latest
|
||||
restart: always
|
||||
|
||||
backend:
|
||||
image: git.aranroig.com/syndria98/dragonroll-backend:latest
|
||||
restart: always
|
||||
1
frontend/.env.production
Normal file
1
frontend/.env.production
Normal file
@@ -0,0 +1 @@
|
||||
NUXT_PUBLIC_API_BASE_URL=https://dragonroll.aranroig.com/api
|
||||
2
frontend/.gitignore
vendored
2
frontend/.gitignore
vendored
@@ -21,4 +21,4 @@ logs
|
||||
# Local env files
|
||||
.env
|
||||
.env.*
|
||||
!.env.example
|
||||
!.env.production
|
||||
@@ -1,6 +1,14 @@
|
||||
# ---------- Build Stage ----------
|
||||
FROM node:20-alpine AS builder
|
||||
|
||||
ARG NUXT_PUBLIC_GIT_COMMIT
|
||||
ARG NUXT_PUBLIC_GIT_TAG
|
||||
ARG NUXT_PUBLIC_GIT_BRANCH
|
||||
|
||||
ENV NUXT_PUBLIC_GIT_COMMIT=$NUXT_PUBLIC_GIT_COMMIT
|
||||
ENV NUXT_PUBLIC_GIT_TAG=$NUXT_PUBLIC_GIT_TAG
|
||||
ENV NUXT_PUBLIC_GIT_BRANCH=$NUXT_PUBLIC_GIT_BRANCH
|
||||
|
||||
WORKDIR /app
|
||||
|
||||
# Copy package files
|
||||
|
||||
@@ -1,75 +1,19 @@
|
||||
# Nuxt Minimal Starter
|
||||
# Frontend
|
||||
|
||||
Look at the [Nuxt documentation](https://nuxt.com/docs/getting-started/introduction) to learn more.
|
||||
This folder contains the React frontend for Dragonroll, an open-source role-playing game helper.
|
||||
|
||||
## Setup
|
||||
## Features
|
||||
|
||||
Make sure to install dependencies:
|
||||
- Campaign management with character tracking
|
||||
- Player note sharing (markdown)
|
||||
- Audio for in-person campaigns
|
||||
- Encounter planning
|
||||
- Item and spell management
|
||||
|
||||
```bash
|
||||
# npm
|
||||
npm install
|
||||
## Files
|
||||
|
||||
# pnpm
|
||||
pnpm install
|
||||
- `app/` - Application components and services
|
||||
- `app/services/` - Frontend services (Campaign, User, Window, etc.)
|
||||
- Styling and React components
|
||||
|
||||
# yarn
|
||||
yarn install
|
||||
|
||||
# bun
|
||||
bun install
|
||||
```
|
||||
|
||||
## Development Server
|
||||
|
||||
Start the development server on `http://localhost:3000`:
|
||||
|
||||
```bash
|
||||
# npm
|
||||
npm run dev
|
||||
|
||||
# pnpm
|
||||
pnpm dev
|
||||
|
||||
# yarn
|
||||
yarn dev
|
||||
|
||||
# bun
|
||||
bun run dev
|
||||
```
|
||||
|
||||
## Production
|
||||
|
||||
Build the application for production:
|
||||
|
||||
```bash
|
||||
# npm
|
||||
npm run build
|
||||
|
||||
# pnpm
|
||||
pnpm build
|
||||
|
||||
# yarn
|
||||
yarn build
|
||||
|
||||
# bun
|
||||
bun run build
|
||||
```
|
||||
|
||||
Locally preview production build:
|
||||
|
||||
```bash
|
||||
# npm
|
||||
npm run preview
|
||||
|
||||
# pnpm
|
||||
pnpm preview
|
||||
|
||||
# yarn
|
||||
yarn preview
|
||||
|
||||
# bun
|
||||
bun run preview
|
||||
```
|
||||
|
||||
Check out the [deployment documentation](https://nuxt.com/docs/getting-started/deployment) for more information.
|
||||
See the main README for complete information.
|
||||
|
||||
@@ -7,10 +7,16 @@ import { CreateWindow } from '@/services/Windows'
|
||||
import { GetUser, HasAdmin } from './services/User';
|
||||
import TooltipManager from './components/managers/TooltipManager.vue';
|
||||
import ContextMenuManager from './components/managers/ContextMenuManager.vue';
|
||||
import { useCampaignService } from './services/Campaign';
|
||||
|
||||
const { RestoreCampaign } = useCampaignService();
|
||||
|
||||
async function start(){
|
||||
if(GetUser()){
|
||||
CreateWindow('main_menu');
|
||||
const restoredCampaign = await RestoreCampaign();
|
||||
if (!restoredCampaign) {
|
||||
CreateWindow('main_menu');
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
@@ -56,4 +62,4 @@ onMounted(() => {
|
||||
width: 100%;
|
||||
height: 100vh;
|
||||
}
|
||||
</style>
|
||||
</style>
|
||||
|
||||
@@ -6,6 +6,7 @@ $themes: (
|
||||
background-light: #202020,
|
||||
background-line: #202324,
|
||||
background-fore: #10141f,
|
||||
background-soft: #20202077,
|
||||
|
||||
window-handle-background: #191919,
|
||||
window-background: #141414,
|
||||
@@ -18,6 +19,8 @@ $themes: (
|
||||
|
||||
toast-background: #202020,
|
||||
|
||||
note-border: #202324,
|
||||
|
||||
hover: #21262d,
|
||||
selected: #4a4a4b,
|
||||
border-color: #819796,
|
||||
@@ -28,6 +31,7 @@ $themes: (
|
||||
|
||||
red: #e06c75,
|
||||
green: #98c379,
|
||||
gray: #cccccc,
|
||||
|
||||
icon-invert: 100%
|
||||
),
|
||||
@@ -48,6 +52,8 @@ $themes: (
|
||||
|
||||
toast-background: #f0f0f0,
|
||||
|
||||
note-border: #e0e0e0,
|
||||
|
||||
border-color: #e0e0e0,
|
||||
border: #f0f0f0,
|
||||
hover: #e9e9e9,
|
||||
@@ -58,6 +64,7 @@ $themes: (
|
||||
|
||||
red: #e06c75,
|
||||
green: #98c379,
|
||||
gray: #cccccc,
|
||||
|
||||
icon-invert: 0%
|
||||
)
|
||||
|
||||
@@ -106,7 +106,8 @@ textarea {
|
||||
background-color: var(--color-background-softer);
|
||||
padding: 12px;
|
||||
color: var(--color-text);
|
||||
border: none;
|
||||
border-radius: 6px;
|
||||
border: solid 1px var(--color-border);;
|
||||
}
|
||||
|
||||
input[type=text]:focus, input[type=password]:focus, input[type=email]:focus {
|
||||
|
||||
62
frontend/app/components/layouts/ColorPicker.vue
Normal file
62
frontend/app/components/layouts/ColorPicker.vue
Normal file
@@ -0,0 +1,62 @@
|
||||
<script setup>
|
||||
import { onMounted, ref } from 'vue';
|
||||
|
||||
const color = ref("");
|
||||
const colorValue = ref(null);
|
||||
const colorPicker = ref(null);
|
||||
|
||||
const selectedColorCode = ref(null);
|
||||
|
||||
onMounted(() => {
|
||||
colorValue.value.addEventListener('click', () => {
|
||||
colorPicker.value.click();
|
||||
})
|
||||
|
||||
colorPicker.value.addEventListener('input', (event) => {
|
||||
let newColor = event.target.value;
|
||||
colorValue.value.classList.remove('unselected');
|
||||
colorValue.value.style.backgroundColor = newColor;
|
||||
color.value = newColor;
|
||||
selectedColorCode.value.textContent = color.value.toUpperCase();
|
||||
});
|
||||
});
|
||||
|
||||
let GetColor = () => color.value;
|
||||
|
||||
defineExpose({ GetColor });
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<input type="color" id="colorPicker" ref="colorPicker">
|
||||
<div class="color-value unselected" ref="colorValue">
|
||||
<span class="selected-color-code" ref="selectedColorCode"></span>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style lang="scss">
|
||||
#colorPicker {
|
||||
display: none;
|
||||
}
|
||||
|
||||
.color-value {
|
||||
width: 100px;
|
||||
text-align: center;
|
||||
font-weight: bold;
|
||||
font-size: 16px;
|
||||
height: 20px;
|
||||
border-radius: 10px;
|
||||
|
||||
&.unselected {
|
||||
background-image: linear-gradient(45deg, #808080 25%, transparent 25%), linear-gradient(-45deg, #808080 25%, transparent 25%), linear-gradient(45deg, transparent 75%, #808080 75%), linear-gradient(-45deg, transparent 75%, #808080 75%);
|
||||
background-size: 10px 10px;
|
||||
background-position: 0 0, 0 5px, 5px -5px, -5px 0px;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
.selected-color-code {
|
||||
font-size: 12px;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
}
|
||||
</style>
|
||||
@@ -4,17 +4,16 @@ import { AddContextMenu, HideContextMenu } from '@/services/ContextMenu';
|
||||
const props = defineProps(['options', 'onselect', 'selected', 'keyFunc']);
|
||||
const options = props.options;
|
||||
const selectCallback = props.onselect;
|
||||
|
||||
const initialSelect = props.selected;
|
||||
const dropdown = ref(null);
|
||||
|
||||
const selected = ref(initialSelect);
|
||||
const selected = ref(null);
|
||||
|
||||
onMounted(() => {
|
||||
if(props.keyFunc == undefined) props.keyFunc = (option) => option;
|
||||
else selected.value = props.keyFunc(initialSelect);
|
||||
|
||||
selected.value = props.keyFunc(initialSelect);
|
||||
|
||||
let context = [];
|
||||
if(props.selected == undefined) selected.value = "undefined";
|
||||
watch(() => props.selected, () => {
|
||||
selected.value = props.keyFunc(props.selected);
|
||||
});
|
||||
|
||||
@@ -1,18 +1,41 @@
|
||||
<script setup>
|
||||
import { watch } from 'vue';
|
||||
import Content from '../viewer/content/Content.vue';
|
||||
import StatusBar from '../viewer/statusbar/StatusBar.vue';
|
||||
import TopBar from '../viewer/TopBar.vue';
|
||||
|
||||
import { ShowContent } from '../../services/Content.js';
|
||||
import { useCampaignService } from '~/services/Campaign.js';
|
||||
import ContentSidebar from '../partials/ContentSidebar.vue';
|
||||
|
||||
const { Campaign } = useCampaignService();
|
||||
|
||||
watch(Campaign, () => {
|
||||
if(Campaign.value) ShowContent.value = true;
|
||||
}, { immediate: true });
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div v-show="ShowContent">
|
||||
<div v-show="ShowContent" class="content-manager">
|
||||
<TopBar></TopBar>
|
||||
<Content></Content>
|
||||
<div class="content-layout">
|
||||
<ContentSidebar></ContentSidebar>
|
||||
<Content></Content>
|
||||
</div>
|
||||
<StatusBar></StatusBar>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
</style>
|
||||
.content-manager {
|
||||
height: 100%;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.content-layout {
|
||||
min-height: 0;
|
||||
flex: 1;
|
||||
display: flex;
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -4,13 +4,9 @@ import { windows, getComponent } from '@/services/Windows';
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="window-container">
|
||||
<TransitionGroup name="window" tag="div">
|
||||
<div v-for="win in windows" :key="win.id">
|
||||
<component :is="getComponent(win.type)" :data="win" />
|
||||
</div>
|
||||
</TransitionGroup>
|
||||
</div>
|
||||
<TransitionGroup name="window" tag="div">
|
||||
<component v-for="win in windows" :key="win.id" :is="getComponent(win.type)" :data="win" />
|
||||
</TransitionGroup>
|
||||
</template>
|
||||
|
||||
|
||||
@@ -21,16 +17,18 @@ import { windows, getComponent } from '@/services/Windows';
|
||||
}
|
||||
.window-enter-from,
|
||||
.window-leave-to {
|
||||
transition: all 0.15s ease;
|
||||
opacity: 0;
|
||||
transform: translateY(15px);
|
||||
}
|
||||
.window-move {
|
||||
transition: transform 0.15s ease;
|
||||
}
|
||||
|
||||
.window-wrapper {
|
||||
background-color: var(--color-window-background);
|
||||
|
||||
/* backdrop-filter: blur(10px); */
|
||||
position: fixed;
|
||||
|
||||
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
|
||||
86
frontend/app/components/partials/CampaignEntry.vue
Normal file
86
frontend/app/components/partials/CampaignEntry.vue
Normal file
@@ -0,0 +1,86 @@
|
||||
<script setup>
|
||||
|
||||
import { onMounted, ref } from 'vue';
|
||||
|
||||
import { AddSound } from '../../services/Sound';
|
||||
import { useCampaignService } from '~/services/Campaign';
|
||||
import { ClearWindow } from '~/services/Windows';
|
||||
const { SetCampaign } = useCampaignService();
|
||||
|
||||
const props = defineProps(['data']);
|
||||
const data = props.data;
|
||||
|
||||
const title = ref("");
|
||||
|
||||
const container = ref(null);
|
||||
|
||||
onMounted(() => {
|
||||
title.value = data.name;
|
||||
if (data.color && container.value) {
|
||||
container.value.style.background = `linear-gradient(90deg, ${data.color}, ${data.color}44)`;
|
||||
}
|
||||
|
||||
AddSound(container.value)
|
||||
});
|
||||
|
||||
function ViewCampaign(){
|
||||
SetCampaign(data);
|
||||
ClearWindow({type: "main_menu"});
|
||||
}
|
||||
</script>
|
||||
|
||||
|
||||
<template>
|
||||
<div class="campaign-entry-container" ref="container">
|
||||
<div class="main-campaign-entry-container-inner">
|
||||
<img class="campaign-icon" src="/img/def-avatar.jpg" draggable="false">
|
||||
<div class="campaign-info">
|
||||
<b>{{ title }}</b>
|
||||
</div>
|
||||
|
||||
<div class="campaign-user-actions">
|
||||
<button class="btn-primary button-small sound-click" v-on:click.prevent="ViewCampaign">{{ $t('general.open')}}</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style lang="scss" scoped>
|
||||
.button-small {
|
||||
height: 32px;
|
||||
padding: 10px;
|
||||
}
|
||||
|
||||
.campaign-entry-container {
|
||||
background-color: var(--color-background-softer);
|
||||
width: 100%;
|
||||
user-select: none;
|
||||
|
||||
border-bottom: 1px solid var(--color-border);
|
||||
&:first-child {
|
||||
border-top: 1px solid var(--color-border);
|
||||
}
|
||||
}
|
||||
|
||||
.main-campaign-entry-container-inner {
|
||||
padding: 10px;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
|
||||
justify-content: center;
|
||||
}
|
||||
|
||||
.campaign-info {
|
||||
text-align: left;
|
||||
margin-left: 10px;
|
||||
}
|
||||
|
||||
.campaign-icon {
|
||||
width: 40px;
|
||||
height: 40px;
|
||||
}
|
||||
|
||||
.campaign-user-actions {
|
||||
margin-left: auto;
|
||||
}
|
||||
</style>
|
||||
370
frontend/app/components/partials/ContentSidebar.vue
Normal file
370
frontend/app/components/partials/ContentSidebar.vue
Normal file
@@ -0,0 +1,370 @@
|
||||
<script setup>
|
||||
import { computed, onMounted, onUnmounted, ref, watch } from 'vue';
|
||||
|
||||
import { useCampaignService } from '~/services/Campaign.js';
|
||||
import { emitter } from '~/services/Emitter';
|
||||
import Server from '~/services/Server';
|
||||
|
||||
const { Campaign } = useCampaignService();
|
||||
const notes = ref([]);
|
||||
const loadingNotes = ref(false);
|
||||
const notesError = ref('');
|
||||
const sidebarCollapsed = ref(false);
|
||||
|
||||
const campaignId = computed(() => {
|
||||
return Campaign.value?._id ?? Campaign.value?.id ?? null;
|
||||
});
|
||||
|
||||
const notesMeta = computed(() => {
|
||||
const count = notes.value.length;
|
||||
return `${count} ${count === 1 ? 'note' : 'notes'}`;
|
||||
});
|
||||
|
||||
async function fetchCampaignNotes() {
|
||||
if (!campaignId.value) {
|
||||
notes.value = [];
|
||||
notesError.value = '';
|
||||
return;
|
||||
}
|
||||
|
||||
loadingNotes.value = true;
|
||||
notesError.value = '';
|
||||
|
||||
try {
|
||||
const response = await Server().get('/note/list', {
|
||||
params: {
|
||||
campaign: campaignId.value
|
||||
}
|
||||
});
|
||||
|
||||
if (response.data.status !== 'ok') {
|
||||
notes.value = [];
|
||||
notesError.value = response.data.msg ?? 'Unable to load notes.';
|
||||
return;
|
||||
}
|
||||
|
||||
notes.value = response.data.notes.map((note) => {
|
||||
return {
|
||||
key: note._id,
|
||||
title: note.title,
|
||||
text: note.content ?? '',
|
||||
date: note.date
|
||||
};
|
||||
});
|
||||
} catch (error) {
|
||||
notes.value = [];
|
||||
notesError.value = 'Unable to load notes.';
|
||||
} finally {
|
||||
loadingNotes.value = false;
|
||||
}
|
||||
}
|
||||
|
||||
function toggleSidebar() {
|
||||
sidebarCollapsed.value = !sidebarCollapsed.value;
|
||||
}
|
||||
|
||||
async function createNote() {
|
||||
if (!Campaign.value) {
|
||||
return;
|
||||
}
|
||||
|
||||
const campaignId = Campaign.value?._id
|
||||
|
||||
try {
|
||||
const response = await Server().post('/note/create', {
|
||||
title: 'New note',
|
||||
content: "",
|
||||
campaign: campaignId
|
||||
});
|
||||
|
||||
if (response.data.status !== 'ok') {
|
||||
return;
|
||||
}
|
||||
|
||||
emitter.emit('note-created', response.data.note);
|
||||
} catch (err) {
|
||||
console.log(err);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
function openNote(note) {
|
||||
emitter.emit('push-note', note);
|
||||
}
|
||||
|
||||
function handleNoteCreated(note) {
|
||||
if (!note) {
|
||||
return;
|
||||
}
|
||||
|
||||
const noteCampaignId = note.campaign?._id ?? note.campaign ?? null;
|
||||
if (campaignId.value && noteCampaignId && noteCampaignId !== campaignId.value) {
|
||||
return;
|
||||
}
|
||||
|
||||
const createdNote = {
|
||||
key: note._id,
|
||||
title: note.title,
|
||||
text: note.content ?? '',
|
||||
date: note.date
|
||||
};
|
||||
|
||||
notes.value = notes.value.filter((currentNote) => {
|
||||
return currentNote.key !== createdNote.key;
|
||||
});
|
||||
notes.value.unshift(createdNote);
|
||||
openNote(createdNote);
|
||||
}
|
||||
|
||||
onMounted(() => {
|
||||
emitter.on('note-created', handleNoteCreated);
|
||||
});
|
||||
|
||||
onUnmounted(() => {
|
||||
emitter.off('note-created', handleNoteCreated);
|
||||
});
|
||||
|
||||
watch(Campaign, () => {
|
||||
fetchCampaignNotes();
|
||||
}, { immediate: true });
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="sidebar-shell">
|
||||
<nav class="sidebar-actions" aria-label="Campaign tools">
|
||||
<button
|
||||
class="sidebar-action"
|
||||
type="button"
|
||||
@click="toggleSidebar"
|
||||
:aria-expanded="(!sidebarCollapsed).toString()"
|
||||
aria-controls="campaign-notes-list"
|
||||
:title="sidebarCollapsed ? 'Expand notes' : 'Collapse notes'"
|
||||
:aria-label="sidebarCollapsed ? 'Expand notes' : 'Collapse notes'"
|
||||
>
|
||||
<img
|
||||
class="sidebar-action-icon"
|
||||
:src="sidebarCollapsed ? '/icons/iconoir/regular/nav-arrow-right.svg' : '/icons/iconoir/regular/nav-arrow-left.svg'"
|
||||
alt=""
|
||||
aria-hidden="true"
|
||||
>
|
||||
</button>
|
||||
|
||||
<button
|
||||
class="sidebar-action"
|
||||
type="button"
|
||||
@click="createNote"
|
||||
:disabled="!Campaign"
|
||||
title="New note"
|
||||
aria-label="New note"
|
||||
>
|
||||
<img class="sidebar-action-icon" src="/icons/iconoir/regular/plus.svg" alt="" aria-hidden="true">
|
||||
</button>
|
||||
|
||||
<button
|
||||
class="sidebar-action"
|
||||
type="button"
|
||||
@click="fetchCampaignNotes"
|
||||
:disabled="!Campaign || loadingNotes"
|
||||
title="Refresh notes"
|
||||
aria-label="Refresh notes"
|
||||
>
|
||||
<img class="sidebar-action-icon" src="/icons/iconoir/regular/refresh.svg" alt="" aria-hidden="true">
|
||||
</button>
|
||||
</nav>
|
||||
|
||||
<aside class="notes-sidebar" :class="{ collapsed: sidebarCollapsed }">
|
||||
<div class="sidebar-header">
|
||||
<div class="sidebar-copy">
|
||||
<span class="sidebar-eyebrow">Campaign</span>
|
||||
<strong class="sidebar-title">Notes</strong>
|
||||
<span class="sidebar-meta">{{ notesMeta }}</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="campaign-notes-list" class="sidebar-list">
|
||||
<div v-if="loadingNotes" class="sidebar-state">
|
||||
Loading notes...
|
||||
</div>
|
||||
|
||||
<div v-else-if="notesError" class="sidebar-state error">
|
||||
{{ notesError }}
|
||||
</div>
|
||||
|
||||
<div v-else-if="notes.length === 0" class="sidebar-state">
|
||||
No notes in this campaign yet.
|
||||
</div>
|
||||
|
||||
<template v-else>
|
||||
<button
|
||||
v-for="note in notes"
|
||||
:key="note.key"
|
||||
type="button"
|
||||
class="note-link"
|
||||
@click="openNote(note)"
|
||||
>
|
||||
<span class="note-link-title">{{ note.title }}</span>
|
||||
</button>
|
||||
</template>
|
||||
</div>
|
||||
</aside>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
.sidebar-shell {
|
||||
min-height: 0;
|
||||
flex-shrink: 0;
|
||||
display: flex;
|
||||
}
|
||||
|
||||
.sidebar-actions {
|
||||
width: 32px;
|
||||
min-width: 32px;
|
||||
padding: 8px 6px;
|
||||
border-right: 1px solid var(--color-border);
|
||||
background-color: var(--color-background-light);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: center;
|
||||
gap: 4px;
|
||||
}
|
||||
|
||||
.sidebar-action {
|
||||
width: 34px;
|
||||
height: 34px;
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
border: 1px solid var(--color-border);
|
||||
border-radius: 8px;
|
||||
background: var(--color-background-soft);
|
||||
box-shadow: none;
|
||||
display: flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.sidebar-action:hover {
|
||||
background: var(--color-button-hover);
|
||||
}
|
||||
|
||||
.sidebar-action:disabled {
|
||||
opacity: 0.45;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.sidebar-action-icon {
|
||||
width: 18px;
|
||||
height: 18px;
|
||||
filter: invert(var(--color-icon-invert));
|
||||
}
|
||||
|
||||
.notes-sidebar {
|
||||
width: 280px;
|
||||
min-width: 280px;
|
||||
border-right: 1px solid var(--color-border);
|
||||
background-color: var(--color-background-light);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
overflow: hidden;
|
||||
transition: width 0.2s ease, min-width 0.2s ease, border-color 0.2s ease;
|
||||
}
|
||||
|
||||
.notes-sidebar.collapsed {
|
||||
width: 0;
|
||||
min-width: 0;
|
||||
border-right-color: transparent;
|
||||
}
|
||||
|
||||
.sidebar-header {
|
||||
width: 280px;
|
||||
min-width: 280px;
|
||||
box-sizing: border-box;
|
||||
padding: 12px;
|
||||
display: flex;
|
||||
align-items: flex-start;
|
||||
gap: 10px;
|
||||
border-bottom: 1px solid var(--color-border);
|
||||
}
|
||||
|
||||
.sidebar-copy {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 2px;
|
||||
min-width: 0;
|
||||
}
|
||||
|
||||
.sidebar-eyebrow,
|
||||
.sidebar-meta {
|
||||
font-size: 12px;
|
||||
opacity: 0.7;
|
||||
}
|
||||
|
||||
.sidebar-title {
|
||||
line-height: 1.2;
|
||||
word-break: break-word;
|
||||
}
|
||||
|
||||
.sidebar-list {
|
||||
width: 280px;
|
||||
min-width: 280px;
|
||||
box-sizing: border-box;
|
||||
padding: 10px;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
overflow-y: auto;
|
||||
}
|
||||
|
||||
.sidebar-state {
|
||||
padding: 12px;
|
||||
border-radius: 10px;
|
||||
background: var(--color-background-soft);
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.sidebar-state.error {
|
||||
color: #9e2a2b;
|
||||
}
|
||||
|
||||
.note-link {
|
||||
width: 100%;
|
||||
padding: 6px;
|
||||
margin: 0;
|
||||
box-shadow: none;
|
||||
border: none;
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
align-items: flex-start;
|
||||
text-align: left;
|
||||
cursor: pointer;
|
||||
transition: transform 0.15s ease, background-color 0.15s ease;
|
||||
}
|
||||
|
||||
.note-link:hover {
|
||||
transform: translateX(2px);
|
||||
background: var(--color-background-light);
|
||||
}
|
||||
|
||||
.note-link-title {
|
||||
font-weight: 600;
|
||||
word-break: break-word;
|
||||
}
|
||||
|
||||
.note-link-date {
|
||||
font-size: 12px;
|
||||
opacity: 0.7;
|
||||
}
|
||||
|
||||
@media (max-width: 900px) {
|
||||
.notes-sidebar {
|
||||
width: 220px;
|
||||
min-width: 220px;
|
||||
}
|
||||
|
||||
.sidebar-header,
|
||||
.sidebar-list {
|
||||
width: 220px;
|
||||
min-width: 220px;
|
||||
}
|
||||
}
|
||||
</style>
|
||||
@@ -2,10 +2,9 @@
|
||||
import { onMounted, ref } from 'vue';
|
||||
import { GetUser, LogoutUser } from '@/services/User'
|
||||
|
||||
import Server from '@/services/Server'
|
||||
import Server, { getBaseUrl } from '@/services/Server'
|
||||
|
||||
import { CreateWindow, CreateChildWindow, ClearWindow, GetFirstWindowId } from '../../services/Windows';
|
||||
import { backendUrl } from '../../services/BackendURL';
|
||||
import Spinner from './Spinner.vue';
|
||||
const loadedIcon = ref(false);
|
||||
|
||||
@@ -21,7 +20,7 @@ function retrieveAvatar(){
|
||||
Server().get('/user/retrieve-avatar?username=' + GetUser().username)
|
||||
.then((response) => {
|
||||
if(response.data.image){
|
||||
const imgUrl = backendUrl + "public/" + response.data.image;
|
||||
const imgUrl = getBaseUrl() + "/public/" + response.data.image;
|
||||
|
||||
// Wait for the image to fully load
|
||||
const img = new Image();
|
||||
@@ -48,12 +47,6 @@ function LogOut(){
|
||||
CreateWindow('login');
|
||||
}
|
||||
|
||||
function EditProfile(){
|
||||
CreateChildWindow(GetFirstWindowId('main_menu'), 'edit_profile', {
|
||||
user: GetUser()
|
||||
});
|
||||
}
|
||||
|
||||
function EditSettings(){
|
||||
CreateChildWindow(GetFirstWindowId('main_menu'), 'settings', {
|
||||
user: GetUser()
|
||||
@@ -102,7 +95,6 @@ onMounted(() => {
|
||||
</div>
|
||||
|
||||
<div class="main-user-actions">
|
||||
<button class="btn-primary button-small sound-click" v-on:click.prevent="EditProfile">{{ $t("main-menu.edit-profile") }}</button>
|
||||
<button class="btn-primary button-small sound-click" v-on:click.prevent="EditSettings">{{ $t("main-menu.settings") }}</button>
|
||||
<button class="btn-primary button-small sound-click" v-on:click.prevent="LogOut">{{ $t("main-menu.log-out") }}</button>
|
||||
</div>
|
||||
|
||||
@@ -17,6 +17,7 @@ const config = useRuntimeConfig()
|
||||
bottom: 0;
|
||||
left: 0;
|
||||
user-select: none;
|
||||
font-size: 14px;
|
||||
}
|
||||
span{
|
||||
color: rgb(59, 59, 59);
|
||||
|
||||
@@ -1,17 +1,70 @@
|
||||
<script setup>
|
||||
import TopSearchBar from './topbar/TopSearchBar.vue';
|
||||
import { useCampaignService } from '~/services/Campaign';
|
||||
import { CreateWindow } from '~/services/Windows';
|
||||
import { SetShowContent } from '~/services/Content';
|
||||
|
||||
const { Campaign, SetCampaign } = useCampaignService();
|
||||
|
||||
const campaignName = computed(() => {
|
||||
return Campaign.value?.name ?? 'Campaign';
|
||||
});
|
||||
|
||||
async function createNote() {
|
||||
if (!Campaign.value) {
|
||||
return;
|
||||
}
|
||||
|
||||
const campaignId = Campaign.value?._id
|
||||
|
||||
try {
|
||||
const response = await Server().post('/note/create', {
|
||||
title: 'New note',
|
||||
content: content.value,
|
||||
campaign: campaignId
|
||||
});
|
||||
|
||||
if (response.data.status !== 'ok') {
|
||||
return;
|
||||
}
|
||||
|
||||
emitter.emit('note-created', response.data.note);
|
||||
} catch (err) {
|
||||
error.value = 'Unable to create note.';
|
||||
} finally {
|
||||
isSaving.value = false;
|
||||
}
|
||||
}
|
||||
|
||||
function exitToMainMenu() {
|
||||
SetCampaign(null);
|
||||
SetShowContent(false);
|
||||
CreateWindow('main_menu');
|
||||
}
|
||||
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="top-bar">
|
||||
<div class="left">
|
||||
<span class="top-bar-title"></span>
|
||||
<span class="top-bar-title">
|
||||
<img src="/img/logo.png" alt="Dragonroll Logo" class="logo">
|
||||
<span>{{ campaignName }}</span>
|
||||
</span>
|
||||
</div>
|
||||
<div class="center">
|
||||
<TopSearchBar></TopSearchBar>
|
||||
</div>
|
||||
<div class="right"></div>
|
||||
<div class="right">
|
||||
<button class="top-bar-button sound-click" type="button" @click="exitToMainMenu">
|
||||
<img class="top-bar-button-icon" src="/icons/iconoir/regular/nav-arrow-left.svg" alt="" aria-hidden="true">
|
||||
<span>Main Menu</span>
|
||||
</button>
|
||||
<button class="note-button sound-click" type="button" @click="createNote" :disabled="!Campaign">
|
||||
<img class="note-button-icon" src="/icons/iconoir/regular/plus.svg" alt="" aria-hidden="true">
|
||||
<span>New Note</span>
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@@ -25,17 +78,54 @@ import TopSearchBar from './topbar/TopSearchBar.vue';
|
||||
display: flex;
|
||||
}
|
||||
|
||||
.logo {
|
||||
height: 32px;
|
||||
width: 32px;
|
||||
position: absolute;
|
||||
top: 0px;
|
||||
left: 6px;
|
||||
}
|
||||
|
||||
.left, .right {
|
||||
flex: 1;
|
||||
}
|
||||
|
||||
.right {
|
||||
text-align: right;
|
||||
display: flex;
|
||||
justify-content: flex-end;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
padding-right: 10px;
|
||||
}
|
||||
|
||||
.top-bar-title {
|
||||
padding: 10px;
|
||||
display: flex;
|
||||
margin-left: 48px;
|
||||
font-weight: bold;
|
||||
}
|
||||
</style>
|
||||
|
||||
.top-bar-button,
|
||||
.note-button {
|
||||
height: 30px;
|
||||
padding: 0 12px;
|
||||
border: 1px solid var(--color-border);
|
||||
border-radius: 8px;
|
||||
background: var(--color-background-soft);
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 8px;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.note-button:disabled {
|
||||
opacity: 0.5;
|
||||
cursor: not-allowed;
|
||||
}
|
||||
|
||||
.top-bar-button-icon,
|
||||
.note-button-icon {
|
||||
width: 16px;
|
||||
height: 16px;
|
||||
}
|
||||
</style>
|
||||
|
||||
@@ -1,16 +1,13 @@
|
||||
<script setup>
|
||||
import NoteContainer from './NoteContainer.vue';
|
||||
|
||||
const emitter = useEmitter();
|
||||
|
||||
function hideSearch(){
|
||||
emitter.emit("hide-search-container");
|
||||
}
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="content" v-on:click="hideSearch">
|
||||
<NoteContainer></NoteContainer>
|
||||
<div class="content">
|
||||
<NoteContainer>
|
||||
|
||||
</NoteContainer>
|
||||
<!-- PowerMod -->
|
||||
</div>
|
||||
</template>
|
||||
@@ -18,6 +15,7 @@ function hideSearch(){
|
||||
<style scoped>
|
||||
.content {
|
||||
flex-grow: 1;
|
||||
min-width: 0; /* 👈 important */
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
align-items: center;
|
||||
|
||||
@@ -1,56 +1,104 @@
|
||||
<script setup>
|
||||
import { onMounted, ref } from 'vue';
|
||||
import { onMounted, onUnmounted, ref, createApp } from 'vue';
|
||||
import ToastManager from '~/components/managers/ToastManager.vue';
|
||||
import { emitter } from '~/services/Emitter';
|
||||
import { GetWidget, ParseMarkdown } from '~/services/Marker';
|
||||
import Server from '~/services/Server';
|
||||
import { DisplayToast } from '~/services/Toaster';
|
||||
import TestWidget from '../widgets/TestWidget.vue';
|
||||
|
||||
// import { GetNote, GetContent } from '@/services/Content';
|
||||
const props = defineProps(['text', 'title', 'noteKey']);
|
||||
const noteContent = ref(null); // Markdown text
|
||||
|
||||
const noteContent = ref(null);
|
||||
const sourceText = ref(''); // Original markdown source, used for editing
|
||||
const displayText = ref(''); // Compiled HTML from markdown
|
||||
|
||||
const emitter = useEmitter();
|
||||
|
||||
function gotoNote(){
|
||||
// emitter.emit('goto-note', props.noteKey);
|
||||
}
|
||||
|
||||
function closeNote(){
|
||||
// emitter.emit('delete-note', props.noteKey);
|
||||
}
|
||||
|
||||
|
||||
/*
|
||||
onMounted(() => {
|
||||
let content = GetContent();
|
||||
let elements = noteContent.value.getElementsByTagName('a');
|
||||
for(let i = 0, len = elements.length; i < len; i++) {
|
||||
let link = elements[i].pathname.split('/').slice(1).join('');
|
||||
link = decodeURIComponent(link);
|
||||
if(content[link] !== undefined){
|
||||
elements[i].onclick = function (event) {
|
||||
event.preventDefault();
|
||||
|
||||
GetNote(link, (result) => {
|
||||
emitter.emit("push-note", {key: link, text: "<h1>" + result.title + "</h1>" + result.html, title: result.title});
|
||||
});
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
elements[i].classList.add("error-link");
|
||||
elements[i].onclick = function (event) {
|
||||
event.preventDefault();
|
||||
return false;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
setTimeout(() => setupCallout(), 0);
|
||||
});
|
||||
const editingMode = ref(false);
|
||||
const editableTitle = ref(null);
|
||||
const title = ref(props.title);
|
||||
const displayTitle = ref('');
|
||||
|
||||
function closeNote(){
|
||||
emitter.emit('delete-note', props.noteKey);
|
||||
}
|
||||
|
||||
function gotoNote(){
|
||||
// emitter.emit('goto-note', props.noteKey);
|
||||
const compiledMarkdown = computed(() => {
|
||||
return ParseMarkdown(sourceText.value);
|
||||
});
|
||||
|
||||
function mountComponents() {
|
||||
// Should no need more
|
||||
const widget_types = ['display', 'inline'];
|
||||
widget_types.forEach((widget_type) => {
|
||||
const nodes = document.querySelectorAll('.vue-component-' + widget_type);
|
||||
nodes.forEach(el => {
|
||||
const app = createApp(GetWidget(widget_type, el.dataset.component), { content: el.dataset.content });
|
||||
app.mount(el);
|
||||
});
|
||||
});
|
||||
}
|
||||
///
|
||||
|
||||
function update(){
|
||||
displayText.value = compiledMarkdown.value;
|
||||
setTimeout(() => {
|
||||
setupCallout()
|
||||
mountComponents();
|
||||
}, 0);
|
||||
}
|
||||
|
||||
watch(sourceText, (newText) => {
|
||||
// update();
|
||||
});
|
||||
|
||||
onMounted(() => {
|
||||
sourceText.value = props.text;
|
||||
title.value = props.title;
|
||||
displayTitle.value = props.title;
|
||||
// window.addEventListener('keydown', handleKeydown);
|
||||
setTimeout(() => setupCallout(), 0);
|
||||
update();
|
||||
});
|
||||
|
||||
onUnmounted(() => {
|
||||
// window.removeEventListener('keydown', handleKeydown);
|
||||
});
|
||||
|
||||
function handleKeydown(e) {
|
||||
// Check for Ctrl + E (or Cmd + E on Mac)
|
||||
if ((e.ctrlKey || e.metaKey) && e.key.toLowerCase() === 'e') {
|
||||
e.preventDefault(); // prevent browser default behavior
|
||||
editingMode.value = !editingMode.value;
|
||||
if(!editingMode.value){
|
||||
update();
|
||||
SaveNote(); // Save when switching to display mode
|
||||
}
|
||||
return;
|
||||
}
|
||||
|
||||
if ((e.ctrlKey || e.metaKey) && e.key.toLowerCase() === 's') {
|
||||
e.preventDefault(); // prevent browser default behavior
|
||||
// Save the note (you can emit an event or call a method here)
|
||||
SaveNote();
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
function SaveNote(){
|
||||
Server().post('/note/update', {
|
||||
id: props.noteKey,
|
||||
content: sourceText.value,
|
||||
title: title.value,
|
||||
}).then((response) => {
|
||||
if(response.data.status !== 'ok'){
|
||||
// Handle error (e.g., show a notification)
|
||||
return;
|
||||
}
|
||||
// DisplayToast('green', "Note saved successfully.", 500);
|
||||
}).catch((error) => {
|
||||
// Handle error (e.g., show a notification)
|
||||
});
|
||||
}
|
||||
|
||||
function toggleCallout() {
|
||||
@@ -78,6 +126,10 @@ function toggleCallout() {
|
||||
}
|
||||
|
||||
function setupCallout() {
|
||||
if (!noteContent.value) {
|
||||
return;
|
||||
}
|
||||
|
||||
const collapsible = noteContent.value.getElementsByClassName(
|
||||
`callout is-collapsible`,
|
||||
);
|
||||
@@ -93,21 +145,29 @@ function setupCallout() {
|
||||
}
|
||||
}
|
||||
}
|
||||
*/
|
||||
|
||||
|
||||
const editTitle = (e) => {
|
||||
title.value = e.target.innerText;
|
||||
}
|
||||
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="note">
|
||||
<div class="note-stunt" v-on:click="gotoNote">
|
||||
<div class="note" @keydown="handleKeydown" tabindex="0">
|
||||
<div class="note-stunt">
|
||||
<div class="close-button" v-on:click="closeNote">
|
||||
<img class="icon" src="/icons/Pixelarticons/white/close.svg" alt="My Happy SVG"/>
|
||||
</div>
|
||||
<span>{{ title }}</span>
|
||||
</div>
|
||||
<div class="note-content-container">
|
||||
<div class="note-content" ref="noteContent" v-html="text"></div>
|
||||
<textarea v-model="sourceText" class="full-editor" v-if="editingMode"></textarea>
|
||||
<div v-else class="note-content" ref="noteContent">
|
||||
<h1 contenteditable="true" ref="editableTitle" @input="editTitle">{{ displayTitle }}</h1>
|
||||
<div ref="noteContent" v-html="displayText"></div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
@@ -124,6 +184,22 @@ function setupCallout() {
|
||||
user-select: none;
|
||||
}
|
||||
|
||||
.full-editor {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
box-sizing: border-box;
|
||||
|
||||
border: none;
|
||||
outline: none;
|
||||
resize: none;
|
||||
|
||||
padding: 20px;
|
||||
font-size: 16px;
|
||||
font-family: monospace; /* optional, gives document/editor feel */
|
||||
|
||||
padding-bottom: 400px; /* Small bottom margin */
|
||||
}
|
||||
|
||||
.close-button {
|
||||
height: 20px;
|
||||
width: 20px;
|
||||
@@ -131,6 +207,7 @@ function setupCallout() {
|
||||
display: flex;
|
||||
justify-content: center;
|
||||
cursor: pointer;
|
||||
filter: invert(var(--color-icon-invert));
|
||||
}
|
||||
|
||||
.note {
|
||||
@@ -138,34 +215,32 @@ function setupCallout() {
|
||||
max-width: 700px;
|
||||
overflow-y: auto;
|
||||
|
||||
border-color: var(--note-border-color);
|
||||
border-color: var(--color-note-border);
|
||||
border-width: 0px;
|
||||
border-right-width: 1px;
|
||||
border-style: solid;
|
||||
display: flex;
|
||||
|
||||
background-color: var(--background-color);
|
||||
background-color: var(--color-background);
|
||||
position: sticky;
|
||||
top: 0px;
|
||||
}
|
||||
|
||||
/* Contingut de cada nota */
|
||||
.note-content {
|
||||
padding-bottom: 60px;
|
||||
padding-bottom: 400px;
|
||||
overflow-y: auto;
|
||||
max-width: 600px;
|
||||
padding: 20px;
|
||||
}
|
||||
|
||||
.note-content-container {
|
||||
margin: 20px;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.note-content :deep(img) {
|
||||
max-width: 100%;
|
||||
height: auto;
|
||||
display: block; /* optional: avoids inline spacing issues */
|
||||
}
|
||||
</style>
|
||||
|
||||
<style>
|
||||
.note-content > h1 {
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.note-content .katex-display {
|
||||
max-width: 600px;
|
||||
}
|
||||
</style>
|
||||
@@ -1,5 +1,5 @@
|
||||
<script setup>
|
||||
import { ref, onMounted } from 'vue';
|
||||
import { onMounted, onUnmounted, ref } from 'vue';
|
||||
import Note from './Note.vue';
|
||||
import { emitter } from '~/services/Emitter';
|
||||
|
||||
@@ -9,6 +9,10 @@ const noteContainer = ref(null);
|
||||
|
||||
function calculateContainerWidth(){
|
||||
let dom = noteContainer.value;
|
||||
if (!dom) {
|
||||
return;
|
||||
}
|
||||
|
||||
dom.style.width = noteData.value.length * 701 + "px";
|
||||
|
||||
setTimeout(() => {
|
||||
@@ -22,19 +26,33 @@ function calculateContainerWidth(){
|
||||
}
|
||||
|
||||
function pushNote(note){
|
||||
noteData.value = noteData.value.filter((currentNote) => {
|
||||
return currentNote.key !== note.key;
|
||||
});
|
||||
noteData.value.push(note);
|
||||
calculateContainerWidth();
|
||||
}
|
||||
|
||||
emitter.on("push-note", (note) => {
|
||||
function handlePushNote(note) {
|
||||
pushNote(note);
|
||||
})
|
||||
}
|
||||
|
||||
emitter.on("delete-note", (key) => {
|
||||
function handleDeleteNote(key) {
|
||||
noteData.value = noteData.value.filter((note) => {
|
||||
return note.key !== key;
|
||||
});
|
||||
calculateContainerWidth();
|
||||
}
|
||||
|
||||
// Moure aixo
|
||||
onMounted(() => {
|
||||
emitter.on("push-note", handlePushNote);
|
||||
emitter.on("delete-note", handleDeleteNote);
|
||||
});
|
||||
|
||||
onUnmounted(() => {
|
||||
emitter.off("push-note", handlePushNote);
|
||||
emitter.off("delete-note", handleDeleteNote);
|
||||
});
|
||||
|
||||
</script>
|
||||
@@ -59,10 +77,9 @@ emitter.on("delete-note", (key) => {
|
||||
display: flex;
|
||||
height: 100%;
|
||||
margin: 0;
|
||||
height: 100%;
|
||||
background-color: var(--color-background);
|
||||
}
|
||||
</style>
|
||||
|
||||
<style>
|
||||
</style>
|
||||
</style>
|
||||
|
||||
3
frontend/app/components/viewer/widgets/TableWidget.vue
Normal file
3
frontend/app/components/viewer/widgets/TableWidget.vue
Normal file
@@ -0,0 +1,3 @@
|
||||
<template>
|
||||
|
||||
</template>
|
||||
13
frontend/app/components/viewer/widgets/TestWidget.vue
Normal file
13
frontend/app/components/viewer/widgets/TestWidget.vue
Normal file
@@ -0,0 +1,13 @@
|
||||
<script setup>
|
||||
const props = defineProps(['content']);
|
||||
|
||||
const name = ref('');
|
||||
|
||||
onMounted(() => {
|
||||
name.value = props.content || 'No content';
|
||||
});
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<h2>This is a {{name}} widget</h2>
|
||||
</template>
|
||||
@@ -0,0 +1,177 @@
|
||||
<script setup>
|
||||
const props = defineProps(['content']);
|
||||
|
||||
import { parse } from '~/services/widgets/DiceParser';
|
||||
import { AddSound } from '~/services/Sound';
|
||||
|
||||
const container = ref(null);
|
||||
const resultText = ref("");
|
||||
const steps = ref(null);
|
||||
const stepsHtml = ref("");
|
||||
|
||||
const rollDice = () => {
|
||||
const result = parse(props.content);
|
||||
resultText.value = result.total;
|
||||
stepsHtml.value = result.steps.map(renderStep).join('');
|
||||
};
|
||||
|
||||
const renderStep = (s) => {
|
||||
if (s.type === 'op') {
|
||||
const label = s.op === '*' ? '×' : s.op === '/' ? '÷' : s.op;
|
||||
return `<span class="step-op">${label}</span>`;
|
||||
}
|
||||
|
||||
if (s.type === 'const') {
|
||||
return `<span class="step-const">${s.value}</span>`;
|
||||
}
|
||||
|
||||
if (s.type === 'dice') {
|
||||
const { entry } = s;
|
||||
const { rawRolls, kept, sides, mod, value } = entry;
|
||||
|
||||
const keptCopy = [...kept];
|
||||
const tagged = rawRolls.map(v => {
|
||||
const i = keptCopy.indexOf(v);
|
||||
if (i !== -1) { keptCopy.splice(i, 1); return { v, kept: true }; }
|
||||
return { v, kept: false };
|
||||
});
|
||||
|
||||
let html = '';
|
||||
|
||||
tagged.forEach(({ v, kept }) => {
|
||||
const isMax = v === sides, isMin = v === 1;
|
||||
const cls = !kept ? 'roll-val dropped'
|
||||
: isMax ? 'roll-val max-val'
|
||||
: isMin ? 'roll-val min-val'
|
||||
: 'roll-val kept';
|
||||
html += `<span class="${cls}">${v}</span>`;
|
||||
});
|
||||
|
||||
if (kept.length < rawRolls.length || kept.length > 1) {
|
||||
html += `<span class="step-sum">=${value}</span>`;
|
||||
}
|
||||
|
||||
return html;
|
||||
}
|
||||
|
||||
return '';
|
||||
};
|
||||
|
||||
onMounted(() => {
|
||||
AddSound(container.value);
|
||||
});
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="roll-widget" ref="container">
|
||||
<div class="roll-widget-body">
|
||||
<span class="result-text">{{ resultText || '-' }}</span>
|
||||
<button class="btn-primary roll-btn sound-click" @click="rollDice">
|
||||
<span class="dice-content">
|
||||
<!-- Dice icon (SVG) -->
|
||||
<img class="icon" src="/icons/iconoir/regular/dice-three.svg" draggable="false">
|
||||
</span>
|
||||
</button>
|
||||
</div>
|
||||
<div class="roll-widget-results">
|
||||
<span class="formula">{{ "[" + props.content + "]" }}</span>
|
||||
<div class="steps" v-html="stepsHtml"></div>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
.steps {
|
||||
margin-left: 8px;
|
||||
height: 22px;
|
||||
> {
|
||||
font-size: 12px;
|
||||
}
|
||||
}
|
||||
.steps :deep(.roll-val){
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
justify-content: center;
|
||||
min-width: 18px;
|
||||
padding: 1px 2px;
|
||||
border-radius: 3px;
|
||||
margin: 2px;
|
||||
font-size: 12px;
|
||||
border: 1px solid;
|
||||
}
|
||||
.steps :deep(.roll-val.kept){
|
||||
background: rgba(93,184,122,0.12);
|
||||
border-color: #4a8c5c;
|
||||
color: #a8d4b4;
|
||||
}
|
||||
|
||||
.steps :deep(.roll-val.dropped) {
|
||||
background: transparent;
|
||||
border-color: var(--color-border);
|
||||
color: var(--color-text-tertiary);
|
||||
text-decoration: line-through;
|
||||
}
|
||||
|
||||
.steps :deep(.roll-val.max-val) {
|
||||
background: rgba(93,184,122,0.2);
|
||||
border-color: #5db87a;
|
||||
color: #5db87a;
|
||||
}
|
||||
|
||||
.steps :deep(.roll-val.min-val) {
|
||||
background: rgba(201,95,95,0.12);
|
||||
border-color: #c95f5f;
|
||||
color: #c95f5f;
|
||||
}
|
||||
|
||||
.steps :deep(.step-op) { color: var(--color-text-secondary); padding: 0 4px; font-size: 13px; }
|
||||
.steps :deep(.step-const) { font-size: 13px; color: var(--color-text-primary); padding: 0 2px; }
|
||||
.steps :deep(.step-dice-label) { font-size: 11px; color: var(--color-text-tertiary); margin-right: 2px; }
|
||||
.steps :deep(.step-sum) { font-size: 11px; color: var(--color-text-tertiary); margin-left: 2px; }
|
||||
.steps :deep(.dice-mod) { font-style: normal; margin-left: 2px; }
|
||||
|
||||
.result-text {
|
||||
font-size: 24px;
|
||||
vertical-align: center;
|
||||
}
|
||||
|
||||
.roll-widget-body {
|
||||
width: 100%;
|
||||
display: flex;
|
||||
flex-direction: row;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.result-text {
|
||||
margin-left: 12px;
|
||||
}
|
||||
|
||||
.formula {
|
||||
margin-left: 12px;
|
||||
font-family: ui-monospace, monospace;
|
||||
font-size: 12px;
|
||||
color: var(--color-gray);
|
||||
}
|
||||
|
||||
.roll-widget-results {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
padding-bottom: 10px;
|
||||
}
|
||||
|
||||
.roll-widget {
|
||||
width: 100%;
|
||||
background-color: var(--color-background-light);
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
margin-bottom: 8px;
|
||||
border-radius: 6px;
|
||||
}
|
||||
|
||||
.roll-btn {
|
||||
padding: 8px;
|
||||
margin-right: 8px;
|
||||
}
|
||||
|
||||
</style>
|
||||
@@ -0,0 +1,59 @@
|
||||
<script setup>
|
||||
const props = defineProps(['content']);
|
||||
|
||||
import { parse } from '~/services/widgets/DiceParser';
|
||||
import { AddSound } from '~/services/Sound';
|
||||
|
||||
const container = ref(null);
|
||||
const resultText = ref("");
|
||||
|
||||
const rollDice = () => {
|
||||
console.log(props.content);
|
||||
const result = parse(props.content);
|
||||
console.log(result);
|
||||
resultText.value = result.total;
|
||||
};
|
||||
|
||||
onMounted(() => {
|
||||
AddSound(container.value);
|
||||
});
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div class="roll-widget" ref="container">
|
||||
<div class="roll-widget-body">
|
||||
<button class="btn-primary btn-inline sound-click" @click="rollDice">
|
||||
<span class="dice-content">
|
||||
<!-- Dice icon (SVG) -->
|
||||
<img class="icon" src="/icons/iconoir/regular/dice-three.svg" draggable="false">
|
||||
|
||||
<!-- Result text -->
|
||||
<span class="result-text">
|
||||
{{ resultText || props.content }}
|
||||
</span>
|
||||
</span>
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
.roll-widget {
|
||||
display: inline-flex; /* or inline-block */
|
||||
vertical-align: middle; /* keeps it aligned nicely with text */
|
||||
}
|
||||
|
||||
.btn-inline {
|
||||
padding: 2px 6px;
|
||||
}
|
||||
|
||||
.dice-content {
|
||||
display: inline-flex;
|
||||
align-items: center;
|
||||
gap: 6px;
|
||||
}
|
||||
|
||||
.result-text {
|
||||
font-weight: 500;
|
||||
}
|
||||
</style>
|
||||
144
frontend/app/components/windows/CreateCampaignWindow.vue
Normal file
144
frontend/app/components/windows/CreateCampaignWindow.vue
Normal file
@@ -0,0 +1,144 @@
|
||||
<script setup>
|
||||
import { onMounted, ref } from 'vue';
|
||||
import { SetupHandle, SetSize, ResetPosition, Top, ClearWindow } from '@/services/Windows';
|
||||
|
||||
import WindowHandle from './partials/WindowHandle.vue';
|
||||
import ColorPicker from '../layouts/ColorPicker.vue';
|
||||
import Server from '~/services/Server';
|
||||
import { DisplayToast } from '~/services/Toaster';
|
||||
|
||||
const handle = ref(null);
|
||||
const wrapper = ref(null);
|
||||
|
||||
const props = defineProps(['data']);
|
||||
const data = props.data;
|
||||
const loading = ref(false);
|
||||
|
||||
let id = data.id;
|
||||
|
||||
onMounted(() => {
|
||||
Top(wrapper);
|
||||
SetupHandle(id, handle);
|
||||
SetSize(id, {width: 500, height: 400});
|
||||
ResetPosition(id, "center");
|
||||
});
|
||||
|
||||
const campaignName = ref("");
|
||||
const campaignDescription = ref("");
|
||||
const colorPicker = ref(null);
|
||||
|
||||
function NewCampaign(){
|
||||
const color = colorPicker.value.GetColor();
|
||||
console.log(color);
|
||||
loading.value = true;
|
||||
Server().post('/campaign/create', {
|
||||
name: campaignName.value,
|
||||
description: campaignDescription.value,
|
||||
color: colorPicker.value.GetColor(),
|
||||
}).then((response) => {
|
||||
loading.value = false;
|
||||
console.log(response.data);
|
||||
DisplayToast('green', $t('campaigns.create.success'), 3000);
|
||||
ClearWindow({id});
|
||||
});
|
||||
}
|
||||
</script>
|
||||
|
||||
|
||||
<template>
|
||||
<div class="window-wrapper" :id="'window-wrapper-' + id" ref="wrapper">
|
||||
<WindowHandle :window="id" ref="handle"></WindowHandle>
|
||||
|
||||
<div class="body">
|
||||
<!-- Body -->
|
||||
<form v-on:submit.prevent="NewCampaign">
|
||||
<div class="form-field">
|
||||
<label>{{ $t('campaigns.create.name') }}</label>
|
||||
<input type="text" :placeholder="$t('campaigns.create.enter')" name="campaignName" v-model="campaignName" autocomplete="off" >
|
||||
</div>
|
||||
<div class="form-field">
|
||||
<label>{{ $t('campaigns.create.description') }}</label>
|
||||
<textarea type="text" :placeholder="$t('campaigns.create.description-placeholder')" name="campaignDescription" v-model="campaignDescription" autocomplete="off" ></textarea>
|
||||
</div>
|
||||
<div class="form-field">
|
||||
<label>{{ $t('campaigns.create.color') }}</label>
|
||||
<ColorPicker ref="colorPicker"></ColorPicker>
|
||||
</div>
|
||||
<div class="form-actions">
|
||||
<button class="btn-primary sound-click">
|
||||
<span v-if="loading">
|
||||
<Spinner />
|
||||
</span>
|
||||
<span v-else>
|
||||
{{ $t("general.create") }}
|
||||
</span>
|
||||
</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</template>
|
||||
|
||||
|
||||
<style scoped>
|
||||
.window-wrapper {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
.body {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.window-second-header {
|
||||
width: 100%;
|
||||
h2 {
|
||||
font-family: MrEavesRemake;
|
||||
}
|
||||
}
|
||||
|
||||
form {
|
||||
margin-top: 10px;
|
||||
margin-left: 10px;
|
||||
margin-right: 10px;
|
||||
}
|
||||
|
||||
.form-field {
|
||||
padding: 2px;
|
||||
display: flex;
|
||||
align-items: left;
|
||||
flex-direction: row;
|
||||
justify-content: space-between;
|
||||
width: 100%;
|
||||
|
||||
> * {
|
||||
flex-grow: 1;
|
||||
}
|
||||
}
|
||||
|
||||
label {
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
textarea {
|
||||
resize: none;
|
||||
height: 200px;
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.form-actions {
|
||||
display: flex;
|
||||
justify-content: center; /* centers horizontally */
|
||||
margin-top: 10px;
|
||||
}
|
||||
|
||||
.form-actions button {
|
||||
width: 100%; /* makes it expand */
|
||||
max-width: 300px; /* optional: prevents it from being too wide */
|
||||
}
|
||||
|
||||
</style>
|
||||
|
||||
|
||||
@@ -1,91 +0,0 @@
|
||||
<script setup>
|
||||
import { onMounted, ref } from 'vue';
|
||||
import { SetupHandle, SetSize, ResetPosition, Top } from '@/services/Windows';
|
||||
|
||||
import Server from '@/services/Server'
|
||||
import { SetMinSize, SetResizable } from '@/services/Windows';
|
||||
import { backendUrl } from '@/services/BackendURL';
|
||||
import { GetUser } from '@/services/User';
|
||||
|
||||
import WindowHandle from './partials/WindowHandle.vue';
|
||||
import BigIconTemplate from './partials/BigIconTemplate.vue';
|
||||
import FixedBottomButtons from './partials/FixedBottomButtons.vue';
|
||||
|
||||
const props = defineProps(['data']);
|
||||
const data = props.data;
|
||||
const userIcon = ref("");
|
||||
|
||||
const handle = ref(null);
|
||||
const wrapper = ref(null);
|
||||
const isAdmin = ref(false);
|
||||
|
||||
let id = data.id;
|
||||
console.log(data);
|
||||
|
||||
onMounted(() => {
|
||||
Top(wrapper);
|
||||
SetupHandle(id, handle);
|
||||
SetSize(id, {width: 500, height: 480});
|
||||
ResetPosition(id, "center");
|
||||
|
||||
SetResizable(id, true);
|
||||
SetMinSize(id, {width: 350, height: 280});
|
||||
|
||||
isAdmin.value = GetUser().admin;
|
||||
|
||||
Server().get('/user/retrieve-avatar?username=' + data.user.username).then((response) => {
|
||||
if(response.data.image) userIcon.value = backendUrl + "public/" + response.data.image;
|
||||
else userIcon.value = "public/img/def-avatar.jpg";
|
||||
}).catch((err) => console.log("Internal error"));
|
||||
});
|
||||
|
||||
function RemoveUser(){
|
||||
alert("Remove")
|
||||
}
|
||||
</script>
|
||||
|
||||
|
||||
<template>
|
||||
<div class="window-wrapper" :id="'window-wrapper-' + id" ref="wrapper">
|
||||
<WindowHandle :window="id" ref="handle"></WindowHandle>
|
||||
|
||||
<BigIconTemplate :title="data.user.username" :img="userIcon">
|
||||
<div v-if="props.data.editable || isAdmin">
|
||||
|
||||
</div>
|
||||
<div v-else>
|
||||
|
||||
</div>
|
||||
</BigIconTemplate>
|
||||
|
||||
<FixedBottomButtons v-if="isAdmin" :remove="RemoveUser"></FixedBottomButtons>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
|
||||
<style scoped>
|
||||
.window-wrapper {
|
||||
|
||||
display: flex;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
.splash-image {
|
||||
width: 600px;
|
||||
height: 250px;
|
||||
}
|
||||
|
||||
.form-field {
|
||||
padding: 10px;
|
||||
display: flex;
|
||||
align-items: left;
|
||||
flex-direction: column;
|
||||
justify-content: left;
|
||||
width: 600px;
|
||||
}
|
||||
|
||||
label {
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
</style>
|
||||
@@ -1,10 +1,12 @@
|
||||
<script setup>
|
||||
import { onMounted, ref } from 'vue';
|
||||
import { SetupHandle, SetSize, ResetPosition, Top } from '@/services/Windows';
|
||||
import { SetupHandle, SetSize, ResetPosition, Top, CreateChildWindow, GetFirstWindowId } from '@/services/Windows';
|
||||
|
||||
import WindowHandle from './partials/WindowHandle.vue';
|
||||
import VersionRender from '../partials/VersionRender.vue';
|
||||
import EditUserPartial from '../partials/EditUserPartial.vue';
|
||||
import CampaignEntry from '../partials/CampaignEntry.vue';
|
||||
import Server from '~/services/Server';
|
||||
|
||||
const handle = ref(null);
|
||||
const wrapper = ref(null);
|
||||
@@ -12,13 +14,30 @@ const wrapper = ref(null);
|
||||
const props = defineProps(['data']);
|
||||
const data = props.data;
|
||||
|
||||
const campaings = ref([]);
|
||||
|
||||
let id = data.id;
|
||||
|
||||
function CreateCampaignWindow(){
|
||||
CreateChildWindow(GetFirstWindowId('main_menu'), 'create_campaign');
|
||||
}
|
||||
|
||||
function RefreshCampaigns(){
|
||||
Server().get('/campaign/list').then((response) => {
|
||||
if(response.data.status !== "ok") return;
|
||||
response.data.campaigns.forEach((camp) => {
|
||||
campaings.value.push(camp);
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
onMounted(() => {
|
||||
Top(wrapper);
|
||||
SetupHandle(id, handle);
|
||||
SetSize(id, {width: 580, height: 760});
|
||||
SetSize(id, {width: 880, height: 760});
|
||||
ResetPosition(id, "center");
|
||||
|
||||
RefreshCampaigns();
|
||||
});
|
||||
</script>
|
||||
|
||||
@@ -27,14 +46,41 @@ onMounted(() => {
|
||||
<div class="window-wrapper" :id="'window-wrapper-' + id" ref="wrapper">
|
||||
<WindowHandle :window="id" ref="handle"></WindowHandle>
|
||||
|
||||
<EditUserPartial></EditUserPartial>
|
||||
<!-- Body -->
|
||||
<h1>{{ $t("main-menu.main-menu")}}</h1>
|
||||
<div class="two-column">
|
||||
<div class="vert-expand secondary">
|
||||
<div class="image-container">
|
||||
<img alt="Dragonroll logo" src="/img/logo-splash.png" draggable="false" width="100%">
|
||||
</div>
|
||||
<div class="patch-notes-container">
|
||||
<h1>Welcome to dragonroll!</h1>
|
||||
<h2>Version 0.1</h2>
|
||||
<p>This is totally under construction. This is a review of how the patch notes will be displayed.</p>
|
||||
<p>There is also a lot of heavy development here.</p>
|
||||
</div>
|
||||
<VersionRender></VersionRender>
|
||||
</div>
|
||||
|
||||
<div class="button-container">
|
||||
<button class="btn-primary button-expand sound-click" v-on:click="OpenCampaigns" ref="campaignButton">{{ $t("main-menu.campaigns") }}</button>
|
||||
<div class="vert-expand" style="max-width: 450px;">
|
||||
<EditUserPartial></EditUserPartial>
|
||||
<!-- Body -->
|
||||
<div class="vert-expand">
|
||||
<div class="vert top">
|
||||
<h1>{{ $t("main-menu.main-menu")}}</h1>
|
||||
|
||||
<!-- HERE -->
|
||||
<div class="campaign-list">
|
||||
<CampaignEntry v-for="camp in campaings" :key="camp._id" :data="camp"></CampaignEntry>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="vert bot">
|
||||
<div class="button-container">
|
||||
<button class="btn-primary button-expand sound-click" v-on:click="CreateCampaignWindow" ref="campaignButton">{{ $t("main-menu.create-campaign") }}</button>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<VersionRender></VersionRender>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@@ -43,16 +89,58 @@ onMounted(() => {
|
||||
h1 {
|
||||
margin-top: 20px;
|
||||
font-family: MrEavesRemake;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
h2, h3 {
|
||||
font-family: MrEavesRemake;
|
||||
}
|
||||
|
||||
.patch-notes-container {
|
||||
margin: 10px;
|
||||
padding: 10px;
|
||||
border-radius: 5px;
|
||||
flex-grow: 1;
|
||||
}
|
||||
|
||||
.two-column {
|
||||
display: flex;
|
||||
flex-direction: row;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
}
|
||||
|
||||
.secondary {
|
||||
background-color: var(--color-background-soft);
|
||||
}
|
||||
|
||||
.expand {
|
||||
width: 100%;
|
||||
|
||||
display: flex;
|
||||
flex-direction: row;
|
||||
|
||||
> * {
|
||||
flex-grow: 1;
|
||||
}
|
||||
}
|
||||
|
||||
.vert-expand {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
justify-content: space-between;
|
||||
flex-grow: 1;
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
|
||||
.button-expand {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.button-container {
|
||||
display: flex;
|
||||
width: 100%;
|
||||
padding: 20px;
|
||||
margin: 20px;
|
||||
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
@@ -6,6 +6,7 @@ import Tabs from '../layouts/Tabs.vue';
|
||||
import Dropdown from '../layouts/Dropdown.vue';
|
||||
import { GetUser, GetUserSetting, SetUserSetting } from '@/services/User';
|
||||
import { SetupHandle, SetSize, ResetPosition, Top, ClearWindow, CreateWindow, SetMinSize, SetResizable } from '@/services/Windows';
|
||||
import { locales } from '~~/i18n/locales';
|
||||
|
||||
const handle = ref(null);
|
||||
const wrapper = ref(null);
|
||||
@@ -13,11 +14,12 @@ const wrapper = ref(null);
|
||||
const props = defineProps(['data']);
|
||||
const data = props.data;
|
||||
|
||||
const { locales, setLocale, locale } = useI18n();
|
||||
const { locale } = useI18n();
|
||||
|
||||
|
||||
const changeLocale = (lang) => {
|
||||
console.log(lang);
|
||||
setLocale(lang.code);
|
||||
locale.value = lang.code;
|
||||
SetUserSetting('lang', lang.code);
|
||||
}
|
||||
|
||||
@@ -28,12 +30,18 @@ const rows = ref([{id: "account-settings", value: "settings.tabs.account-setting
|
||||
/* TODO
|
||||
const languageOptions = ref(["English", "Spanish", "Catalan"])
|
||||
const langSelector = ref(null);
|
||||
const currentLanguage = ref("");
|
||||
*/
|
||||
function getLocaleFromCode(code){
|
||||
for(let i = 0; i < locales.length; i++){
|
||||
if(locales[i].code == code) return locales[i];
|
||||
}
|
||||
}
|
||||
|
||||
const selectedLocale = ref("");
|
||||
onBeforeMount(() => {
|
||||
GetUserSetting('lang').then(value => {
|
||||
currentLanguage.value = codes[value ?? 'en']
|
||||
console.log(currentLanguage.value)
|
||||
locale.value = value;
|
||||
selectedLocale.value = getLocaleFromCode(value); // Set selected in dropdown
|
||||
});
|
||||
if(GetUser().admin) rows.value.push({
|
||||
id: "site-administration",
|
||||
@@ -44,11 +52,11 @@ onBeforeMount(() => {
|
||||
onMounted(() => {
|
||||
Top(wrapper);
|
||||
SetupHandle(id, handle);
|
||||
SetSize(id, {width: 400, height: 480});
|
||||
SetSize(id, {width: 600, height: 480});
|
||||
ResetPosition(id, "center");
|
||||
|
||||
SetResizable(id, true);
|
||||
SetMinSize(id, {width: 350, height: 280});
|
||||
SetMinSize(id, {width: 450, height: 280});
|
||||
});
|
||||
|
||||
function OpenManageAccounts(){
|
||||
@@ -86,7 +94,7 @@ const getLocaleName = (locale) => {
|
||||
<div class="form-container">
|
||||
<div class="form-element">
|
||||
<label>{{ $t('settings.account-settings.language') }}</label>
|
||||
<Dropdown :options="locales" :keyFunc="getLocaleName" :onselect="changeLocale" :selected="locale"></Dropdown>
|
||||
<Dropdown :options="locales" :keyFunc="getLocaleName" :onselect="changeLocale" :selected="selectedLocale"></Dropdown>
|
||||
</div>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
@@ -1,5 +1,5 @@
|
||||
<script setup>
|
||||
import IconButton from '~/components/partials/IconButton.vue';
|
||||
import IconButton from '~/components/layouts/IconButton.vue';
|
||||
|
||||
const props = defineProps(['plus', 'edit', 'view', 'remove']);
|
||||
|
||||
|
||||
@@ -126,6 +126,11 @@ defineExpose({
|
||||
justify-content: right;
|
||||
}
|
||||
|
||||
.center {
|
||||
display: flex;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
span {
|
||||
font-family: MrEavesRemake;
|
||||
}
|
||||
|
||||
6
frontend/app/plugins/api.ts
Normal file
6
frontend/app/plugins/api.ts
Normal file
@@ -0,0 +1,6 @@
|
||||
import { initApi } from '../services/Server';
|
||||
|
||||
export default defineNuxtPlugin(() => {
|
||||
const config = useRuntimeConfig();
|
||||
initApi(config.public.apiBaseUrl);
|
||||
});
|
||||
31
frontend/app/plugins/i18n.ts
Normal file
31
frontend/app/plugins/i18n.ts
Normal file
@@ -0,0 +1,31 @@
|
||||
import { createI18n } from 'vue-i18n'
|
||||
|
||||
function loadLocaleMessages() {
|
||||
const locales = import.meta.glob('../../i18n/locales/**/*.json', { eager: true })
|
||||
const messages: Record<string, any> = {}
|
||||
|
||||
for (const path in locales) {
|
||||
const matched = path.match(/i18n\/locales\/(\w+)\/(.*)\.json$/)
|
||||
if (!matched) continue
|
||||
|
||||
const [, locale, file] = matched
|
||||
|
||||
if (!messages[locale]) {
|
||||
messages[locale] = {}
|
||||
}
|
||||
|
||||
messages[locale][file] = (locales[path] as any).default
|
||||
}
|
||||
|
||||
return messages
|
||||
}
|
||||
|
||||
export default defineNuxtPlugin((nuxtApp) => {
|
||||
const i18n = createI18n({
|
||||
legacy: false,
|
||||
locale: 'en',
|
||||
messages: loadLocaleMessages()
|
||||
})
|
||||
|
||||
nuxtApp.vueApp.use(i18n)
|
||||
})
|
||||
@@ -1,11 +0,0 @@
|
||||
var backendUrl = ''
|
||||
if (import.meta.env.PROD) {
|
||||
backendUrl = 'https://api.aranroig.com/';
|
||||
} else {
|
||||
backendUrl = 'http://localhost:5000/'
|
||||
}
|
||||
|
||||
|
||||
export {
|
||||
backendUrl
|
||||
};
|
||||
41
frontend/app/services/Campaign.js
Normal file
41
frontend/app/services/Campaign.js
Normal file
@@ -0,0 +1,41 @@
|
||||
import Server from './Server';
|
||||
|
||||
const SELECTED_CAMPAIGN_KEY = 'selectedCampaignId';
|
||||
|
||||
export const useCampaignService = () => {
|
||||
const Campaign = useState('campaign', () => null)
|
||||
|
||||
const SetCampaign = (data) => {
|
||||
Campaign.value = data;
|
||||
if (data?._id) {
|
||||
localStorage.setItem(SELECTED_CAMPAIGN_KEY, data._id);
|
||||
} else {
|
||||
localStorage.removeItem(SELECTED_CAMPAIGN_KEY);
|
||||
}
|
||||
}
|
||||
|
||||
const RestoreCampaign = async () => {
|
||||
const campaignId = localStorage.getItem(SELECTED_CAMPAIGN_KEY);
|
||||
if (!campaignId) return false;
|
||||
|
||||
try {
|
||||
const response = await Server().get(`/campaign/retrieve/${campaignId}`);
|
||||
if (response.data.status !== 'ok') {
|
||||
SetCampaign(null);
|
||||
return false;
|
||||
}
|
||||
|
||||
SetCampaign(response.data.campaign);
|
||||
return true;
|
||||
} catch (error) {
|
||||
SetCampaign(null);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
return {
|
||||
Campaign,
|
||||
SetCampaign,
|
||||
RestoreCampaign
|
||||
}
|
||||
}
|
||||
109
frontend/app/services/Marker.js
Normal file
109
frontend/app/services/Marker.js
Normal file
@@ -0,0 +1,109 @@
|
||||
import { Marked } from "marked";
|
||||
const widget_map = {
|
||||
inline: {
|
||||
roll: () => import("~/components/viewer/widgets/inline/RollWidgetInline.vue"),
|
||||
},
|
||||
display: {
|
||||
roll: () => import("~/components/viewer/widgets/display/RollWidgetDisplay.vue"),
|
||||
}
|
||||
};
|
||||
|
||||
const componentCache = {
|
||||
inline: {},
|
||||
display: {}
|
||||
}
|
||||
|
||||
const GetWidget = (type, name) => {
|
||||
if (!componentCache[type][name]) {
|
||||
componentCache[type][name] = defineAsyncComponent(
|
||||
widget_map[type][name]
|
||||
)
|
||||
}
|
||||
|
||||
return componentCache[type][name]
|
||||
}
|
||||
|
||||
|
||||
const marker = new Marked();
|
||||
|
||||
const extension = {
|
||||
name: "widget",
|
||||
level: "block",
|
||||
|
||||
tokenizer(src) {
|
||||
const rule = /^@(\w+)\n([\s\S]+?)\n@end/;
|
||||
const match = rule.exec(src);
|
||||
|
||||
if (!match) return;
|
||||
|
||||
return {
|
||||
type: "widget",
|
||||
raw: match[0],
|
||||
name: match[1],
|
||||
text: match[2],
|
||||
};
|
||||
},
|
||||
|
||||
renderer(token) {
|
||||
return `<div class="vue-component-display" data-component="${token.name}" data-content="${token.text}"></div>`;
|
||||
},
|
||||
};
|
||||
|
||||
const inlineExtension = {
|
||||
name: "widget_inline",
|
||||
level: "inline",
|
||||
start(src) {
|
||||
return src.indexOf("@");
|
||||
},
|
||||
|
||||
tokenizer(src) {
|
||||
const rule = /^@(\w+)\s*\[([^\]]*)\]/;
|
||||
const match = rule.exec(src);
|
||||
if (!match) return;
|
||||
|
||||
return {
|
||||
type: "widget_inline",
|
||||
raw: match[0],
|
||||
name: match[1],
|
||||
text: match[2],
|
||||
};
|
||||
},
|
||||
|
||||
renderer(token) {
|
||||
return `<span class="vue-component-inline" data-component="${token.name}" data-content="${token.text}"></span>`;
|
||||
},
|
||||
};
|
||||
|
||||
const linkExtension = {
|
||||
name: "link_to",
|
||||
level: "inline",
|
||||
start(src) {
|
||||
return src.indexOf("[[");
|
||||
},
|
||||
|
||||
tokenizer(src) {
|
||||
const rule = /^\[\[([^\n]*)\]\]/;
|
||||
const match = rule.exec(src);
|
||||
if (!match) return;
|
||||
|
||||
return {
|
||||
type: "link_to",
|
||||
raw: match[0],
|
||||
link: match[1],
|
||||
};
|
||||
},
|
||||
|
||||
renderer(token) {
|
||||
return `<span class="vue-link" data-href="${token.link}"></span>`;
|
||||
},
|
||||
};
|
||||
|
||||
marker.use({
|
||||
extensions: [extension, inlineExtension, linkExtension],
|
||||
});
|
||||
|
||||
function ParseMarkdown(source) {
|
||||
return marker.parse(source || "");
|
||||
}
|
||||
|
||||
export { ParseMarkdown, GetWidget };
|
||||
@@ -1,15 +1,16 @@
|
||||
import axios from 'axios';
|
||||
|
||||
import { backendUrl } from './BackendURL';
|
||||
|
||||
const server = axios.create({
|
||||
baseURL: backendUrl,
|
||||
baseURL: 'http://localhost:5000/api', // fallback only
|
||||
headers: {
|
||||
"Access-Control-Allow-Origin": "*",
|
||||
}
|
||||
});
|
||||
|
||||
// Attach token dynamically on each request via interceptor
|
||||
export const initApi = (baseURL) => {
|
||||
server.defaults.baseURL = baseURL;
|
||||
};
|
||||
|
||||
server.interceptors.request.use((config) => {
|
||||
const token = localStorage.getItem('token');
|
||||
if (token) {
|
||||
@@ -18,4 +19,6 @@ server.interceptors.request.use((config) => {
|
||||
return config;
|
||||
});
|
||||
|
||||
export default () => server;
|
||||
export default () => server;
|
||||
|
||||
export const getBaseUrl = () => server.defaults.baseURL;
|
||||
@@ -64,6 +64,7 @@ function LoadUser(){
|
||||
|
||||
function LogoutUser(){
|
||||
localStorage.removeItem("token");
|
||||
localStorage.removeItem("selectedCampaignId");
|
||||
UserStatus.value = 0;
|
||||
}
|
||||
|
||||
@@ -77,4 +78,4 @@ export {
|
||||
HasAdmin,
|
||||
GetUserSetting,
|
||||
SetUserSetting
|
||||
}
|
||||
}
|
||||
|
||||
@@ -21,11 +21,6 @@ const defWindows = {
|
||||
main_menu: {
|
||||
title: 'windows.main-menu',
|
||||
component: () => import('~/components/windows/MainMenuWindow.vue'),
|
||||
},
|
||||
edit_profile: {
|
||||
title: "windows.edit-profile",
|
||||
component: () => import('~/components/windows/EditProfileWindow.vue'),
|
||||
close: () => ClearWindow({type: 'edit_profile'}),
|
||||
movable: true
|
||||
},
|
||||
settings: {
|
||||
@@ -33,9 +28,15 @@ const defWindows = {
|
||||
component: () => import('~/components/windows/SettingsWindow.vue'),
|
||||
close: () => ClearWindow({type: 'settings'}),
|
||||
movable: true
|
||||
},
|
||||
create_campaign: {
|
||||
title: "windows.create-campaign",
|
||||
component: () => import('~/components/windows/CreateCampaignWindow.vue'),
|
||||
close: () => ClearWindow({type: 'create_campaign'}),
|
||||
movable: true
|
||||
}
|
||||
}
|
||||
|
||||
export {
|
||||
defWindows
|
||||
}
|
||||
}
|
||||
|
||||
@@ -224,7 +224,6 @@ function CreateChildWindow(parentId, type, data = {}) {
|
||||
if (parent.children) parent.children.push(newId); // We will create the child window right now
|
||||
else parent.children = [newId];
|
||||
CreateWindow(type, data);
|
||||
console.log(windows.value);
|
||||
}
|
||||
|
||||
function GetFirstWindowId(type) {
|
||||
|
||||
134
frontend/app/services/widgets/DiceParser.js
Normal file
134
frontend/app/services/widgets/DiceParser.js
Normal file
@@ -0,0 +1,134 @@
|
||||
// DiceParser.js
|
||||
|
||||
function roll(sides) {
|
||||
return Math.floor(Math.random() * sides) + 1;
|
||||
}
|
||||
|
||||
function tokenize(expr) {
|
||||
const re = /(\d*d\d+(?:adv|dis|kh\d+|kl\d+)?|\d+|[+\-*\/()])/gi;
|
||||
const tokens = [];
|
||||
let m;
|
||||
while ((m = re.exec(expr)) !== null) tokens.push(m[0].toLowerCase());
|
||||
return tokens;
|
||||
}
|
||||
|
||||
function parseDiceToken(tok) {
|
||||
const m = tok.match(/^(\d*)d(\d+)(adv|dis|kh(\d+)|kl(\d+))?$/i);
|
||||
if (!m) return null;
|
||||
const count = parseInt(m[1] || '1');
|
||||
const sides = parseInt(m[2]);
|
||||
const mod = (m[3] || '').toLowerCase();
|
||||
if (count < 1 || count > 1000 || sides < 2 || sides > 10000)
|
||||
throw new Error(`Invalid dice: ${tok}`);
|
||||
return { count, sides, mod };
|
||||
}
|
||||
|
||||
export function parse(expr) {
|
||||
const tokens = tokenize(expr);
|
||||
if (!tokens.length) throw new Error('Empty expression');
|
||||
let pos = 0;
|
||||
|
||||
const peek = () => tokens[pos];
|
||||
const consume = () => tokens[pos++];
|
||||
|
||||
function parseExpr() { return parseAddSub(); }
|
||||
|
||||
function parseAddSub() {
|
||||
let left = parseMulDiv();
|
||||
while (peek() === '+' || peek() === '-') {
|
||||
const op = consume();
|
||||
const right = parseMulDiv();
|
||||
left = {
|
||||
value: op === '+' ? left.value + right.value : left.value - right.value,
|
||||
steps: [...left.steps, { type: 'op', op }, ...right.steps],
|
||||
};
|
||||
}
|
||||
return left;
|
||||
}
|
||||
|
||||
function parseMulDiv() {
|
||||
let left = parseUnary();
|
||||
while (peek() === '*' || peek() === '/') {
|
||||
const op = consume();
|
||||
const right = parseUnary();
|
||||
if (op === '/' && right.value === 0) throw new Error('Division by zero');
|
||||
left = {
|
||||
value: op === '*' ? left.value * right.value : Math.floor(left.value / right.value),
|
||||
steps: [...left.steps, { type: 'op', op }, ...right.steps],
|
||||
};
|
||||
}
|
||||
return left;
|
||||
}
|
||||
|
||||
function parseUnary() {
|
||||
if (peek() === '-') {
|
||||
consume();
|
||||
const r = parsePrimary();
|
||||
return { value: -r.value, steps: [{ type: 'op', op: '-' }, ...r.steps] };
|
||||
}
|
||||
return parsePrimary();
|
||||
}
|
||||
|
||||
function parsePrimary() {
|
||||
const tok = peek();
|
||||
if (!tok) throw new Error('Unexpected end of expression');
|
||||
|
||||
if (tok === '(') {
|
||||
consume();
|
||||
const inner = parseExpr();
|
||||
if (peek() !== ')') throw new Error('Missing closing )');
|
||||
consume();
|
||||
return {
|
||||
value: inner.value,
|
||||
steps: [{ type: 'op', op: '(' }, ...inner.steps, { type: 'op', op: ')' }],
|
||||
};
|
||||
}
|
||||
|
||||
const diceInfo = parseDiceToken(tok);
|
||||
if (diceInfo) {
|
||||
consume();
|
||||
return rollDice(diceInfo);
|
||||
}
|
||||
|
||||
if (/^\d+$/.test(tok)) {
|
||||
consume();
|
||||
const v = parseInt(tok);
|
||||
return { value: v, steps: [{ type: 'const', value: v }] };
|
||||
}
|
||||
|
||||
throw new Error(`Unexpected token: ${tok}`);
|
||||
}
|
||||
|
||||
function rollDice({ count, sides, mod }) {
|
||||
let rawRolls, kept;
|
||||
|
||||
if (mod === 'adv') {
|
||||
rawRolls = [roll(sides), roll(sides)];
|
||||
kept = [Math.max(...rawRolls)];
|
||||
} else if (mod === 'dis') {
|
||||
rawRolls = [roll(sides), roll(sides)];
|
||||
kept = [Math.min(...rawRolls)];
|
||||
} else if (mod.startsWith('kh')) {
|
||||
const k = parseInt(mod.slice(2));
|
||||
rawRolls = Array.from({ length: count }, () => roll(sides));
|
||||
kept = [...rawRolls].sort((a, b) => b - a).slice(0, k);
|
||||
} else if (mod.startsWith('kl')) {
|
||||
const k = parseInt(mod.slice(2));
|
||||
rawRolls = Array.from({ length: count }, () => roll(sides));
|
||||
kept = [...rawRolls].sort((a, b) => a - b).slice(0, k);
|
||||
} else {
|
||||
rawRolls = Array.from({ length: count }, () => roll(sides));
|
||||
kept = rawRolls.slice();
|
||||
}
|
||||
|
||||
const entry = { sides, rawRolls, kept, mod, value: kept.reduce((a, b) => a + b, 0) };
|
||||
return {
|
||||
value: entry.value,
|
||||
steps: [{ type: 'dice', entry }],
|
||||
};
|
||||
}
|
||||
|
||||
const result = parseExpr();
|
||||
if (pos < tokens.length) throw new Error(`Unexpected token: ${tokens[pos]}`);
|
||||
return { total: result.value, steps: result.steps };
|
||||
}
|
||||
17
frontend/i18n/locales.ts
Normal file
17
frontend/i18n/locales.ts
Normal file
@@ -0,0 +1,17 @@
|
||||
export const locales = [
|
||||
{
|
||||
code: 'en',
|
||||
name: 'English',
|
||||
flag: '🇬🇧'
|
||||
},
|
||||
{
|
||||
code: 'es',
|
||||
name: 'Español',
|
||||
flag: '🇪🇸'
|
||||
},
|
||||
{
|
||||
code: 'ca',
|
||||
name: 'Català',
|
||||
flag: '<27>🇸'
|
||||
}
|
||||
]
|
||||
@@ -1,74 +1 @@
|
||||
{
|
||||
"windows": {
|
||||
"login": "Inicia sessió",
|
||||
"register": "Registra't",
|
||||
"main-menu": "Dragonroll",
|
||||
"example": "Finestra d'exemple",
|
||||
"edit-profile": "Edita el perfil",
|
||||
"settings": "Configuració"
|
||||
},
|
||||
"login": {
|
||||
"username": "Usuari o correu electrònic",
|
||||
"username-placeholder": "Introdueix el teu usuari o correu electrònic...",
|
||||
"password": "Contrasenya",
|
||||
"password-placeholder": "Introdueix la teva contrasenya...",
|
||||
"log-in": "Inicia sessió",
|
||||
"no-account": "No tens un compte?",
|
||||
"register": "Registra't",
|
||||
"errors": {
|
||||
"invalid-credentials": "Usuari/correu o contrasenya incorrectes.",
|
||||
"params": "Si us plau, introdueix usuari/correu i contrasenya."
|
||||
},
|
||||
"success": "Inici de sessió correcte!"
|
||||
},
|
||||
"register": {
|
||||
"name": "Nom",
|
||||
"name-placeholder": "Introdueix el teu nom...",
|
||||
"email": "Correu electrònic",
|
||||
"email-placeholder": "Introdueix el teu correu electrònic...",
|
||||
"username": "Usuari",
|
||||
"username-placeholder": "Introdueix el teu nom d'usuari...",
|
||||
"password": "Contrasenya",
|
||||
"password-placeholder": "Introdueix la teva contrasenya...",
|
||||
"confirm-password": "Confirma la contrasenya",
|
||||
"confirm-password-placeholder": "Torna a introduir la contrasenya...",
|
||||
"register": "Registra't",
|
||||
"have-account": "Ja tens un compte?",
|
||||
"login": "Inicia sessió",
|
||||
"password-confirm-placeholder": "Confirma la teva contrasenya...",
|
||||
"welcome": "Benvingut a DragonRoll!",
|
||||
"message": "Si us plau, introdueix el nom d'usuari i la contrasenya que desitges per crear un compte.",
|
||||
"first-register-message": "Estàs a punt de crear el primer compte en aquesta instància de DragonRoll. Aquest compte tindrà privilegis d'administrador.",
|
||||
"errors": {
|
||||
"name-empty": "Si us plau, introdueix el teu nom.",
|
||||
"email-empty": "Si us plau, introdueix un correu electrònic vàlid.",
|
||||
"username-empty": "Si us plau, introdueix un nom d'usuari.",
|
||||
"passwords-no-match": "Les contrasenyes no coincideixen.",
|
||||
"email-username-exists": "Ja existeix un compte amb aquest correu electrònic o nom d'usuari."
|
||||
},
|
||||
"success": "Registre correcte! Ara pots iniciar sessió."
|
||||
},
|
||||
"errors": {
|
||||
"internal": "S'ha produït un error intern."
|
||||
},
|
||||
"main-menu": {
|
||||
"main-menu": "Menú principal",
|
||||
"edit-profile": "Edita el perfil",
|
||||
"campaigns": "Campanyes",
|
||||
"log-out": "Tanca la sessió",
|
||||
"settings": "Configuració"
|
||||
},
|
||||
"settings": {
|
||||
"tabs": {
|
||||
"account-settings": "Configuració del compte",
|
||||
"site-administration": "Administració del lloc"
|
||||
},
|
||||
"account-settings": {
|
||||
"appearance": "Aparença",
|
||||
"language": "Idioma"
|
||||
},
|
||||
"site-administration": {
|
||||
"manage-accounts": "Gestiona els comptes"
|
||||
}
|
||||
}
|
||||
}
|
||||
{}
|
||||
1
frontend/i18n/locales/ca/general.json
Normal file
1
frontend/i18n/locales/ca/general.json
Normal file
@@ -0,0 +1 @@
|
||||
{}
|
||||
14
frontend/i18n/locales/ca/login.json
Normal file
14
frontend/i18n/locales/ca/login.json
Normal file
@@ -0,0 +1,14 @@
|
||||
{
|
||||
"username": "Usuari o correu electrònic",
|
||||
"username-placeholder": "Introdueix el teu usuari o correu electrònic...",
|
||||
"password": "Contrasenya",
|
||||
"password-placeholder": "Introdueix la teva contrasenya...",
|
||||
"log-in": "Inicia sessió",
|
||||
"no-account": "No tens un compte?",
|
||||
"register": "Registra't",
|
||||
"errors": {
|
||||
"invalid-credentials": "Usuari/correu o contrasenya incorrectes.",
|
||||
"params": "Si us plau, introdueix usuari/correu i contrasenya."
|
||||
},
|
||||
"success": "Inici de sessió correcte!"
|
||||
}
|
||||
7
frontend/i18n/locales/ca/main-menu.json
Normal file
7
frontend/i18n/locales/ca/main-menu.json
Normal file
@@ -0,0 +1,7 @@
|
||||
{
|
||||
"main-menu": "Menú principal",
|
||||
"edit-profile": "Editar perfil",
|
||||
"create-campaign": "Crear campanya",
|
||||
"log-out": "Tanca la sessió",
|
||||
"settings": "Configuració"
|
||||
}
|
||||
27
frontend/i18n/locales/ca/register.json
Normal file
27
frontend/i18n/locales/ca/register.json
Normal file
@@ -0,0 +1,27 @@
|
||||
{
|
||||
"name": "Nom",
|
||||
"name-placeholder": "Introdueix el teu nom...",
|
||||
"email": "Correu electrònic",
|
||||
"email-placeholder": "Introdueix el teu correu electrònic...",
|
||||
"username": "Usuari",
|
||||
"username-placeholder": "Introdueix el teu nom d'usuari...",
|
||||
"password": "Contrasenya",
|
||||
"password-placeholder": "Introdueix la teva contrasenya...",
|
||||
"confirm-password": "Confirma la contrasenya",
|
||||
"confirm-password-placeholder": "Torna a introduir la contrasenya...",
|
||||
"register": "Registra't",
|
||||
"have-account": "Ja tens un compte?",
|
||||
"login": "Inicia sessió",
|
||||
"password-confirm-placeholder": "Confirma la teva contrasenya...",
|
||||
"welcome": "Benvingut a DragonRoll!",
|
||||
"message": "Si us plau, introdueix el nom d'usuari i la contrasenya que desitges per crear un compte.",
|
||||
"first-register-message": "Estàs a punt de crear el primer compte en aquesta instància de DragonRoll. Aquest compte tindrà privilegis d'administrador.",
|
||||
"errors": {
|
||||
"name-empty": "Si us plau, introdueix el teu nom.",
|
||||
"email-empty": "Si us plau, introdueix un correu electrònic vàlid.",
|
||||
"username-empty": "Si us plau, introdueix un nom d'usuari.",
|
||||
"passwords-no-match": "Les contrasenyes no coincideixen.",
|
||||
"email-username-exists": "Ja existeix un compte amb aquest correu electrònic o nom d'usuari."
|
||||
},
|
||||
"success": "Registre correcte! Ara pots iniciar sessió."
|
||||
}
|
||||
13
frontend/i18n/locales/ca/settings.json
Normal file
13
frontend/i18n/locales/ca/settings.json
Normal file
@@ -0,0 +1,13 @@
|
||||
{
|
||||
"tabs": {
|
||||
"account-settings": "Configuració del compte",
|
||||
"site-administration": "Administració del lloc"
|
||||
},
|
||||
"account-settings": {
|
||||
"appearance": "Aparença",
|
||||
"language": "Idioma"
|
||||
},
|
||||
"site-administration": {
|
||||
"manage-accounts": "Gestiona els comptes"
|
||||
}
|
||||
}
|
||||
9
frontend/i18n/locales/ca/windows.json
Normal file
9
frontend/i18n/locales/ca/windows.json
Normal file
@@ -0,0 +1,9 @@
|
||||
{
|
||||
"login": "Inicia sessió",
|
||||
"register": "Registra't",
|
||||
"main-menu": "Dragonroll",
|
||||
"example": "Finestra d'exemple",
|
||||
"edit-profile": "Editar perfil",
|
||||
"settings": "Configuració",
|
||||
"create-campaign": "Crear campanya"
|
||||
}
|
||||
@@ -1,74 +1 @@
|
||||
{
|
||||
"windows": {
|
||||
"login": "Login",
|
||||
"register": "Register",
|
||||
"main-menu": "Dragonroll",
|
||||
"example": "Example Window",
|
||||
"edit-profile": "Edit Profile",
|
||||
"settings": "Settings"
|
||||
},
|
||||
"login": {
|
||||
"username": "Username or email",
|
||||
"username-placeholder": "Enter your username or email here...",
|
||||
"password": "Password",
|
||||
"password-placeholder": "Enter your password...",
|
||||
"log-in": "Log in",
|
||||
"no-account": "You don't have an account?",
|
||||
"register": "Register",
|
||||
"errors": {
|
||||
"invalid-credentials": "Invalid username/email or password.",
|
||||
"params": "Please enter both username/email and password."
|
||||
},
|
||||
"success": "Login successful!"
|
||||
},
|
||||
"register": {
|
||||
"name": "Name",
|
||||
"name-placeholder": "Enter your name here...",
|
||||
"email": "Email",
|
||||
"email-placeholder": "Enter your email here...",
|
||||
"username": "Username",
|
||||
"username-placeholder": "Enter your username here...",
|
||||
"password": "Password",
|
||||
"password-placeholder": "Enter your password...",
|
||||
"confirm-password": "Confirm Password",
|
||||
"confirm-password-placeholder": "Re-enter your password...",
|
||||
"register": "Register",
|
||||
"have-account": "Already have an account?",
|
||||
"login": "Login",
|
||||
"password-confirm-placeholder": "Confirm your password...",
|
||||
"welcome": "Welcome to DragonRoll!",
|
||||
"message": "Please enter your desired username and password to create an account.",
|
||||
"first-register-message": "You are about to create the first account on this DragonRoll instance. This account will be granted administrator privileges.",
|
||||
"errors": {
|
||||
"name-empty": "Please enter your name.",
|
||||
"email-empty": "Please enter a valid email address.",
|
||||
"username-empty": "Please enter a username.",
|
||||
"passwords-no-match": "The passwords you entered do not match.",
|
||||
"email-username-exists": "An account with this email or username already exists."
|
||||
},
|
||||
"success": "Registration successful! You can now log in."
|
||||
},
|
||||
"errors": {
|
||||
"internal": "An internal error occurred."
|
||||
},
|
||||
"main-menu": {
|
||||
"main-menu": "Main menu",
|
||||
"edit-profile": "Edit profile",
|
||||
"campaigns": "Campaigns",
|
||||
"log-out": "Log out",
|
||||
"settings": "Settings"
|
||||
},
|
||||
"settings": {
|
||||
"tabs": {
|
||||
"account-settings": "Account settings",
|
||||
"site-administration": "Site administration"
|
||||
},
|
||||
"account-settings": {
|
||||
"appearance": "Appearance",
|
||||
"language": "Language"
|
||||
},
|
||||
"site-administration": {
|
||||
"manage-accounts": "Manage accounts"
|
||||
}
|
||||
}
|
||||
}
|
||||
{}
|
||||
10
frontend/i18n/locales/en/campaigns.json
Normal file
10
frontend/i18n/locales/en/campaigns.json
Normal file
@@ -0,0 +1,10 @@
|
||||
{
|
||||
"create": {
|
||||
"name": "Name",
|
||||
"description": "Description",
|
||||
"description-placeholder": "Enter a brief description for your campaign...",
|
||||
"enter": "Enter campaign name here...",
|
||||
"color": "Accent color",
|
||||
"success": "Campaign created successfully!"
|
||||
}
|
||||
}
|
||||
10
frontend/i18n/locales/en/general.json
Normal file
10
frontend/i18n/locales/en/general.json
Normal file
@@ -0,0 +1,10 @@
|
||||
{
|
||||
"errors": {
|
||||
"internal": "An internal error occurred."
|
||||
},
|
||||
"create": "Create",
|
||||
"save": "Save",
|
||||
"cancel": "Cancel",
|
||||
"delete": "Delete",
|
||||
"open": "Open"
|
||||
}
|
||||
15
frontend/i18n/locales/en/login.json
Normal file
15
frontend/i18n/locales/en/login.json
Normal file
@@ -0,0 +1,15 @@
|
||||
|
||||
{
|
||||
"username": "Username or email",
|
||||
"username-placeholder": "Enter your username or email here...",
|
||||
"password": "Password",
|
||||
"password-placeholder": "Enter your password...",
|
||||
"log-in": "Log in",
|
||||
"no-account": "You don't have an account?",
|
||||
"register": "Register",
|
||||
"errors": {
|
||||
"invalid-credentials": "Invalid username/email or password.",
|
||||
"params": "Please enter both username/email and password."
|
||||
},
|
||||
"success": "Login successful!"
|
||||
}
|
||||
7
frontend/i18n/locales/en/main-menu.json
Normal file
7
frontend/i18n/locales/en/main-menu.json
Normal file
@@ -0,0 +1,7 @@
|
||||
{
|
||||
"main-menu": "Main menu",
|
||||
"edit-profile": "Edit profile",
|
||||
"create-campaign": "Create Campaign",
|
||||
"log-out": "Log out",
|
||||
"settings": "Settings"
|
||||
}
|
||||
27
frontend/i18n/locales/en/register.json
Normal file
27
frontend/i18n/locales/en/register.json
Normal file
@@ -0,0 +1,27 @@
|
||||
{
|
||||
"name": "Name",
|
||||
"name-placeholder": "Enter your name here...",
|
||||
"email": "Email",
|
||||
"email-placeholder": "Enter your email here...",
|
||||
"username": "Username",
|
||||
"username-placeholder": "Enter your username here...",
|
||||
"password": "Password",
|
||||
"password-placeholder": "Enter your password...",
|
||||
"confirm-password": "Confirm Password",
|
||||
"confirm-password-placeholder": "Re-enter your password...",
|
||||
"register": "Register",
|
||||
"have-account": "Already have an account?",
|
||||
"login": "Login",
|
||||
"password-confirm-placeholder": "Confirm your password...",
|
||||
"welcome": "Welcome to DragonRoll!",
|
||||
"message": "Please enter your desired username and password to create an account.",
|
||||
"first-register-message": "You are about to create the first account on this DragonRoll instance. This account will be granted administrator privileges.",
|
||||
"errors": {
|
||||
"name-empty": "Please enter your name.",
|
||||
"email-empty": "Please enter a valid email address.",
|
||||
"username-empty": "Please enter a username.",
|
||||
"passwords-no-match": "The passwords you entered do not match.",
|
||||
"email-username-exists": "An account with this email or username already exists."
|
||||
},
|
||||
"success": "Registration successful! You can now log in."
|
||||
}
|
||||
13
frontend/i18n/locales/en/settings.json
Normal file
13
frontend/i18n/locales/en/settings.json
Normal file
@@ -0,0 +1,13 @@
|
||||
{
|
||||
"tabs": {
|
||||
"account-settings": "Account settings",
|
||||
"site-administration": "Site administration"
|
||||
},
|
||||
"account-settings": {
|
||||
"appearance": "Appearance",
|
||||
"language": "Language"
|
||||
},
|
||||
"site-administration": {
|
||||
"manage-accounts": "Manage accounts"
|
||||
}
|
||||
}
|
||||
9
frontend/i18n/locales/en/windows.json
Normal file
9
frontend/i18n/locales/en/windows.json
Normal file
@@ -0,0 +1,9 @@
|
||||
{
|
||||
"login": "Login",
|
||||
"register": "Register",
|
||||
"main-menu": "Dragonroll",
|
||||
"example": "Example Window",
|
||||
"edit-profile": "Edit Profile",
|
||||
"settings": "Settings",
|
||||
"create-campaign": "Create Campaign"
|
||||
}
|
||||
@@ -1,74 +1 @@
|
||||
{
|
||||
"windows": {
|
||||
"login": "Iniciar sesión",
|
||||
"register": "Registrarse",
|
||||
"main-menu": "Dragonroll",
|
||||
"example": "Ventana de ejemplo",
|
||||
"edit-profile": "Editar perfil",
|
||||
"settings": "Configuración"
|
||||
},
|
||||
"login": {
|
||||
"username": "Usuario o correo electrónico",
|
||||
"username-placeholder": "Introduce tu usuario o correo electrónico...",
|
||||
"password": "Contraseña",
|
||||
"password-placeholder": "Introduce tu contraseña...",
|
||||
"log-in": "Iniciar sesión",
|
||||
"no-account": "¿No tienes una cuenta?",
|
||||
"register": "Registrarse",
|
||||
"errors": {
|
||||
"invalid-credentials": "Usuario/correo o contraseña incorrectos.",
|
||||
"params": "Por favor, introduce usuario/correo y contraseña."
|
||||
},
|
||||
"success": "¡Inicio de sesión exitoso!"
|
||||
},
|
||||
"register": {
|
||||
"name": "Nombre",
|
||||
"name-placeholder": "Introduce tu nombre...",
|
||||
"email": "Correo electrónico",
|
||||
"email-placeholder": "Introduce tu correo electrónico...",
|
||||
"username": "Usuario",
|
||||
"username-placeholder": "Introduce tu nombre de usuario...",
|
||||
"password": "Contraseña",
|
||||
"password-placeholder": "Introduce tu contraseña...",
|
||||
"confirm-password": "Confirmar contraseña",
|
||||
"confirm-password-placeholder": "Vuelve a introducir tu contraseña...",
|
||||
"register": "Registrarse",
|
||||
"have-account": "¿Ya tienes una cuenta?",
|
||||
"login": "Iniciar sesión",
|
||||
"password-confirm-placeholder": "Confirma tu contraseña...",
|
||||
"welcome": "¡Bienvenido a DragonRoll!",
|
||||
"message": "Por favor, introduce el usuario y la contraseña que deseas para crear una cuenta.",
|
||||
"first-register-message": "Estás a punto de crear la primera cuenta en esta instancia de DragonRoll. Esta cuenta tendrá privilegios de administrador.",
|
||||
"errors": {
|
||||
"name-empty": "Por favor, introduce tu nombre.",
|
||||
"email-empty": "Por favor, introduce un correo electrónico válido.",
|
||||
"username-empty": "Por favor, introduce un nombre de usuario.",
|
||||
"passwords-no-match": "Las contraseñas no coinciden.",
|
||||
"email-username-exists": "Ya existe una cuenta con este correo electrónico o nombre de usuario."
|
||||
},
|
||||
"success": "¡Registro exitoso! Ahora puedes iniciar sesión."
|
||||
},
|
||||
"errors": {
|
||||
"internal": "Ha ocurrido un error interno."
|
||||
},
|
||||
"main-menu": {
|
||||
"main-menu": "Menú principal",
|
||||
"edit-profile": "Editar perfil",
|
||||
"campaigns": "Campañas",
|
||||
"log-out": "Cerrar sesión",
|
||||
"settings": "Configuración"
|
||||
},
|
||||
"settings": {
|
||||
"tabs": {
|
||||
"account-settings": "Configuración de la cuenta",
|
||||
"site-administration": "Administración del sitio"
|
||||
},
|
||||
"account-settings": {
|
||||
"appearance": "Apariencia",
|
||||
"language": "Idioma"
|
||||
},
|
||||
"site-administration": {
|
||||
"manage-accounts": "Gestionar cuentas"
|
||||
}
|
||||
}
|
||||
}
|
||||
{}
|
||||
1
frontend/i18n/locales/es/general.json
Normal file
1
frontend/i18n/locales/es/general.json
Normal file
@@ -0,0 +1 @@
|
||||
{}
|
||||
14
frontend/i18n/locales/es/login.json
Normal file
14
frontend/i18n/locales/es/login.json
Normal file
@@ -0,0 +1,14 @@
|
||||
{
|
||||
"username": "Usuario o correo electrónico",
|
||||
"username-placeholder": "Introduce tu usuario o correo electrónico...",
|
||||
"password": "Contraseña",
|
||||
"password-placeholder": "Introduce tu contraseña...",
|
||||
"log-in": "Iniciar sesión",
|
||||
"no-account": "¿No tienes una cuenta?",
|
||||
"register": "Registrarse",
|
||||
"errors": {
|
||||
"invalid-credentials": "Usuario/correo o contraseña incorrectos.",
|
||||
"params": "Por favor, introduce usuario/correo y contraseña."
|
||||
},
|
||||
"success": "¡Inicio de sesión exitoso!"
|
||||
}
|
||||
7
frontend/i18n/locales/es/main-menu.json
Normal file
7
frontend/i18n/locales/es/main-menu.json
Normal file
@@ -0,0 +1,7 @@
|
||||
{
|
||||
"main-menu": "Menú principal",
|
||||
"edit-profile": "Editar perfil",
|
||||
"create-campaign": "Crear campanya",
|
||||
"log-out": "Cerrar sesión",
|
||||
"settings": "Configuración"
|
||||
}
|
||||
27
frontend/i18n/locales/es/register.json
Normal file
27
frontend/i18n/locales/es/register.json
Normal file
@@ -0,0 +1,27 @@
|
||||
{
|
||||
"name": "Nombre",
|
||||
"name-placeholder": "Introduce tu nombre...",
|
||||
"email": "Correo electrónico",
|
||||
"email-placeholder": "Introduce tu correo electrónico...",
|
||||
"username": "Usuario",
|
||||
"username-placeholder": "Introduce tu nombre de usuario...",
|
||||
"password": "Contraseña",
|
||||
"password-placeholder": "Introduce tu contraseña...",
|
||||
"confirm-password": "Confirmar contraseña",
|
||||
"confirm-password-placeholder": "Vuelve a introducir tu contraseña...",
|
||||
"register": "Registrarse",
|
||||
"have-account": "¿Ya tienes una cuenta?",
|
||||
"login": "Iniciar sesión",
|
||||
"password-confirm-placeholder": "Confirma tu contraseña...",
|
||||
"welcome": "¡Bienvenido a DragonRoll!",
|
||||
"message": "Por favor, introduce el usuario y la contraseña que deseas para crear una cuenta.",
|
||||
"first-register-message": "Estás a punto de crear la primera cuenta en esta instancia de DragonRoll. Esta cuenta tendrá privilegios de administrador.",
|
||||
"errors": {
|
||||
"name-empty": "Por favor, introduce tu nombre.",
|
||||
"email-empty": "Por favor, introduce un correo electrónico válido.",
|
||||
"username-empty": "Por favor, introduce un nombre de usuario.",
|
||||
"passwords-no-match": "Las contraseñas no coinciden.",
|
||||
"email-username-exists": "Ya existe una cuenta con este correo electrónico o nombre de usuario."
|
||||
},
|
||||
"success": "¡Registro exitoso! Ahora puedes iniciar sesión."
|
||||
}
|
||||
13
frontend/i18n/locales/es/settings.json
Normal file
13
frontend/i18n/locales/es/settings.json
Normal file
@@ -0,0 +1,13 @@
|
||||
{
|
||||
"tabs": {
|
||||
"account-settings": "Configuración de la cuenta",
|
||||
"site-administration": "Administración del sitio"
|
||||
},
|
||||
"account-settings": {
|
||||
"appearance": "Apariencia",
|
||||
"language": "Idioma"
|
||||
},
|
||||
"site-administration": {
|
||||
"manage-accounts": "Gestionar cuentas"
|
||||
}
|
||||
}
|
||||
9
frontend/i18n/locales/es/windows.json
Normal file
9
frontend/i18n/locales/es/windows.json
Normal file
@@ -0,0 +1,9 @@
|
||||
{
|
||||
"login": "Iniciar sesión",
|
||||
"register": "Registrarse",
|
||||
"main-menu": "Dragonroll",
|
||||
"example": "Ventana de ejemplo",
|
||||
"edit-profile": "Editar perfil",
|
||||
"settings": "Configuración",
|
||||
"create-campaign": "Crear campanya"
|
||||
}
|
||||
@@ -43,7 +43,7 @@ export default defineNuxtConfig({
|
||||
|
||||
runtimeConfig: {
|
||||
public: {
|
||||
apiBaseUrl: process.env.API_BASE_URL || 'http://localhost:5000/api',
|
||||
apiBaseUrl: process.env.NUXT_PUBLIC_API_BASE_URL || 'http://localhost:5000/api',
|
||||
gitCommit: git.commit,
|
||||
gitTag: git.tag,
|
||||
gitBranch: git.branch,
|
||||
|
||||
13
frontend/package-lock.json
generated
13
frontend/package-lock.json
generated
@@ -9,6 +9,7 @@
|
||||
"dependencies": {
|
||||
"@nuxtjs/i18n": "^10.3.0",
|
||||
"axios": "^1.15.2",
|
||||
"marked": "^18.0.2",
|
||||
"mitt": "^3.0.1",
|
||||
"motion": "^12.38.0",
|
||||
"nuxt": "^4.4.2",
|
||||
@@ -8886,6 +8887,18 @@
|
||||
"source-map-js": "^1.2.1"
|
||||
}
|
||||
},
|
||||
"node_modules/marked": {
|
||||
"version": "18.0.2",
|
||||
"resolved": "https://registry.npmjs.org/marked/-/marked-18.0.2.tgz",
|
||||
"integrity": "sha512-NsmlUYBS/Zg57rgDWMYdnre6OTj4e+qq/JS2ot3KrYLSoHLw+sDu0Nm1ZGpRgYAq6c+b1ekaY5NzVchMCQnzcg==",
|
||||
"license": "MIT",
|
||||
"bin": {
|
||||
"marked": "bin/marked.js"
|
||||
},
|
||||
"engines": {
|
||||
"node": ">= 20"
|
||||
}
|
||||
},
|
||||
"node_modules/math-intrinsics": {
|
||||
"version": "1.1.0",
|
||||
"resolved": "https://registry.npmjs.org/math-intrinsics/-/math-intrinsics-1.1.0.tgz",
|
||||
|
||||
@@ -3,7 +3,7 @@
|
||||
"type": "module",
|
||||
"private": true,
|
||||
"scripts": {
|
||||
"build": "nuxt build",
|
||||
"build": "nuxt build --dotenv .env.production",
|
||||
"dev": "nuxt dev",
|
||||
"generate": "nuxt generate",
|
||||
"preview": "nuxt preview",
|
||||
@@ -12,6 +12,7 @@
|
||||
"dependencies": {
|
||||
"@nuxtjs/i18n": "^10.3.0",
|
||||
"axios": "^1.15.2",
|
||||
"marked": "^18.0.2",
|
||||
"mitt": "^3.0.1",
|
||||
"motion": "^12.38.0",
|
||||
"nuxt": "^4.4.2",
|
||||
|
||||
Reference in New Issue
Block a user