All checks were successful
Test Asgard Runner / test (push) Successful in 3s
- GatherManager: 2-tab editor (Resource Rates with 1x-10x presets, Advanced with Pickup/Quarry/Excavator/Survey modifiers), 9 resource types with slider+number inputs, CRUD + deploy + import via NATS - AutoDoors: Global settings (delay sliders, 6 toggles), 7 door type toggles, permission group overrides table, CRUD + deploy + import - DB: migrations 015 (gather_configs) + 018 (autodoors_configs) - Backend: GatherModule + AutoDoorsModule registered in app.module.ts - Frontend: Pinia stores, Vue views, router routes, sidebar nav items - Icons: Pickaxe (gather), DoorOpen (autodoors) - All type checks pass: tsc + vue-tsc zero errors Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
535 lines
21 KiB
Vue
535 lines
21 KiB
Vue
<script setup lang="ts">
|
|
import { ref, onMounted } from 'vue'
|
|
import { useGatherStore } from '@/stores/gather'
|
|
import {
|
|
Save,
|
|
Play,
|
|
Download,
|
|
Plus,
|
|
Trash2,
|
|
Pickaxe,
|
|
Settings as SettingsIcon,
|
|
} from 'lucide-vue-next'
|
|
|
|
const store = useGatherStore()
|
|
|
|
const activeTab = ref<'resources' | 'advanced'>('resources')
|
|
const showCreateModal = ref(false)
|
|
const showImportModal = ref(false)
|
|
const newConfigName = ref('')
|
|
const newConfigDesc = ref('')
|
|
const importConfigName = ref('')
|
|
|
|
const tabs = [
|
|
{ key: 'resources', label: 'Resource Rates', icon: Pickaxe },
|
|
{ key: 'advanced', label: 'Advanced', icon: SettingsIcon },
|
|
]
|
|
|
|
// Resource definitions for the main gather tab
|
|
const gatherResources = [
|
|
{ key: 'Wood', label: 'Wood' },
|
|
{ key: 'Stones', label: 'Stones' },
|
|
{ key: 'Metal Ore', label: 'Metal Ore' },
|
|
{ key: 'Sulfur Ore', label: 'Sulfur Ore' },
|
|
{ key: 'HQM Ore', label: 'HQM Ore' },
|
|
{ key: 'Cloth', label: 'Cloth' },
|
|
{ key: 'Leather', label: 'Leather' },
|
|
{ key: 'Animal Fat', label: 'Animal Fat' },
|
|
{ key: 'Bone Fragments', label: 'Bone Fragments' },
|
|
]
|
|
|
|
// Advanced resource categories
|
|
const pickupResources = [
|
|
{ key: 'Wood', label: 'Wood' },
|
|
{ key: 'Stones', label: 'Stones' },
|
|
{ key: 'Metal Ore', label: 'Metal Ore' },
|
|
{ key: 'Sulfur Ore', label: 'Sulfur Ore' },
|
|
]
|
|
|
|
const quarryResources = [
|
|
{ key: 'HQM Ore', label: 'HQM Ore' },
|
|
{ key: 'Metal Ore', label: 'Metal Ore' },
|
|
{ key: 'Sulfur Ore', label: 'Sulfur Ore' },
|
|
{ key: 'Stones', label: 'Stones' },
|
|
]
|
|
|
|
const excavatorResources = [
|
|
{ key: 'HQM Ore', label: 'HQM Ore' },
|
|
{ key: 'Metal Ore', label: 'Metal Ore' },
|
|
{ key: 'Sulfur Ore', label: 'Sulfur Ore' },
|
|
{ key: 'Stones', label: 'Stones' },
|
|
]
|
|
|
|
const surveyResources = [
|
|
{ key: 'Metal Ore', label: 'Metal Ore' },
|
|
{ key: 'Sulfur Ore', label: 'Sulfur Ore' },
|
|
{ key: 'Stones', label: 'Stones' },
|
|
{ key: 'HQM Ore', label: 'HQM Ore' },
|
|
]
|
|
|
|
const presets = [
|
|
{ label: '1x', value: 1 },
|
|
{ label: '2x', value: 2 },
|
|
{ label: '3x', value: 3 },
|
|
{ label: '5x', value: 5 },
|
|
{ label: '10x', value: 10 },
|
|
]
|
|
|
|
onMounted(async () => {
|
|
await store.fetchConfigs()
|
|
if (store.configs.length > 0 && store.configs[0]) {
|
|
await store.loadConfig(store.configs[0].id)
|
|
}
|
|
})
|
|
|
|
// --- Config data helpers ---
|
|
|
|
function getConfigValue(path: string, defaultValue: any = 1): any {
|
|
if (!store.currentConfig?.config_data) return defaultValue
|
|
const parts = path.split('.')
|
|
let current: any = store.currentConfig.config_data
|
|
for (const part of parts) {
|
|
if (current == null || typeof current !== 'object') return defaultValue
|
|
current = current[part]
|
|
}
|
|
return current ?? defaultValue
|
|
}
|
|
|
|
function setConfigValue(path: string, value: any) {
|
|
if (!store.currentConfig) return
|
|
if (!store.currentConfig.config_data) store.currentConfig.config_data = {}
|
|
const parts = path.split('.')
|
|
let current: any = store.currentConfig.config_data
|
|
for (let i = 0; i < parts.length - 1; i++) {
|
|
const part = parts[i]!
|
|
if (current[part] == null || typeof current[part] !== 'object') {
|
|
current[part] = {}
|
|
}
|
|
current = current[part]
|
|
}
|
|
current[parts[parts.length - 1]!] = value
|
|
store.markDirty()
|
|
}
|
|
|
|
// --- Preset handler ---
|
|
|
|
function applyPreset(multiplier: number) {
|
|
for (const resource of gatherResources) {
|
|
setConfigValue(`GatherResourceModifiers.${resource.key}`, multiplier)
|
|
}
|
|
}
|
|
|
|
// --- Action handlers ---
|
|
|
|
async function handleConfigChange(id: string) {
|
|
if (store.isDirty) {
|
|
if (!confirm('Unsaved changes will be lost. Continue?')) return
|
|
}
|
|
await store.loadConfig(id)
|
|
}
|
|
|
|
async function handleCreateConfig() {
|
|
if (!newConfigName.value.trim()) return
|
|
const config = await store.createConfig(
|
|
newConfigName.value.trim(),
|
|
newConfigDesc.value.trim() || undefined,
|
|
)
|
|
if (config) {
|
|
showCreateModal.value = false
|
|
newConfigName.value = ''
|
|
newConfigDesc.value = ''
|
|
}
|
|
}
|
|
|
|
async function handleDeleteConfig() {
|
|
if (!store.currentConfig) return
|
|
if (!confirm(`Delete "${store.currentConfig.config_name}"? This cannot be undone.`)) return
|
|
await store.deleteConfig(store.currentConfig.id)
|
|
}
|
|
|
|
async function handleApply() {
|
|
if (!store.currentConfig) return
|
|
if (!confirm('Apply this gather config to the server? This will overwrite the current GatherManager config.')) return
|
|
if (store.isDirty) {
|
|
await store.saveCurrentConfig()
|
|
}
|
|
await store.applyToServer(store.currentConfig.id)
|
|
}
|
|
|
|
async function handleImport() {
|
|
if (!importConfigName.value.trim()) return
|
|
const config = await store.importFromServer(importConfigName.value.trim())
|
|
if (config) {
|
|
showImportModal.value = false
|
|
importConfigName.value = ''
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<template>
|
|
<div class="p-6 space-y-6">
|
|
<!-- Header -->
|
|
<div class="flex items-center justify-between">
|
|
<h1 class="text-2xl font-bold text-white">Gather Rates</h1>
|
|
<div class="flex items-center gap-3">
|
|
<button
|
|
@click="showCreateModal = true"
|
|
class="flex items-center gap-2 px-3 py-2 bg-neutral-800 text-neutral-300 rounded-lg hover:bg-neutral-700 text-sm"
|
|
>
|
|
<Plus class="w-4 h-4" />
|
|
New Config
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Config Selector + Action Bar -->
|
|
<div class="bg-neutral-900 border border-neutral-800 rounded-xl p-4">
|
|
<div class="flex items-center gap-3 flex-wrap">
|
|
<!-- Config Selector -->
|
|
<select
|
|
v-if="store.configs.length > 0"
|
|
:value="store.currentConfig?.id || ''"
|
|
@change="handleConfigChange(($event.target as HTMLSelectElement).value)"
|
|
class="bg-neutral-800 border border-neutral-700 text-neutral-200 rounded-lg px-3 py-2 text-sm min-w-[200px]"
|
|
>
|
|
<option v-for="c in store.configs" :key="c.id" :value="c.id">
|
|
{{ c.config_name }}
|
|
<template v-if="c.is_active"> (Active)</template>
|
|
</option>
|
|
</select>
|
|
<span v-else class="text-neutral-500 text-sm">No configs yet</span>
|
|
|
|
<!-- Save -->
|
|
<button
|
|
@click="store.saveCurrentConfig()"
|
|
:disabled="!store.currentConfig || !store.isDirty || store.isSaving"
|
|
class="flex items-center gap-2 px-3 py-2 bg-oxide-500 text-white rounded-lg hover:bg-oxide-600 disabled:opacity-50 disabled:cursor-not-allowed text-sm"
|
|
>
|
|
<Save class="w-4 h-4" />
|
|
{{ store.isSaving ? 'Saving...' : 'Save' }}
|
|
</button>
|
|
|
|
<!-- Apply to Server -->
|
|
<button
|
|
@click="handleApply"
|
|
:disabled="!store.currentConfig || store.isApplying"
|
|
class="flex items-center gap-2 px-3 py-2 bg-green-600 text-white rounded-lg hover:bg-green-700 disabled:opacity-50 disabled:cursor-not-allowed text-sm"
|
|
>
|
|
<Play class="w-4 h-4" />
|
|
{{ store.isApplying ? 'Applying...' : 'Apply to Server' }}
|
|
</button>
|
|
|
|
<!-- Import from Server -->
|
|
<button
|
|
@click="showImportModal = true"
|
|
class="flex items-center gap-2 px-3 py-2 bg-neutral-800 text-neutral-300 rounded-lg hover:bg-neutral-700 text-sm"
|
|
>
|
|
<Download class="w-4 h-4" />
|
|
Import from Server
|
|
</button>
|
|
|
|
<!-- Delete -->
|
|
<button
|
|
@click="handleDeleteConfig"
|
|
:disabled="!store.currentConfig"
|
|
class="flex items-center gap-2 px-3 py-2 bg-red-500/10 text-red-400 rounded-lg hover:bg-red-500/20 disabled:opacity-50 text-sm ml-auto"
|
|
>
|
|
<Trash2 class="w-4 h-4" />
|
|
Delete
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Loading State -->
|
|
<div v-if="store.isLoading" class="flex items-center justify-center py-20">
|
|
<div class="animate-spin w-8 h-8 border-2 border-oxide-500 border-t-transparent rounded-full" />
|
|
</div>
|
|
|
|
<!-- No Config Selected -->
|
|
<div v-else-if="!store.currentConfig" class="bg-neutral-900 border border-neutral-800 rounded-xl p-12 text-center">
|
|
<Pickaxe class="w-12 h-12 text-neutral-600 mx-auto mb-4" />
|
|
<h2 class="text-lg font-semibold text-neutral-300 mb-2">No Gather Config Selected</h2>
|
|
<p class="text-neutral-500 mb-4">Create a new config, import from server, or select one from the dropdown above.</p>
|
|
<button
|
|
@click="showCreateModal = true"
|
|
class="px-4 py-2 bg-oxide-500 text-white rounded-lg hover:bg-oxide-600"
|
|
>
|
|
Create First Config
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Config Editor -->
|
|
<div v-else class="space-y-6">
|
|
<!-- Tab Bar -->
|
|
<div class="flex border-b border-neutral-800">
|
|
<button
|
|
v-for="tab in tabs"
|
|
:key="tab.key"
|
|
@click="activeTab = tab.key as typeof activeTab"
|
|
class="flex items-center gap-2 px-4 py-2 text-sm font-medium border-b-2 transition-colors"
|
|
:class="activeTab === tab.key
|
|
? 'border-oxide-500 text-oxide-400'
|
|
: 'border-transparent text-neutral-500 hover:text-neutral-300'"
|
|
>
|
|
<component :is="tab.icon" class="w-4 h-4" />
|
|
{{ tab.label }}
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Resource Rates Tab -->
|
|
<div v-if="activeTab === 'resources'" class="bg-neutral-900 border border-neutral-800 rounded-xl p-6 space-y-6">
|
|
<div class="flex items-center justify-between">
|
|
<h3 class="text-sm font-semibold text-neutral-300 uppercase tracking-wider">Gather Resource Modifiers</h3>
|
|
<div class="flex items-center gap-2">
|
|
<span class="text-xs text-neutral-500 mr-2">Presets:</span>
|
|
<button
|
|
v-for="preset in presets"
|
|
:key="preset.value"
|
|
@click="applyPreset(preset.value)"
|
|
class="px-3 py-1 text-xs bg-neutral-800 text-neutral-300 rounded-lg hover:bg-neutral-700 hover:text-white transition-colors"
|
|
>
|
|
{{ preset.label }}
|
|
</button>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="space-y-4">
|
|
<div
|
|
v-for="resource in gatherResources"
|
|
:key="resource.key"
|
|
class="flex items-center gap-4"
|
|
>
|
|
<label class="text-sm text-neutral-200 w-32 flex-shrink-0">{{ resource.label }}</label>
|
|
<input
|
|
type="range"
|
|
:value="getConfigValue(`GatherResourceModifiers.${resource.key}`, 1)"
|
|
@input="setConfigValue(`GatherResourceModifiers.${resource.key}`, Number(($event.target as HTMLInputElement).value))"
|
|
min="0.1"
|
|
max="100"
|
|
step="0.1"
|
|
class="flex-1 accent-oxide-500"
|
|
/>
|
|
<input
|
|
type="number"
|
|
:value="getConfigValue(`GatherResourceModifiers.${resource.key}`, 1)"
|
|
@input="setConfigValue(`GatherResourceModifiers.${resource.key}`, Number(($event.target as HTMLInputElement).value))"
|
|
min="0.1"
|
|
max="1000"
|
|
step="0.1"
|
|
class="w-20 bg-neutral-800 border border-neutral-700 rounded-lg px-2 py-1 text-neutral-200 text-sm text-center"
|
|
/>
|
|
<span class="text-xs text-neutral-500 w-4">x</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Advanced Tab -->
|
|
<div v-else-if="activeTab === 'advanced'" class="space-y-6">
|
|
<!-- Pickup Resource Modifiers -->
|
|
<div class="bg-neutral-900 border border-neutral-800 rounded-xl p-6 space-y-4">
|
|
<h3 class="text-sm font-semibold text-neutral-300 uppercase tracking-wider">Pickup Resource Modifiers</h3>
|
|
<p class="text-xs text-neutral-500">Modify rates for resources picked up from the ground (small rocks, wood piles).</p>
|
|
<div class="space-y-4">
|
|
<div
|
|
v-for="resource in pickupResources"
|
|
:key="resource.key"
|
|
class="flex items-center gap-4"
|
|
>
|
|
<label class="text-sm text-neutral-200 w-32 flex-shrink-0">{{ resource.label }}</label>
|
|
<input
|
|
type="range"
|
|
:value="getConfigValue(`PickupResourceModifiers.${resource.key}`, 1)"
|
|
@input="setConfigValue(`PickupResourceModifiers.${resource.key}`, Number(($event.target as HTMLInputElement).value))"
|
|
min="0.1"
|
|
max="100"
|
|
step="0.1"
|
|
class="flex-1 accent-oxide-500"
|
|
/>
|
|
<input
|
|
type="number"
|
|
:value="getConfigValue(`PickupResourceModifiers.${resource.key}`, 1)"
|
|
@input="setConfigValue(`PickupResourceModifiers.${resource.key}`, Number(($event.target as HTMLInputElement).value))"
|
|
min="0.1"
|
|
max="1000"
|
|
step="0.1"
|
|
class="w-20 bg-neutral-800 border border-neutral-700 rounded-lg px-2 py-1 text-neutral-200 text-sm text-center"
|
|
/>
|
|
<span class="text-xs text-neutral-500 w-4">x</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Quarry Resource Modifiers -->
|
|
<div class="bg-neutral-900 border border-neutral-800 rounded-xl p-6 space-y-4">
|
|
<h3 class="text-sm font-semibold text-neutral-300 uppercase tracking-wider">Quarry Resource Modifiers</h3>
|
|
<p class="text-xs text-neutral-500">Scale resource output from Mining Quarries.</p>
|
|
<div class="space-y-4">
|
|
<div
|
|
v-for="resource in quarryResources"
|
|
:key="resource.key"
|
|
class="flex items-center gap-4"
|
|
>
|
|
<label class="text-sm text-neutral-200 w-32 flex-shrink-0">{{ resource.label }}</label>
|
|
<input
|
|
type="range"
|
|
:value="getConfigValue(`QuarryResourceModifiers.${resource.key}`, 1)"
|
|
@input="setConfigValue(`QuarryResourceModifiers.${resource.key}`, Number(($event.target as HTMLInputElement).value))"
|
|
min="0.1"
|
|
max="100"
|
|
step="0.1"
|
|
class="flex-1 accent-oxide-500"
|
|
/>
|
|
<input
|
|
type="number"
|
|
:value="getConfigValue(`QuarryResourceModifiers.${resource.key}`, 1)"
|
|
@input="setConfigValue(`QuarryResourceModifiers.${resource.key}`, Number(($event.target as HTMLInputElement).value))"
|
|
min="0.1"
|
|
max="1000"
|
|
step="0.1"
|
|
class="w-20 bg-neutral-800 border border-neutral-700 rounded-lg px-2 py-1 text-neutral-200 text-sm text-center"
|
|
/>
|
|
<span class="text-xs text-neutral-500 w-4">x</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Excavator Resource Modifiers -->
|
|
<div class="bg-neutral-900 border border-neutral-800 rounded-xl p-6 space-y-4">
|
|
<h3 class="text-sm font-semibold text-neutral-300 uppercase tracking-wider">Excavator Resource Modifiers</h3>
|
|
<p class="text-xs text-neutral-500">Scale resource output from the Giant Excavator.</p>
|
|
<div class="space-y-4">
|
|
<div
|
|
v-for="resource in excavatorResources"
|
|
:key="resource.key"
|
|
class="flex items-center gap-4"
|
|
>
|
|
<label class="text-sm text-neutral-200 w-32 flex-shrink-0">{{ resource.label }}</label>
|
|
<input
|
|
type="range"
|
|
:value="getConfigValue(`ExcavatorResourceModifiers.${resource.key}`, 1)"
|
|
@input="setConfigValue(`ExcavatorResourceModifiers.${resource.key}`, Number(($event.target as HTMLInputElement).value))"
|
|
min="0.1"
|
|
max="100"
|
|
step="0.1"
|
|
class="flex-1 accent-oxide-500"
|
|
/>
|
|
<input
|
|
type="number"
|
|
:value="getConfigValue(`ExcavatorResourceModifiers.${resource.key}`, 1)"
|
|
@input="setConfigValue(`ExcavatorResourceModifiers.${resource.key}`, Number(($event.target as HTMLInputElement).value))"
|
|
min="0.1"
|
|
max="1000"
|
|
step="0.1"
|
|
class="w-20 bg-neutral-800 border border-neutral-700 rounded-lg px-2 py-1 text-neutral-200 text-sm text-center"
|
|
/>
|
|
<span class="text-xs text-neutral-500 w-4">x</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Survey Resource Modifiers -->
|
|
<div class="bg-neutral-900 border border-neutral-800 rounded-xl p-6 space-y-4">
|
|
<h3 class="text-sm font-semibold text-neutral-300 uppercase tracking-wider">Survey Charge Resource Modifiers</h3>
|
|
<p class="text-xs text-neutral-500">Modify resource amounts from Survey Charge grenades.</p>
|
|
<div class="space-y-4">
|
|
<div
|
|
v-for="resource in surveyResources"
|
|
:key="resource.key"
|
|
class="flex items-center gap-4"
|
|
>
|
|
<label class="text-sm text-neutral-200 w-32 flex-shrink-0">{{ resource.label }}</label>
|
|
<input
|
|
type="range"
|
|
:value="getConfigValue(`SurveyResourceModifiers.${resource.key}`, 1)"
|
|
@input="setConfigValue(`SurveyResourceModifiers.${resource.key}`, Number(($event.target as HTMLInputElement).value))"
|
|
min="0.1"
|
|
max="100"
|
|
step="0.1"
|
|
class="flex-1 accent-oxide-500"
|
|
/>
|
|
<input
|
|
type="number"
|
|
:value="getConfigValue(`SurveyResourceModifiers.${resource.key}`, 1)"
|
|
@input="setConfigValue(`SurveyResourceModifiers.${resource.key}`, Number(($event.target as HTMLInputElement).value))"
|
|
min="0.1"
|
|
max="1000"
|
|
step="0.1"
|
|
class="w-20 bg-neutral-800 border border-neutral-700 rounded-lg px-2 py-1 text-neutral-200 text-sm text-center"
|
|
/>
|
|
<span class="text-xs text-neutral-500 w-4">x</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Create Config Modal -->
|
|
<div v-if="showCreateModal" class="fixed inset-0 bg-black/60 z-50 flex items-center justify-center p-4" @click.self="showCreateModal = false">
|
|
<div class="bg-neutral-900 border border-neutral-800 rounded-xl p-6 w-full max-w-md">
|
|
<h2 class="text-lg font-semibold text-neutral-100 mb-4">New Gather Config</h2>
|
|
<div class="space-y-4">
|
|
<div>
|
|
<label class="block text-sm text-neutral-400 mb-1">Config Name</label>
|
|
<input
|
|
v-model="newConfigName"
|
|
placeholder="e.g. 3x Gather Rates"
|
|
class="w-full bg-neutral-800 border border-neutral-700 rounded-lg px-3 py-2 text-neutral-200 text-sm"
|
|
@keydown.enter="handleCreateConfig"
|
|
/>
|
|
</div>
|
|
<div>
|
|
<label class="block text-sm text-neutral-400 mb-1">Description (optional)</label>
|
|
<textarea
|
|
v-model="newConfigDesc"
|
|
rows="2"
|
|
placeholder="What is this config for?"
|
|
class="w-full bg-neutral-800 border border-neutral-700 rounded-lg px-3 py-2 text-neutral-200 text-sm resize-none"
|
|
/>
|
|
</div>
|
|
<div class="flex justify-end gap-2">
|
|
<button @click="showCreateModal = false" class="px-4 py-2 text-sm text-neutral-400 hover:text-neutral-200">Cancel</button>
|
|
<button
|
|
@click="handleCreateConfig"
|
|
:disabled="!newConfigName.trim()"
|
|
class="px-4 py-2 bg-oxide-500 text-white rounded-lg hover:bg-oxide-600 disabled:opacity-50 text-sm"
|
|
>
|
|
Create
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Import from Server Modal -->
|
|
<div v-if="showImportModal" class="fixed inset-0 bg-black/60 z-50 flex items-center justify-center p-4" @click.self="showImportModal = false">
|
|
<div class="bg-neutral-900 border border-neutral-800 rounded-xl p-6 w-full max-w-md">
|
|
<h2 class="text-lg font-semibold text-neutral-100 mb-4">Import from Server</h2>
|
|
<p class="text-sm text-neutral-400 mb-4">
|
|
Import the current GatherManager config from your live server. This will create a new config profile.
|
|
</p>
|
|
<div class="space-y-4">
|
|
<div>
|
|
<label class="block text-sm text-neutral-400 mb-1">Config Name</label>
|
|
<input
|
|
v-model="importConfigName"
|
|
placeholder="e.g. Imported Server Config"
|
|
class="w-full bg-neutral-800 border border-neutral-700 rounded-lg px-3 py-2 text-neutral-200 text-sm"
|
|
@keydown.enter="handleImport"
|
|
/>
|
|
</div>
|
|
<div class="flex justify-end gap-2">
|
|
<button @click="showImportModal = false" class="px-4 py-2 text-sm text-neutral-400 hover:text-neutral-200">Cancel</button>
|
|
<button
|
|
@click="handleImport"
|
|
:disabled="!importConfigName.trim()"
|
|
class="px-4 py-2 bg-oxide-500 text-white rounded-lg hover:bg-oxide-600 disabled:opacity-50 text-sm"
|
|
>
|
|
Import
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|