feat: GPS workout UI polish and voice guidance improvements
All checks were successful
CI / update (push) Successful in 2m27s
All checks were successful
CI / update (push) Successful in 2m27s
- Start native GPS service in paused state during pre-start (notification shows "Waiting to start..." instead of running timer) - Bump notification importance to IMPORTANCE_DEFAULT for lock screen - Theme-aware glass blur overlay matching header style (dark/light mode) - Dark Nord blue background for activity picker, audio stats panel - Transparent overlay in pre-start, gradient fade for cancel button - Use Toggle component for voice announcements checkbox - Persist voice guidance settings to localStorage - Derive voice language from page language, remove language selector
This commit is contained in:
@@ -19,7 +19,7 @@ class AndroidBridge(private val context: Context) {
|
|||||||
private var ttsForVoices: TextToSpeech? = null
|
private var ttsForVoices: TextToSpeech? = null
|
||||||
|
|
||||||
@JavascriptInterface
|
@JavascriptInterface
|
||||||
fun startLocationService(ttsConfigJson: String) {
|
fun startLocationService(ttsConfigJson: String, startPaused: Boolean) {
|
||||||
if (context is Activity) {
|
if (context is Activity) {
|
||||||
// Request notification permission on Android 13+ (required for foreground service notification)
|
// Request notification permission on Android 13+ (required for foreground service notification)
|
||||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.TIRAMISU) {
|
||||||
@@ -50,6 +50,7 @@ class AndroidBridge(private val context: Context) {
|
|||||||
|
|
||||||
val intent = Intent(context, LocationForegroundService::class.java).apply {
|
val intent = Intent(context, LocationForegroundService::class.java).apply {
|
||||||
putExtra("ttsConfig", ttsConfigJson)
|
putExtra("ttsConfig", ttsConfigJson)
|
||||||
|
putExtra("startPaused", startPaused)
|
||||||
}
|
}
|
||||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {
|
||||||
context.startForegroundService(intent)
|
context.startForegroundService(intent)
|
||||||
@@ -58,10 +59,16 @@ class AndroidBridge(private val context: Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Backwards-compatible overload for calls without TTS config */
|
/** Overload: TTS config only (not paused) */
|
||||||
|
@JavascriptInterface
|
||||||
|
fun startLocationService(ttsConfigJson: String) {
|
||||||
|
startLocationService(ttsConfigJson, false)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** Overload: no args (not paused, no TTS) */
|
||||||
@JavascriptInterface
|
@JavascriptInterface
|
||||||
fun startLocationService() {
|
fun startLocationService() {
|
||||||
startLocationService("{}")
|
startLocationService("{}", false)
|
||||||
}
|
}
|
||||||
|
|
||||||
@JavascriptInterface
|
@JavascriptInterface
|
||||||
|
|||||||
@@ -127,10 +127,11 @@ class LocationForegroundService : Service(), TextToSpeech.OnInitListener {
|
|||||||
}
|
}
|
||||||
|
|
||||||
override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
|
override fun onStartCommand(intent: Intent?, flags: Int, startId: Int): Int {
|
||||||
|
val startPaused = intent?.getBooleanExtra("startPaused", false) ?: false
|
||||||
startTimeMs = System.currentTimeMillis()
|
startTimeMs = System.currentTimeMillis()
|
||||||
pausedAccumulatedMs = 0L
|
pausedAccumulatedMs = 0L
|
||||||
pausedSinceMs = 0L
|
pausedSinceMs = if (startPaused) startTimeMs else 0L
|
||||||
paused = false
|
paused = startPaused
|
||||||
totalDistanceKm = 0.0
|
totalDistanceKm = 0.0
|
||||||
lastLat = Double.NaN
|
lastLat = Double.NaN
|
||||||
lastLng = Double.NaN
|
lastLng = Double.NaN
|
||||||
@@ -151,7 +152,11 @@ class LocationForegroundService : Service(), TextToSpeech.OnInitListener {
|
|||||||
PendingIntent.FLAG_UPDATE_CURRENT or PendingIntent.FLAG_IMMUTABLE
|
PendingIntent.FLAG_UPDATE_CURRENT or PendingIntent.FLAG_IMMUTABLE
|
||||||
)
|
)
|
||||||
|
|
||||||
val notification = buildNotification("0:00", "0.00 km", "")
|
val notification = if (startPaused) {
|
||||||
|
buildNotification("Waiting to start...", "", "")
|
||||||
|
} else {
|
||||||
|
buildNotification("0:00", "0.00 km", "")
|
||||||
|
}
|
||||||
|
|
||||||
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.Q) {
|
||||||
startForeground(NOTIFICATION_ID, notification, android.content.pm.ServiceInfo.FOREGROUND_SERVICE_TYPE_LOCATION)
|
startForeground(NOTIFICATION_ID, notification, android.content.pm.ServiceInfo.FOREGROUND_SERVICE_TYPE_LOCATION)
|
||||||
@@ -488,7 +493,7 @@ class LocationForegroundService : Service(), TextToSpeech.OnInitListener {
|
|||||||
val channel = NotificationChannel(
|
val channel = NotificationChannel(
|
||||||
CHANNEL_ID,
|
CHANNEL_ID,
|
||||||
"GPS Tracking",
|
"GPS Tracking",
|
||||||
NotificationManager.IMPORTANCE_LOW
|
NotificationManager.IMPORTANCE_DEFAULT
|
||||||
).apply {
|
).apply {
|
||||||
description = "Shows while GPS is recording your workout"
|
description = "Shows while GPS is recording your workout"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -23,7 +23,7 @@ export interface VoiceGuidanceConfig {
|
|||||||
}
|
}
|
||||||
|
|
||||||
interface AndroidBridge {
|
interface AndroidBridge {
|
||||||
startLocationService(ttsConfigJson: string): void;
|
startLocationService(ttsConfigJson: string, startPaused: boolean): void;
|
||||||
stopLocationService(): void;
|
stopLocationService(): void;
|
||||||
getPoints(): string;
|
getPoints(): string;
|
||||||
isTracking(): boolean;
|
isTracking(): boolean;
|
||||||
@@ -112,7 +112,7 @@ export function createGpsTracker() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function start(voiceGuidance?: VoiceGuidanceConfig) {
|
async function start(voiceGuidance?: VoiceGuidanceConfig, startPaused = false) {
|
||||||
_debugMsg = 'starting...';
|
_debugMsg = 'starting...';
|
||||||
if (!checkTauri() || isTracking) {
|
if (!checkTauri() || isTracking) {
|
||||||
_debugMsg = `bail: tauri=${checkTauri()} tracking=${isTracking}`;
|
_debugMsg = `bail: tauri=${checkTauri()} tracking=${isTracking}`;
|
||||||
@@ -145,7 +145,7 @@ export function createGpsTracker() {
|
|||||||
if (bridge) {
|
if (bridge) {
|
||||||
_debugMsg = 'starting native GPS service...';
|
_debugMsg = 'starting native GPS service...';
|
||||||
const ttsConfig = JSON.stringify(voiceGuidance ?? {});
|
const ttsConfig = JSON.stringify(voiceGuidance ?? {});
|
||||||
bridge.startLocationService(ttsConfig);
|
bridge.startLocationService(ttsConfig, startPaused);
|
||||||
// Poll the native side for collected points
|
// Poll the native side for collected points
|
||||||
_pollTimer = setInterval(pollPoints, POLL_INTERVAL_MS);
|
_pollTimer = setInterval(pollPoints, POLL_INTERVAL_MS);
|
||||||
_debugMsg = 'native GPS service started, polling...';
|
_debugMsg = 'native GPS service started, polling...';
|
||||||
|
|||||||
@@ -18,6 +18,7 @@
|
|||||||
import SetTable from '$lib/components/fitness/SetTable.svelte';
|
import SetTable from '$lib/components/fitness/SetTable.svelte';
|
||||||
import ExercisePicker from '$lib/components/fitness/ExercisePicker.svelte';
|
import ExercisePicker from '$lib/components/fitness/ExercisePicker.svelte';
|
||||||
import SyncIndicator from '$lib/components/fitness/SyncIndicator.svelte';
|
import SyncIndicator from '$lib/components/fitness/SyncIndicator.svelte';
|
||||||
|
import Toggle from '$lib/components/Toggle.svelte';
|
||||||
import { onMount } from 'svelte';
|
import { onMount } from 'svelte';
|
||||||
|
|
||||||
const workout = getWorkout();
|
const workout = getWorkout();
|
||||||
@@ -41,13 +42,26 @@
|
|||||||
|
|
||||||
let useGps = $state(gps.isTracking);
|
let useGps = $state(gps.isTracking);
|
||||||
|
|
||||||
// Voice guidance config
|
// Voice guidance config (defaults, overridden from localStorage in onMount)
|
||||||
let vgEnabled = $state(false);
|
let vgEnabled = $state(false);
|
||||||
let vgTriggerType = $state('distance');
|
let vgTriggerType = $state('distance');
|
||||||
let vgTriggerValue = $state(1);
|
let vgTriggerValue = $state(1);
|
||||||
let vgMetrics = $state(['totalTime', 'totalDistance', 'avgPace']);
|
let vgMetrics = $state(['totalTime', 'totalDistance', 'avgPace']);
|
||||||
let vgLanguage = $state('en');
|
const vgLanguage = $derived(lang);
|
||||||
let vgShowPanel = $state(false);
|
let vgShowPanel = $state(false);
|
||||||
|
let vgLoaded = $state(false);
|
||||||
|
|
||||||
|
// Persist voice guidance settings to localStorage
|
||||||
|
$effect(() => {
|
||||||
|
const settings = {
|
||||||
|
enabled: vgEnabled,
|
||||||
|
triggerType: vgTriggerType,
|
||||||
|
triggerValue: vgTriggerValue,
|
||||||
|
metrics: vgMetrics,
|
||||||
|
};
|
||||||
|
if (!vgLoaded) return;
|
||||||
|
localStorage.setItem('vg_settings', JSON.stringify(settings));
|
||||||
|
});
|
||||||
|
|
||||||
// GPS workout mode state — if we're restoring a GPS workout that was already tracking, it's started
|
// GPS workout mode state — if we're restoring a GPS workout that was already tracking, it's started
|
||||||
let gpsStarted = $state(gps.isTracking && workout.mode === 'gps' && !workout.paused);
|
let gpsStarted = $state(gps.isTracking && workout.mode === 'gps' && !workout.paused);
|
||||||
@@ -250,11 +264,24 @@
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Restore voice guidance settings from localStorage
|
||||||
|
try {
|
||||||
|
const saved = localStorage.getItem('vg_settings');
|
||||||
|
if (saved) {
|
||||||
|
const s = JSON.parse(saved);
|
||||||
|
if (typeof s.enabled === 'boolean') vgEnabled = s.enabled;
|
||||||
|
if (s.triggerType === 'distance' || s.triggerType === 'time') vgTriggerType = s.triggerType;
|
||||||
|
if (typeof s.triggerValue === 'number' && s.triggerValue > 0) vgTriggerValue = s.triggerValue;
|
||||||
|
if (Array.isArray(s.metrics)) vgMetrics = s.metrics;
|
||||||
|
}
|
||||||
|
} catch {}
|
||||||
|
vgLoaded = true;
|
||||||
|
|
||||||
// For GPS workouts in pre-start: start GPS immediately so the map
|
// For GPS workouts in pre-start: start GPS immediately so the map
|
||||||
// shows the user's position while they configure activity/audio.
|
// shows the user's position while they configure activity/audio.
|
||||||
if (workout.mode === 'gps' && !gpsStarted && !gps.isTracking) {
|
if (workout.mode === 'gps' && !gpsStarted && !gps.isTracking) {
|
||||||
_prestartGps = true;
|
_prestartGps = true;
|
||||||
gps.start();
|
gps.start(undefined, true);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -873,7 +900,7 @@
|
|||||||
<div class="gps-workout-map" use:mountMap></div>
|
<div class="gps-workout-map" use:mountMap></div>
|
||||||
|
|
||||||
<!-- Overlay: sits on top of the map at the bottom -->
|
<!-- Overlay: sits on top of the map at the bottom -->
|
||||||
<div class="gps-overlay">
|
<div class="gps-overlay" class:gps-overlay-prestart={!gpsStarted}>
|
||||||
{#if gpsStarted}
|
{#if gpsStarted}
|
||||||
<div class="gps-workout-stats">
|
<div class="gps-workout-stats">
|
||||||
<div class="gps-stat">
|
<div class="gps-stat">
|
||||||
@@ -950,10 +977,7 @@
|
|||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
{:else}
|
{:else}
|
||||||
<label class="vg-row">
|
<Toggle bind:checked={vgEnabled} label="Enable voice announcements" />
|
||||||
<input type="checkbox" bind:checked={vgEnabled} />
|
|
||||||
<span>Enable voice announcements</span>
|
|
||||||
</label>
|
|
||||||
|
|
||||||
{#if vgEnabled}
|
{#if vgEnabled}
|
||||||
<div class="vg-group">
|
<div class="vg-group">
|
||||||
@@ -989,13 +1013,6 @@
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="vg-group">
|
|
||||||
<span class="vg-label">Language</span>
|
|
||||||
<select class="vg-select" bind:value={vgLanguage}>
|
|
||||||
<option value="en">English</option>
|
|
||||||
<option value="de">Deutsch</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
{/if}
|
{/if}
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
@@ -1059,10 +1076,7 @@
|
|||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
{:else}
|
{:else}
|
||||||
<label class="vg-row">
|
<Toggle bind:checked={vgEnabled} label="Enable voice announcements" />
|
||||||
<input type="checkbox" bind:checked={vgEnabled} />
|
|
||||||
<span>Enable voice announcements</span>
|
|
||||||
</label>
|
|
||||||
|
|
||||||
{#if vgEnabled}
|
{#if vgEnabled}
|
||||||
<div class="vg-group">
|
<div class="vg-group">
|
||||||
@@ -1097,14 +1111,6 @@
|
|||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="vg-group">
|
|
||||||
<span class="vg-label">Language</span>
|
|
||||||
<select class="vg-select" bind:value={vgLanguage}>
|
|
||||||
<option value="en">English</option>
|
|
||||||
<option value="de">Deutsch</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
{/if}
|
{/if}
|
||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
@@ -1814,10 +1820,10 @@
|
|||||||
color: var(--nord7);
|
color: var(--nord7);
|
||||||
}
|
}
|
||||||
.gps-overlay .vg-panel {
|
.gps-overlay .vg-panel {
|
||||||
background: rgba(255,255,255,0.1);
|
background: rgba(46, 52, 64, 0.82);
|
||||||
backdrop-filter: blur(8px);
|
backdrop-filter: blur(8px);
|
||||||
-webkit-backdrop-filter: blur(8px);
|
-webkit-backdrop-filter: blur(8px);
|
||||||
border: 1px solid rgba(255,255,255,0.15);
|
border: 1px solid rgba(255,255,255,0.12);
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
padding: 0.6rem;
|
padding: 0.6rem;
|
||||||
border-top: none;
|
border-top: none;
|
||||||
@@ -1885,10 +1891,45 @@
|
|||||||
gap: 0.5rem;
|
gap: 0.5rem;
|
||||||
padding: 0.75rem;
|
padding: 0.75rem;
|
||||||
padding-bottom: calc(0.75rem + env(safe-area-inset-bottom, 0px));
|
padding-bottom: calc(0.75rem + env(safe-area-inset-bottom, 0px));
|
||||||
background: linear-gradient(to top, rgba(0,0,0,0.7) 60%, transparent);
|
background: var(--nav-bg, rgba(46, 52, 64, 0.82));
|
||||||
color: #fff;
|
backdrop-filter: blur(16px);
|
||||||
|
-webkit-backdrop-filter: blur(16px);
|
||||||
|
border-top: 1px solid var(--nav-border, rgba(255,255,255,0.08));
|
||||||
|
box-shadow: 0 -4px 24px var(--nav-shadow, rgba(0,0,0,0.25));
|
||||||
|
color: var(--nav-text-active, #fff);
|
||||||
pointer-events: none;
|
pointer-events: none;
|
||||||
}
|
}
|
||||||
|
@media (prefers-color-scheme: dark) {
|
||||||
|
.gps-overlay {
|
||||||
|
--nav-bg: rgba(20, 20, 20, 0.78);
|
||||||
|
--nav-border: rgba(255,255,255,0.06);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
:global(:root[data-theme="dark"]) .gps-overlay {
|
||||||
|
--nav-bg: rgba(20, 20, 20, 0.78);
|
||||||
|
--nav-border: rgba(255,255,255,0.06);
|
||||||
|
}
|
||||||
|
:global(:root[data-theme="light"]) .gps-overlay {
|
||||||
|
--nav-bg: rgba(255, 255, 255, 0.82);
|
||||||
|
--nav-border: rgba(0,0,0,0.08);
|
||||||
|
--nav-shadow: rgba(0,0,0,0.1);
|
||||||
|
--nav-text-active: var(--nord0);
|
||||||
|
}
|
||||||
|
@media (prefers-color-scheme: light) {
|
||||||
|
:global(:root:not([data-theme])) .gps-overlay {
|
||||||
|
--nav-bg: rgba(255, 255, 255, 0.82);
|
||||||
|
--nav-border: rgba(0,0,0,0.08);
|
||||||
|
--nav-shadow: rgba(0,0,0,0.1);
|
||||||
|
--nav-text-active: var(--nord0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.gps-overlay-prestart {
|
||||||
|
background: none !important;
|
||||||
|
backdrop-filter: none !important;
|
||||||
|
-webkit-backdrop-filter: none !important;
|
||||||
|
border-top: none !important;
|
||||||
|
box-shadow: none !important;
|
||||||
|
}
|
||||||
.gps-overlay > * {
|
.gps-overlay > * {
|
||||||
pointer-events: auto;
|
pointer-events: auto;
|
||||||
}
|
}
|
||||||
@@ -1906,14 +1947,13 @@
|
|||||||
font-size: 1.8rem;
|
font-size: 1.8rem;
|
||||||
font-weight: 800;
|
font-weight: 800;
|
||||||
font-variant-numeric: tabular-nums;
|
font-variant-numeric: tabular-nums;
|
||||||
color: #fff;
|
color: inherit;
|
||||||
text-shadow: 0 1px 4px rgba(0,0,0,0.5);
|
|
||||||
}
|
}
|
||||||
.gps-stat-unit {
|
.gps-stat-unit {
|
||||||
font-size: 0.7rem;
|
font-size: 0.7rem;
|
||||||
text-transform: uppercase;
|
text-transform: uppercase;
|
||||||
letter-spacing: 0.06em;
|
letter-spacing: 0.06em;
|
||||||
color: rgba(255,255,255,0.75);
|
opacity: 0.65;
|
||||||
}
|
}
|
||||||
.gps-options-grid {
|
.gps-options-grid {
|
||||||
display: grid;
|
display: grid;
|
||||||
@@ -1926,10 +1966,10 @@
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 0.25rem;
|
gap: 0.25rem;
|
||||||
padding: 0.65rem 0.5rem;
|
padding: 0.65rem 0.5rem;
|
||||||
background: rgba(255,255,255,0.12);
|
background: rgba(46, 52, 64, 0.82);
|
||||||
backdrop-filter: blur(8px);
|
backdrop-filter: blur(8px);
|
||||||
-webkit-backdrop-filter: blur(8px);
|
-webkit-backdrop-filter: blur(8px);
|
||||||
border: 1px solid rgba(255,255,255,0.2);
|
border: 1px solid rgba(255,255,255,0.12);
|
||||||
border-radius: 10px;
|
border-radius: 10px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font: inherit;
|
font: inherit;
|
||||||
@@ -1962,10 +2002,10 @@
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 0.5rem;
|
gap: 0.5rem;
|
||||||
padding: 0.55rem 0.75rem;
|
padding: 0.55rem 0.75rem;
|
||||||
background: rgba(255,255,255,0.1);
|
background: rgba(46, 52, 64, 0.82);
|
||||||
backdrop-filter: blur(8px);
|
backdrop-filter: blur(8px);
|
||||||
-webkit-backdrop-filter: blur(8px);
|
-webkit-backdrop-filter: blur(8px);
|
||||||
border: 1px solid rgba(255,255,255,0.2);
|
border: 1px solid rgba(255,255,255,0.12);
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
font: inherit;
|
font: inherit;
|
||||||
@@ -1976,8 +2016,9 @@
|
|||||||
}
|
}
|
||||||
.gps-activity-choice.active {
|
.gps-activity-choice.active {
|
||||||
border-color: var(--nord8);
|
border-color: var(--nord8);
|
||||||
background: rgba(136,192,208,0.25);
|
background: rgba(46, 52, 64, 0.9);
|
||||||
color: var(--nord8);
|
color: var(--nord8);
|
||||||
|
box-shadow: inset 0 0 0 1px rgba(136,192,208,0.25);
|
||||||
}
|
}
|
||||||
.gps-activity-choice:hover:not(.active) {
|
.gps-activity-choice:hover:not(.active) {
|
||||||
border-color: rgba(255,255,255,0.4);
|
border-color: rgba(255,255,255,0.4);
|
||||||
@@ -2010,13 +2051,15 @@
|
|||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
gap: 0.3rem;
|
gap: 0.3rem;
|
||||||
background: none;
|
|
||||||
border: none;
|
border: none;
|
||||||
color: rgba(255,255,255,0.5);
|
color: #fff;
|
||||||
|
text-shadow: 0 1px 4px rgba(0,0,0,0.6);
|
||||||
font: inherit;
|
font: inherit;
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
padding: 0.25rem;
|
padding: 1rem 0.75rem calc(0.75rem + env(safe-area-inset-bottom, 0px));
|
||||||
|
margin: 0 -0.75rem calc(-0.75rem - env(safe-area-inset-bottom, 0px));
|
||||||
|
background: linear-gradient(to top, rgba(0,0,0,0.55) 0%, rgba(0,0,0,0.3) 50%, transparent 100%);
|
||||||
}
|
}
|
||||||
.gps-cancel-link:hover {
|
.gps-cancel-link:hover {
|
||||||
color: var(--nord11);
|
color: var(--nord11);
|
||||||
@@ -2032,17 +2075,15 @@
|
|||||||
justify-content: center;
|
justify-content: center;
|
||||||
width: 3rem;
|
width: 3rem;
|
||||||
height: 3rem;
|
height: 3rem;
|
||||||
background: rgba(255,255,255,0.15);
|
background: rgba(128,128,128,0.12);
|
||||||
backdrop-filter: blur(8px);
|
border: 1px solid var(--nav-border, rgba(255,255,255,0.25));
|
||||||
-webkit-backdrop-filter: blur(8px);
|
|
||||||
border: 1px solid rgba(255,255,255,0.25);
|
|
||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
color: #fff;
|
color: inherit;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
flex-shrink: 0;
|
flex-shrink: 0;
|
||||||
}
|
}
|
||||||
.gps-overlay-pause:hover {
|
.gps-overlay-pause:hover {
|
||||||
background: rgba(255,255,255,0.25);
|
background: rgba(128,128,128,0.2);
|
||||||
}
|
}
|
||||||
.gps-overlay-cancel {
|
.gps-overlay-cancel {
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -2051,8 +2092,6 @@
|
|||||||
width: 3rem;
|
width: 3rem;
|
||||||
height: 3rem;
|
height: 3rem;
|
||||||
background: rgba(191,97,106,0.25);
|
background: rgba(191,97,106,0.25);
|
||||||
backdrop-filter: blur(8px);
|
|
||||||
-webkit-backdrop-filter: blur(8px);
|
|
||||||
border: 1px solid var(--nord11);
|
border: 1px solid var(--nord11);
|
||||||
border-radius: 50%;
|
border-radius: 50%;
|
||||||
color: var(--nord11);
|
color: var(--nord11);
|
||||||
|
|||||||
Reference in New Issue
Block a user