mirror of
https://github.com/9ParsonsB/Pulsar.git
synced 2025-04-05 17:39:39 -04:00
Update Flag handling
Status now shows readable flag names Updated Power display Added more missing Journal events
This commit is contained in:
parent
2ac1a927ca
commit
d4f3476ad3
@ -43,7 +43,7 @@ public class Scan : ScanBaryCentre
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
Parent = ParentList.ToImmutableList();
|
Parent = ParentList.ToImmutableList();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// "Parents" object rearranged into more intuitive structure for ease of use.
|
/// "Parents" object rearranged into more intuitive structure for ease of use.
|
||||||
|
@ -12,7 +12,6 @@ namespace Pulsar.Utils;
|
|||||||
|
|
||||||
using Observatory.Framework.Files.Journal;
|
using Observatory.Framework.Files.Journal;
|
||||||
|
|
||||||
|
|
||||||
[Flags]
|
[Flags]
|
||||||
public enum JournalReaderState
|
public enum JournalReaderState
|
||||||
{
|
{
|
||||||
@ -20,14 +19,17 @@ public enum JournalReaderState
|
|||||||
/// Have read the first character of the object
|
/// Have read the first character of the object
|
||||||
/// </summary>
|
/// </summary>
|
||||||
Start,
|
Start,
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Have read the timestamp. Generally the first property in a journal entry.
|
/// Have read the timestamp. Generally the first property in a journal entry.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
Timestamp,
|
Timestamp,
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// have read the event name. Generally the second property in a journal entry.
|
/// have read the event name. Generally the second property in a journal entry.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
Event,
|
Event,
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Have read the last character of the object, the next character should be a newline, whitespace, EOF, or another object.
|
/// Have read the last character of the object, the next character should be a newline, whitespace, EOF, or another object.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
@ -40,9 +42,10 @@ public enum JournalReaderState
|
|||||||
/// all Journals can be deserialized into a JournalBase object for identification
|
/// all Journals can be deserialized into a JournalBase object for identification
|
||||||
/// and then deserialized into their respective types.
|
/// and then deserialized into their respective types.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public class JournalConverter(ILogger logger) : JsonConverter<JournalBase>
|
public class JournalConverter(ILogger<JournalConverter> logger) : JsonConverter<JournalBase>
|
||||||
{
|
{
|
||||||
private JournalReaderState state = JournalReaderState.Start;
|
private JournalReaderState state = JournalReaderState.Start;
|
||||||
|
|
||||||
public override JournalBase? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
|
public override JournalBase? Read(ref Utf8JsonReader reader, Type typeToConvert, JsonSerializerOptions options)
|
||||||
{
|
{
|
||||||
Utf8JsonReader clone = reader;
|
Utf8JsonReader clone = reader;
|
||||||
@ -86,12 +89,13 @@ public class JournalConverter(ILogger logger) : JsonConverter<JournalBase>
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ((state & JournalReaderState.Event) != 0)
|
if ((state & JournalReaderState.Event) != 0)
|
||||||
{
|
{
|
||||||
// create destination type
|
// create destination type
|
||||||
return GetDestinationType(ref reader, eventName!);
|
return GetDestinationType(ref reader, eventName!);
|
||||||
}
|
}
|
||||||
|
|
||||||
break;
|
break;
|
||||||
case JsonTokenType.Comment:
|
case JsonTokenType.Comment:
|
||||||
continue;
|
continue;
|
||||||
@ -108,7 +112,6 @@ public class JournalConverter(ILogger logger) : JsonConverter<JournalBase>
|
|||||||
default:
|
default:
|
||||||
throw new ArgumentOutOfRangeException();
|
throw new ArgumentOutOfRangeException();
|
||||||
}
|
}
|
||||||
|
|
||||||
} while (clone.Read());
|
} while (clone.Read());
|
||||||
|
|
||||||
return new() { Timestamp = timestamp!.Value, Event = eventName! };
|
return new() { Timestamp = timestamp!.Value, Event = eventName! };
|
||||||
@ -340,7 +343,20 @@ public class JournalConverter(ILogger logger) : JsonConverter<JournalBase>
|
|||||||
return JsonSerializer.Deserialize<CommitCrime>(ref reader)!;
|
return JsonSerializer.Deserialize<CommitCrime>(ref reader)!;
|
||||||
case "modulestore":
|
case "modulestore":
|
||||||
return JsonSerializer.Deserialize<ModuleStore>(ref reader)!;
|
return JsonSerializer.Deserialize<ModuleStore>(ref reader)!;
|
||||||
|
case "factionkillbond":
|
||||||
|
return JsonSerializer.Deserialize<FactionKillBond>(ref reader)!;
|
||||||
|
case "rebootrepair":
|
||||||
|
return JsonSerializer.Deserialize<RebootRepair>(ref reader)!;
|
||||||
|
case "launchdrone":
|
||||||
|
return JsonSerializer.Deserialize<LaunchDrone>(ref reader)!;
|
||||||
|
case "sellmicroresources":
|
||||||
|
return JsonSerializer.Deserialize<SellMicroResources>(ref reader)!;
|
||||||
|
case "navbeaconscan":
|
||||||
|
return JsonSerializer.Deserialize<NavBeaconScan>(ref reader)!;
|
||||||
|
case "searchandrescue":
|
||||||
|
return JsonSerializer.Deserialize<SearchAndRescue>(ref reader)!;
|
||||||
|
case "marketsell":
|
||||||
|
return JsonSerializer.Deserialize<MarketSell>(ref reader)!;
|
||||||
default:
|
default:
|
||||||
logger.LogWarning("Unknown Journal event type {EventName}", eventName);
|
logger.LogWarning("Unknown Journal event type {EventName}", eventName);
|
||||||
return JsonSerializer.Deserialize<JournalBase>(ref reader)!;
|
return JsonSerializer.Deserialize<JournalBase>(ref reader)!;
|
||||||
|
@ -2,9 +2,12 @@
|
|||||||
import { onMount } from "svelte";
|
import { onMount } from "svelte";
|
||||||
import { statusStore } from "./stores/Status.store";
|
import { statusStore } from "./stores/Status.store";
|
||||||
import connection from "./stores/Connection.store";
|
import connection from "./stores/Connection.store";
|
||||||
import { StatusFlags, StatusFlags2 } from "../types/api/enums";
|
import { FocusStatus, StatusFlags, StatusFlags2 } from "../types/api/enums";
|
||||||
import type JournalBase from "../types/api/JournalBase";
|
import type JournalBase from "../types/api/JournalBase";
|
||||||
import { slide } from "svelte/transition";
|
import { fly, scale, slide } from "svelte/transition";
|
||||||
|
import { getEnumFlags, getEnumNameFromValue, getEnumNamesFromFlag, getEnumPairsFromValue } from "../types/flags";
|
||||||
|
import type Status from "../types/api/Status";
|
||||||
|
import { HubConnectionState } from "@microsoft/signalr";
|
||||||
|
|
||||||
const x: string | null = $state(null);
|
const x: string | null = $state(null);
|
||||||
|
|
||||||
@ -16,7 +19,11 @@
|
|||||||
let alert: JournalBase[] = $state([]);
|
let alert: JournalBase[] = $state([]);
|
||||||
|
|
||||||
onMount(async () => {
|
onMount(async () => {
|
||||||
await $connection.start();
|
|
||||||
|
if ($connection.state === HubConnectionState.Disconnected)
|
||||||
|
{
|
||||||
|
await $connection.start();
|
||||||
|
}
|
||||||
|
|
||||||
loading = false;
|
loading = false;
|
||||||
|
|
||||||
@ -36,9 +43,9 @@
|
|||||||
change.push(last[i] - last[i - 1]);
|
change.push(last[i] - last[i - 1]);
|
||||||
}
|
}
|
||||||
|
|
||||||
const avg = change.reduce((a, b) => a + b, 0) / change.length;
|
const avg = change.length ? change.reduce((a, b) => a + b, 0) / change.length : 0;
|
||||||
const max = 32;
|
const max = 32;
|
||||||
if ($statusStore.fuel?.fuelMain) {
|
if ($statusStore.fuel?.fuelMain && avg) {
|
||||||
timeToMax = (max - $statusStore.fuel?.fuelMain) / avg;
|
timeToMax = (max - $statusStore.fuel?.fuelMain) / avg;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -57,9 +64,15 @@
|
|||||||
});
|
});
|
||||||
|
|
||||||
if (!$statusStore.pips) {
|
if (!$statusStore.pips) {
|
||||||
statusStore.set(
|
const value = (await (
|
||||||
await (await fetch("http://localhost:5000/api/status")).json()
|
await fetch("http://localhost:5000/api/status")
|
||||||
);
|
).json()) as Status;
|
||||||
|
|
||||||
|
console.log(value);
|
||||||
|
|
||||||
|
statusStore.set({
|
||||||
|
...value,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
</script>
|
</script>
|
||||||
@ -88,33 +101,33 @@
|
|||||||
>
|
>
|
||||||
<div class="power">
|
<div class="power">
|
||||||
<div class="sys">
|
<div class="sys">
|
||||||
{#each [...Array($statusStore?.pips?.sys ?? 0)] as sys}
|
|
||||||
<div class="pip" transition:slide></div>
|
|
||||||
{/each}
|
|
||||||
<div>{$statusStore?.pips?.sys ?? "?"}</div>
|
<div>{$statusStore?.pips?.sys ?? "?"}</div>
|
||||||
<div>Sys</div>
|
<div>Sys</div>
|
||||||
</div>
|
{#each [...Array($statusStore?.pips?.sys ?? 0)].map((_,i) => i) as sys (sys)}
|
||||||
<div class="eng">
|
<div class="pip" transition:scale|global ></div>
|
||||||
{#each [...Array($statusStore?.pips?.eng ?? 0)] as eng}
|
|
||||||
<div class="pip" transition:slide></div>
|
|
||||||
{/each}
|
{/each}
|
||||||
<div>{$statusStore?.pips?.eng ?? "?"}</div>
|
</div>
|
||||||
|
<div class="eng">
|
||||||
|
<div>{$statusStore?.pips?.eng ?? "?"}</div>
|
||||||
<div>Eng</div>
|
<div>Eng</div>
|
||||||
</div>
|
{#each [...Array($statusStore?.pips?.eng ?? 0)].map((_,i) => i) as eng (eng)}
|
||||||
<div class="wep">
|
<div class="pip" transition:scale|global></div>
|
||||||
{#each [...Array($statusStore?.pips?.wep ?? 0)] as wep}
|
|
||||||
<div class="pip" transition:slide></div>
|
|
||||||
{/each}
|
{/each}
|
||||||
|
</div>
|
||||||
|
<div class="wep">
|
||||||
<div>{$statusStore?.pips?.wep ?? "?"}</div>
|
<div>{$statusStore?.pips?.wep ?? "?"}</div>
|
||||||
<div>Wep</div>
|
<div>Wep</div>
|
||||||
|
{#each [...Array($statusStore?.pips?.wep ?? 0)].map((_,i) => i) as wep (wep)}
|
||||||
|
<div class="pip" transition:scale|global></div>
|
||||||
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<span>dest?: {$statusStore?.destination?.name}</span>
|
<span>dest?: {$statusStore?.destination?.name}</span>
|
||||||
<span>gui focus: {$statusStore.guiFocus}</span>
|
<span>gui focus: {getEnumNameFromValue(FocusStatus, $statusStore.guiFocus!)}</span>
|
||||||
<span>cargo: {$statusStore.cargo}</span>
|
<span>cargo: {$statusStore.cargo}</span>
|
||||||
<span>flag1: {$statusStore.flags}</span>
|
<span>flag1: {getEnumNamesFromFlag(StatusFlags, $statusStore.flags!)}</span>
|
||||||
<span>flag2: {$statusStore.flags2}</span>
|
<span>flag2: {getEnumNamesFromFlag(StatusFlags2, $statusStore.flags2!)}</span>
|
||||||
{:else}
|
{:else}
|
||||||
<span>No data :(</span>
|
<span>No data :(</span>
|
||||||
{/if}
|
{/if}
|
||||||
@ -133,7 +146,7 @@
|
|||||||
div {
|
div {
|
||||||
display: flex;
|
display: flex;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
flex-direction: column;
|
flex-direction: column-reverse;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
div.pip {
|
div.pip {
|
||||||
min-width: 2vw;
|
min-width: 2vw;
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
export enum StatusFlags {
|
export enum StatusFlags {
|
||||||
Docked = 1,
|
None = 0,
|
||||||
|
Docked = 1 << 0,
|
||||||
Landed = 1 << 1,
|
Landed = 1 << 1,
|
||||||
LandingGear = 1 << 2,
|
LandingGear = 1 << 2,
|
||||||
Shields = 1 << 3,
|
Shields = 1 << 3,
|
||||||
@ -34,7 +35,8 @@ export enum StatusFlags {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export enum StatusFlags2 {
|
export enum StatusFlags2 {
|
||||||
OnFoot = 1,
|
None = 0,
|
||||||
|
OnFoot = 1 << 0,
|
||||||
InTaxi = 1 << 1,
|
InTaxi = 1 << 1,
|
||||||
InMulticrew = 1 << 2,
|
InMulticrew = 1 << 2,
|
||||||
OnFootInStation = 1 << 3,
|
OnFootInStation = 1 << 3,
|
||||||
|
52
Pulsar/WebApp/src/types/flags.ts
Normal file
52
Pulsar/WebApp/src/types/flags.ts
Normal file
@ -0,0 +1,52 @@
|
|||||||
|
const isPowerOfTwo = (x: number): boolean => {
|
||||||
|
return x !== 0 && (x & (x - 1)) === 0;
|
||||||
|
};
|
||||||
|
|
||||||
|
/***
|
||||||
|
* returns all possible flags for an enum
|
||||||
|
* @param obj the Enum type
|
||||||
|
*/
|
||||||
|
export function getEnumFlags<
|
||||||
|
O extends object,
|
||||||
|
F extends O[keyof O] = O[keyof O],
|
||||||
|
>(obj: O): [number, string][] {
|
||||||
|
const isFlag = (arg: string | number | F): arg is F => {
|
||||||
|
const nArg = Number(arg);
|
||||||
|
const isNumber = !Number.isNaN(nArg);
|
||||||
|
return isNumber && isPowerOfTwo(nArg);
|
||||||
|
};
|
||||||
|
|
||||||
|
if (!obj) return [];
|
||||||
|
|
||||||
|
return Object.entries(obj)
|
||||||
|
.filter((k) => Number.parseInt(k[0]))
|
||||||
|
.map((k) => [Number(k[0]), k[1]]);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getEnumPairsFromValue<E extends object>(e: E, flag: number) {
|
||||||
|
const flags = getEnumFlags(e);
|
||||||
|
const active = [];
|
||||||
|
for (const f of flags) {
|
||||||
|
if (f[0] & flag) active.push(f[1]);
|
||||||
|
}
|
||||||
|
return active;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getEnumNamesFromFlag<E extends Record<number, string>>(
|
||||||
|
e: E,
|
||||||
|
flag: number,
|
||||||
|
) {
|
||||||
|
const flags = getEnumFlags(e);
|
||||||
|
const active = [];
|
||||||
|
for (const f of flags) {
|
||||||
|
if (f[0] & flag) active.push(e[f[0]]);
|
||||||
|
}
|
||||||
|
return active;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getEnumNameFromValue<E extends Record<number, string>>(
|
||||||
|
e: E,
|
||||||
|
value: number,
|
||||||
|
) {
|
||||||
|
return e[value];
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user