ba39b08e09
apple / swift (push) Successful in 1m6s
ci / rust (push) Successful in 5m51s
android / android (push) Successful in 6m21s
ci / web (push) Successful in 49s
ci / docs-site (push) Successful in 58s
windows-host / package (push) Successful in 8m6s
release / apple (push) Successful in 8m17s
deb / build-publish (push) Successful in 3m26s
decky / build-publish (push) Successful in 25s
docker / build-push (--build-arg FEDORA_VERSION=44, ci, ci/fedora-rpm.Dockerfile, punktfunk-fedora44-rpm) (push) Successful in 5s
ci / bench (push) Successful in 4m42s
docker / build-push (., web/Dockerfile, punktfunk-web) (push) Successful in 30s
docker / build-push (ci, ci/fedora-rpm.Dockerfile, punktfunk-fedora-rpm) (push) Successful in 2m36s
docker / build-push (ci, ci/rust-ci.Dockerfile, punktfunk-rust-ci) (push) Successful in 2m17s
rpm / build-publish (fedora-44, punktfunk-fedora44-rpm) (push) Failing after 19s
docker / build-push (docs-site, docs-site/Dockerfile, punktfunk-docs) (push) Successful in 51s
apple / screenshots (push) Successful in 5m45s
docker / deploy-docs (push) Successful in 22s
rpm / build-publish (bazzite, punktfunk-fedora-rpm) (push) Failing after 22s
Web console - Pairing/Library/Stats refactored into self-contained subsections that each own their own queries + mutations; a shared slot-based layout (view.tsx) is filled by the live page (containers) and Storybook (pure cards + fixtures) so the layout can't drift. - All paired devices in one list on Pairing with a protocol column (punktfunk/1 + Moonlight), routing each unpair to the right endpoint; the redundant Clients page is removed. - Library: overview grid split from the add/edit form into separate files. - Login screen links out to the docs. Docs - "Console login password" section on every host page (apt/RPM/Bazzite/SteamOS/Windows) plus a new "Forgot your Password?" troubleshooting page, linked from the login screen. - Console served as HTTP/1.1 over TLS (drop the unusable HTTP/3 advertising) across the Bun entry, launchers, systemd units, and packaging. Tooling - Biome now respects .gitignore (stops linting generated code), config migrated to 2.5.1; all lint issues fixed cleanly. Also includes this branch's in-progress host, Apple client, packaging, and CI changes. Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
143 lines
4.2 KiB
TypeScript
143 lines
4.2 KiB
TypeScript
import Section from "@unom/ui/section";
|
|
import type { FC } from "react";
|
|
import type { AvailableCompositor } from "@/api/gen/model/availableCompositor";
|
|
import type { HostInfo } from "@/api/gen/model/hostInfo";
|
|
import { QueryState } from "@/components/query-state";
|
|
import { Badge } from "@/components/ui/badge";
|
|
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card";
|
|
import type { Loadable } from "@/lib/query";
|
|
import { m } from "@/paraglide/messages";
|
|
|
|
export const HostView: FC<{
|
|
host: Loadable<HostInfo>;
|
|
compositors: Loadable<AvailableCompositor[]>;
|
|
}> = ({ host, compositors }) => {
|
|
const h = host.data;
|
|
return (
|
|
<Section maxWidth={false}>
|
|
<div className="flex flex-col gap-card">
|
|
<h1 className="text-2xl font-semibold">{m.nav_host()}</h1>
|
|
|
|
<QueryState
|
|
isLoading={host.isLoading}
|
|
error={host.error}
|
|
refetch={host.refetch}
|
|
>
|
|
{h && (
|
|
<div className="grid gap-card lg:grid-cols-2">
|
|
<Card>
|
|
<CardHeader>
|
|
<CardTitle>{m.host_identity()}</CardTitle>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<dl className="grid grid-cols-1 gap-3">
|
|
<Row label={m.host_hostname()} value={h.hostname} />
|
|
<Row label={m.host_local_ip()} value={h.local_ip} mono />
|
|
<Row
|
|
label={m.host_version()}
|
|
value={`${h.app_version} (${h.version})`}
|
|
/>
|
|
<Row label={m.host_abi()} value={String(h.abi_version)} />
|
|
<Row label={m.host_uniqueid()} value={h.uniqueid} mono />
|
|
</dl>
|
|
</CardContent>
|
|
</Card>
|
|
<div className="space-y-card">
|
|
<Card>
|
|
<CardHeader>
|
|
<CardTitle>{m.host_codecs()}</CardTitle>
|
|
</CardHeader>
|
|
<CardContent className="flex flex-wrap gap-2">
|
|
{h.codecs.map((c) => (
|
|
<Badge key={c} variant="secondary">
|
|
{c.toUpperCase()}
|
|
</Badge>
|
|
))}
|
|
</CardContent>
|
|
</Card>
|
|
<Card>
|
|
<CardHeader>
|
|
<CardTitle>{m.host_ports()}</CardTitle>
|
|
</CardHeader>
|
|
<CardContent>
|
|
<dl className="grid grid-cols-2 gap-x-6 gap-y-2 text-sm tabular-nums">
|
|
{Object.entries(h.ports).map(([k, v]) => (
|
|
<div key={k} className="flex justify-between">
|
|
<dt className="text-muted-foreground uppercase">
|
|
{k}
|
|
</dt>
|
|
<dd className="font-medium">{v as number}</dd>
|
|
</div>
|
|
))}
|
|
</dl>
|
|
</CardContent>
|
|
</Card>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</QueryState>
|
|
|
|
<Card>
|
|
<CardHeader>
|
|
<CardTitle>{m.host_compositors()}</CardTitle>
|
|
</CardHeader>
|
|
<CardContent className="space-y-4">
|
|
<p className="text-sm text-muted-foreground">
|
|
{m.host_compositors_help()}
|
|
</p>
|
|
<QueryState
|
|
isLoading={compositors.isLoading}
|
|
error={compositors.error}
|
|
refetch={compositors.refetch}
|
|
>
|
|
<ul className="divide-y rounded-md border">
|
|
{compositors.data?.map((c) => (
|
|
<li
|
|
key={c.id}
|
|
className="flex items-center justify-between gap-4 px-4 py-3"
|
|
>
|
|
<div className="min-w-0">
|
|
<div className="flex items-center gap-2">
|
|
<span className="font-medium">{c.label}</span>
|
|
{c.default && (
|
|
<Badge variant="secondary">
|
|
{m.compositor_default()}
|
|
</Badge>
|
|
)}
|
|
</div>
|
|
<code className="text-xs text-muted-foreground">
|
|
{c.id}
|
|
</code>
|
|
</div>
|
|
<Badge variant={c.available ? "default" : "outline"}>
|
|
{c.available
|
|
? m.compositor_available()
|
|
: m.compositor_unavailable()}
|
|
</Badge>
|
|
</li>
|
|
))}
|
|
</ul>
|
|
</QueryState>
|
|
</CardContent>
|
|
</Card>
|
|
</div>
|
|
</Section>
|
|
);
|
|
};
|
|
|
|
const Row: FC<{ label: string; value: string; mono?: boolean }> = ({
|
|
label,
|
|
value,
|
|
mono,
|
|
}) => (
|
|
<div className="flex items-baseline justify-between gap-4">
|
|
<dt className="text-sm text-muted-foreground">{label}</dt>
|
|
<dd
|
|
className={mono ? "truncate font-mono text-xs" : "font-medium"}
|
|
title={value}
|
|
>
|
|
{value}
|
|
</dd>
|
|
</div>
|
|
);
|