Initial commit - Fire alarm management application
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
This commit is contained in:
436
app/templates/job_detail.html
Normal file
436
app/templates/job_detail.html
Normal file
@@ -0,0 +1,436 @@
|
||||
{% extends "base.html" %}
|
||||
|
||||
{% block title %}{{ job.job_name }} - Fire Alarm Management{% endblock %}
|
||||
|
||||
{% block content %}
|
||||
<div class="row mb-4">
|
||||
<div class="col">
|
||||
<nav aria-label="breadcrumb">
|
||||
<ol class="breadcrumb">
|
||||
<li class="breadcrumb-item"><a href="{{ url_for('main.jobs_list') }}">Jobs</a></li>
|
||||
<li class="breadcrumb-item active">{{ job.job_name }}</li>
|
||||
</ol>
|
||||
</nav>
|
||||
<h1>
|
||||
<i class="bi bi-briefcase"></i> {{ job.job_name }}
|
||||
<small class="text-muted">#{{ job.job_number }}</small>
|
||||
</h1>
|
||||
</div>
|
||||
<div class="col-auto">
|
||||
<a href="{{ url_for('main.job_edit', job_id=job.id) }}" class="btn btn-primary">
|
||||
<i class="bi bi-pencil"></i> Edit
|
||||
</a>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Progress Bar -->
|
||||
<div class="row mb-4">
|
||||
<div class="col">
|
||||
<div class="card">
|
||||
<div class="card-body">
|
||||
<h5>Overall Progress: {{ (job.percent_complete * 100)|round(0) }}%</h5>
|
||||
<div class="progress" style="height: 30px;">
|
||||
<div class="progress-bar {{ 'bg-success' if job.percent_complete >= 1 else ('bg-info' if job.percent_complete > 0.5 else ('bg-warning' if job.percent_complete > 0 else 'bg-secondary')) }}"
|
||||
role="progressbar" style="width: {{ job.percent_complete * 100 }}%;">
|
||||
{{ (job.percent_complete * 100)|round(0) }}%
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="row">
|
||||
<!-- Basic Info -->
|
||||
<div class="col-md-6">
|
||||
<div class="card mb-4">
|
||||
<div class="card-header">Basic Information</div>
|
||||
<div class="card-body">
|
||||
<table class="table table-borderless">
|
||||
<tr><th>Location:</th><td>{{ job.location or '-' }}</td></tr>
|
||||
<tr><th>Est. Starting Qtr:</th><td>{{ job.est_starting_qtr or '-' }}</td></tr>
|
||||
<tr><th>Number of Units:</th><td>{{ job.number_of_units or '-' }}</td></tr>
|
||||
<tr><th>Sep Club House:</th><td>{{ job.sep_club_house or '-' }}</td></tr>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Budget -->
|
||||
<div class="col-md-6">
|
||||
<div class="card mb-4">
|
||||
<div class="card-header">Budget & Estimates</div>
|
||||
<div class="card-body">
|
||||
<table class="table table-borderless">
|
||||
<tr><th>Fire Alarm Budget:</th><td class="text-success">${{ '{:,.2f}'.format(job.fire_alarm_budget or 0) }}</td></tr>
|
||||
<tr><th>Labor Estimate:</th><td>${{ '{:,.2f}'.format(job.labor_estimate or 0) }}</td></tr>
|
||||
<tr><th>Material Estimate:</th><td>${{ '{:,.2f}'.format(job.material_estimate or 0) }}</td></tr>
|
||||
<tr><th>Amount Left:</th><td class="text-warning">${{ '{:,.2f}'.format(job.amount_left_on_contract or 0) }}</td></tr>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Assignments -->
|
||||
<div class="col-md-6">
|
||||
<div class="card mb-4">
|
||||
<div class="card-header">Assignments</div>
|
||||
<div class="card-body">
|
||||
<table class="table table-borderless">
|
||||
<tr><th>PM Assigned:</th><td>{{ job.pm_assigned or '-' }}</td></tr>
|
||||
<tr><th>AOR:</th><td>{{ job.aor or '-' }}</td></tr>
|
||||
<tr><th>Fire Vendor:</th><td>{{ job.fire_vendor or '-' }}</td></tr>
|
||||
<tr><th>Install Partner:</th><td>{{ job.install_partner or '-' }}</td></tr>
|
||||
<tr><th>P/S or Install:</th><td>{{ job.ps_or_install or '-' }}</td></tr>
|
||||
<tr><th>Subcontractor:</th><td>{{ job.subcontractor or '-' }}</td></tr>
|
||||
<tr><th>PCI:</th><td>{{ job.pci or '-' }}</td></tr>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Communication -->
|
||||
<div class="col-md-6">
|
||||
<div class="card mb-4">
|
||||
<div class="card-header">Communication & Plans</div>
|
||||
<div class="card-body">
|
||||
<table class="table table-borderless">
|
||||
<tr><th>VOIP/Phone:</th><td>{{ job.voip_or_phone or '-' }}</td></tr>
|
||||
<tr><th>Plans:</th><td>{{ job.plans or '-' }}</td></tr>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Key Dates -->
|
||||
<div class="col-md-6">
|
||||
<div class="card mb-4">
|
||||
<div class="card-header">Key Dates</div>
|
||||
<div class="card-body">
|
||||
<table class="table table-borderless">
|
||||
<tr><th>Elevator Final:</th><td>{{ job.elevator_final or '-' }}</td></tr>
|
||||
<tr><th>Pretest:</th><td>{{ job.pretest or '-' }}</td></tr>
|
||||
<tr><th>Final:</th><td>{{ job.final_date or '-' }}</td></tr>
|
||||
<tr><th>C/O Drop Dead:</th><td>{{ job.co_drop_dead_date or '-' }}</td></tr>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Milestones -->
|
||||
<div class="col-md-6">
|
||||
<div class="card mb-4">
|
||||
<div class="card-header">Milestones</div>
|
||||
<div class="card-body">
|
||||
<table class="table table-borderless">
|
||||
{% for i in range(1, 8) %}
|
||||
{% set milestone = job['milestone_' ~ i] %}
|
||||
{% if milestone %}
|
||||
<tr><th>{{ i }}{{ 'st' if i == 1 else ('nd' if i == 2 else ('rd' if i == 3 else 'th')) }}:</th><td>{{ milestone }}</td></tr>
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Notes & Issues -->
|
||||
<div class="col-md-6">
|
||||
<div class="card mb-4">
|
||||
<div class="card-header">Notes</div>
|
||||
<div class="card-body">
|
||||
<p>{{ job.notes or 'No notes' }}</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="col-md-6">
|
||||
<div class="card mb-4">
|
||||
<div class="card-header {% if job.issues %}bg-warning{% endif %}">Issues</div>
|
||||
<div class="card-body">
|
||||
<p>{{ job.issues or 'No issues' }}</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Phases Section -->
|
||||
<div class="row mt-4">
|
||||
<div class="col">
|
||||
<div class="card">
|
||||
<div class="card-header d-flex justify-content-between align-items-center">
|
||||
<span><i class="bi bi-list-check"></i> Phases</span>
|
||||
<button class="btn btn-sm btn-primary" data-bs-toggle="modal" data-bs-target="#addPhaseModal">
|
||||
<i class="bi bi-plus-lg"></i> Add Phase
|
||||
</button>
|
||||
</div>
|
||||
<div class="card-body">
|
||||
<div class="table-responsive">
|
||||
<table class="table table-striped">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>Type</th>
|
||||
<th>Phase #</th>
|
||||
<th>Points</th>
|
||||
<th>Start Date</th>
|
||||
<th>Due Date</th>
|
||||
<th>Men on Site</th>
|
||||
<th>Status</th>
|
||||
<th>Actions</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody id="phasesTable">
|
||||
<tr><td colspan="8" class="text-center">Loading...</td></tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Materials Section -->
|
||||
<div class="row mt-4">
|
||||
<div class="col">
|
||||
<div class="card">
|
||||
<div class="card-header d-flex justify-content-between align-items-center">
|
||||
<span><i class="bi bi-box-seam"></i> Materials</span>
|
||||
<button class="btn btn-sm btn-primary" data-bs-toggle="modal" data-bs-target="#addMaterialModal">
|
||||
<i class="bi bi-plus-lg"></i> Add Material
|
||||
</button>
|
||||
</div>
|
||||
<div class="card-body">
|
||||
<div class="table-responsive">
|
||||
<table class="table table-striped">
|
||||
<thead>
|
||||
<tr>
|
||||
<th>Part #</th>
|
||||
<th>Qty</th>
|
||||
<th>Ordered</th>
|
||||
<th>Received</th>
|
||||
<th>Received Qty</th>
|
||||
<th>Delivered Qty</th>
|
||||
<th>Actions</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody id="materialsTable">
|
||||
<tr><td colspan="7" class="text-center">Loading...</td></tr>
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Add Phase Modal -->
|
||||
<div class="modal fade" id="addPhaseModal" tabindex="-1">
|
||||
<div class="modal-dialog">
|
||||
<div class="modal-content">
|
||||
<div class="modal-header">
|
||||
<h5 class="modal-title">Add Phase</h5>
|
||||
<button type="button" class="btn-close" data-bs-dismiss="modal"></button>
|
||||
</div>
|
||||
<form id="addPhaseForm">
|
||||
<div class="modal-body">
|
||||
<div class="mb-3">
|
||||
<label class="form-label">Phase Type</label>
|
||||
<select class="form-select" name="phase_type" required>
|
||||
<option value="rough_in">Rough-in</option>
|
||||
<option value="trim">Trim</option>
|
||||
<option value="commissioning">Commissioning</option>
|
||||
<option value="final">Final</option>
|
||||
<option value="turnover">Turnover</option>
|
||||
</select>
|
||||
</div>
|
||||
<div class="mb-3">
|
||||
<label class="form-label">Phase Number</label>
|
||||
<input type="number" class="form-control" name="phase_number" min="1" required>
|
||||
</div>
|
||||
<div class="mb-3">
|
||||
<label class="form-label">Points</label>
|
||||
<input type="number" class="form-control" name="points">
|
||||
</div>
|
||||
<div class="row">
|
||||
<div class="col-md-6 mb-3">
|
||||
<label class="form-label">Start Date</label>
|
||||
<input type="date" class="form-control" name="start_date">
|
||||
</div>
|
||||
<div class="col-md-6 mb-3">
|
||||
<label class="form-label">Due Date</label>
|
||||
<input type="date" class="form-control" name="due_date">
|
||||
</div>
|
||||
</div>
|
||||
<div class="mb-3">
|
||||
<label class="form-label">Men on Site</label>
|
||||
<input type="number" class="form-control" name="men_on_site">
|
||||
</div>
|
||||
</div>
|
||||
<div class="modal-footer">
|
||||
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Cancel</button>
|
||||
<button type="submit" class="btn btn-primary">Add Phase</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Add Material Modal -->
|
||||
<div class="modal fade" id="addMaterialModal" tabindex="-1">
|
||||
<div class="modal-dialog">
|
||||
<div class="modal-content">
|
||||
<div class="modal-header">
|
||||
<h5 class="modal-title">Add Material</h5>
|
||||
<button type="button" class="btn-close" data-bs-dismiss="modal"></button>
|
||||
</div>
|
||||
<form id="addMaterialForm">
|
||||
<div class="modal-body">
|
||||
<div class="mb-3">
|
||||
<label class="form-label">Part Number</label>
|
||||
<input type="text" class="form-control" name="part_number" required>
|
||||
</div>
|
||||
<div class="mb-3">
|
||||
<label class="form-label">Quantity</label>
|
||||
<input type="number" class="form-control" name="quantity" min="1" required>
|
||||
</div>
|
||||
<div class="mb-3">
|
||||
<label class="form-label">Delivered To</label>
|
||||
<input type="text" class="form-control" name="delivered_to">
|
||||
</div>
|
||||
</div>
|
||||
<div class="modal-footer">
|
||||
<button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Cancel</button>
|
||||
<button type="submit" class="btn btn-primary">Add Material</button>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{% endblock %}
|
||||
|
||||
{% block scripts %}
|
||||
<script>
|
||||
const jobId = {{ job.id }};
|
||||
|
||||
async function loadPhases() {
|
||||
try {
|
||||
const response = await fetch(`/api/jobs/${jobId}/phases`);
|
||||
const phases = await response.json();
|
||||
|
||||
const tbody = document.getElementById('phasesTable');
|
||||
if (phases.length === 0) {
|
||||
tbody.innerHTML = '<tr><td colspan="8" class="text-center">No phases added yet</td></tr>';
|
||||
return;
|
||||
}
|
||||
|
||||
tbody.innerHTML = phases.map(p => `
|
||||
<tr>
|
||||
<td>${p.phase_type.replace('_', ' ').toUpperCase()}</td>
|
||||
<td>${p.phase_number}</td>
|
||||
<td>${p.points || '-'}</td>
|
||||
<td>${p.start_date || '-'}</td>
|
||||
<td>${p.due_date || '-'}</td>
|
||||
<td>${p.men_on_site || '-'}</td>
|
||||
<td>
|
||||
<span class="badge ${p.completed ? 'bg-success' : 'bg-secondary'}">
|
||||
${p.completed ? 'Completed' : 'In Progress'}
|
||||
</span>
|
||||
</td>
|
||||
<td>
|
||||
<button class="btn btn-sm btn-outline-danger" onclick="deletePhase(${p.id})">
|
||||
<i class="bi bi-trash"></i>
|
||||
</button>
|
||||
</td>
|
||||
</tr>
|
||||
`).join('');
|
||||
} catch (error) {
|
||||
console.error('Error loading phases:', error);
|
||||
}
|
||||
}
|
||||
|
||||
async function loadMaterials() {
|
||||
try {
|
||||
const response = await fetch(`/api/jobs/${jobId}/materials`);
|
||||
const materials = await response.json();
|
||||
|
||||
const tbody = document.getElementById('materialsTable');
|
||||
if (materials.length === 0) {
|
||||
tbody.innerHTML = '<tr><td colspan="7" class="text-center">No materials added yet</td></tr>';
|
||||
return;
|
||||
}
|
||||
|
||||
tbody.innerHTML = materials.map(m => `
|
||||
<tr>
|
||||
<td>${m.part_number}</td>
|
||||
<td>${m.quantity}</td>
|
||||
<td><i class="bi bi-${m.ordered ? 'check-circle text-success' : 'x-circle text-danger'}"></i></td>
|
||||
<td><i class="bi bi-${m.received ? 'check-circle text-success' : 'x-circle text-danger'}"></i></td>
|
||||
<td>${m.received_qty}</td>
|
||||
<td>${m.delivered_qty}</td>
|
||||
<td>
|
||||
<button class="btn btn-sm btn-outline-danger" onclick="deleteMaterial(${m.id})">
|
||||
<i class="bi bi-trash"></i>
|
||||
</button>
|
||||
</td>
|
||||
</tr>
|
||||
`).join('');
|
||||
} catch (error) {
|
||||
console.error('Error loading materials:', error);
|
||||
}
|
||||
}
|
||||
|
||||
document.getElementById('addPhaseForm').addEventListener('submit', async (e) => {
|
||||
e.preventDefault();
|
||||
const formData = new FormData(e.target);
|
||||
const data = Object.fromEntries(formData);
|
||||
|
||||
try {
|
||||
await fetch(`/api/jobs/${jobId}/phases`, {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify(data)
|
||||
});
|
||||
bootstrap.Modal.getInstance(document.getElementById('addPhaseModal')).hide();
|
||||
e.target.reset();
|
||||
loadPhases();
|
||||
} catch (error) {
|
||||
console.error('Error adding phase:', error);
|
||||
}
|
||||
});
|
||||
|
||||
document.getElementById('addMaterialForm').addEventListener('submit', async (e) => {
|
||||
e.preventDefault();
|
||||
const formData = new FormData(e.target);
|
||||
const data = Object.fromEntries(formData);
|
||||
|
||||
try {
|
||||
await fetch(`/api/jobs/${jobId}/materials`, {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify(data)
|
||||
});
|
||||
bootstrap.Modal.getInstance(document.getElementById('addMaterialModal')).hide();
|
||||
e.target.reset();
|
||||
loadMaterials();
|
||||
} catch (error) {
|
||||
console.error('Error adding material:', error);
|
||||
}
|
||||
});
|
||||
|
||||
async function deletePhase(phaseId) {
|
||||
if (!confirm('Delete this phase?')) return;
|
||||
await fetch(`/api/phases/${phaseId}`, { method: 'DELETE' });
|
||||
loadPhases();
|
||||
}
|
||||
|
||||
async function deleteMaterial(materialId) {
|
||||
if (!confirm('Delete this material?')) return;
|
||||
await fetch(`/api/materials/${materialId}`, { method: 'DELETE' });
|
||||
loadMaterials();
|
||||
}
|
||||
|
||||
document.addEventListener('DOMContentLoaded', () => {
|
||||
loadPhases();
|
||||
loadMaterials();
|
||||
});
|
||||
</script>
|
||||
{% endblock %}
|
||||
Reference in New Issue
Block a user