Refonte mini-player flottant, nettoyage GrateWizard, corrections UI

- PlayerPersistent: widget compact pill + panneau extensible, aligné au contenu
- BookPlayer: ajustements scroll mode, suppression bordures boutons
- UnoCSS: ajout border-none au shortcut btn-ghost
- GrateWizard: suppression composants, services et utils obsolètes
- Ajout du PDF source des paroles (media/)
- Mises à jour config et dépendances

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
Yvv
2026-02-22 22:43:41 +01:00
parent 0e1e704319
commit ac2b8040b1
25 changed files with 232 additions and 2268 deletions

View File

@@ -1,97 +0,0 @@
const CESIUM_PODS = [
'https://g1.data.brussels.ovh/user/profile/_search',
'https://g1.data.le-sou.org/user/profile/_search',
'https://g1.data.e-is.pro/user/profile/_search',
];
const BATCH_SIZE = 500;
export type GeoMember = {
pubkey: string;
title: string;
city: string;
lat: number;
lon: number;
};
/** Find the first Cesium+ pod that responds successfully. */
async function findWorkingPod(): Promise<string> {
for (const url of CESIUM_PODS) {
try {
const res = await fetch(url, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({ size: 0, query: { match_all: {} } }),
});
if (res.ok) return url;
} catch {
// try next pod
}
}
throw new Error('Aucun pod Cesium+ disponible');
}
/**
* Fetch Cesium+ profiles for a given list of v1 pubkeys.
* Uses Elasticsearch `ids` query with batches of 500, filtered to geolocated profiles only.
* Pass `null` to skip fetching (e.g. while pubkeys are still loading).
*/
export function useCesiumProfiles(v1Pubkeys: Ref<string[] | null>) {
const geoMembers = ref<GeoMember[]>([]);
const loading = ref(true);
const error = ref<string | null>(null);
watch(v1Pubkeys, async (pubkeys) => {
if (pubkeys === null) return;
loading.value = true;
error.value = null;
try {
const podUrl = await findWorkingPod();
const allMembers: GeoMember[] = [];
for (let i = 0; i < pubkeys.length; i += BATCH_SIZE) {
const batch = pubkeys.slice(i, i + BATCH_SIZE);
const res = await fetch(podUrl, {
method: 'POST',
headers: { 'Content-Type': 'application/json' },
body: JSON.stringify({
size: batch.length,
_source: ['title', 'city', 'geoPoint'],
query: {
bool: {
must: [
{ ids: { values: batch } },
{ exists: { field: 'geoPoint' } },
],
},
},
}),
});
if (!res.ok) throw new Error(`Cesium+ HTTP ${res.status}`);
const json = await res.json();
for (const hit of json.hits?.hits ?? []) {
const s = hit._source;
if (!s?.geoPoint?.lat || !s?.geoPoint?.lon) continue;
allMembers.push({
pubkey: hit._id,
title: s.title || '',
city: s.city || '',
lat: s.geoPoint.lat,
lon: s.geoPoint.lon,
});
}
}
geoMembers.value = allMembers;
} catch (e: any) {
error.value = e.message;
} finally {
loading.value = false;
}
}, { immediate: true });
return { geoMembers, loading, error };
}

View File

@@ -8,7 +8,7 @@ export function useGrateWizard() {
const win = window.open(
url,
'grateWizard',
`width=${popup.width},height=${popup.height},left=${left},top=${top},scrollbars=yes,resizable=yes`,
`width=${popup.width},height=${popup.height},left=${left},top=${top},menubar=no,toolbar=no,location=no,status=no,scrollbars=yes,resizable=yes`,
)
if (win) e?.preventDefault()
}

View File

@@ -1,24 +0,0 @@
export type SavedPerimeter = {
name: string;
polygon: [number, number][];
createdAt: string;
};
const STORAGE_KEY = 'gw-saved-perimeters';
export function useSavedPerimeters() {
const perimeters = useLocalStorage<SavedPerimeter[]>(STORAGE_KEY, []);
function savePerimeter(name: string, polygon: [number, number][]) {
perimeters.value = [
...perimeters.value.filter((p) => p.name !== name),
{ name, polygon, createdAt: new Date().toISOString() },
];
}
function deletePerimeter(name: string) {
perimeters.value = perimeters.value.filter((p) => p.name !== name);
}
return { perimeters, savePerimeter, deletePerimeter };
}