|
<!DOCTYPE html> |
|
<html lang="en"> |
|
<head> |
|
<meta charset="UTF-8"> |
|
<title>Frame Viewer</title> |
|
<style> |
|
|
|
* { |
|
box-sizing: border-box; |
|
margin: 0; |
|
padding: 0; |
|
} |
|
|
|
body { |
|
font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif; |
|
background-color: #f5f5f5; |
|
color: #333; |
|
line-height: 1.6; |
|
display: flex; |
|
flex-direction: column; |
|
min-height: 100vh; |
|
} |
|
|
|
|
|
header { |
|
background-color: #2c3e50; |
|
color: #ecf0f1; |
|
padding: 15px 20px; |
|
text-align: center; |
|
font-size: 1.5em; |
|
font-weight: bold; |
|
box-shadow: 0 2px 4px rgba(0,0,0,0.1); |
|
margin-bottom: 20px; |
|
position: relative; |
|
} |
|
|
|
|
|
#sort-container { |
|
position: absolute; |
|
top: 15px; |
|
right: 20px; |
|
display: flex; |
|
align-items: center; |
|
gap: 10px; |
|
} |
|
|
|
#sort-container label { |
|
color: #ecf0f1; |
|
font-size: 0.9em; |
|
} |
|
|
|
#sort-select { |
|
padding: 5px 10px; |
|
border: none; |
|
border-radius: 4px; |
|
font-size: 0.9em; |
|
cursor: pointer; |
|
} |
|
|
|
|
|
#main-container { |
|
display: flex; |
|
flex-direction: column; |
|
align-items: center; |
|
padding: 20px; |
|
gap: 20px; |
|
max-width: 1200px; |
|
margin: 0 auto; |
|
flex: 1; |
|
} |
|
|
|
|
|
#image-section { |
|
width: 90%; |
|
max-width: 900px; |
|
background-color: #fff; |
|
padding: 15px; |
|
border-radius: 8px; |
|
box-shadow: 0 4px 8px rgba(0,0,0,0.1); |
|
text-align: center; |
|
margin-bottom: 10px; |
|
} |
|
|
|
#image-canvas { |
|
width: 100%; |
|
height: auto; |
|
border-radius: 5px; |
|
border: 1px solid #ddd; |
|
display: block; |
|
} |
|
|
|
|
|
#metadata-section { |
|
width: 90%; |
|
max-width: 900px; |
|
background-color: #fff; |
|
padding: 20px; |
|
border-radius: 8px; |
|
box-shadow: 0 4px 8px rgba(0,0,0,0.1); |
|
} |
|
|
|
#metadata-form h2 { |
|
margin-bottom: 15px; |
|
font-size: 1.3em; |
|
border-bottom: 1px solid #ccc; |
|
padding-bottom: 10px; |
|
} |
|
|
|
|
|
#metadata-form label { |
|
display: block; |
|
margin-top: 10px; |
|
font-weight: 500; |
|
color: #555; |
|
} |
|
|
|
#metadata-form input, |
|
#metadata-form textarea, |
|
#metadata-form select { |
|
width: 100%; |
|
padding: 8px 10px; |
|
margin-top: 5px; |
|
border: 1px solid #ccc; |
|
border-radius: 4px; |
|
background-color: #fafafa; |
|
transition: border-color 0.3s; |
|
font-size: 0.95em; |
|
} |
|
|
|
#metadata-form input:focus, |
|
#metadata-form textarea:focus, |
|
#metadata-form select:focus { |
|
border-color: #7f8c8d; |
|
outline: none; |
|
} |
|
|
|
|
|
#button-container { |
|
margin-top: 10px; |
|
display: flex; |
|
flex-wrap: wrap; |
|
gap: 10px; |
|
justify-content: center; |
|
} |
|
|
|
|
|
button { |
|
padding: 10px 15px; |
|
font-size: 0.95em; |
|
cursor: pointer; |
|
border: none; |
|
border-radius: 4px; |
|
transition: background-color 0.3s, transform 0.2s; |
|
color: #fff; |
|
min-width: 100px; |
|
} |
|
|
|
button:hover { |
|
transform: translateY(-2px); |
|
} |
|
|
|
|
|
#prev-btn, #next-btn { |
|
background-color: #3498db; |
|
flex: 1; |
|
} |
|
|
|
#prev-btn:disabled, #next-btn:disabled { |
|
background-color: #95a5a6; |
|
cursor: not-allowed; |
|
} |
|
|
|
#save-btn { |
|
background-color: #2980b9; |
|
flex: 1; |
|
} |
|
|
|
#export-btn { |
|
background-color: #27ae60; |
|
flex: 1; |
|
} |
|
|
|
#delete-btn { |
|
background-color: #e74c3c; |
|
flex: 1; |
|
} |
|
|
|
|
|
footer { |
|
background-color: #2c3e50; |
|
color: #ecf0f1; |
|
text-align: center; |
|
padding: 10px 0; |
|
margin-top: 20px; |
|
} |
|
</style> |
|
</head> |
|
<body> |
|
<header> |
|
KeepTrack Frame Viewer |
|
|
|
<div id="sort-container"> |
|
<label for="sort-select">Sort By:</label> |
|
<select id="sort-select"> |
|
<option value="item_number">Item Number</option> |
|
<option value="estimated_worth">Estimated Worth</option> |
|
<option value="overall_certainty_flag">Overall Certainty Flag</option> |
|
</select> |
|
</div> |
|
</header> |
|
<div id="main-container"> |
|
<div id="image-section"> |
|
<canvas id="image-canvas"></canvas> |
|
<div id="button-container"> |
|
<button id="prev-btn" title="Previous Frame">Previous</button> |
|
<button id="next-btn" title="Next Frame">Next</button> |
|
</div> |
|
</div> |
|
|
|
<div id="metadata-section"> |
|
<form id="metadata-form"> |
|
<h2>Metadata</h2> |
|
<div id="form-fields"></div> |
|
</form> |
|
<div id="button-container"> |
|
<button type="button" id="save-btn">Save Changes</button> |
|
<button type="button" id="export-btn">Export Metadata</button> |
|
<button type="button" id="delete-btn">Delete Item</button> |
|
</div> |
|
</div> |
|
</div> |
|
<footer> |
|
© 2024 KeepTrack Frame Viewer App |
|
</footer> |
|
|
|
<script> |
|
|
|
let frames = []; |
|
let currentIndex = 0; |
|
const canvas = document.getElementById('image-canvas'); |
|
const ctx = canvas.getContext('2d'); |
|
const prevBtn = document.getElementById('prev-btn'); |
|
const nextBtn = document.getElementById('next-btn'); |
|
const formFields = document.getElementById('form-fields'); |
|
const saveBtn = document.getElementById('save-btn'); |
|
const exportBtn = document.getElementById('export-btn'); |
|
const deleteBtn = document.getElementById('delete-btn'); |
|
const sortSelect = document.getElementById('sort-select'); |
|
|
|
|
|
fetch('frames_metadata_with_boxes.json') |
|
.then(response => response.json()) |
|
.then(data => { |
|
frames = data; |
|
if (frames.length > 0) { |
|
sortFrames(); |
|
renderFrame(currentIndex); |
|
} else { |
|
alert('No frames found in the JSON file.'); |
|
} |
|
}) |
|
.catch(error => { |
|
console.error('Error loading JSON:', error); |
|
alert('Failed to load frames_metadata_with_boxes.json. Please ensure the file exists and is correctly formatted.'); |
|
}); |
|
|
|
|
|
prevBtn.addEventListener('click', () => { |
|
if (currentIndex > 0) { |
|
currentIndex--; |
|
renderFrame(currentIndex); |
|
} |
|
}); |
|
|
|
nextBtn.addEventListener('click', () => { |
|
if (currentIndex < frames.length - 1) { |
|
currentIndex++; |
|
renderFrame(currentIndex); |
|
} |
|
}); |
|
|
|
|
|
saveBtn.addEventListener('click', () => { |
|
saveFormData(); |
|
alert('Changes saved in memory. To export, click "Export Metadata".'); |
|
}); |
|
|
|
|
|
exportBtn.addEventListener('click', () => { |
|
const dataStr = "data:text/json;charset=utf-8," + encodeURIComponent(JSON.stringify(frames, null, 2)); |
|
const downloadAnchor = document.createElement('a'); |
|
downloadAnchor.setAttribute("href", dataStr); |
|
downloadAnchor.setAttribute("download", "modified_frames_metadata_with_boxes.json"); |
|
document.body.appendChild(downloadAnchor); |
|
downloadAnchor.click(); |
|
downloadAnchor.remove(); |
|
}); |
|
|
|
|
|
deleteBtn.addEventListener('click', () => { |
|
if (confirm('Are you sure you want to delete this item?')) { |
|
frames.splice(currentIndex, 1); |
|
if (frames.length === 0) { |
|
alert('All items have been deleted.'); |
|
ctx.clearRect(0, 0, canvas.width, canvas.height); |
|
formFields.innerHTML = ''; |
|
prevBtn.disabled = true; |
|
nextBtn.disabled = true; |
|
saveBtn.disabled = true; |
|
exportBtn.disabled = true; |
|
deleteBtn.disabled = true; |
|
} else { |
|
if (currentIndex >= frames.length) { |
|
currentIndex = frames.length - 1; |
|
} |
|
renderFrame(currentIndex); |
|
} |
|
} |
|
}); |
|
|
|
|
|
sortSelect.addEventListener('change', () => { |
|
sortFrames(); |
|
currentIndex = 0; |
|
renderFrame(currentIndex); |
|
}); |
|
|
|
|
|
function renderFrame(index) { |
|
const frame = frames[index]; |
|
const img = new Image(); |
|
img.src = frame.frame_filename; |
|
img.onload = () => { |
|
|
|
canvas.width = img.width; |
|
canvas.height = img.height; |
|
|
|
ctx.clearRect(0, 0, canvas.width, canvas.height); |
|
ctx.drawImage(img, 0, 0); |
|
|
|
if (frame.boxes_converted && Array.isArray(frame.boxes_converted)) { |
|
frame.boxes_converted.forEach(box => { |
|
const [ymin, xmin, ymax, xmax] = box; |
|
const width = xmax - xmin; |
|
const height = ymax - ymin; |
|
ctx.strokeStyle = 'green'; |
|
ctx.lineWidth = 4; |
|
ctx.strokeRect(xmin, ymin, width, height); |
|
}); |
|
} |
|
}; |
|
img.onerror = () => { |
|
alert(`Failed to load image: ${frame.frame_filename}`); |
|
}; |
|
|
|
populateForm(frame); |
|
|
|
updateNavButtons(); |
|
} |
|
|
|
|
|
function updateNavButtons() { |
|
prevBtn.disabled = currentIndex === 0; |
|
nextBtn.disabled = currentIndex === frames.length - 1; |
|
} |
|
|
|
|
|
function populateForm(frame) { |
|
formFields.innerHTML = ''; |
|
|
|
const fields = [ |
|
{ key: 'item_number', label: 'Item Number', type: 'number' }, |
|
{ key: 'item_name', label: 'Item Name', type: 'text' }, |
|
{ key: 'item_type', label: 'Item Type', type: 'text' }, |
|
{ key: 'item_description', label: 'Item Description', type: 'textarea' }, |
|
{ key: 'item_brand', label: 'Item Brand', type: 'text' }, |
|
{ key: 'item_condition', label: 'Item Condition', type: 'text' }, |
|
{ key: 'number_of_items', label: 'Number of Items', type: 'number' }, |
|
{ key: 'estimated_worth', label: 'Estimated Worth', type: 'number', step: '0.01' }, |
|
{ key: 'estimated_worth_flag', label: 'Estimated Worth Flag', type: 'number' }, |
|
{ key: 'mentioned_worth', label: 'Mentioned Worth', type: 'text' }, |
|
{ key: 'room', label: 'Room', type: 'text' }, |
|
{ key: 'timestamp', label: 'Timestamp', type: 'text' }, |
|
{ key: 'overall_certainty_flag', label: 'Overall Certainty Flag', type: 'number' }, |
|
{ key: 'is_similar_to', label: 'Is Similar To', type: 'text' }, |
|
{ key: 'frame_filename', label: 'Frame Filename', type: 'text', disabled: true } |
|
]; |
|
|
|
fields.forEach(field => { |
|
const label = document.createElement('label'); |
|
label.textContent = field.label; |
|
const input = field.type === 'textarea' ? document.createElement('textarea') : document.createElement('input'); |
|
if (field.type !== 'textarea') { |
|
input.type = field.type; |
|
} |
|
input.value = frame[field.key] !== 'nan' ? frame[field.key] : ''; |
|
input.id = field.key; |
|
input.name = field.key; |
|
if (field.step) input.step = field.step; |
|
if (field.disabled) input.disabled = true; |
|
label.appendChild(input); |
|
formFields.appendChild(label); |
|
}); |
|
} |
|
|
|
|
|
function saveFormData() { |
|
const frame = frames[currentIndex]; |
|
const inputs = formFields.querySelectorAll('input, textarea'); |
|
inputs.forEach(input => { |
|
if (input.disabled) return; |
|
const key = input.name; |
|
let value = input.value; |
|
|
|
const numericFields = [ |
|
'item_number', |
|
'number_of_items', |
|
'estimated_worth', |
|
'estimated_worth_flag', |
|
'overall_certainty_flag' |
|
]; |
|
if (numericFields.includes(key)) { |
|
value = value === '' ? 'nan' : Number(value); |
|
if (isNaN(value)) value = 'nan'; |
|
} |
|
frame[key] = value; |
|
}); |
|
} |
|
|
|
|
|
function sortFrames() { |
|
const sortBy = sortSelect.value; |
|
frames.sort((a, b) => { |
|
let aValue = a[sortBy]; |
|
let bValue = b[sortBy]; |
|
|
|
|
|
if (aValue === 'nan') return 1; |
|
if (bValue === 'nan') return -1; |
|
|
|
|
|
if (sortBy === 'item_number' || sortBy === 'estimated_worth' || sortBy === 'overall_certainty_flag') { |
|
aValue = Number(aValue); |
|
bValue = Number(bValue); |
|
if (isNaN(aValue)) aValue = -Infinity; |
|
if (isNaN(bValue)) bValue = -Infinity; |
|
} |
|
|
|
if (aValue < bValue) return -1; |
|
if (aValue > bValue) return 1; |
|
return 0; |
|
}); |
|
} |
|
</script> |
|
</body> |
|
</html> |
|
|