- Enhanced Flutter web app management in PowerShell scripts - Added Flutter web server to start-services.ps1 as 4th service - Updated stop-services.ps1 to stop Flutter web server - Improved service orchestration and startup sequence - Implemented server caching for improved resilience - Added ServerCacheService for browser localStorage caching - Server lists persist across service restarts - Automatic fallback to cached data when API unavailable - Action picker categories always visible regardless of server status - Redesigned Action Mappings view with compact table layout - Replaced card-based ListView with DataTable for higher density - Added real-time search across name, input, output, description - Implemented multi-filter support (status: enabled/disabled) - Added column sorting (name, input, output, status, executions) - Batch operations: select all/multiple, batch delete - Reduced row height from ~120px to 56px for better overview - Redesigned Servers Management view with compact table layout - Replaced card-based ListView with DataTable - Added search by alias, host, user - Multi-filter support (type: all/G-Core/GeViScope, status: all/enabled/disabled) - Column sorting (alias, host, user, type, status) - Batch operations: select all/multiple, batch delete - Color-coded type and status badges - Improved action picker dialog for GSC/G-Core actions - GSC and G-Core categories always visible - Server validation with clear error messages - Fixed duplicate checkbox issue in table headers - Debug logging for troubleshooting server parameter issues 🤖 Generated with [Claude Code](https://claude.com/claude-code) Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
770 lines
28 KiB
Dart
770 lines
28 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
|
import 'package:go_router/go_router.dart';
|
|
import '../../../domain/entities/server.dart';
|
|
import '../../blocs/auth/auth_bloc.dart';
|
|
import '../../blocs/auth/auth_event.dart';
|
|
import '../../blocs/auth/auth_state.dart';
|
|
import '../../blocs/server/server_bloc.dart';
|
|
import '../../blocs/server/server_event.dart';
|
|
import '../../blocs/server/server_state.dart';
|
|
import '../../widgets/app_drawer.dart';
|
|
|
|
class ServersManagementScreen extends StatefulWidget {
|
|
const ServersManagementScreen({super.key});
|
|
|
|
@override
|
|
State<ServersManagementScreen> createState() => _ServersManagementScreenState();
|
|
}
|
|
|
|
class _ServersManagementScreenState extends State<ServersManagementScreen> {
|
|
final TextEditingController _searchController = TextEditingController();
|
|
final Set<String> _selectedServers = {};
|
|
bool _selectAll = false;
|
|
|
|
// Filter states
|
|
String _filterType = 'all'; // 'all', 'gcore', 'geviscope'
|
|
bool? _filterEnabled; // null = all, true = enabled only, false = disabled only
|
|
String _searchQuery = '';
|
|
|
|
// Sort states
|
|
int? _sortColumnIndex;
|
|
bool _sortAscending = true;
|
|
|
|
@override
|
|
void dispose() {
|
|
_searchController.dispose();
|
|
super.dispose();
|
|
}
|
|
|
|
List<Server> _getFilteredServers(List<Server> allServers) {
|
|
var filtered = allServers;
|
|
|
|
// Apply type filter
|
|
if (_filterType == 'gcore') {
|
|
filtered = filtered.where((s) => s.type == ServerType.gcore).toList();
|
|
} else if (_filterType == 'geviscope') {
|
|
filtered = filtered.where((s) => s.type == ServerType.geviscope).toList();
|
|
}
|
|
|
|
// Apply enabled/disabled filter
|
|
if (_filterEnabled != null) {
|
|
filtered = filtered.where((s) => s.enabled == _filterEnabled).toList();
|
|
}
|
|
|
|
// Apply search filter
|
|
if (_searchQuery.isNotEmpty) {
|
|
final query = _searchQuery.toLowerCase();
|
|
filtered = filtered.where((s) {
|
|
return s.alias.toLowerCase().contains(query) ||
|
|
s.host.toLowerCase().contains(query) ||
|
|
s.user.toLowerCase().contains(query);
|
|
}).toList();
|
|
}
|
|
|
|
// Apply sorting
|
|
if (_sortColumnIndex != null) {
|
|
filtered.sort((a, b) {
|
|
int comparison = 0;
|
|
switch (_sortColumnIndex) {
|
|
case 0: // Alias
|
|
comparison = a.alias.toLowerCase().compareTo(b.alias.toLowerCase());
|
|
break;
|
|
case 1: // Host
|
|
comparison = a.host.toLowerCase().compareTo(b.host.toLowerCase());
|
|
break;
|
|
case 2: // User
|
|
comparison = a.user.toLowerCase().compareTo(b.user.toLowerCase());
|
|
break;
|
|
case 3: // Type
|
|
comparison = a.type.toString().compareTo(b.type.toString());
|
|
break;
|
|
case 4: // Status
|
|
comparison = a.enabled == b.enabled ? 0 : (a.enabled ? -1 : 1);
|
|
break;
|
|
}
|
|
return _sortAscending ? comparison : -comparison;
|
|
});
|
|
}
|
|
|
|
return filtered;
|
|
}
|
|
|
|
void _toggleSelectAll(List<Server> servers) {
|
|
setState(() {
|
|
if (_selectAll) {
|
|
_selectedServers.clear();
|
|
} else {
|
|
_selectedServers.addAll(servers.map((s) => s.id));
|
|
}
|
|
_selectAll = !_selectAll;
|
|
});
|
|
}
|
|
|
|
void _showBatchDeleteDialog(BuildContext context) {
|
|
if (_selectedServers.isEmpty) return;
|
|
|
|
showDialog(
|
|
context: context,
|
|
builder: (dialogContext) => AlertDialog(
|
|
title: Text('Delete ${_selectedServers.length} Servers'),
|
|
content: Text(
|
|
'Are you sure you want to delete ${_selectedServers.length} selected server${_selectedServers.length != 1 ? 's' : ''}?'
|
|
),
|
|
actions: [
|
|
TextButton(
|
|
onPressed: () => Navigator.of(dialogContext).pop(),
|
|
child: const Text('Cancel'),
|
|
),
|
|
TextButton(
|
|
onPressed: () {
|
|
// Get all selected servers to determine their types
|
|
final bloc = context.read<ServerBloc>();
|
|
final state = bloc.state;
|
|
if (state is ServerLoaded) {
|
|
for (final id in _selectedServers) {
|
|
final server = state.servers.firstWhere((s) => s.id == id);
|
|
bloc.add(DeleteServerEvent(id, server.type));
|
|
}
|
|
}
|
|
setState(() {
|
|
_selectedServers.clear();
|
|
_selectAll = false;
|
|
});
|
|
Navigator.of(dialogContext).pop();
|
|
},
|
|
child: const Text('Delete All', style: TextStyle(color: Colors.red)),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return Scaffold(
|
|
drawer: const AppDrawer(currentRoute: '/servers'),
|
|
appBar: AppBar(
|
|
title: Row(
|
|
children: [
|
|
const Icon(Icons.dns, size: 24),
|
|
const SizedBox(width: 8),
|
|
const Text('Server Management'),
|
|
],
|
|
),
|
|
actions: [
|
|
// Sync button with dirty count badge
|
|
BlocBuilder<ServerBloc, ServerState>(
|
|
builder: (context, state) {
|
|
final dirtyCount = state is ServerLoaded ? state.dirtyCount : 0;
|
|
return Stack(
|
|
children: [
|
|
IconButton(
|
|
icon: const Icon(Icons.sync),
|
|
onPressed: dirtyCount > 0
|
|
? () {
|
|
context.read<ServerBloc>().add(const SyncServersEvent());
|
|
}
|
|
: null,
|
|
tooltip: dirtyCount > 0
|
|
? 'Sync $dirtyCount unsaved change${dirtyCount != 1 ? 's' : ''}'
|
|
: 'No changes to sync',
|
|
),
|
|
if (dirtyCount > 0)
|
|
Positioned(
|
|
right: 4,
|
|
top: 4,
|
|
child: Container(
|
|
padding: const EdgeInsets.all(4),
|
|
decoration: const BoxDecoration(
|
|
color: Colors.red,
|
|
shape: BoxShape.circle,
|
|
),
|
|
child: Text(
|
|
'$dirtyCount',
|
|
style: const TextStyle(
|
|
color: Colors.white,
|
|
fontSize: 10,
|
|
fontWeight: FontWeight.bold,
|
|
),
|
|
),
|
|
),
|
|
),
|
|
],
|
|
);
|
|
},
|
|
),
|
|
// Download/refresh button
|
|
Builder(
|
|
builder: (context) => IconButton(
|
|
icon: const Icon(Icons.cloud_download),
|
|
onPressed: () {
|
|
context.read<ServerBloc>().add(const DownloadServersEvent());
|
|
},
|
|
tooltip: 'Download latest from server',
|
|
),
|
|
),
|
|
const SizedBox(width: 8),
|
|
BlocBuilder<AuthBloc, AuthState>(
|
|
builder: (context, state) {
|
|
if (state is Authenticated) {
|
|
return Padding(
|
|
padding: const EdgeInsets.symmetric(horizontal: 16.0),
|
|
child: Row(
|
|
children: [
|
|
Icon(
|
|
state.user.role == 'Administrator'
|
|
? Icons.admin_panel_settings
|
|
: Icons.person,
|
|
size: 20,
|
|
),
|
|
const SizedBox(width: 8),
|
|
Text(state.user.username),
|
|
const SizedBox(width: 16),
|
|
IconButton(
|
|
icon: const Icon(Icons.logout),
|
|
onPressed: () {
|
|
context.read<AuthBloc>().add(const LogoutRequested());
|
|
},
|
|
tooltip: 'Logout',
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
return const SizedBox.shrink();
|
|
},
|
|
),
|
|
],
|
|
),
|
|
body: Column(
|
|
children: [
|
|
// Toolbar with search, filters, and batch actions
|
|
Container(
|
|
padding: const EdgeInsets.all(16.0),
|
|
color: Colors.grey[100],
|
|
child: Column(
|
|
children: [
|
|
// Search and Add button row
|
|
Row(
|
|
children: [
|
|
Expanded(
|
|
child: TextField(
|
|
controller: _searchController,
|
|
decoration: InputDecoration(
|
|
hintText: 'Search by alias, host, or user...',
|
|
prefixIcon: const Icon(Icons.search),
|
|
suffixIcon: _searchQuery.isNotEmpty
|
|
? IconButton(
|
|
icon: const Icon(Icons.clear),
|
|
onPressed: () {
|
|
setState(() {
|
|
_searchController.clear();
|
|
_searchQuery = '';
|
|
});
|
|
},
|
|
)
|
|
: null,
|
|
border: const OutlineInputBorder(),
|
|
filled: true,
|
|
fillColor: Colors.white,
|
|
contentPadding: const EdgeInsets.symmetric(horizontal: 12, vertical: 8),
|
|
),
|
|
onChanged: (value) {
|
|
setState(() {
|
|
_searchQuery = value;
|
|
});
|
|
},
|
|
),
|
|
),
|
|
const SizedBox(width: 16),
|
|
ElevatedButton.icon(
|
|
onPressed: () {
|
|
_showAddServerDialog(context);
|
|
},
|
|
icon: const Icon(Icons.add),
|
|
label: const Text('Add Server'),
|
|
style: ElevatedButton.styleFrom(
|
|
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 12),
|
|
),
|
|
),
|
|
],
|
|
),
|
|
const SizedBox(height: 12),
|
|
|
|
// Filter chips and batch actions
|
|
Row(
|
|
children: [
|
|
// Type filter chips
|
|
const Text('Type: ', style: TextStyle(fontWeight: FontWeight.w500)),
|
|
const SizedBox(width: 8),
|
|
ChoiceChip(
|
|
label: const Text('All'),
|
|
selected: _filterType == 'all',
|
|
onSelected: (selected) {
|
|
setState(() {
|
|
_filterType = 'all';
|
|
});
|
|
},
|
|
),
|
|
const SizedBox(width: 8),
|
|
ChoiceChip(
|
|
label: const Text('G-Core'),
|
|
selected: _filterType == 'gcore',
|
|
onSelected: (selected) {
|
|
setState(() {
|
|
_filterType = selected ? 'gcore' : 'all';
|
|
});
|
|
},
|
|
),
|
|
const SizedBox(width: 8),
|
|
ChoiceChip(
|
|
label: const Text('GeViScope'),
|
|
selected: _filterType == 'geviscope',
|
|
onSelected: (selected) {
|
|
setState(() {
|
|
_filterType = selected ? 'geviscope' : 'all';
|
|
});
|
|
},
|
|
),
|
|
const SizedBox(width: 24),
|
|
|
|
// Status filter chips
|
|
const Text('Status: ', style: TextStyle(fontWeight: FontWeight.w500)),
|
|
const SizedBox(width: 8),
|
|
ChoiceChip(
|
|
label: const Text('All'),
|
|
selected: _filterEnabled == null,
|
|
onSelected: (selected) {
|
|
setState(() {
|
|
_filterEnabled = null;
|
|
});
|
|
},
|
|
),
|
|
const SizedBox(width: 8),
|
|
ChoiceChip(
|
|
label: const Text('Enabled'),
|
|
selected: _filterEnabled == true,
|
|
onSelected: (selected) {
|
|
setState(() {
|
|
_filterEnabled = selected ? true : null;
|
|
});
|
|
},
|
|
),
|
|
const SizedBox(width: 8),
|
|
ChoiceChip(
|
|
label: const Text('Disabled'),
|
|
selected: _filterEnabled == false,
|
|
onSelected: (selected) {
|
|
setState(() {
|
|
_filterEnabled = selected ? false : null;
|
|
});
|
|
},
|
|
),
|
|
const Spacer(),
|
|
|
|
// Batch actions
|
|
if (_selectedServers.isNotEmpty) ...[
|
|
Text(
|
|
'${_selectedServers.length} selected',
|
|
style: const TextStyle(fontWeight: FontWeight.w500),
|
|
),
|
|
const SizedBox(width: 8),
|
|
TextButton(
|
|
onPressed: () {
|
|
setState(() {
|
|
_selectedServers.clear();
|
|
_selectAll = false;
|
|
});
|
|
},
|
|
child: const Text('Clear'),
|
|
),
|
|
const SizedBox(width: 16),
|
|
OutlinedButton.icon(
|
|
onPressed: () => _showBatchDeleteDialog(context),
|
|
icon: const Icon(Icons.delete, size: 18),
|
|
label: const Text('Delete Selected'),
|
|
style: OutlinedButton.styleFrom(
|
|
foregroundColor: Colors.red,
|
|
),
|
|
),
|
|
],
|
|
],
|
|
),
|
|
],
|
|
),
|
|
),
|
|
|
|
// Table
|
|
Expanded(
|
|
child: BlocConsumer<ServerBloc, ServerState>(
|
|
listener: (context, state) {
|
|
if (state is ServerError) {
|
|
ScaffoldMessenger.of(context).showSnackBar(
|
|
SnackBar(
|
|
content: Text(state.message),
|
|
backgroundColor: Colors.red,
|
|
),
|
|
);
|
|
} else if (state is ServerOperationSuccess) {
|
|
ScaffoldMessenger.of(context).showSnackBar(
|
|
SnackBar(
|
|
content: Text(state.message),
|
|
backgroundColor: Colors.green,
|
|
),
|
|
);
|
|
} else if (state is ServerSyncSuccess) {
|
|
ScaffoldMessenger.of(context).showSnackBar(
|
|
SnackBar(
|
|
content: Text(state.message),
|
|
backgroundColor: Colors.green,
|
|
),
|
|
);
|
|
}
|
|
},
|
|
builder: (context, state) {
|
|
if (state is ServerLoading) {
|
|
return const Center(child: CircularProgressIndicator());
|
|
} else if (state is ServerSyncing) {
|
|
return Center(
|
|
child: Column(
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
children: [
|
|
const CircularProgressIndicator(),
|
|
const SizedBox(height: 16),
|
|
Text(
|
|
state.message,
|
|
style: Theme.of(context).textTheme.titleMedium,
|
|
),
|
|
],
|
|
),
|
|
);
|
|
} else if (state is ServerDownloading) {
|
|
return const Center(
|
|
child: Column(
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
children: [
|
|
CircularProgressIndicator(),
|
|
SizedBox(height: 16),
|
|
Text('Downloading servers...'),
|
|
],
|
|
),
|
|
);
|
|
} else if (state is ServerLoaded) {
|
|
final filteredServers = _getFilteredServers(state.servers);
|
|
|
|
if (filteredServers.isEmpty) {
|
|
return Center(
|
|
child: Column(
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
children: [
|
|
Icon(Icons.dns_outlined, size: 64, color: Colors.grey[400]),
|
|
const SizedBox(height: 16),
|
|
Text(
|
|
_searchQuery.isNotEmpty || _filterEnabled != null || _filterType != 'all'
|
|
? 'No matching servers'
|
|
: 'No servers found',
|
|
style: Theme.of(context).textTheme.titleLarge?.copyWith(
|
|
color: Colors.grey[600],
|
|
),
|
|
),
|
|
const SizedBox(height: 8),
|
|
Text(
|
|
_searchQuery.isNotEmpty || _filterEnabled != null || _filterType != 'all'
|
|
? 'Try adjusting your search or filters'
|
|
: 'Add a server to get started',
|
|
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
|
|
color: Colors.grey[500],
|
|
),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
|
|
return _buildCompactTable(context, filteredServers);
|
|
} else if (state is ServerError) {
|
|
return Center(
|
|
child: Column(
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
children: [
|
|
Icon(Icons.error_outline, size: 64, color: Colors.red[300]),
|
|
const SizedBox(height: 16),
|
|
Text(
|
|
'Error loading servers',
|
|
style: Theme.of(context).textTheme.titleLarge,
|
|
),
|
|
const SizedBox(height: 8),
|
|
Text(
|
|
state.message,
|
|
style: Theme.of(context).textTheme.bodyMedium?.copyWith(
|
|
color: Colors.grey[600],
|
|
),
|
|
textAlign: TextAlign.center,
|
|
),
|
|
const SizedBox(height: 24),
|
|
ElevatedButton.icon(
|
|
onPressed: () {
|
|
context.read<ServerBloc>().add(const LoadServers());
|
|
},
|
|
icon: const Icon(Icons.refresh),
|
|
label: const Text('Retry'),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
|
|
return const Center(child: CircularProgressIndicator());
|
|
},
|
|
),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
|
|
Widget _buildCompactTable(BuildContext context, List<Server> servers) {
|
|
return SingleChildScrollView(
|
|
child: Padding(
|
|
padding: const EdgeInsets.all(16.0),
|
|
child: DataTable(
|
|
columnSpacing: 24,
|
|
horizontalMargin: 12,
|
|
headingRowHeight: 48,
|
|
dataRowHeight: 56,
|
|
showCheckboxColumn: true,
|
|
sortColumnIndex: _sortColumnIndex,
|
|
sortAscending: _sortAscending,
|
|
columns: [
|
|
DataColumn(
|
|
label: const Text('Alias', style: TextStyle(fontWeight: FontWeight.bold)),
|
|
onSort: (columnIndex, ascending) {
|
|
setState(() {
|
|
_sortColumnIndex = columnIndex;
|
|
_sortAscending = ascending;
|
|
});
|
|
},
|
|
),
|
|
DataColumn(
|
|
label: const Text('Host', style: TextStyle(fontWeight: FontWeight.bold)),
|
|
onSort: (columnIndex, ascending) {
|
|
setState(() {
|
|
_sortColumnIndex = columnIndex;
|
|
_sortAscending = ascending;
|
|
});
|
|
},
|
|
),
|
|
DataColumn(
|
|
label: const Text('User', style: TextStyle(fontWeight: FontWeight.bold)),
|
|
onSort: (columnIndex, ascending) {
|
|
setState(() {
|
|
_sortColumnIndex = columnIndex;
|
|
_sortAscending = ascending;
|
|
});
|
|
},
|
|
),
|
|
DataColumn(
|
|
label: const Text('Type', style: TextStyle(fontWeight: FontWeight.bold)),
|
|
onSort: (columnIndex, ascending) {
|
|
setState(() {
|
|
_sortColumnIndex = columnIndex;
|
|
_sortAscending = ascending;
|
|
});
|
|
},
|
|
),
|
|
DataColumn(
|
|
label: const Text('Status', style: TextStyle(fontWeight: FontWeight.bold)),
|
|
onSort: (columnIndex, ascending) {
|
|
setState(() {
|
|
_sortColumnIndex = columnIndex;
|
|
_sortAscending = ascending;
|
|
});
|
|
},
|
|
),
|
|
const DataColumn(
|
|
label: Text('Actions', style: TextStyle(fontWeight: FontWeight.bold)),
|
|
),
|
|
],
|
|
rows: servers.map((server) {
|
|
final isSelected = _selectedServers.contains(server.id);
|
|
|
|
return DataRow(
|
|
selected: isSelected,
|
|
onSelectChanged: (selected) {
|
|
setState(() {
|
|
if (selected == true) {
|
|
_selectedServers.add(server.id);
|
|
} else {
|
|
_selectedServers.remove(server.id);
|
|
}
|
|
_selectAll = _selectedServers.length == servers.length;
|
|
});
|
|
},
|
|
cells: [
|
|
// Alias
|
|
DataCell(
|
|
Text(
|
|
server.alias,
|
|
style: const TextStyle(fontWeight: FontWeight.w500),
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis,
|
|
),
|
|
),
|
|
|
|
// Host
|
|
DataCell(
|
|
Text(
|
|
server.host,
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis,
|
|
),
|
|
),
|
|
|
|
// User
|
|
DataCell(
|
|
Text(
|
|
server.user,
|
|
maxLines: 1,
|
|
overflow: TextOverflow.ellipsis,
|
|
),
|
|
),
|
|
|
|
// Type
|
|
DataCell(
|
|
Container(
|
|
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 4),
|
|
decoration: BoxDecoration(
|
|
color: server.type == ServerType.gcore
|
|
? Colors.green.withOpacity(0.1)
|
|
: Colors.purple.withOpacity(0.1),
|
|
border: Border.all(
|
|
color: server.type == ServerType.gcore ? Colors.green : Colors.purple,
|
|
),
|
|
borderRadius: BorderRadius.circular(12),
|
|
),
|
|
child: Text(
|
|
server.type == ServerType.gcore ? 'G-Core' : 'GeViScope',
|
|
style: TextStyle(
|
|
color: server.type == ServerType.gcore ? Colors.green[700] : Colors.purple[700],
|
|
fontSize: 12,
|
|
fontWeight: FontWeight.w500,
|
|
),
|
|
),
|
|
),
|
|
),
|
|
|
|
// Status
|
|
DataCell(
|
|
Container(
|
|
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 4),
|
|
decoration: BoxDecoration(
|
|
color: server.enabled ? Colors.green.withOpacity(0.1) : Colors.grey.withOpacity(0.1),
|
|
border: Border.all(
|
|
color: server.enabled ? Colors.green : Colors.grey,
|
|
),
|
|
borderRadius: BorderRadius.circular(12),
|
|
),
|
|
child: Text(
|
|
server.enabled ? 'Enabled' : 'Disabled',
|
|
style: TextStyle(
|
|
color: server.enabled ? Colors.green[700] : Colors.grey[700],
|
|
fontSize: 12,
|
|
fontWeight: FontWeight.w500,
|
|
),
|
|
),
|
|
),
|
|
),
|
|
|
|
// Actions
|
|
DataCell(
|
|
Row(
|
|
mainAxisSize: MainAxisSize.min,
|
|
children: [
|
|
IconButton(
|
|
icon: const Icon(Icons.edit, size: 20),
|
|
onPressed: () {
|
|
context.push('/servers/edit/${server.id}', extra: server);
|
|
},
|
|
tooltip: 'Edit',
|
|
padding: EdgeInsets.zero,
|
|
constraints: const BoxConstraints(),
|
|
),
|
|
const SizedBox(width: 8),
|
|
IconButton(
|
|
icon: const Icon(Icons.delete, size: 20, color: Colors.red),
|
|
onPressed: () {
|
|
_showDeleteConfirmation(context, server);
|
|
},
|
|
tooltip: 'Delete',
|
|
padding: EdgeInsets.zero,
|
|
constraints: const BoxConstraints(),
|
|
),
|
|
],
|
|
),
|
|
),
|
|
],
|
|
);
|
|
}).toList(),
|
|
),
|
|
),
|
|
);
|
|
}
|
|
|
|
void _showDeleteConfirmation(BuildContext context, Server server) {
|
|
showDialog(
|
|
context: context,
|
|
builder: (dialogContext) => AlertDialog(
|
|
title: const Text('Delete Server'),
|
|
content: Text('Are you sure you want to delete "${server.alias}"?'),
|
|
actions: [
|
|
TextButton(
|
|
onPressed: () => Navigator.of(dialogContext).pop(),
|
|
child: const Text('Cancel'),
|
|
),
|
|
TextButton(
|
|
onPressed: () {
|
|
context.read<ServerBloc>().add(DeleteServerEvent(server.id, server.type));
|
|
setState(() {
|
|
_selectedServers.remove(server.id);
|
|
});
|
|
Navigator.of(dialogContext).pop();
|
|
},
|
|
child: const Text('Delete', style: TextStyle(color: Colors.red)),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
|
|
void _showAddServerDialog(BuildContext context) {
|
|
showDialog(
|
|
context: context,
|
|
builder: (dialogContext) => AlertDialog(
|
|
title: const Text('Add Server'),
|
|
content: const Text('Choose the server type:'),
|
|
actions: [
|
|
TextButton(
|
|
onPressed: () {
|
|
Navigator.of(dialogContext).pop();
|
|
context.push('/servers/create?type=gcore');
|
|
},
|
|
child: const Text('G-Core Server'),
|
|
),
|
|
TextButton(
|
|
onPressed: () {
|
|
Navigator.of(dialogContext).pop();
|
|
context.push('/servers/create?type=geviscope');
|
|
},
|
|
child: const Text('GeViScope Server'),
|
|
),
|
|
TextButton(
|
|
onPressed: () => Navigator.of(dialogContext).pop(),
|
|
child: const Text('Cancel'),
|
|
),
|
|
],
|
|
),
|
|
);
|
|
}
|
|
}
|