13 Commits
0.4.2 ... main

Author SHA1 Message Date
9953e2765a Merge pull request 'homepage/welcome' (#113) from homepage/welcome into main
All checks were successful
Continuous Integration / Update Development (push) Successful in 3m7s
Reviewed-on: #113
2025-12-16 23:59:17 -06:00
f2acf86dd8 reduced log spam from client polling 2025-12-17 00:58:59 -05:00
2127a48a83 hid member controls from non member accounts in the calendar 2025-12-17 00:34:21 -05:00
6cb53deee3 added bg image 2025-12-17 00:26:52 -05:00
b6a9b6f855 fixed an error with loading the welcome message when logged in 2025-12-17 00:26:46 -05:00
2f7eb37771 Merge branch 'main' into homepage/welcome 2025-12-16 23:44:50 -05:00
0cfc5ce12b Started applying bg image (no work yet) 2025-12-16 23:44:37 -05:00
044dc78122 disabled WIP navigation options 2025-12-16 23:44:19 -05:00
2e31b9bce4 Redesigned marketing/landing page 2025-12-16 23:31:42 -05:00
fd6a1822f4 tweaked button primary hover state 2025-12-16 23:31:30 -05:00
d484c357fb tweaked background color slightly 2025-12-16 23:31:12 -05:00
5f03820891 added first pass of homepage 2025-12-16 11:49:14 -05:00
e9cce2571a fixed router redirect for homepage join button 2025-12-16 11:11:14 -05:00
11 changed files with 183 additions and 35 deletions

View File

@@ -7,7 +7,7 @@ import morgan = require('morgan');
const app = express() const app = express()
app.use(morgan('dev', { app.use(morgan('dev', {
skip: (req) => { skip: (req) => {
return req.path === '/members/me'; return req.originalUrl === '/members/me';
} }
})) }))
@@ -64,6 +64,7 @@ import { authRouter } from './routes/auth';
import { roles, memberRoles } from './routes/roles'; import { roles, memberRoles } from './routes/roles';
import { courseRouter, eventRouter } from './routes/course'; import { courseRouter, eventRouter } from './routes/course';
import { calendarRouter } from './routes/calendar'; import { calendarRouter } from './routes/calendar';
import { docsRouter } from './routes/docs';
app.use('/application', applicationRouter); app.use('/application', applicationRouter);
app.use('/ranks', ranks); app.use('/ranks', ranks);
@@ -77,6 +78,7 @@ app.use('/memberRoles', memberRoles)
app.use('/course', courseRouter) app.use('/course', courseRouter)
app.use('/courseEvent', eventRouter) app.use('/courseEvent', eventRouter)
app.use('/calendar', calendarRouter) app.use('/calendar', calendarRouter)
app.use('/docs', docsRouter)
app.use('/', authRouter) app.use('/', authRouter)
app.get('/ping', (req, res) => { app.get('/ping', (req, res) => {

24
api/src/routes/docs.ts Normal file
View File

@@ -0,0 +1,24 @@
const express = require('express');
const router = express.Router();
import { Request, Response } from 'express';
import { requireLogin } from '../middleware/auth';
router.get('/welcome', [requireLogin], async (req: Request, res: Response) => {
const output = await fetch(`${process.env.DOC_HOST}/api/pages/717`, {
headers: {
Authorization: `Token ${process.env.DOC_TOKEN_ID}:${process.env.DOC_TOKEN_SECRET}`,
}
})
if (output.ok) {
const out = await output.json();
res.status(200).json(out.html);
} else {
console.error("Failed to fetch LOA policy from bookstack");
res.sendStatus(500);
}
})
export const docsRouter = router;

BIN
ui/public/bg.jpg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 543 KiB

View File

@@ -22,7 +22,10 @@ const environment = import.meta.env.VITE_ENVIRONMENT;
</script> </script>
<template> <template>
<div class="flex flex-col min-h-screen"> <div class="flex flex-col min-h-screen" style="background-image: linear-gradient(rgba(0, 0, 0, 0.25), rgba(0, 0, 0, 0.25)), url('/bg.jpg');
background-size: contain;
background-attachment: fixed;
background-position: center;">
<div class="sticky top-0 bg-background z-50"> <div class="sticky top-0 bg-background z-50">
<Navbar class="flex"></Navbar> <Navbar class="flex"></Navbar>
<Alert v-if="environment == 'dev'" class="m-2 mx-auto w-5xl" variant="info"> <Alert v-if="environment == 'dev'" class="m-2 mx-auto w-5xl" variant="info">
@@ -32,8 +35,10 @@ const environment = import.meta.env.VITE_ENVIRONMENT;
</Alert> </Alert>
<Alert v-if="userStore.user?.LOAs?.[0]" class="m-2 mx-auto w-5xl" variant="info"> <Alert v-if="userStore.user?.LOAs?.[0]" class="m-2 mx-auto w-5xl" variant="info">
<AlertDescription class="flex flex-row items-center text-nowrap gap-5 mx-auto"> <AlertDescription class="flex flex-row items-center text-nowrap gap-5 mx-auto">
<p>You are on LOA until <strong>{{ formatDate(userStore.user?.LOAs?.[0].extended_till || userStore.user?.LOAs?.[0].end_date) }}</strong></p> <p>You are on LOA until <strong>{{ formatDate(userStore.user?.LOAs?.[0].extended_till ||
<Button variant="secondary" @click="async () => { await cancelLOA(userStore.user.LOAs?.[0].id); userStore.loadUser(); }">End userStore.user?.LOAs?.[0].end_date) }}</strong></p>
<Button variant="secondary"
@click="async () => { await cancelLOA(userStore.user.LOAs?.[0].id); userStore.loadUser(); }">End
LOA</Button> LOA</Button>
</AlertDescription> </AlertDescription>
</Alert> </Alert>

18
ui/src/api/docs.ts Normal file
View File

@@ -0,0 +1,18 @@
// @ts-ignore
const addr = import.meta.env.VITE_APIHOST;
export async function getWelcomeMessage(): Promise<string> {
const res = await fetch(`${addr}/docs/welcome`, {
method: "GET",
credentials: 'include',
});
if (res.ok) {
const out = res.json();
if (!out) {
return null;
}
return out;
} else {
return null;
}
}

View File

@@ -4,7 +4,7 @@
@custom-variant dark (&:is(.dark *)); @custom-variant dark (&:is(.dark *));
:root { :root {
--background: oklch(0.2046 0 0); --background: oklch(19.125% 0.00002 271.152);
--foreground: oklch(0.9219 0 0); --foreground: oklch(0.9219 0 0);
--card: oklch(23.075% 0.00003 271.152); --card: oklch(23.075% 0.00003 271.152);
--card-foreground: oklch(0.9219 0 0); --card-foreground: oklch(0.9219 0 0);

View File

@@ -85,11 +85,11 @@ function blurAfter() {
</RouterLink> </RouterLink>
</NavigationMenuLink> </NavigationMenuLink>
<NavigationMenuLink as-child :class="navigationMenuTriggerStyle()"> <!-- <NavigationMenuLink as-child :class="navigationMenuTriggerStyle()">
<RouterLink to="/transfer" @click="blurAfter"> <RouterLink to="/transfer" @click="blurAfter">
Transfer Request Transfer Request
</RouterLink> </RouterLink>
</NavigationMenuLink> </NavigationMenuLink> -->
<NavigationMenuLink as-child :class="navigationMenuTriggerStyle()"> <NavigationMenuLink as-child :class="navigationMenuTriggerStyle()">
<RouterLink to="/trainingReport" @click="blurAfter"> <RouterLink to="/trainingReport" @click="blurAfter">
@@ -107,13 +107,13 @@ function blurAfter() {
<NavigationMenuContent <NavigationMenuContent
class="grid gap-1 p-2 text-left [&_a]:w-full [&_a]:block [&_a]:whitespace-nowrap *:bg-transparent"> class="grid gap-1 p-2 text-left [&_a]:w-full [&_a]:block [&_a]:whitespace-nowrap *:bg-transparent">
<NavigationMenuLink <!-- <NavigationMenuLink
v-if="auth.hasAnyRole(['17th Administrator', '17th HQ', '17th Command'])" v-if="auth.hasAnyRole(['17th Administrator', '17th HQ', '17th Command'])"
as-child :class="navigationMenuTriggerStyle()"> as-child :class="navigationMenuTriggerStyle()">
<RouterLink to="/administration/rankChange" @click="blurAfter"> <RouterLink to="/administration/rankChange" @click="blurAfter">
Promotions Promotions
</RouterLink> </RouterLink>
</NavigationMenuLink> </NavigationMenuLink> -->
<NavigationMenuLink <NavigationMenuLink
v-if="auth.hasAnyRole(['17th Administrator', '17th HQ', '17th Command'])" v-if="auth.hasAnyRole(['17th Administrator', '17th HQ', '17th Command'])"
@@ -123,13 +123,13 @@ function blurAfter() {
</RouterLink> </RouterLink>
</NavigationMenuLink> </NavigationMenuLink>
<NavigationMenuLink <!-- <NavigationMenuLink
v-if="auth.hasAnyRole(['17th Administrator', '17th HQ', '17th Command'])" v-if="auth.hasAnyRole(['17th Administrator', '17th HQ', '17th Command'])"
as-child :class="navigationMenuTriggerStyle()"> as-child :class="navigationMenuTriggerStyle()">
<RouterLink to="/administration/transfer" @click="blurAfter"> <RouterLink to="/administration/transfer" @click="blurAfter">
Transfer Requests Transfer Requests
</RouterLink> </RouterLink>
</NavigationMenuLink> </NavigationMenuLink> -->
<NavigationMenuLink v-if="auth.hasRole('Recruiter')" as-child <NavigationMenuLink v-if="auth.hasRole('Recruiter')" as-child
:class="navigationMenuTriggerStyle()"> :class="navigationMenuTriggerStyle()">
@@ -147,11 +147,11 @@ function blurAfter() {
</NavigationMenuContent> </NavigationMenuContent>
</NavigationMenuItem> </NavigationMenuItem>
<NavigationMenuItem as-child :class="navigationMenuTriggerStyle()"> <!-- <NavigationMenuItem as-child :class="navigationMenuTriggerStyle()">
<RouterLink to="/members" @click="blurAfter"> <RouterLink to="/members" @click="blurAfter">
Members (debug) Members (debug)
</RouterLink> </RouterLink>
</NavigationMenuItem> </NavigationMenuItem> -->
</NavigationMenuList> </NavigationMenuList>
</NavigationMenu> </NavigationMenu>

View File

@@ -80,6 +80,7 @@ async function setAttendance(state: CalendarAttendance) {
const canEditEvent = computed(() => { const canEditEvent = computed(() => {
if (!userStore.isLoggedIn) return false; if (!userStore.isLoggedIn) return false;
if (userStore.state !== 'member') return false;
if (userStore.user.member.member_id == activeEvent.value.creator_id) if (userStore.user.member.member_id == activeEvent.value.creator_id)
return true; return true;
}); });
@@ -215,7 +216,7 @@ defineExpose({ forceReload })
<CircleAlert></CircleAlert> This event has been cancelled <CircleAlert></CircleAlert> This event has been cancelled
</div> </div>
</section> </section>
<section v-if="isPast && userStore.isLoggedIn" class="w-full"> <section v-if="isPast && userStore.state === 'member'" class="w-full">
<ButtonGroup class="flex w-full"> <ButtonGroup class="flex w-full">
<Button variant="outline" <Button variant="outline"
:class="myAttendance?.status === CalendarAttendance.Attending ? 'border-2 border-primary text-primary' : ''" :class="myAttendance?.status === CalendarAttendance.Attending ? 'border-2 border-primary text-primary' : ''"

View File

@@ -8,7 +8,7 @@ export const buttonVariants = cva(
variants: { variants: {
variant: { variant: {
default: default:
"bg-primary text-primary-foreground shadow-xs hover:bg-primary/90", "bg-primary text-primary-foreground shadow-xs hover:bg-primary/70",
destructive: destructive:
"bg-destructive text-white shadow-xs hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60", "bg-destructive text-white shadow-xs hover:bg-destructive/90 focus-visible:ring-destructive/20 dark:focus-visible:ring-destructive/40 dark:bg-destructive/60",
outline: outline:

View File

@@ -49,15 +49,8 @@ const dialogRef = ref<any>(null)
// NEW: handle day/time slot clicks to start creating an event // NEW: handle day/time slot clicks to start creating an event
function onDateClick(arg: { dateStr: string }) { function onDateClick(arg: { dateStr: string }) {
if (!userStore.isLoggedIn) return; if (!userStore.isLoggedIn) return;
if (userStore.state !== 'member') return;
dialogRef.value?.openDialog(arg.dateStr); dialogRef.value?.openDialog(arg.dateStr);
// For now, just open the panel with a draft payload.
// activeEvent.value = {
// id: '__draft__',
// title: 'New event',
// start: arg.dateStr,
// extendedProps: { draft: true }
// }
// panelOpen.value = true
} }
const calendarOptions = ref({ const calendarOptions = ref({
@@ -203,7 +196,7 @@ onMounted(() => {
@click="goToday"> @click="goToday">
Today Today
</button> </button>
<button v-if="userStore.isLoggedIn" <button v-if="userStore.isLoggedIn && userStore.state === 'member'"
class="cursor-pointer ml-1 inline-flex items-center gap-1.5 rounded-md bg-primary px-3 py-1.5 text-sm text-primary-foreground hover:opacity-90" class="cursor-pointer ml-1 inline-flex items-center gap-1.5 rounded-md bg-primary px-3 py-1.5 text-sm text-primary-foreground hover:opacity-90"
@click="onCreateEvent"> @click="onCreateEvent">
<Plus class="h-4 w-4" /> <Plus class="h-4 w-4" />

View File

@@ -1,6 +1,8 @@
<script setup lang="ts"> <script setup lang="ts">
import { getWelcomeMessage } from '@/api/docs';
import { Button } from '@/components/ui/button' import { Button } from '@/components/ui/button'
import { useUserStore } from '@/stores/user' import { useUserStore } from '@/stores/user'
import { onMounted, ref } from 'vue';
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
const router = useRouter() const router = useRouter()
@@ -8,23 +10,126 @@ const router = useRouter()
const user = useUserStore(); const user = useUserStore();
function goToApplication() { function goToApplication() {
router.push('/apply') // change to your form route router.push('/join') // change to your form route
} }
onMounted(async () => {
if (user.state == 'member') {
let policy = await getWelcomeMessage() as any;
welcomeRef.value.innerHTML = policy;
}
})
const welcomeRef = ref<HTMLElement>(null);
</script> </script>
<template> <template>
<div> <div>
<div v-if="user.state == 'guest'" class="flex flex-col items-center justify-center"> <div v-if="user.state == 'member'" class="mt-10">
<h1 class="text-4xl font-bold mb-4">Welcome to the 17th</h1> <div ref="welcomeRef" class="bookstack-container">
<p class="text-neutral-400 mb-8 max-w-md"> <!-- bookstack -->
To join our unit, please fill out an application to continue. </div>
</div>
<div v-else class="text-foreground px-6 py-12 selection:bg-primary/10">
<div class="max-w-5xl mx-auto space-y-8">
<header class="space-y-4">
<div class="flex flex-col md:flex-row md:items-end justify-between gap-4">
<div>
<h1 class="text-4xl font-semibold tracking-tight">17th Ranger Battalion</h1>
</div>
</div>
<div class="h-px bg-border w-full"></div>
</header>
<div class="grid grid-cols-1 lg:grid-cols-12 gap-12">
<div class="lg:col-span-7 space-y-6">
<div class="space-y-2">
<h2 class="text-sm font-medium uppercase tracking-wider text-primary">Unit Philosophy</h2>
<p class="text-lg leading-relaxed font-normal">
The 17th RBN emphasizes high-skill gameplay through real-world tactics, stripped of
traditional military formalities. We prioritize effective coordination over enforced
etiquette.
</p> </p>
<Button @click="goToApplication" class="px-6 py-3 text-lg"> </div>
<p class="text-muted-foreground leading-relaxed">
Our "Real Life First" mindset ensures participation remains a hobby, not a second job. With
a consistent roster of 4050 members for Saturday operations, we focus on effective
coordination and mission success without the requirement of "Yes sir, no sir" protocols.
</p>
<div class="flex flex-wrap items-center gap-6 pt-4">
<Button size="lg" @click="goToApplication" class="font-medium">
Begin Application Begin Application
</Button> </Button>
<div class="flex flex-col">
<span class="text-xs uppercase tracking-tighter text-muted-foreground">Age
Requirement</span>
<span class="text-sm font-medium">18+</span>
</div>
</div>
</div>
<div class="lg:col-span-5 space-y-8">
<section class="space-y-3">
<h3 class="text-xs font-semibold uppercase tracking-widest text-muted-foreground">
Operational Schedule</h3>
<div class="rounded-lg border bg-card p-4 shadow-sm">
<div class="flex justify-between items-center">
<span class="text-sm font-medium">Main Operation</span>
<span class="text-sm font-mono text-primary">Sat 19:00 CST</span>
</div>
</div>
</section>
<section class="space-y-4">
<h3 class="text-xs font-semibold uppercase tracking-[0.2em] text-muted-foreground">
Force Structure
</h3>
<ul class="space-y-3">
<li class="flex items-start gap-4">
<span class="h-1.5 w-1.5 rounded-full bg-primary mt-2 shrink-0"></span>
<div class="space-y-1">
<p class="text-sm font-medium leading-none">Alpha Company</p>
<p class="text-[13px] text-muted-foreground leading-relaxed">
Rifleman, Medic, CLS, Anti-Tank, RTO, Leadership
</p>
</div>
</li>
<li class="flex items-start gap-4">
<span class="h-1.5 w-1.5 rounded-full bg-primary mt-2 shrink-0"></span>
<div class="space-y-1">
<p class="text-sm font-medium leading-none">Echo Company</p>
<p class="text-[13px] text-muted-foreground leading-relaxed">
Logistics, CAS Pilot, Armor, Artillery, JTAC, Forward Observer
</p>
</div>
</li>
</ul>
<p class="text-[12px] text-muted-foreground/70 border-t pt-2 border-border">
Roles are fluid; specialization or weekly rotation is supported.
</p>
</section>
</div>
</div>
<section class="space-y-4">
<div class="relative rounded-xl border bg-card shadow-sm overflow-hidden">
<div class="aspect-video">
<iframe class="w-full h-full"
src="https://www.youtube.com/embed/61L397HwmrU?si=oY9qf6vFv6hXo6Fk&controls=1&mute=1&start=102&end=152"
title="YouTube video player" frameborder="0"
allow="accelerometer; autoplay; clipboard-write; encrypted-media; gyroscope; picture-in-picture; web-share"
allowfullscreen>
</iframe>
</div>
</div>
</section>
</div> </div>
<div v-else>
HOMEPAGEEEEEEEEEEEEEEEEEEE
</div> </div>
</div> </div>
</template> </template>