215 lines
7.5 KiB
HTML
215 lines
7.5 KiB
HTML
<!DOCTYPE html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="UTF-8">
|
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
<title>Test Bulk Status Update</title>
|
|
<style>
|
|
body {
|
|
font-family: Arial, sans-serif;
|
|
max-width: 1200px;
|
|
margin: 20px auto;
|
|
padding: 20px;
|
|
}
|
|
.test-section {
|
|
margin: 20px 0;
|
|
padding: 15px;
|
|
border: 1px solid #ddd;
|
|
border-radius: 5px;
|
|
}
|
|
.success {
|
|
color: green;
|
|
}
|
|
.error {
|
|
color: red;
|
|
}
|
|
button {
|
|
padding: 10px 20px;
|
|
margin: 5px;
|
|
cursor: pointer;
|
|
}
|
|
.task-item {
|
|
padding: 10px;
|
|
margin: 5px 0;
|
|
border: 1px solid #eee;
|
|
border-radius: 3px;
|
|
}
|
|
.task-item.selected {
|
|
background-color: #e3f2fd;
|
|
border-color: #2196f3;
|
|
}
|
|
</style>
|
|
</head>
|
|
<body>
|
|
<h1>Bulk Status Update Test</h1>
|
|
|
|
<div class="test-section">
|
|
<h2>1. Login</h2>
|
|
<button onclick="login()">Login as Admin</button>
|
|
<div id="login-result"></div>
|
|
</div>
|
|
|
|
<div class="test-section">
|
|
<h2>2. Fetch Tasks</h2>
|
|
<button onclick="fetchTasks()">Fetch Tasks</button>
|
|
<div id="tasks-result"></div>
|
|
</div>
|
|
|
|
<div class="test-section">
|
|
<h2>3. Select Tasks and Update Status</h2>
|
|
<p>Click on tasks to select them (they will highlight), then click a status button to update all selected tasks.</p>
|
|
<div id="task-list"></div>
|
|
<div style="margin-top: 15px;">
|
|
<button onclick="bulkUpdateStatus('not_started')">Set to Not Started</button>
|
|
<button onclick="bulkUpdateStatus('in_progress')">Set to In Progress</button>
|
|
<button onclick="bulkUpdateStatus('submitted')">Set to Submitted</button>
|
|
<button onclick="bulkUpdateStatus('approved')">Set to Approved</button>
|
|
<button onclick="bulkUpdateStatus('retake')">Set to Retake</button>
|
|
</div>
|
|
<div id="update-result"></div>
|
|
</div>
|
|
|
|
<script>
|
|
const BASE_URL = 'http://localhost:8000';
|
|
let token = null;
|
|
let tasks = [];
|
|
let selectedTaskIds = new Set();
|
|
|
|
async function login() {
|
|
const resultDiv = document.getElementById('login-result');
|
|
resultDiv.innerHTML = 'Logging in...';
|
|
|
|
try {
|
|
const response = await fetch(`${BASE_URL}/auth/login`, {
|
|
method: 'POST',
|
|
headers: {
|
|
'Content-Type': 'application/json',
|
|
},
|
|
body: JSON.stringify({
|
|
email: 'admin@vfx.com',
|
|
password: 'admin123'
|
|
})
|
|
});
|
|
|
|
if (response.ok) {
|
|
const data = await response.json();
|
|
token = data.access_token;
|
|
resultDiv.innerHTML = '<span class="success">✓ Login successful!</span>';
|
|
} else {
|
|
const error = await response.text();
|
|
resultDiv.innerHTML = `<span class="error">✗ Login failed: ${error}</span>`;
|
|
}
|
|
} catch (error) {
|
|
resultDiv.innerHTML = `<span class="error">✗ Error: ${error.message}</span>`;
|
|
}
|
|
}
|
|
|
|
async function fetchTasks() {
|
|
const resultDiv = document.getElementById('tasks-result');
|
|
|
|
if (!token) {
|
|
resultDiv.innerHTML = '<span class="error">Please login first</span>';
|
|
return;
|
|
}
|
|
|
|
resultDiv.innerHTML = 'Fetching tasks...';
|
|
|
|
try {
|
|
const response = await fetch(`${BASE_URL}/tasks/?project_id=1`, {
|
|
headers: {
|
|
'Authorization': `Bearer ${token}`
|
|
}
|
|
});
|
|
|
|
if (response.ok) {
|
|
tasks = await response.json();
|
|
resultDiv.innerHTML = `<span class="success">✓ Fetched ${tasks.length} tasks</span>`;
|
|
displayTasks();
|
|
} else {
|
|
const error = await response.text();
|
|
resultDiv.innerHTML = `<span class="error">✗ Failed to fetch tasks: ${error}</span>`;
|
|
}
|
|
} catch (error) {
|
|
resultDiv.innerHTML = `<span class="error">✗ Error: ${error.message}</span>`;
|
|
}
|
|
}
|
|
|
|
function displayTasks() {
|
|
const taskListDiv = document.getElementById('task-list');
|
|
|
|
if (tasks.length === 0) {
|
|
taskListDiv.innerHTML = '<p>No tasks found</p>';
|
|
return;
|
|
}
|
|
|
|
taskListDiv.innerHTML = tasks.slice(0, 10).map(task => `
|
|
<div class="task-item ${selectedTaskIds.has(task.id) ? 'selected' : ''}"
|
|
onclick="toggleTaskSelection(${task.id})">
|
|
<strong>Task ${task.id}:</strong> ${task.name}
|
|
<span style="color: #666;">(Status: ${task.status})</span>
|
|
</div>
|
|
`).join('');
|
|
}
|
|
|
|
function toggleTaskSelection(taskId) {
|
|
if (selectedTaskIds.has(taskId)) {
|
|
selectedTaskIds.delete(taskId);
|
|
} else {
|
|
selectedTaskIds.add(taskId);
|
|
}
|
|
displayTasks();
|
|
}
|
|
|
|
async function bulkUpdateStatus(status) {
|
|
const resultDiv = document.getElementById('update-result');
|
|
|
|
if (!token) {
|
|
resultDiv.innerHTML = '<span class="error">Please login first</span>';
|
|
return;
|
|
}
|
|
|
|
if (selectedTaskIds.size === 0) {
|
|
resultDiv.innerHTML = '<span class="error">Please select at least one task</span>';
|
|
return;
|
|
}
|
|
|
|
const taskIds = Array.from(selectedTaskIds);
|
|
resultDiv.innerHTML = `Updating ${taskIds.length} tasks to status "${status}"...`;
|
|
|
|
try {
|
|
const response = await fetch(`${BASE_URL}/tasks/bulk/status`, {
|
|
method: 'PUT',
|
|
headers: {
|
|
'Authorization': `Bearer ${token}`,
|
|
'Content-Type': 'application/json'
|
|
},
|
|
body: JSON.stringify({
|
|
task_ids: taskIds,
|
|
status: status
|
|
})
|
|
});
|
|
|
|
if (response.ok) {
|
|
const result = await response.json();
|
|
resultDiv.innerHTML = `
|
|
<span class="success">
|
|
✓ Successfully updated ${result.success_count} tasks!
|
|
${result.failed_count > 0 ? `<br>Failed: ${result.failed_count}` : ''}
|
|
</span>
|
|
`;
|
|
|
|
// Clear selection and refresh tasks
|
|
selectedTaskIds.clear();
|
|
await fetchTasks();
|
|
} else {
|
|
const error = await response.text();
|
|
resultDiv.innerHTML = `<span class="error">✗ Failed to update tasks: ${error}</span>`;
|
|
}
|
|
} catch (error) {
|
|
resultDiv.innerHTML = `<span class="error">✗ Error: ${error.message}</span>`;
|
|
}
|
|
}
|
|
</script>
|
|
</body>
|
|
</html>
|