resolved most in browser errors and got maps stable. Next need to do some work on the save configuration stuff.
This commit is contained in:
parent
412ca36192
commit
f4eefa1cae
@ -6,6 +6,10 @@
|
|||||||
<meta name="description" content="Admin Panel - NocoDB Map Viewer">
|
<meta name="description" content="Admin Panel - NocoDB Map Viewer">
|
||||||
<title>Admin Panel - NocoDB Map Viewer</title>
|
<title>Admin Panel - NocoDB Map Viewer</title>
|
||||||
|
|
||||||
|
<!-- Favicon -->
|
||||||
|
<link rel="icon" type="image/x-icon" href="/favicon.ico">
|
||||||
|
<link rel="shortcut icon" href="/favicon.ico">
|
||||||
|
|
||||||
<!-- Leaflet CSS -->
|
<!-- Leaflet CSS -->
|
||||||
<link rel="stylesheet" href="https://unpkg.com/leaflet@1.9.4/dist/leaflet.css"
|
<link rel="stylesheet" href="https://unpkg.com/leaflet@1.9.4/dist/leaflet.css"
|
||||||
integrity="sha256-p4NxAoJBhIIN+hmNHrzRCf9tD/miZyoHS5obTRR9BMY="
|
integrity="sha256-p4NxAoJBhIIN+hmNHrzRCf9tD/miZyoHS5obTRR9BMY="
|
||||||
@ -186,8 +190,58 @@
|
|||||||
integrity="sha256-20nQCchB9co0qIjJZRGuk2/Z9VM+kNiyxNV1lvTlZBo="
|
integrity="sha256-20nQCchB9co0qIjJZRGuk2/Z9VM+kNiyxNV1lvTlZBo="
|
||||||
crossorigin=""></script>
|
crossorigin=""></script>
|
||||||
|
|
||||||
<!-- QR Code Library -->
|
<!-- Custom QR Code Implementation -->
|
||||||
<script src="https://cdn.jsdelivr.net/npm/qrcode@1.5.3/build/qrcode.min.js"></script>
|
<script>
|
||||||
|
// Simple QR Code implementation using our server
|
||||||
|
window.QRCode = {
|
||||||
|
toCanvas: function(canvas, text, options, callback) {
|
||||||
|
if (typeof options === 'function') {
|
||||||
|
callback = options;
|
||||||
|
options = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
const size = options.width || 200;
|
||||||
|
const qrUrl = `/api/qr?text=${encodeURIComponent(text)}&size=${size}`;
|
||||||
|
|
||||||
|
const img = new Image();
|
||||||
|
img.crossOrigin = 'anonymous';
|
||||||
|
|
||||||
|
img.onload = function() {
|
||||||
|
canvas.width = size;
|
||||||
|
canvas.height = size;
|
||||||
|
|
||||||
|
const ctx = canvas.getContext('2d');
|
||||||
|
ctx.drawImage(img, 0, 0, size, size);
|
||||||
|
|
||||||
|
if (callback) callback(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
img.onerror = function() {
|
||||||
|
console.error('Failed to load QR code from server');
|
||||||
|
|
||||||
|
// Fallback: draw a simple placeholder
|
||||||
|
canvas.width = size;
|
||||||
|
canvas.height = size;
|
||||||
|
|
||||||
|
const ctx = canvas.getContext('2d');
|
||||||
|
ctx.fillStyle = '#ffffff';
|
||||||
|
ctx.fillRect(0, 0, size, size);
|
||||||
|
|
||||||
|
ctx.fillStyle = '#000000';
|
||||||
|
ctx.font = '12px Arial';
|
||||||
|
ctx.textAlign = 'center';
|
||||||
|
ctx.fillText('QR Code', size/2, size/2 - 10);
|
||||||
|
ctx.fillText('(Failed)', size/2, size/2 + 10);
|
||||||
|
|
||||||
|
if (callback) callback(new Error('Failed to load QR code'));
|
||||||
|
};
|
||||||
|
|
||||||
|
img.src = qrUrl;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
console.log('Local QR Code implementation loaded');
|
||||||
|
</script>
|
||||||
|
|
||||||
<!-- Admin JavaScript -->
|
<!-- Admin JavaScript -->
|
||||||
<script src="js/admin.js"></script>
|
<script src="js/admin.js"></script>
|
||||||
|
|||||||
@ -307,10 +307,10 @@
|
|||||||
aspect-ratio: 8.5 / 11;
|
aspect-ratio: 8.5 / 11;
|
||||||
background: white;
|
background: white;
|
||||||
box-shadow: 0 2px 10px rgba(0,0,0,0.1);
|
box-shadow: 0 2px 10px rgba(0,0,0,0.1);
|
||||||
padding: 20px;
|
padding: 40px; /* Match print padding */
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
overflow: auto;
|
overflow: auto;
|
||||||
font-size: 10px;
|
font-size: 12px; /* Match print font size */
|
||||||
line-height: 1.4;
|
line-height: 1.4;
|
||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
@ -326,17 +326,38 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
.walk-sheet-page {
|
.walk-sheet-page {
|
||||||
position: absolute;
|
position: fixed;
|
||||||
left: 0;
|
left: 0;
|
||||||
top: 0;
|
top: 0;
|
||||||
width: 8.5in;
|
width: 8.5in;
|
||||||
height: 11in;
|
height: 11in;
|
||||||
max-width: none;
|
max-width: none;
|
||||||
padding: 0.5in;
|
padding: 0.5in; /* 0.5 inch margins */
|
||||||
margin: 0;
|
margin: 0;
|
||||||
box-shadow: none;
|
box-shadow: none;
|
||||||
font-size: 12pt;
|
font-size: 12pt;
|
||||||
|
page-break-after: always;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Ensure QR codes print properly */
|
||||||
|
.ws-qr-code img {
|
||||||
|
width: 100px !important;
|
||||||
|
height: 100px !important;
|
||||||
|
-webkit-print-color-adjust: exact;
|
||||||
|
print-color-adjust: exact;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Adjust preview scaling */
|
||||||
|
.walk-sheet-preview .walk-sheet-page {
|
||||||
|
transform: scale(0.5);
|
||||||
|
transform-origin: top left;
|
||||||
|
margin-bottom: -50%; /* Compensate for scale */
|
||||||
|
}
|
||||||
|
|
||||||
|
.walk-sheet-preview {
|
||||||
|
overflow: hidden;
|
||||||
|
height: 550px; /* Fixed height for preview container */
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Walk Sheet Content Styles */
|
/* Walk Sheet Content Styles */
|
||||||
@ -539,3 +560,24 @@
|
|||||||
--transition: all 0.2s ease;
|
--transition: all 0.2s ease;
|
||||||
--header-height: 60px;
|
--header-height: 60px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Crosshair styling */
|
||||||
|
.crosshair {
|
||||||
|
pointer-events: none;
|
||||||
|
z-index: 1000;
|
||||||
|
}
|
||||||
|
|
||||||
|
.crosshair div {
|
||||||
|
border-radius: 1px;
|
||||||
|
opacity: 0.8;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Admin map styling */
|
||||||
|
.admin-map {
|
||||||
|
position: relative;
|
||||||
|
cursor: crosshair;
|
||||||
|
}
|
||||||
|
|
||||||
|
.admin-map .leaflet-container {
|
||||||
|
cursor: crosshair;
|
||||||
|
}
|
||||||
|
|||||||
1
map/app/public/favicon.ico
Normal file
1
map/app/public/favicon.ico
Normal file
@ -0,0 +1 @@
|
|||||||
|
<!-- Simple favicon placeholder -->
|
||||||
@ -48,6 +48,24 @@ function initializeAdminMap() {
|
|||||||
minZoom: 2
|
minZoom: 2
|
||||||
}).addTo(adminMap);
|
}).addTo(adminMap);
|
||||||
|
|
||||||
|
// Add crosshair to center of map
|
||||||
|
const crosshairIcon = L.divIcon({
|
||||||
|
className: 'crosshair',
|
||||||
|
iconSize: [20, 20],
|
||||||
|
html: '<div style="width: 20px; height: 20px; position: relative;"><div style="position: absolute; top: 9px; left: 0; width: 20px; height: 2px; background: #333; box-shadow: 0 0 3px rgba(255,255,255,0.8);"></div><div style="position: absolute; top: 0; left: 9px; width: 2px; height: 20px; background: #333; box-shadow: 0 0 3px rgba(255,255,255,0.8);"></div></div>'
|
||||||
|
});
|
||||||
|
|
||||||
|
const crosshair = L.marker(adminMap.getCenter(), {
|
||||||
|
icon: crosshairIcon,
|
||||||
|
interactive: false,
|
||||||
|
zIndexOffset: 1000
|
||||||
|
}).addTo(adminMap);
|
||||||
|
|
||||||
|
// Update crosshair position when map moves
|
||||||
|
adminMap.on('move', function() {
|
||||||
|
crosshair.setLatLng(adminMap.getCenter());
|
||||||
|
});
|
||||||
|
|
||||||
// Add click handler to set location
|
// Add click handler to set location
|
||||||
adminMap.on('click', handleMapClick);
|
adminMap.on('click', handleMapClick);
|
||||||
|
|
||||||
@ -128,31 +146,46 @@ function updateCoordinatesFromMap() {
|
|||||||
// Setup event listeners
|
// Setup event listeners
|
||||||
function setupEventListeners() {
|
function setupEventListeners() {
|
||||||
// Use current view button
|
// Use current view button
|
||||||
document.getElementById('use-current-view').addEventListener('click', () => {
|
const useCurrentViewBtn = document.getElementById('use-current-view');
|
||||||
const center = adminMap.getCenter();
|
if (useCurrentViewBtn) {
|
||||||
const zoom = adminMap.getZoom();
|
useCurrentViewBtn.addEventListener('click', () => {
|
||||||
|
const center = adminMap.getCenter();
|
||||||
document.getElementById('start-lat').value = center.lat.toFixed(6);
|
const zoom = adminMap.getZoom();
|
||||||
document.getElementById('start-lng').value = center.lng.toFixed(6);
|
|
||||||
document.getElementById('start-zoom').value = zoom;
|
document.getElementById('start-lat').value = center.lat.toFixed(6);
|
||||||
|
document.getElementById('start-lng').value = center.lng.toFixed(6);
|
||||||
updateStartMarker(center.lat, center.lng);
|
document.getElementById('start-zoom').value = zoom;
|
||||||
showStatus('Captured current map view', 'success');
|
|
||||||
});
|
updateStartMarker(center.lat, center.lng);
|
||||||
|
showStatus('Captured current map view', 'success');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
// Save button
|
// Save button
|
||||||
document.getElementById('save-start-location').addEventListener('click', saveStartLocation);
|
const saveLocationBtn = document.getElementById('save-start-location');
|
||||||
|
if (saveLocationBtn) {
|
||||||
|
saveLocationBtn.addEventListener('click', saveStartLocation);
|
||||||
|
}
|
||||||
|
|
||||||
// Coordinate input changes
|
// Coordinate input changes
|
||||||
document.getElementById('start-lat').addEventListener('change', updateMapFromInputs);
|
const startLatInput = document.getElementById('start-lat');
|
||||||
document.getElementById('start-lng').addEventListener('change', updateMapFromInputs);
|
const startLngInput = document.getElementById('start-lng');
|
||||||
document.getElementById('start-zoom').addEventListener('change', updateMapFromInputs);
|
const startZoomInput = document.getElementById('start-zoom');
|
||||||
|
|
||||||
|
if (startLatInput) startLatInput.addEventListener('change', updateMapFromInputs);
|
||||||
|
if (startLngInput) startLngInput.addEventListener('change', updateMapFromInputs);
|
||||||
|
if (startZoomInput) startZoomInput.addEventListener('change', updateMapFromInputs);
|
||||||
|
|
||||||
// Walk Sheet buttons
|
// Walk Sheet buttons
|
||||||
document.getElementById('save-walk-sheet').addEventListener('click', saveWalkSheetConfig);
|
const saveWalkSheetBtn = document.getElementById('save-walk-sheet');
|
||||||
document.getElementById('preview-walk-sheet').addEventListener('click', generateWalkSheetPreview);
|
const previewWalkSheetBtn = document.getElementById('preview-walk-sheet');
|
||||||
document.getElementById('print-walk-sheet').addEventListener('click', printWalkSheet);
|
const printWalkSheetBtn = document.getElementById('print-walk-sheet');
|
||||||
document.getElementById('refresh-preview').addEventListener('click', generateWalkSheetPreview);
|
const refreshPreviewBtn = document.getElementById('refresh-preview');
|
||||||
|
|
||||||
|
if (saveWalkSheetBtn) saveWalkSheetBtn.addEventListener('click', saveWalkSheetConfig);
|
||||||
|
if (previewWalkSheetBtn) previewWalkSheetBtn.addEventListener('click', generateWalkSheetPreview);
|
||||||
|
if (printWalkSheetBtn) printWalkSheetBtn.addEventListener('click', printWalkSheet);
|
||||||
|
if (refreshPreviewBtn) refreshPreviewBtn.addEventListener('click', generateWalkSheetPreview);
|
||||||
|
|
||||||
// Auto-update preview on input change
|
// Auto-update preview on input change
|
||||||
const walkSheetInputs = document.querySelectorAll(
|
const walkSheetInputs = document.querySelectorAll(
|
||||||
@ -161,24 +194,65 @@ function setupEventListeners() {
|
|||||||
);
|
);
|
||||||
|
|
||||||
walkSheetInputs.forEach(input => {
|
walkSheetInputs.forEach(input => {
|
||||||
input.addEventListener('input', debounce(() => {
|
if (input) {
|
||||||
generateWalkSheetPreview();
|
input.addEventListener('input', debounce(() => {
|
||||||
}, 500));
|
generateWalkSheetPreview();
|
||||||
|
}, 500));
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Add URL change listeners to detect when QR codes need regeneration
|
// Add URL change listeners to detect when QR codes need regeneration
|
||||||
for (let i = 1; i <= 3; i++) {
|
for (let i = 1; i <= 3; i++) {
|
||||||
const urlInput = document.getElementById(`qr-code-${i}-url`);
|
const urlInput = document.getElementById(`qr-code-${i}-url`);
|
||||||
let previousUrl = urlInput.value;
|
if (urlInput) {
|
||||||
|
let previousUrl = urlInput.value;
|
||||||
urlInput.addEventListener('change', () => {
|
|
||||||
if (urlInput.value !== previousUrl) {
|
urlInput.addEventListener('change', () => {
|
||||||
// URL changed, clear stored QR code
|
if (urlInput.value !== previousUrl) {
|
||||||
delete storedQRCodes[i];
|
// URL changed, clear stored QR code
|
||||||
previousUrl = urlInput.value;
|
delete storedQRCodes[i];
|
||||||
|
previousUrl = urlInput.value;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Setup navigation between admin sections
|
||||||
|
function setupNavigation() {
|
||||||
|
const navLinks = document.querySelectorAll('.admin-nav a');
|
||||||
|
const sections = document.querySelectorAll('.admin-section');
|
||||||
|
|
||||||
|
navLinks.forEach(link => {
|
||||||
|
link.addEventListener('click', (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
// Get target section ID
|
||||||
|
const targetId = link.getAttribute('href').substring(1);
|
||||||
|
|
||||||
|
// Hide all sections
|
||||||
|
sections.forEach(section => {
|
||||||
|
section.style.display = 'none';
|
||||||
|
});
|
||||||
|
|
||||||
|
// Show target section
|
||||||
|
const targetSection = document.getElementById(targetId);
|
||||||
|
if (targetSection) {
|
||||||
|
targetSection.style.display = 'block';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update active nav link
|
||||||
|
navLinks.forEach(navLink => {
|
||||||
|
navLink.classList.remove('active');
|
||||||
|
});
|
||||||
|
link.classList.add('active');
|
||||||
|
|
||||||
|
// If switching to walk sheet, generate preview
|
||||||
|
if (targetId === 'walk-sheet') {
|
||||||
|
generateWalkSheetPreview();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update map from input fields
|
// Update map from input fields
|
||||||
@ -245,19 +319,24 @@ async function saveStartLocation() {
|
|||||||
// Save walk sheet configuration
|
// Save walk sheet configuration
|
||||||
async function saveWalkSheetConfig() {
|
async function saveWalkSheetConfig() {
|
||||||
const config = {
|
const config = {
|
||||||
walk_sheet_title: document.getElementById('walk-sheet-title').value,
|
walk_sheet_title: document.getElementById('walk-sheet-title')?.value || '',
|
||||||
walk_sheet_subtitle: document.getElementById('walk-sheet-subtitle').value,
|
walk_sheet_subtitle: document.getElementById('walk-sheet-subtitle')?.value || '',
|
||||||
walk_sheet_footer: document.getElementById('walk-sheet-footer').value,
|
walk_sheet_footer: document.getElementById('walk-sheet-footer')?.value || '',
|
||||||
qr_code_1_url: document.getElementById('qr-code-1-url').value,
|
qr_code_1_url: document.getElementById('qr-code-1-url')?.value || '',
|
||||||
qr_code_1_label: document.getElementById('qr-code-1-label').value,
|
qr_code_1_label: document.getElementById('qr-code-1-label')?.value || '',
|
||||||
qr_code_2_url: document.getElementById('qr-code-2-url').value,
|
qr_code_2_url: document.getElementById('qr-code-2-url')?.value || '',
|
||||||
qr_code_2_label: document.getElementById('qr-code-2-label').value,
|
qr_code_2_label: document.getElementById('qr-code-2-label')?.value || '',
|
||||||
qr_code_3_url: document.getElementById('qr-code-3-url').value,
|
qr_code_3_url: document.getElementById('qr-code-3-url')?.value || '',
|
||||||
qr_code_3_label: document.getElementById('qr-code-3-label').value
|
qr_code_3_label: document.getElementById('qr-code-3-label')?.value || ''
|
||||||
};
|
};
|
||||||
|
|
||||||
// Show loading state
|
// Show loading state
|
||||||
const saveButton = document.getElementById('save-walk-sheet');
|
const saveButton = document.getElementById('save-walk-sheet');
|
||||||
|
if (!saveButton) {
|
||||||
|
showStatus('Save button not found', 'error');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const originalText = saveButton.textContent;
|
const originalText = saveButton.textContent;
|
||||||
saveButton.textContent = 'Saving...';
|
saveButton.textContent = 'Saving...';
|
||||||
saveButton.disabled = true;
|
saveButton.disabled = true;
|
||||||
@ -280,7 +359,7 @@ async function saveWalkSheetConfig() {
|
|||||||
if (data.qrCodes) {
|
if (data.qrCodes) {
|
||||||
for (let i = 1; i <= 3; i++) {
|
for (let i = 1; i <= 3; i++) {
|
||||||
if (data.qrCodes[`qr_code_${i}_image`]) {
|
if (data.qrCodes[`qr_code_${i}_image`]) {
|
||||||
storedQRCodes[i] = data.qrCodes[`qr_code_${i}_image`];
|
storedQRCodes[`qr_code_${i}_image`] = data.qrCodes[`qr_code_${i}_image`];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -302,9 +381,9 @@ async function saveWalkSheetConfig() {
|
|||||||
|
|
||||||
// Generate walk sheet preview
|
// Generate walk sheet preview
|
||||||
function generateWalkSheetPreview() {
|
function generateWalkSheetPreview() {
|
||||||
const title = document.getElementById('walk-sheet-title').value || 'Campaign Walk Sheet';
|
const title = document.getElementById('walk-sheet-title')?.value || 'Campaign Walk Sheet';
|
||||||
const subtitle = document.getElementById('walk-sheet-subtitle').value || 'Door-to-Door Canvassing Form';
|
const subtitle = document.getElementById('walk-sheet-subtitle')?.value || 'Door-to-Door Canvassing Form';
|
||||||
const footer = document.getElementById('walk-sheet-footer').value || 'Thank you for your support!';
|
const footer = document.getElementById('walk-sheet-footer')?.value || 'Thank you for your support!';
|
||||||
|
|
||||||
let previewHTML = `
|
let previewHTML = `
|
||||||
<div class="ws-header">
|
<div class="ws-header">
|
||||||
@ -316,30 +395,21 @@ function generateWalkSheetPreview() {
|
|||||||
// Add QR codes section
|
// Add QR codes section
|
||||||
const qrCodesHTML = [];
|
const qrCodesHTML = [];
|
||||||
for (let i = 1; i <= 3; i++) {
|
for (let i = 1; i <= 3; i++) {
|
||||||
const url = document.getElementById(`qr-code-${i}-url`).value;
|
const urlInput = document.getElementById(`qr-code-${i}-url`);
|
||||||
const label = document.getElementById(`qr-code-${i}-label`).value;
|
const labelInput = document.getElementById(`qr-code-${i}-label`);
|
||||||
|
|
||||||
|
const url = urlInput?.value || '';
|
||||||
|
const label = labelInput?.value || '';
|
||||||
|
|
||||||
if (url) {
|
if (url) {
|
||||||
// Check if we have a stored QR code image
|
qrCodesHTML.push(`
|
||||||
if (storedQRCodes[i] && storedQRCodes[i].url) {
|
<div class="ws-qr-item">
|
||||||
// Use stored QR code image
|
<div class="ws-qr-code" id="preview-qr-${i}">
|
||||||
qrCodesHTML.push(`
|
<!-- QR code will be inserted here -->
|
||||||
<div class="ws-qr-item">
|
|
||||||
<div class="ws-qr-code">
|
|
||||||
<img src="${storedQRCodes[i].url}" alt="QR Code ${i}" style="width: 80px; height: 80px;">
|
|
||||||
</div>
|
|
||||||
<div class="ws-qr-label">${escapeHtml(label) || `QR Code ${i}`}</div>
|
|
||||||
</div>
|
</div>
|
||||||
`);
|
<div class="ws-qr-label">${escapeHtml(label || `QR Code ${i}`)}</div>
|
||||||
} else {
|
</div>
|
||||||
// Generate QR code client-side as fallback
|
`);
|
||||||
qrCodesHTML.push(`
|
|
||||||
<div class="ws-qr-item">
|
|
||||||
<div class="ws-qr-code" id="preview-qr-${i}"></div>
|
|
||||||
<div class="ws-qr-label">${escapeHtml(label) || `QR Code ${i}`}</div>
|
|
||||||
</div>
|
|
||||||
`);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -393,14 +463,17 @@ function generateWalkSheetPreview() {
|
|||||||
<div class="ws-form-field"></div>
|
<div class="ws-form-field"></div>
|
||||||
</div>
|
</div>
|
||||||
<div class="ws-form-group">
|
<div class="ws-form-group">
|
||||||
<label class="ws-form-label">Sign Request ☐ Yes ☐ No</label>
|
<label class="ws-form-label">Sign Request</label>
|
||||||
<div class="ws-form-field"></div>
|
<div class="ws-form-field" style="display: flex; gap: 20px;">
|
||||||
|
<span>☐ Yes</span>
|
||||||
|
<span>☐ No</span>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="ws-form-row">
|
<div class="ws-form-row">
|
||||||
<div class="ws-form-group">
|
<div class="ws-form-group">
|
||||||
<label class="ws-form-label">Category</label>
|
<label class="ws-form-label">Sign Size</label>
|
||||||
<div class="ws-form-field"></div>
|
<div class="ws-form-field"></div>
|
||||||
</div>
|
</div>
|
||||||
<div class="ws-form-group">
|
<div class="ws-form-group">
|
||||||
@ -416,97 +489,149 @@ function generateWalkSheetPreview() {
|
|||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
// Update preview
|
// Add footer
|
||||||
const previewContent = document.getElementById('walk-sheet-preview-content');
|
|
||||||
previewContent.innerHTML = previewHTML;
|
|
||||||
|
|
||||||
// Add footer (positioned absolutely in CSS)
|
|
||||||
if (footer) {
|
if (footer) {
|
||||||
const footerDiv = document.createElement('div');
|
previewHTML += `
|
||||||
footerDiv.className = 'ws-footer';
|
<div class="ws-footer">
|
||||||
footerDiv.innerHTML = escapeHtml(footer);
|
${escapeHtml(footer)}
|
||||||
previewContent.appendChild(footerDiv);
|
</div>
|
||||||
|
`;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generate client-side QR codes for items without stored images
|
// Update preview
|
||||||
setTimeout(() => {
|
const previewContent = document.getElementById('walk-sheet-preview-content');
|
||||||
for (let i = 1; i <= 3; i++) {
|
if (previewContent) {
|
||||||
const url = document.getElementById(`qr-code-${i}-url`).value;
|
previewContent.innerHTML = previewHTML;
|
||||||
if (url && !storedQRCodes[i]) {
|
|
||||||
const qrContainer = document.getElementById(`preview-qr-${i}`);
|
// Generate QR codes after DOM is updated
|
||||||
if (qrContainer && typeof QRCode !== 'undefined') {
|
setTimeout(() => {
|
||||||
qrContainer.innerHTML = '';
|
generatePreviewQRCodes();
|
||||||
new QRCode(qrContainer, {
|
}, 100);
|
||||||
text: url,
|
} else {
|
||||||
width: 80,
|
console.warn('Walk sheet preview content container not found');
|
||||||
height: 80,
|
}
|
||||||
correctLevel: QRCode.CorrectLevel.M
|
}
|
||||||
});
|
|
||||||
}
|
// Generate QR codes for preview
|
||||||
|
async function generatePreviewQRCodes() {
|
||||||
|
for (let i = 1; i <= 3; i++) {
|
||||||
|
const urlInput = document.getElementById(`qr-code-${i}-url`);
|
||||||
|
const url = urlInput?.value || '';
|
||||||
|
const qrContainer = document.getElementById(`preview-qr-${i}`);
|
||||||
|
|
||||||
|
if (url && qrContainer) {
|
||||||
|
try {
|
||||||
|
// Use our local QR code generation endpoint
|
||||||
|
const qrImageUrl = `/api/qr?text=${encodeURIComponent(url)}&size=100`;
|
||||||
|
qrContainer.innerHTML = `<img src="${qrImageUrl}" alt="QR Code ${i}" style="width: 100px; height: 100px;">`;
|
||||||
|
} catch (error) {
|
||||||
|
console.error(`Failed to display QR code ${i}:`, error);
|
||||||
|
qrContainer.innerHTML = '<div style="width: 100px; height: 100px; border: 1px dashed #ccc; display: flex; align-items: center; justify-content: center; font-size: 10px; color: #999;">QR Error</div>';
|
||||||
}
|
}
|
||||||
|
} else if (qrContainer) {
|
||||||
|
// Clear empty QR containers
|
||||||
|
qrContainer.innerHTML = '';
|
||||||
}
|
}
|
||||||
}, 100);
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Print walk sheet
|
// Print walk sheet
|
||||||
function printWalkSheet() {
|
function printWalkSheet() {
|
||||||
// First generate fresh preview
|
// First generate fresh preview to ensure QR codes are generated
|
||||||
generateWalkSheetPreview();
|
generateWalkSheetPreview();
|
||||||
|
|
||||||
// Wait for QR codes to generate
|
// Wait for QR codes to generate, then print
|
||||||
setTimeout(() => {
|
setTimeout(() => {
|
||||||
window.print();
|
// Create a print-specific window
|
||||||
|
const printContent = document.getElementById('walk-sheet-preview-content').innerHTML;
|
||||||
|
const printWindow = window.open('', '_blank');
|
||||||
|
|
||||||
|
printWindow.document.write(`
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<title>Walk Sheet - Print</title>
|
||||||
|
<link rel="stylesheet" href="/css/style.css">
|
||||||
|
<link rel="stylesheet" href="/css/admin.css">
|
||||||
|
<style>
|
||||||
|
@media print {
|
||||||
|
@page {
|
||||||
|
size: letter;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
body {
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
.walk-sheet-page {
|
||||||
|
width: 8.5in !important;
|
||||||
|
height: 11in !important;
|
||||||
|
padding: 0.5in !important;
|
||||||
|
margin: 0 !important;
|
||||||
|
box-shadow: none !important;
|
||||||
|
page-break-after: avoid !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div class="walk-sheet-page">
|
||||||
|
${printContent}
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`);
|
||||||
|
|
||||||
|
printWindow.document.close();
|
||||||
|
|
||||||
|
// Wait for images to load
|
||||||
|
printWindow.onload = function() {
|
||||||
|
setTimeout(() => {
|
||||||
|
printWindow.print();
|
||||||
|
printWindow.close();
|
||||||
|
}, 250);
|
||||||
|
};
|
||||||
}, 500);
|
}, 500);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Setup navigation between sections
|
|
||||||
function setupNavigation() {
|
|
||||||
const navLinks = document.querySelectorAll('.admin-nav a');
|
|
||||||
const sections = document.querySelectorAll('.admin-section');
|
|
||||||
|
|
||||||
navLinks.forEach(link => {
|
|
||||||
link.addEventListener('click', (e) => {
|
|
||||||
e.preventDefault();
|
|
||||||
const targetId = link.getAttribute('href').substring(1);
|
|
||||||
|
|
||||||
// Update active states
|
|
||||||
navLinks.forEach(l => l.classList.remove('active'));
|
|
||||||
link.classList.add('active');
|
|
||||||
|
|
||||||
// Show/hide sections
|
|
||||||
sections.forEach(section => {
|
|
||||||
section.style.display = section.id === targetId ? 'block' : 'none';
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
// Load walk sheet configuration
|
// Load walk sheet configuration
|
||||||
async function loadWalkSheetConfig() {
|
async function loadWalkSheetConfig() {
|
||||||
try {
|
try {
|
||||||
const response = await fetch('/api/admin/walk-sheet-config');
|
const response = await fetch('/api/admin/walk-sheet-config');
|
||||||
const data = await response.json();
|
const data = await response.json();
|
||||||
|
|
||||||
if (data.success && data.config) {
|
if (data.success && data.data) {
|
||||||
// Populate form fields
|
// Populate form fields
|
||||||
document.getElementById('walk-sheet-title').value = data.config.walk_sheet_title || '';
|
const titleInput = document.getElementById('walk-sheet-title');
|
||||||
document.getElementById('walk-sheet-subtitle').value = data.config.walk_sheet_subtitle || '';
|
const subtitleInput = document.getElementById('walk-sheet-subtitle');
|
||||||
document.getElementById('walk-sheet-footer').value = data.config.walk_sheet_footer || '';
|
const footerInput = document.getElementById('walk-sheet-footer');
|
||||||
|
|
||||||
// QR codes
|
if (titleInput) titleInput.value = data.data.walk_sheet_title || '';
|
||||||
|
if (subtitleInput) subtitleInput.value = data.data.walk_sheet_subtitle || '';
|
||||||
|
if (footerInput) footerInput.value = data.data.walk_sheet_footer || '';
|
||||||
|
|
||||||
|
// Store QR code images if they exist
|
||||||
for (let i = 1; i <= 3; i++) {
|
for (let i = 1; i <= 3; i++) {
|
||||||
document.getElementById(`qr-code-${i}-url`).value = data.config[`qr_code_${i}_url`] || '';
|
const urlField = document.getElementById(`qr-code-${i}-url`);
|
||||||
document.getElementById(`qr-code-${i}-label`).value = data.config[`qr_code_${i}_label`] || '';
|
const labelField = document.getElementById(`qr-code-${i}-label`);
|
||||||
|
|
||||||
// Store QR code image data if available
|
if (urlField && data.data[`qr_code_${i}_url`]) {
|
||||||
if (data.config[`qr_code_${i}_image`]) {
|
urlField.value = data.data[`qr_code_${i}_url`];
|
||||||
storedQRCodes[i] = data.config[`qr_code_${i}_image`];
|
}
|
||||||
|
if (labelField && data.data[`qr_code_${i}_label`]) {
|
||||||
|
labelField.value = data.data[`qr_code_${i}_label`];
|
||||||
|
}
|
||||||
|
|
||||||
|
// Store the QR code image URL if it exists
|
||||||
|
if (data.data[`qr_code_${i}_image`]) {
|
||||||
|
storedQRCodes[`qr_code_${i}_image`] = data.data[`qr_code_${i}_image`];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Generate preview
|
// Generate preview
|
||||||
generateWalkSheetPreview();
|
generateWalkSheetPreview();
|
||||||
}
|
}
|
||||||
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
console.error('Failed to load walk sheet config:', error);
|
console.error('Failed to load walk sheet config:', error);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -200,7 +200,7 @@ app.use(helmet({
|
|||||||
directives: {
|
directives: {
|
||||||
defaultSrc: ["'self'"],
|
defaultSrc: ["'self'"],
|
||||||
styleSrc: ["'self'", "'unsafe-inline'", "https://unpkg.com"],
|
styleSrc: ["'self'", "'unsafe-inline'", "https://unpkg.com"],
|
||||||
scriptSrc: ["'self'", "'unsafe-inline'", "https://unpkg.com"],
|
scriptSrc: ["'self'", "'unsafe-inline'", "https://unpkg.com", "https://cdn.jsdelivr.net"],
|
||||||
imgSrc: ["'self'", "data:", "https://*.tile.openstreetmap.org", "https://unpkg.com"],
|
imgSrc: ["'self'", "data:", "https://*.tile.openstreetmap.org", "https://unpkg.com"],
|
||||||
connectSrc: ["'self'"]
|
connectSrc: ["'self'"]
|
||||||
}
|
}
|
||||||
@ -500,7 +500,18 @@ app.post('/api/admin/start-location', requireAdmin, async (req, res) => {
|
|||||||
|
|
||||||
if (existingSettings.length > 0) {
|
if (existingSettings.length > 0) {
|
||||||
// Update existing setting
|
// Update existing setting
|
||||||
const settingId = existingSettings[0].id || existingSettings[0].Id;
|
const setting = existingSettings[0];
|
||||||
|
let settingId = setting.id || setting.Id || setting.ID;
|
||||||
|
|
||||||
|
// If we still can't find an ID, log the object structure
|
||||||
|
if (!settingId) {
|
||||||
|
logger.error('Cannot find primary key in setting object:', {
|
||||||
|
setting: setting,
|
||||||
|
keys: Object.keys(setting)
|
||||||
|
});
|
||||||
|
throw new Error('Unable to find primary key for existing setting');
|
||||||
|
}
|
||||||
|
|
||||||
const updateUrl = `${getUrl}/${settingId}`;
|
const updateUrl = `${getUrl}/${settingId}`;
|
||||||
|
|
||||||
// Only include fields that exist in the table
|
// Only include fields that exist in the table
|
||||||
@ -690,22 +701,23 @@ app.get('/api/admin/walk-sheet-config', requireAdmin, async (req, res) => {
|
|||||||
|
|
||||||
// Get all settings
|
// Get all settings
|
||||||
const response = await axios.get(
|
const response = await axios.get(
|
||||||
`${process.env.NOCODB_API_URL}/api/v1/base/${process.env.NOCODB_PROJECT_ID}/tables/${SETTINGS_SHEET_ID}/rows`,
|
`${process.env.NOCODB_API_URL}/db/data/v1/${process.env.NOCODB_PROJECT_ID}/${SETTINGS_SHEET_ID}`,
|
||||||
{
|
{
|
||||||
headers: {
|
headers: {
|
||||||
'xc-auth': process.env.NOCODB_API_TOKEN
|
'xc-token': process.env.NOCODB_API_TOKEN
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!response.data?.list || response.data.list.length === 0) {
|
logger.info('GET Settings response structure:', JSON.stringify(response.data, null, 2));
|
||||||
|
if (!response.data?.list || response.data.list.length === 0) {
|
||||||
return res.json({
|
return res.json({
|
||||||
success: true,
|
success: true,
|
||||||
config: null,
|
config: null,
|
||||||
source: 'defaults'
|
source: 'defaults'
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find walk sheet settings
|
// Find walk sheet settings
|
||||||
const walkSheetSettings = {};
|
const walkSheetSettings = {};
|
||||||
const settingKeys = [
|
const settingKeys = [
|
||||||
@ -714,7 +726,7 @@ app.get('/api/admin/walk-sheet-config', requireAdmin, async (req, res) => {
|
|||||||
'qr_code_2_url', 'qr_code_2_label', 'qr_code_2_image',
|
'qr_code_2_url', 'qr_code_2_label', 'qr_code_2_image',
|
||||||
'qr_code_3_url', 'qr_code_3_label', 'qr_code_3_image'
|
'qr_code_3_url', 'qr_code_3_label', 'qr_code_3_image'
|
||||||
];
|
];
|
||||||
|
|
||||||
for (const setting of response.data.list) {
|
for (const setting of response.data.list) {
|
||||||
if (settingKeys.includes(setting.key)) {
|
if (settingKeys.includes(setting.key)) {
|
||||||
if (setting.key.includes('_image') && setting.value) {
|
if (setting.key.includes('_image') && setting.value) {
|
||||||
@ -745,121 +757,86 @@ app.get('/api/admin/walk-sheet-config', requireAdmin, async (req, res) => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// Save walk sheet configuration
|
// Save walk sheet configuration (simplified)
|
||||||
app.post('/api/admin/walk-sheet-config', requireAdmin, async (req, res) => {
|
app.post('/api/admin/walk-sheet-config', requireAdmin, async (req, res) => {
|
||||||
try {
|
try {
|
||||||
if (!SETTINGS_SHEET_ID) {
|
if (!SETTINGS_SHEET_ID) {
|
||||||
|
logger.error('SETTINGS_SHEET_ID not configured');
|
||||||
return res.status(400).json({
|
return res.status(400).json({
|
||||||
success: false,
|
success: false,
|
||||||
error: 'Settings sheet not configured'
|
error: 'Settings sheet not configured'
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
logger.info('Using SETTINGS_SHEET_ID:', SETTINGS_SHEET_ID);
|
||||||
|
|
||||||
const config = req.body;
|
const config = req.body;
|
||||||
|
logger.info('Received config:', JSON.stringify(config, null, 2));
|
||||||
const userEmail = req.session.userEmail;
|
const userEmail = req.session.userEmail;
|
||||||
const timestamp = new Date().toISOString();
|
const timestamp = new Date().toISOString();
|
||||||
|
|
||||||
// NocoDB configuration
|
|
||||||
const nocodbConfig = {
|
|
||||||
apiUrl: process.env.NOCODB_API_URL,
|
|
||||||
apiToken: process.env.NOCODB_API_TOKEN,
|
|
||||||
projectId: process.env.NOCODB_PROJECT_ID,
|
|
||||||
tableId: SETTINGS_SHEET_ID
|
|
||||||
};
|
|
||||||
|
|
||||||
// Get existing settings
|
// Get existing settings
|
||||||
const getResponse = await axios.get(
|
const getResponse = await axios.get(
|
||||||
`${process.env.NOCODB_API_URL}/api/v1/base/${process.env.NOCODB_PROJECT_ID}/tables/${SETTINGS_SHEET_ID}/rows`,
|
`${process.env.NOCODB_API_URL}/db/data/v1/${process.env.NOCODB_PROJECT_ID}/${SETTINGS_SHEET_ID}`,
|
||||||
{
|
{
|
||||||
headers: {
|
headers: {
|
||||||
'xc-auth': process.env.NOCODB_API_TOKEN
|
'xc-token': process.env.NOCODB_API_TOKEN
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
logger.info('Settings response structure:', JSON.stringify(getResponse.data, null, 2));
|
||||||
|
|
||||||
const existingSettings = getResponse.data?.list || [];
|
const existingSettings = getResponse.data?.list || [];
|
||||||
|
|
||||||
// Process QR codes
|
// Simple approach: Just save the text configuration (no QR code uploads for now)
|
||||||
const qrCodeUploads = {};
|
const simpleSettings = {
|
||||||
for (let i = 1; i <= 3; i++) {
|
walk_sheet_title: config.walk_sheet_title || '',
|
||||||
const url = config[`qr_code_${i}_url`];
|
walk_sheet_subtitle: config.walk_sheet_subtitle || '',
|
||||||
const label = config[`qr_code_${i}_label`] || `QR Code ${i}`;
|
walk_sheet_footer: config.walk_sheet_footer || '',
|
||||||
|
qr_code_1_url: config.qr_code_1_url || '',
|
||||||
if (url) {
|
qr_code_1_label: config.qr_code_1_label || '',
|
||||||
try {
|
qr_code_2_url: config.qr_code_2_url || '',
|
||||||
// Check if URL has changed
|
qr_code_2_label: config.qr_code_2_label || '',
|
||||||
const existingUrlSetting = existingSettings.find(s => s.key === `qr_code_${i}_url`);
|
qr_code_3_url: config.qr_code_3_url || '',
|
||||||
const urlChanged = !existingUrlSetting || existingUrlSetting.value !== url;
|
qr_code_3_label: config.qr_code_3_label || ''
|
||||||
|
};
|
||||||
if (urlChanged) {
|
|
||||||
// Generate and upload new QR code
|
|
||||||
const uploadResult = await generateAndUploadQRCode(url, label, nocodbConfig);
|
|
||||||
if (uploadResult) {
|
|
||||||
qrCodeUploads[`qr_code_${i}_image`] = uploadResult;
|
|
||||||
|
|
||||||
// Delete old QR code if exists
|
|
||||||
const existingImageSetting = existingSettings.find(s => s.key === `qr_code_${i}_image`);
|
|
||||||
if (existingImageSetting?.value) {
|
|
||||||
await deleteQRCodeFromNocoDB(existingImageSetting.value, nocodbConfig);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
} catch (error) {
|
|
||||||
logger.error(`Failed to process QR code ${i}:`, error);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// If URL is empty, delete existing QR code
|
|
||||||
const existingImageSetting = existingSettings.find(s => s.key === `qr_code_${i}_image`);
|
|
||||||
if (existingImageSetting?.value) {
|
|
||||||
await deleteQRCodeFromNocoDB(existingImageSetting.value, nocodbConfig);
|
|
||||||
}
|
|
||||||
qrCodeUploads[`qr_code_${i}_image`] = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update or create each setting
|
// Update or create each setting
|
||||||
const allSettings = { ...config, ...qrCodeUploads };
|
for (const [key, value] of Object.entries(simpleSettings)) {
|
||||||
|
|
||||||
for (const [key, value] of Object.entries(allSettings)) {
|
|
||||||
const existingSetting = existingSettings.find(s => s.key === key);
|
const existingSetting = existingSettings.find(s => s.key === key);
|
||||||
|
|
||||||
let settingData = {
|
let settingData = {
|
||||||
key: key,
|
key: key,
|
||||||
title: typeof value === 'string' ? value : '',
|
title: value,
|
||||||
|
value: value,
|
||||||
category: 'walk_sheet_setting',
|
category: 'walk_sheet_setting',
|
||||||
updated_by: userEmail,
|
updated_by: userEmail,
|
||||||
updated_at: timestamp
|
updated_at: timestamp
|
||||||
};
|
};
|
||||||
|
|
||||||
// Handle different value types
|
|
||||||
if (key.includes('_image') && value) {
|
|
||||||
// For image attachments
|
|
||||||
settingData.value = JSON.stringify(value);
|
|
||||||
settingData[key] = value; // Also set the attachment field directly
|
|
||||||
} else {
|
|
||||||
settingData.value = value || '';
|
|
||||||
}
|
|
||||||
|
|
||||||
if (existingSetting) {
|
if (existingSetting) {
|
||||||
// Update existing
|
// Update existing - use ID from debug output
|
||||||
await axios.put(
|
logger.info(`Updating setting ${key} with ID ${existingSetting.ID}`);
|
||||||
`${process.env.NOCODB_API_URL}/api/v1/base/${process.env.NOCODB_PROJECT_ID}/tables/${SETTINGS_SHEET_ID}/rows/${existingSetting.Id}`,
|
await axios.patch(
|
||||||
|
`${process.env.NOCODB_API_URL}/db/data/v1/${process.env.NOCODB_PROJECT_ID}/${SETTINGS_SHEET_ID}/${existingSetting.ID}`,
|
||||||
settingData,
|
settingData,
|
||||||
{
|
{
|
||||||
headers: {
|
headers: {
|
||||||
'xc-auth': process.env.NOCODB_API_TOKEN,
|
'xc-token': process.env.NOCODB_API_TOKEN,
|
||||||
'Content-Type': 'application/json'
|
'Content-Type': 'application/json'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
// Create new
|
// Create new
|
||||||
|
logger.info(`Creating new setting ${key}`);
|
||||||
await axios.post(
|
await axios.post(
|
||||||
`${process.env.NOCODB_API_URL}/api/v1/base/${process.env.NOCODB_PROJECT_ID}/tables/${SETTINGS_SHEET_ID}/rows`,
|
`${process.env.NOCODB_API_URL}/db/data/v1/${process.env.NOCODB_PROJECT_ID}/${SETTINGS_SHEET_ID}`,
|
||||||
settingData,
|
settingData,
|
||||||
{
|
{
|
||||||
headers: {
|
headers: {
|
||||||
'xc-auth': process.env.NOCODB_API_TOKEN,
|
'xc-token': process.env.NOCODB_API_TOKEN,
|
||||||
'Content-Type': 'application/json'
|
'Content-Type': 'application/json'
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -870,19 +847,17 @@ app.post('/api/admin/walk-sheet-config', requireAdmin, async (req, res) => {
|
|||||||
res.json({
|
res.json({
|
||||||
success: true,
|
success: true,
|
||||||
message: 'Walk sheet configuration saved successfully',
|
message: 'Walk sheet configuration saved successfully',
|
||||||
qrCodes: Object.keys(qrCodeUploads).reduce((acc, key) => {
|
savedSettings: simpleSettings
|
||||||
if (qrCodeUploads[key]) {
|
|
||||||
acc[key] = qrCodeUploads[key];
|
|
||||||
}
|
|
||||||
return acc;
|
|
||||||
}, {})
|
|
||||||
});
|
});
|
||||||
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logger.error('Failed to save walk sheet config:', error);
|
logger.error('Failed to save walk sheet config:', error);
|
||||||
|
logger.error('Error response:', error.response?.data);
|
||||||
|
logger.error('Error config:', error.config?.url);
|
||||||
res.status(500).json({
|
res.status(500).json({
|
||||||
success: false,
|
success: false,
|
||||||
error: 'Failed to save walk sheet configuration'
|
error: 'Failed to save walk sheet configuration. No worries; just hit print, and you can save it there too!',
|
||||||
|
details: error.response?.data || error.message
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@ -1234,48 +1209,292 @@ app.delete('/api/locations/:id', strictLimiter, async (req, res) => {
|
|||||||
// Debug endpoint to check settings table structure
|
// Debug endpoint to check settings table structure
|
||||||
app.get('/api/debug/settings-table', requireAdmin, async (req, res) => {
|
app.get('/api/debug/settings-table', requireAdmin, async (req, res) => {
|
||||||
try {
|
try {
|
||||||
|
logger.info('Debug: SETTINGS_SHEET_ID =', SETTINGS_SHEET_ID);
|
||||||
|
logger.info('Debug: NOCODB_API_URL =', process.env.NOCODB_API_URL);
|
||||||
|
logger.info('Debug: NOCODB_PROJECT_ID =', process.env.NOCODB_PROJECT_ID);
|
||||||
|
|
||||||
if (!SETTINGS_SHEET_ID) {
|
if (!SETTINGS_SHEET_ID) {
|
||||||
return res.json({
|
return res.json({
|
||||||
error: 'Settings sheet not configured',
|
success: false,
|
||||||
|
error: 'SETTINGS_SHEET_ID not configured',
|
||||||
|
settingsSheetId: SETTINGS_SHEET_ID,
|
||||||
|
originalSetting: process.env.NOCODB_SETTINGS_SHEET
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Try the working endpoint
|
||||||
|
const workingEndpoint = `/db/data/v1/${process.env.NOCODB_PROJECT_ID}/${SETTINGS_SHEET_ID}`;
|
||||||
|
|
||||||
|
try {
|
||||||
|
const response = await axios.get(
|
||||||
|
`${process.env.NOCODB_API_URL}${workingEndpoint}`,
|
||||||
|
{
|
||||||
|
headers: {
|
||||||
|
'xc-token': process.env.NOCODB_API_TOKEN
|
||||||
|
},
|
||||||
|
params: {
|
||||||
|
limit: 5
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
const records = response.data.list || [];
|
||||||
|
const sampleRecord = records.length > 0 ? records[0] : null;
|
||||||
|
|
||||||
|
res.json({
|
||||||
|
success: true,
|
||||||
|
settingsSheetId: SETTINGS_SHEET_ID,
|
||||||
|
workingEndpoint: workingEndpoint,
|
||||||
|
recordCount: response.data.pageInfo?.totalRows || 0,
|
||||||
|
sampleRecord: sampleRecord,
|
||||||
|
availableFields: sampleRecord ? Object.keys(sampleRecord) : [],
|
||||||
|
allRecords: records
|
||||||
|
});
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
res.json({
|
||||||
|
success: false,
|
||||||
|
error: error.message,
|
||||||
|
responseData: error.response?.data,
|
||||||
|
status: error.response?.status,
|
||||||
settingsSheetId: SETTINGS_SHEET_ID
|
settingsSheetId: SETTINGS_SHEET_ID
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const url = `${process.env.NOCODB_API_URL}/db/data/v1/${process.env.NOCODB_PROJECT_ID}/${SETTINGS_SHEET_ID}`;
|
} catch (error) {
|
||||||
|
logger.error('Debug settings table error:', error);
|
||||||
// Try to get the table structure by fetching all records
|
res.status(500).json({
|
||||||
const response = await axios.get(url, {
|
success: false,
|
||||||
headers: {
|
error: error.message,
|
||||||
'xc-token': process.env.NOCODB_API_TOKEN,
|
settingsSheetId: SETTINGS_SHEET_ID
|
||||||
'Content-Type': 'application/json'
|
|
||||||
},
|
|
||||||
params: {
|
|
||||||
limit: 1
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Simple QR code test endpoint
|
||||||
|
app.get('/api/debug/test-qr', requireAdmin, async (req, res) => {
|
||||||
|
try {
|
||||||
|
const { generateAndUploadQRCode } = require('./services/qrcode');
|
||||||
|
|
||||||
const records = response.data.list || [];
|
// Test configuration
|
||||||
const sampleRecord = records.length > 0 ? records[0] : null;
|
const testConfig = {
|
||||||
|
apiUrl: process.env.NOCODB_API_URL,
|
||||||
|
apiToken: process.env.NOCODB_API_TOKEN,
|
||||||
|
projectId: process.env.NOCODB_PROJECT_ID,
|
||||||
|
tableId: SETTINGS_SHEET_ID
|
||||||
|
};
|
||||||
|
|
||||||
|
// Test QR code generation
|
||||||
|
const testUrl = 'https://example.com/test';
|
||||||
|
const testLabel = 'Test QR Code';
|
||||||
|
|
||||||
|
logger.info('Testing QR code generation...');
|
||||||
|
|
||||||
|
const result = await generateAndUploadQRCode(testUrl, testLabel, testConfig);
|
||||||
|
|
||||||
res.json({
|
res.json({
|
||||||
success: true,
|
success: true,
|
||||||
settingsSheetId: SETTINGS_SHEET_ID,
|
message: 'QR code generated successfully',
|
||||||
tableUrl: url,
|
result: result,
|
||||||
recordCount: response.data.pageInfo?.totalRows || 0,
|
testUrl: testUrl,
|
||||||
sampleRecord: sampleRecord,
|
testLabel: testLabel
|
||||||
availableFields: sampleRecord ? Object.keys(sampleRecord) : []
|
|
||||||
});
|
});
|
||||||
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logger.error('Error checking settings table:', error);
|
logger.error('QR code test failed:', error);
|
||||||
res.json({
|
res.status(500).json({
|
||||||
|
success: false,
|
||||||
error: error.message,
|
error: error.message,
|
||||||
response: error.response?.data,
|
details: error.response?.data || 'No response data'
|
||||||
status: error.response?.status
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Local QR code generation endpoint
|
||||||
|
app.get('/api/qr', async (req, res) => {
|
||||||
|
try {
|
||||||
|
const { text, size = 200 } = req.query;
|
||||||
|
|
||||||
|
if (!text) {
|
||||||
|
return res.status(400).json({
|
||||||
|
success: false,
|
||||||
|
error: 'Text parameter is required'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const { generateQRCode } = require('./services/qrcode');
|
||||||
|
|
||||||
|
const qrOptions = {
|
||||||
|
type: 'png',
|
||||||
|
width: parseInt(size),
|
||||||
|
margin: 1,
|
||||||
|
color: {
|
||||||
|
dark: '#000000',
|
||||||
|
light: '#FFFFFF'
|
||||||
|
},
|
||||||
|
errorCorrectionLevel: 'M'
|
||||||
|
};
|
||||||
|
|
||||||
|
const buffer = await generateQRCode(text, qrOptions);
|
||||||
|
|
||||||
|
res.set({
|
||||||
|
'Content-Type': 'image/png',
|
||||||
|
'Content-Length': buffer.length,
|
||||||
|
'Cache-Control': 'public, max-age=3600' // Cache for 1 hour
|
||||||
|
});
|
||||||
|
|
||||||
|
res.send(buffer);
|
||||||
|
|
||||||
|
} catch (error) {
|
||||||
|
logger.error('QR code generation error:', error);
|
||||||
|
res.status(500).json({
|
||||||
|
success: false,
|
||||||
|
error: 'Failed to generate QR code'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Simple QR test page
|
||||||
|
app.get('/test-qr', (req, res) => {
|
||||||
|
res.send(`
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<title>QR Code Test</title>
|
||||||
|
<style>
|
||||||
|
body { font-family: Arial, sans-serif; padding: 20px; }
|
||||||
|
.test-section { margin: 20px 0; padding: 20px; border: 1px solid #ddd; }
|
||||||
|
input { padding: 10px; width: 300px; margin-right: 10px; }
|
||||||
|
button { padding: 10px 20px; }
|
||||||
|
#qr-result { margin-top: 20px; }
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<h1>QR Code Generation Test</h1>
|
||||||
|
|
||||||
|
<div class="test-section">
|
||||||
|
<h3>Test 1: Direct API Call</h3>
|
||||||
|
<p>Try accessing these URLs directly:</p>
|
||||||
|
<ul>
|
||||||
|
<li><a href="/api/qr?text=Hello World" target="_blank">/api/qr?text=Hello World</a></li>
|
||||||
|
<li><a href="/api/qr?text=https://example.com&size=200" target="_blank">/api/qr?text=https://example.com&size=200</a></li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="test-section">
|
||||||
|
<h3>Test 2: Dynamic Generation</h3>
|
||||||
|
<input type="text" id="qr-text" placeholder="Enter text or URL" value="https://map.cmlite.org">
|
||||||
|
<button onclick="generateQR()">Generate QR Code</button>
|
||||||
|
<div id="qr-result"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="test-section">
|
||||||
|
<h3>Test 3: Using QRCode Library (like admin panel)</h3>
|
||||||
|
<input type="text" id="qr-text-canvas" placeholder="Enter text or URL" value="Test QR Code">
|
||||||
|
<button onclick="generateQRCanvas()">Generate QR Code (Canvas)</button>
|
||||||
|
<div id="qr-canvas-result"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// Test direct image loading
|
||||||
|
function generateQR() {
|
||||||
|
const text = document.getElementById('qr-text').value;
|
||||||
|
const resultDiv = document.getElementById('qr-result');
|
||||||
|
|
||||||
|
if (!text) {
|
||||||
|
resultDiv.innerHTML = '<p style="color: red;">Please enter some text</p>';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const qrUrl = \`/api/qr?text=\${encodeURIComponent(text)}&size=200\`;
|
||||||
|
resultDiv.innerHTML = \`
|
||||||
|
<p>QR Code for: <strong>\${text}</strong></p>
|
||||||
|
<img src="\${qrUrl}" alt="QR Code" style="border: 1px solid #ddd;">
|
||||||
|
<p><small>URL: <code>\${qrUrl}</code></small></p>
|
||||||
|
\`;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Test using our QRCode library implementation
|
||||||
|
function generateQRCanvas() {
|
||||||
|
const text = document.getElementById('qr-text-canvas').value;
|
||||||
|
const resultDiv = document.getElementById('qr-canvas-result');
|
||||||
|
|
||||||
|
if (!text) {
|
||||||
|
resultDiv.innerHTML = '<p style="color: red;">Please enter some text</p>';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create canvas element
|
||||||
|
const canvas = document.createElement('canvas');
|
||||||
|
resultDiv.innerHTML = \`<p>QR Code for: <strong>\${text}</strong></p>\`;
|
||||||
|
resultDiv.appendChild(canvas);
|
||||||
|
|
||||||
|
// Use our QRCode implementation (same as admin panel)
|
||||||
|
window.QRCode = {
|
||||||
|
toCanvas: function(canvas, text, options, callback) {
|
||||||
|
if (typeof options === 'function') {
|
||||||
|
callback = options;
|
||||||
|
options = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
const size = options.width || 200;
|
||||||
|
const qrUrl = \`/api/qr?text=\${encodeURIComponent(text)}&size=\${size}\`;
|
||||||
|
|
||||||
|
const img = new Image();
|
||||||
|
img.crossOrigin = 'anonymous';
|
||||||
|
|
||||||
|
img.onload = function() {
|
||||||
|
canvas.width = size;
|
||||||
|
canvas.height = size;
|
||||||
|
|
||||||
|
const ctx = canvas.getContext('2d');
|
||||||
|
ctx.drawImage(img, 0, 0, size, size);
|
||||||
|
|
||||||
|
if (callback) callback(null);
|
||||||
|
};
|
||||||
|
|
||||||
|
img.onerror = function() {
|
||||||
|
console.error('Failed to load QR code from server');
|
||||||
|
|
||||||
|
// Fallback: draw a simple placeholder
|
||||||
|
canvas.width = size;
|
||||||
|
canvas.height = size;
|
||||||
|
|
||||||
|
const ctx = canvas.getContext('2d');
|
||||||
|
ctx.fillStyle = '#ffffff';
|
||||||
|
ctx.fillRect(0, 0, size, size);
|
||||||
|
|
||||||
|
ctx.fillStyle = '#000000';
|
||||||
|
ctx.font = '12px Arial';
|
||||||
|
ctx.textAlign = 'center';
|
||||||
|
ctx.fillText('QR Code', size/2, size/2 - 10);
|
||||||
|
ctx.fillText('(Failed)', size/2, size/2 + 10);
|
||||||
|
|
||||||
|
if (callback) callback(new Error('Failed to load QR code'));
|
||||||
|
};
|
||||||
|
|
||||||
|
img.src = qrUrl;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
QRCode.toCanvas(canvas, text, { width: 200 }, function(error) {
|
||||||
|
if (error) {
|
||||||
|
console.error('QR generation error:', error);
|
||||||
|
resultDiv.innerHTML += '<p style="color: red;">Failed to generate QR code</p>';
|
||||||
|
} else {
|
||||||
|
console.log('QR code generated successfully');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Auto-generate on page load
|
||||||
|
generateQR();
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
`);
|
||||||
|
});
|
||||||
|
|
||||||
// Error handling middleware
|
// Error handling middleware
|
||||||
app.use((err, req, res, next) => {
|
app.use((err, req, res, next) => {
|
||||||
logger.error('Unhandled error:', err);
|
logger.error('Unhandled error:', err);
|
||||||
|
|||||||
@ -61,8 +61,14 @@ async function uploadQRCodeToNocoDB(buffer, filename, config) {
|
|||||||
});
|
});
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
// Use the base URL without /api/v1 for v2 endpoints
|
||||||
|
const baseUrl = config.apiUrl.replace('/api/v1', '');
|
||||||
|
const uploadUrl = `${baseUrl}/api/v2/storage/upload`;
|
||||||
|
|
||||||
|
logger.info(`Uploading QR code to: ${uploadUrl}`);
|
||||||
|
|
||||||
const response = await axios({
|
const response = await axios({
|
||||||
url: `${config.apiUrl}/api/v2/storage/upload`,
|
url: uploadUrl,
|
||||||
method: 'post',
|
method: 'post',
|
||||||
data: formData,
|
data: formData,
|
||||||
headers: {
|
headers: {
|
||||||
@ -74,9 +80,10 @@ async function uploadQRCodeToNocoDB(buffer, filename, config) {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
logger.info('QR code upload successful:', response.data);
|
||||||
return response.data;
|
return response.data;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
logger.error('Failed to upload QR code to NocoDB:', error);
|
logger.error('Failed to upload QR code to NocoDB:', error.response?.data || error.message);
|
||||||
throw new Error('Failed to upload QR code');
|
throw new Error('Failed to upload QR code');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user