stack/apps/dev-launchpad/public/index.html
2024-12-18 11:44:12 -08:00

277 lines
6.9 KiB
HTML

<!DOCTYPE html>
<html>
<head>
<title>Stack Auth Dev Launchpad</title>
<style>
body {
font-family: Arial, sans-serif;
background-color: #e0f0e0;
padding-left: 16px;
padding-right: 16px;
}
.apps-container {
display: flex;
flex-direction: row;
flex-wrap: wrap;
gap: 16px;
}
.apps-container > a {
border: 1px solid #8888;
background-color: #fff;
padding: 0px 4px 8px 4px;
width: 120px;
text-decoration: none;
display: flex;
flex-direction: column;
align-items: center;
gap: 12px;
position: relative;
}
.apps-container > a.important {
background-color: #fee;
}
.apps-container > a.unimportant {
opacity: 0.2;
}
.apps-container > a.unimportant:hover {
opacity: 0.5;
}
.apps-container > a:hover {
border-color: #888;
transition: opacity 0.1s ease-in-out;
}
.apps-container > a > div > img {
height: 68px;
}
.apps-container > a > .description {
text-align: center;
font-size: 12px;
color: #888;
}
.apps-container > a > .port {
padding-right: 0.5px;
padding-top: 1px;
align-self: flex-end;
font-size: 12px;
color: #888;
}
.apps-container > a > .hover-description {
display: none;
position: absolute;
top: 100%;
left: -1px;
pointer-events: none;
z-index: 1000;
white-space: pre;
background-color: #ffc;
border: 1px solid #888;
padding: 2px;
color: #0008;
font-size: 12px;
}
.apps-container > a:hover > .hover-description {
display: block;
}
</style>
</head>
<body>
<h1>Stack Auth Dev Launchpad</h1>
<div class="apps-container"></div>
<hr />
<div class="apps-container"></div>
<hr />
<div class="apps-container"></div>
<h2 style="margin-top: 64px;">Background services</h2>
<ul>
<li>
5432: PostgreSQL
</li>
<li>
2500: Inbucket SMTP
</li>
<li>
1100: Inbucket POP3
</li>
<li>
4318: OTel collector
</li>
</ul>
<noscript>
This page requires JavaScript.
</noscript>
<script>
const apps = [
{
name: "Dashboard",
port: 8101,
description: [
"Src: ./apps/dashboard",
"Prod: https://app.stack-auth.com",
],
img: "https://www.svgrepo.com/show/507260/dashboard.svg",
importance: 2,
},
{
name: "Backend",
port: 8102,
description: [
"Src: ./apps/backend",
"Prod: https://api.stack-auth.com",
],
img: "https://www.svgrepo.com/show/340122/datastore.svg",
importance: 2,
},
{
name: "Demo app",
port: 8103,
description: [
"Src: ./examples/demo",
"Prod: https://demo.stack-auth.com",
],
importance: 2,
},
{
name: "Docs",
port: 8104,
description: [
"Src: ./docs",
"Prod: https://docs.stack-auth.com",
],
img: "https://www.svgrepo.com/show/448400/docs.svg",
importance: 2,
},
{
name: "Inbucket",
port: 8105,
img: "https://www.svgrepo.com/show/533176/at-sign.svg",
importance: 1,
description: [
"Email mock",
],
},
{
name: "Prisma Studio",
port: 8106,
importance: 1,
img: "https://encrypted-tbn0.gstatic.com/images?q=tbn:ANd9GcS95TdAw63YPAPcUpvRl4imIf-VJ1sGHnEvbw&s",
description: [
"Database interface",
],
},
{
name: "Jaeger UI (OTel)",
port: 8107,
description: [
"Performance & tracing",
],
importance: 1,
img: "https://www.jaegertracing.io/img/jaeger-icon-reverse-color.svg",
},
{
name: "examples/docs-examples",
port: 8108,
description: [
"Src: ./examples/docs-examples",
],
},
{
name: "examples/partial-prerendering",
port: 8109,
description: [
"Src: ./examples/partial-prerendering",
],
},
{
name: "examples/cjs-test",
port: 8110,
description: [
"Src: ./examples/cjs-test",
],
},
{
name: "examples/e-commerce",
port: 8111,
description: [
"Src: ./examples/e-commerce",
],
},
{
name: "examples/middleware",
port: 8112,
description: [
"Src: ./examples/middleware",
],
},
{
name: "Svix server",
port: 8113,
importance: 1,
img: "https://yt3.googleusercontent.com/H3gTWq_CsJiRsDcscTFlqDBzZSL5_LcO7HkILLT7G701SL-XUZPYEDIjPu5cExrdP4YO0Lkt=s900-c-k-c0x00ffffff-no-rj",
description: [
"Webhooks",
],
},
{
name: "OAuth mock server",
port: 8114,
description: [
"Src: ./apps/oauth-mock-server",
],
},
{
name: "examples/supabase",
port: 8115,
description: [
"Src: ./examples/supabase",
],
},
{
name: "PgHero",
port: 8116,
description: [
"For database monitoring",
],
importance: 1,
img: "https://pghero.dokkuapp.com/assets/pghero-88a0d052.png",
},
];
const appsContainers = document.querySelectorAll(".apps-container");
for (let i = 0; i < appsContainers.length; i++) {
const appContainer = appsContainers[i];
const importance = appsContainers.length - i - 1;
for (const app of apps) {
if ((app.importance ?? 0) === importance) {
// TODO escape HTML
appContainer.innerHTML += `
<a href="http://localhost:${app.port}" target="_blank" rel="noopener noreferrer" class="${app.importance === 2 ? "important" : app.importance === 1 ? "" : "unimportant"}">
<div class="port">:${app.port}</div>
<div>
<img src=${app.img || `//localhost:${app.port}/favicon.ico`} />
</div>
<span class="description">${app.name}</span>
${app.description ? `<div class="hover-description">${app.description.join("\n")}</div>` : ""}
</a>
`;
}
}
}
</script>
</body>
</html>