Moves agent editing to in-place ajax forms

main
Yasen Pramatarov 2025-01-20 21:42:22 +02:00
parent 5321942da8
commit d45ba62805
2 changed files with 206 additions and 50 deletions

View File

@ -154,20 +154,45 @@ if ($_SERVER['REQUEST_METHOD'] == 'POST') {
} }
// an update to an existing agent // an update to an existing agent
} elseif (isset($_POST['agent'])) { } elseif (isset($_POST['item']) && $_POST['item'] === 'agent') {
$agent_id = $_POST['agent'];
$platform_id = $_POST['platform'];
$updatedAgent = [ $updatedAgent = [
'id' => $agent, 'id' => $agent_id,
'agent_type_id' => $type, 'agent_type_id' => $_POST['agent_type_id'],
'url' => $url, 'url' => $_POST['url'],
'secret_key' => $secret_key, 'secret_key' => $_POST['secret_key'],
'check_period' => $check_period, 'check_period' => $_POST['check_period']
]; ];
$result = $agentObject->editAgent($platform_id, $updatedAgent); $result = $agentObject->editAgent($platform_id, $updatedAgent);
// Check if it's an AJAX request
if (!empty($_SERVER['HTTP_X_REQUESTED_WITH']) &&
strtolower($_SERVER['HTTP_X_REQUESTED_WITH']) == 'xmlhttprequest') {
// Clear any output buffers to ensure clean JSON
while (ob_get_level()) ob_end_clean();
header('Content-Type: application/json');
if ($result === true) { if ($result === true) {
$_SESSION['notice'] = "Agent id \"{$_REQUEST['agent']}\" edited."; echo json_encode(['success' => true]);
} else {
echo json_encode([
'success' => false,
'message' => "Editing the agent failed. Error: $result"
]);
}
exit();
} else {
// Regular form submission
if ($result === true) {
$_SESSION['notice'] = "Agent edited.";
} else { } else {
$_SESSION['error'] = "Editing the agent failed. Error: $result"; $_SESSION['error'] = "Editing the agent failed. Error: $result";
} }
header("Location: $app_root?page=config&item=$item");
exit();
}
// an update to an existing platform // an update to an existing platform
} else { } else {

View File

@ -138,7 +138,7 @@
<h6 class="mb-0">Host id #<?= htmlspecialchars($host['id']) ?> in platform "<?= htmlspecialchars($platform['name']) ?>"</h6> <h6 class="mb-0">Host id #<?= htmlspecialchars($host['id']) ?> in platform "<?= htmlspecialchars($platform['name']) ?>"</h6>
</div> </div>
<div class="ps-4"> <div class="ps-4">
<span class="view-mode"> <span class="host-view-mode">
<div class="row g-2"> <div class="row g-2">
<div class="col-md-6"> <div class="col-md-6">
<div class="small text-muted mb-1">Host description</div> <div class="small text-muted mb-1">Host description</div>
@ -150,7 +150,7 @@
</div> </div>
</div> </div>
</span> </span>
<div class="edit-mode" style="display: none;"> <div class="host-edit-mode" style="display: none;">
<div class="row g-2"> <div class="row g-2">
<div class="col-md-6"> <div class="col-md-6">
<label class="form-label small text-muted">Host description</label> <label class="form-label small text-muted">Host description</label>
@ -170,17 +170,17 @@
</div> </div>
<div class="btn-group host-actions ms-3" data-host-id="<?= htmlspecialchars($host['id']) ?>" <div class="btn-group host-actions ms-3" data-host-id="<?= htmlspecialchars($host['id']) ?>"
data-platform-id="<?= htmlspecialchars($platform['id']) ?>"> data-platform-id="<?= htmlspecialchars($platform['id']) ?>">
<button type="button" class="btn btn-outline-primary btn-sm edit-host"> <button type="button" class="btn btn-outline-primary btn-sm edit-host host-view-mode">
<i class="fas fa-edit me-1"></i>Edit host <i class="fas fa-edit me-1"></i>Edit host
</button> </button>
<button type="button" class="btn btn-outline-primary btn-sm save-host" style="display: none;"> <button type="button" class="btn btn-outline-primary btn-sm save-host host-edit-mode" style="display: none;">
<i class="fas fa-save me-1"></i>Save <i class="fas fa-save me-1"></i>Save
</button> </button>
<button type="button" class="btn btn-outline-secondary btn-sm cancel-host-edit" style="display: none;"> <button type="button" class="btn btn-outline-secondary btn-sm cancel-host-edit host-edit-mode" style="display: none;">
<i class="fas fa-times me-1"></i>Cancel <i class="fas fa-times me-1"></i>Cancel
</button> </button>
<a href="<?= htmlspecialchars($app_root) ?>?page=config&item=host&platform=<?= htmlspecialchars($platform['id']) ?>&host=<?= htmlspecialchars($host['id']) ?>&action=delete" <a href="<?= htmlspecialchars($app_root) ?>?page=config&item=host&platform=<?= htmlspecialchars($platform['id']) ?>&host=<?= htmlspecialchars($host['id']) ?>&action=delete"
class="btn btn-outline-danger btn-sm"> class="btn btn-outline-danger btn-sm host-view-mode">
<i class="fas fa-trash me-1"></i>Delete host <i class="fas fa-trash me-1"></i>Delete host
</a> </a>
</div> </div>
@ -219,25 +219,66 @@
<td> <td>
<div class="d-flex align-items-center"> <div class="d-flex align-items-center">
<i class="fas fa-robot me-2 text-secondary"></i> <i class="fas fa-robot me-2 text-secondary"></i>
<span class="agent-view-mode">
<?= htmlspecialchars($agent['agent_description']) ?> <?= htmlspecialchars($agent['agent_description']) ?>
</span>
<div class="agent-edit-mode" style="display: none;">
<select class="form-select form-select-sm" name="agent_type_id" required>
<?php foreach ($agentObject->getAgentTypes() as $type): ?>
<option value="<?= htmlspecialchars($type['id']) ?>"
data-endpoint="<?= htmlspecialchars($type['endpoint']) ?>"
<?= $type['id'] === $agent['agent_type_id'] ? 'selected' : '' ?>>
<?= htmlspecialchars($type['description']) ?>
</option>
<?php endforeach; ?>
</select>
</div>
</div>
</td>
<td class="text-break">
<span class="agent-view-mode">
<?= htmlspecialchars($agent['url'].$agent['agent_endpoint']) ?>
</span>
<div class="agent-edit-mode" style="display: none;">
<label class="form-label small text-muted">URL</label>
<input type="text" class="form-control form-control-sm text-break mb-2" name="url"
value="<?= htmlspecialchars($agent['url']) ?>"
placeholder="e.g., http://localhost:8080" required>
<label class="form-label small text-muted">Secret Key</label>
<input type="text" class="form-control form-control-sm text-break" name="secret_key"
value="<?= htmlspecialchars($agent['secret_key']) ?>"
placeholder="Secret key for authentication" required>
</div> </div>
</td> </td>
<td class="text-break"><?= htmlspecialchars($agent['url'].$agent['agent_endpoint']) ?></td>
<td> <td>
<span class="agent-view-mode">
<?php if (isset($agent['check_period']) && $agent['check_period'] !== 0): ?> <?php if (isset($agent['check_period']) && $agent['check_period'] !== 0): ?>
<?= htmlspecialchars($agent['check_period']) ?> <?= ($agent['check_period'] == 1 ? 'minute' : 'minutes') ?> <?= htmlspecialchars($agent['check_period']) ?> <?= ($agent['check_period'] == 1 ? 'minute' : 'minutes') ?>
<?php else: ?> <?php else: ?>
<span class="text-muted">-</span> <span class="text-muted">-</span>
<?php endif; ?> <?php endif; ?>
</span>
<div class="agent-edit-mode" style="display: none;">
<input type="number" class="form-control form-control-sm" name="check_period"
value="<?= htmlspecialchars($agent['check_period']) ?>"
min="0" placeholder="Check interval in minutes">
</div>
</td> </td>
<td class="text-end"> <td class="text-end">
<div class="btn-group"> <div class="btn-group agent-actions" data-agent-id="<?= htmlspecialchars($agent['id']) ?>"
<a href="<?= htmlspecialchars($app_root) ?>?page=config&item=agent&action=edit&platform=<?= htmlspecialchars($platform['id']) ?>&host=<?= htmlspecialchars($host['id']) ?>&agent=<?= htmlspecialchars($agent['id']) ?>" data-platform-id="<?= htmlspecialchars($platform['id']) ?>"
class="btn btn-sm btn-outline-primary"> data-host-id="<?= htmlspecialchars($host['id']) ?>">
<button type="button" class="btn btn-outline-primary btn-sm edit-agent agent-view-mode">
<i class="fas fa-edit me-1"></i>Edit <i class="fas fa-edit me-1"></i>Edit
</a> </button>
<button type="button" class="btn btn-outline-primary btn-sm save-agent agent-edit-mode" style="display: none;">
<i class="fas fa-save me-1"></i>Save
</button>
<button type="button" class="btn btn-outline-secondary btn-sm cancel-agent-edit agent-edit-mode" style="display: none;">
<i class="fas fa-times me-1"></i>Cancel
</button>
<a href="<?= htmlspecialchars($app_root) ?>?page=config&item=agent&action=delete&platform=<?= htmlspecialchars($platform['id']) ?>&host=<?= htmlspecialchars($host['id']) ?>&agent=<?= htmlspecialchars($agent['id']) ?>" <a href="<?= htmlspecialchars($app_root) ?>?page=config&item=agent&action=delete&platform=<?= htmlspecialchars($platform['id']) ?>&host=<?= htmlspecialchars($host['id']) ?>&agent=<?= htmlspecialchars($agent['id']) ?>"
class="btn btn-sm btn-outline-danger"> class="btn btn-outline-danger btn-sm agent-view-mode">
<i class="fas fa-trash me-1"></i>Delete <i class="fas fa-trash me-1"></i>Delete
</a> </a>
</div> </div>
@ -438,16 +479,15 @@ $(function() {
// Host editing functionality // Host editing functionality
$('.edit-host').click(function() { $('.edit-host').click(function() {
const hostActions = $(this).closest('.host-actions'); const hostActions = $(this).closest('.host-actions');
const hostId = hostActions.data('host-id');
const card = hostActions.closest('.card'); const card = hostActions.closest('.card');
// Show edit mode // Show edit mode
card.find('.view-mode').hide(); card.find('.host-view-mode:not(.btn)').hide();
card.find('.edit-mode').show(); card.find('.host-edit-mode').show();
// Toggle buttons // Toggle buttons
hostActions.find('.edit-host').hide(); hostActions.find('.host-view-mode').hide();
hostActions.find('.save-host, .cancel-host-edit').show(); hostActions.find('.host-edit-mode').show();
}); });
// Cancel host edit // Cancel host edit
@ -456,12 +496,12 @@ $(function() {
const card = hostActions.closest('.card'); const card = hostActions.closest('.card');
// Show view mode // Show view mode
card.find('.view-mode').show(); card.find('.host-view-mode:not(.btn)').show();
card.find('.edit-mode').hide(); card.find('.host-edit-mode').hide();
// Toggle buttons // Toggle buttons
hostActions.find('.edit-host').show(); hostActions.find('.host-view-mode').show();
hostActions.find('.save-host, .cancel-host-edit').hide(); hostActions.find('.host-edit-mode').hide();
}); });
// Save host // Save host
@ -473,11 +513,11 @@ $(function() {
// Collect form data // Collect form data
const formData = new FormData(); const formData = new FormData();
formData.append('item', 'host');
formData.append('host', hostId); formData.append('host', hostId);
formData.append('platform', platformId); formData.append('platform', platformId);
formData.append('item', 'host');
card.find('.edit-mode input').each(function() { card.find('.host-edit-mode input').each(function() {
formData.append($(this).attr('name'), $(this).val()); formData.append($(this).attr('name'), $(this).val());
}); });
@ -505,9 +545,10 @@ $(function() {
.then(data => { .then(data => {
if (data.success) { if (data.success) {
// Update view mode with new values // Update view mode with new values
const name = card.find('input[name="name"]').val() || 'Unnamed Host'; const name = card.find('input[name="name"]').val() || '(no description)';
const address = card.find('input[name="address"]').val(); const address = card.find('input[name="address"]').val();
card.find('.view-mode').html( const viewContent = card.find('.host-view-mode:not(.btn)').first();
viewContent.html(
`<div class="row g-2"> `<div class="row g-2">
<div class="col-md-6"> <div class="col-md-6">
<div class="small text-muted mb-1">Host description</div> <div class="small text-muted mb-1">Host description</div>
@ -521,12 +562,12 @@ $(function() {
); );
// Switch back to view mode // Switch back to view mode
card.find('.view-mode').show(); card.find('.host-view-mode:not(.btn)').show();
card.find('.edit-mode').hide(); card.find('.host-edit-mode').hide();
// Toggle buttons // Toggle buttons
hostActions.find('.edit-host').show(); hostActions.find('.host-view-mode').show();
hostActions.find('.save-host, .cancel-host-edit').hide(); hostActions.find('.host-edit-mode').hide();
} else { } else {
alert('Error saving host: ' + (data.message || 'Unknown error')); alert('Error saving host: ' + (data.message || 'Unknown error'));
} }
@ -534,9 +575,10 @@ $(function() {
.catch(error => { .catch(error => {
console.error('Error:', error); console.error('Error:', error);
// Since we know the save might work despite JSON errors, update UI anyway // Since we know the save might work despite JSON errors, update UI anyway
const name = card.find('input[name="name"]').val() || 'Unnamed Host'; const name = card.find('input[name="name"]').val() || '(no description)';
const address = card.find('input[name="address"]').val(); const address = card.find('input[name="address"]').val();
card.find('.view-mode').html( const viewContent = card.find('.host-view-mode:not(.btn)').first();
viewContent.html(
`<div class="row g-2"> `<div class="row g-2">
<div class="col-md-6"> <div class="col-md-6">
<div class="small text-muted mb-1">Host description</div> <div class="small text-muted mb-1">Host description</div>
@ -550,12 +592,101 @@ $(function() {
); );
// Switch back to view mode // Switch back to view mode
card.find('.view-mode').show(); card.find('.host-view-mode:not(.btn)').show();
card.find('.edit-mode').hide(); card.find('.host-edit-mode').hide();
// Toggle buttons // Toggle buttons
hostActions.find('.edit-host').show(); hostActions.find('.host-view-mode').show();
hostActions.find('.save-host, .cancel-host-edit').hide(); hostActions.find('.host-edit-mode').hide();
});
});
// Agent editing functionality
$('.edit-agent').click(function() {
const agentActions = $(this).closest('.agent-actions');
const row = agentActions.closest('tr');
// Show edit mode
row.find('.agent-view-mode').hide();
row.find('.agent-edit-mode').show();
});
// Cancel agent edit
$('.cancel-agent-edit').click(function() {
const agentActions = $(this).closest('.agent-actions');
const row = agentActions.closest('tr');
// Show view mode
row.find('.agent-view-mode').show();
row.find('.agent-edit-mode').hide();
});
// Save agent
$('.save-agent').click(function() {
const agentActions = $(this).closest('.agent-actions');
const agentId = agentActions.data('agent-id');
const platformId = agentActions.data('platform-id');
const hostId = agentActions.data('host-id');
const row = agentActions.closest('tr');
// Collect form data
const formData = new FormData();
formData.append('item', 'agent');
formData.append('agent', agentId);
formData.append('platform', platformId);
formData.append('host', hostId);
row.find('.agent-edit-mode input, .agent-edit-mode select').each(function() {
formData.append($(this).attr('name'), $(this).val());
});
// Save via AJAX
fetch('<?= htmlspecialchars($app_root) ?>?page=config&item=agent&action=save', {
method: 'POST',
headers: {
'X-Requested-With': 'XMLHttpRequest'
},
body: formData
})
.then(response => {
if (!response.ok) {
throw new Error('Network response was not ok');
}
return response.text().then(text => {
try {
return JSON.parse(text);
} catch (e) {
console.log('Response text:', text);
return { success: true };
}
});
})
.then(data => {
if (data.success) {
// Update view mode with new values
const type = row.find('select[name="agent_type_id"] option:selected').text();
const url = row.find('input[name="url"]').val();
const endpoint = row.find('select[name="agent_type_id"] option:selected').data('endpoint');
const checkPeriod = row.find('input[name="check_period"]').val();
row.find('td:first-child .agent-view-mode').text(type);
row.find('td:nth-child(2) .agent-view-mode').text(url + endpoint);
row.find('td:nth-child(3) .agent-view-mode').text(
checkPeriod > 0 ?
`${checkPeriod} ${checkPeriod == 1 ? 'minute' : 'minutes'}` :
'-'
);
// Switch back to view mode
row.find('.agent-view-mode').show();
row.find('.agent-edit-mode').hide();
} else {
alert('Error saving agent: ' + (data.message || 'Unknown error'));
}
})
.catch(error => {
console.error('Error:', error);
alert('Error saving agent. Please try again.');
}); });
}); });