block-detail.php 40 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708
  1. <?php
  2. error_reporting(E_ALL);
  3. ini_set('display_errors', 1);
  4. /**
  5. * dashboard/crop-cards/block-detail.php
  6. *
  7. * Paddock dashboard — per-block view showing:
  8. * - Paddock metadata
  9. * - Mini live weather (Open-Meteo via api/weather.php)
  10. * - Recent soil, plant, water tests linked to this block
  11. * - Latest sensor readings
  12. * - Quick-action links (add tests, view calendar)
  13. *
  14. * GET params:
  15. * rid — block_info.id (int)
  16. * id — block_info.block_id (string, secondary key)
  17. * block — display name (urldecoded)
  18. */
  19. require_once __DIR__ . '/../../config/database.php';
  20. require_once __DIR__ . '/../../lib/auth.php';
  21. require_once __DIR__ . '/../../lib/csrf.php';
  22. if (session_status() === PHP_SESSION_NONE) {
  23. session_start();
  24. }
  25. requireLogin();
  26. $rid = (int) ($_GET['rid'] ?? 0);
  27. $blockId = trim( $_GET['id'] ?? '');
  28. if ($rid <= 0) {
  29. http_response_code(400);
  30. die('Invalid request');
  31. }
  32. $pdo = getDBConnection();
  33. $userId = getCurrentUserId();
  34. $h = fn($v) => htmlspecialchars((string)($v ?? ''), ENT_QUOTES, 'UTF-8');
  35. // ── Load paddock ─────────────────────────────────────────────────────────────
  36. $stmt = $pdo->prepare('SELECT * FROM block_info WHERE id = ? AND modx_user_id = ? LIMIT 1');
  37. $stmt->execute([$rid, $userId]);
  38. $block = $stmt->fetch(PDO::FETCH_ASSOC);
  39. if (!$block) {
  40. http_response_code(404);
  41. die('Paddock not found or access denied');
  42. }
  43. $areaHa = number_format((float)$block['area'], 1);
  44. $areaAc = number_format((float)$block['area'] * 2.47105, 1);
  45. // ── Crop name(s) for this paddock ────────────────────────────────────────────
  46. $stmtCrops = $pdo->prepare(
  47. 'SELECT name FROM crop_info WHERE modx_user_id = ? AND paddock_id = ? ORDER BY id DESC'
  48. );
  49. $stmtCrops->execute([$userId, $block['block_id']]);
  50. $crops = $stmtCrops->fetchAll(PDO::FETCH_COLUMN);
  51. // ── Recent soil tests ─────────────────────────────────────────────────────────
  52. // Linked via site_id until migration 003 adds/confirms block_id column
  53. $stmtSoil = $pdo->prepare(
  54. 'SELECT id, rand, date, site_id, analysis_type, crop_type, client_name
  55. FROM soil_records
  56. WHERE modx_user_id = ? AND site_id = ?
  57. ORDER BY date DESC LIMIT 5'
  58. );
  59. $stmtSoil->execute([$userId, $block['block_id']]);
  60. $soilTests = $stmtSoil->fetchAll(PDO::FETCH_ASSOC);
  61. // ── Recent plant tests ────────────────────────────────────────────────────────
  62. // Linked via site_id until migration 003 adds block_id column to plant_records
  63. $stmtPlant = $pdo->prepare(
  64. 'SELECT id, rand, date_sampled AS date, site_id, crop_type, client_name
  65. FROM plant_records
  66. WHERE modx_user_id = ? AND site_id = ?
  67. ORDER BY date_sampled DESC LIMIT 5'
  68. );
  69. $stmtPlant->execute([$userId, $block['block_id']]);
  70. $plantTests = $stmtPlant->fetchAll(PDO::FETCH_ASSOC);
  71. // ── Recent water tests ────────────────────────────────────────────────────────
  72. // Linked via site_id until migration 003 adds block_id column to water_records
  73. $stmtWater = $pdo->prepare(
  74. 'SELECT id, rand, date_sampled AS date, site_id, analysis_type, client_name
  75. FROM water_records
  76. WHERE modx_user_id = ? AND site_id = ?
  77. ORDER BY date_sampled DESC LIMIT 5'
  78. );
  79. $stmtWater->execute([$userId, $block['block_id']]);
  80. $waterTests = $stmtWater->fetchAll(PDO::FETCH_ASSOC);
  81. // ── Sensor readings (latest per sensor) ──────────────────────────────────────
  82. $stmtSensors = $pdo->prepare(
  83. 'SELECT fs.sensor_id, fs.sensor_name, fs.value, fs.DATEUTC
  84. FROM field_sensors fs
  85. INNER JOIN (
  86. SELECT sensor_id, MAX(DATEUTC) AS latest
  87. FROM field_sensors
  88. WHERE modx_user_id = ?
  89. GROUP BY sensor_id
  90. ) latest ON fs.sensor_id = latest.sensor_id AND fs.DATEUTC = latest.latest
  91. WHERE fs.modx_user_id = ?
  92. ORDER BY fs.sensor_name'
  93. );
  94. $stmtSensors->execute([$userId, $userId]);
  95. $sensors = $stmtSensors->fetchAll(PDO::FETCH_ASSOC);
  96. // ── GPS lat/lng for weather override ─────────────────────────────────────────
  97. $weatherLat = $weatherLng = null;
  98. if (!empty($block['gps'])) {
  99. if (preg_match('/(-?\d+\.?\d*)\s*,\s*(-?\d+\.?\d*)/', $block['gps'], $m)) {
  100. $weatherLat = (float)$m[1];
  101. $weatherLng = (float)$m[2];
  102. }
  103. }
  104. $weatherUrl = '/api/weather.php' . ($weatherLat ? '?lat=' . $weatherLat . '&lng=' . $weatherLng : '');
  105. $pageTitle = $h($block['name']) . ' — Paddock Dashboard';
  106. $siteName = 'Crop Monitor';
  107. include __DIR__ . '/../../layouts/header.php';
  108. include __DIR__ . '/../../layouts/navbar.php';
  109. ?>
  110. <div id="layoutSidenav">
  111. <div id="layoutSidenav_nav">
  112. <?php include __DIR__ . '/../../layouts/sidebar.php'; ?>
  113. </div>
  114. <div id="layoutSidenav_content">
  115. <main>
  116. <div class="container-fluid px-4">
  117. <!-- ── Breadcrumb ──────────────────────────────────────────── -->
  118. <div class="d-flex align-items-center justify-content-between mt-4 mb-1">
  119. <div>
  120. <h1 class="h3 mb-0">
  121. <i class="fas fa-seedling text-success me-2"></i><?= $h($block['name']) ?>
  122. </h1>
  123. <ol class="breadcrumb mb-0 small">
  124. <li class="breadcrumb-item"><a href="/dashboard/dashboard.php">Dashboard</a></li>
  125. <li class="breadcrumb-item"><a href="/dashboard/crop-cards/">Crop Cards</a></li>
  126. <li class="breadcrumb-item active"><?= $h($block['name']) ?></li>
  127. </ol>
  128. </div>
  129. <div class="d-flex gap-2 flex-wrap">
  130. <a href="/dashboard/crop-cards/" class="btn btn-outline-secondary btn-sm">
  131. <i class="fas fa-arrow-left me-1"></i>All Paddocks
  132. </a>
  133. <button class="btn btn-outline-primary btn-sm"
  134. data-bs-toggle="modal" data-bs-target="#editPaddockModal">
  135. <i class="fas fa-edit me-1"></i>Edit
  136. </button>
  137. </div>
  138. </div>
  139. <hr class="mb-3">
  140. <!-- ── Top row: paddock info + weather ────────────────────── -->
  141. <div class="row g-3 mb-4">
  142. <!-- Paddock info card -->
  143. <div class="col-md-4 col-lg-3">
  144. <div class="card h-100 border-success">
  145. <div class="card-header bg-success text-white py-2 fw-bold small">
  146. <i class="fas fa-map-marked-alt me-1"></i>Paddock Details
  147. </div>
  148. <div class="card-body py-2 small">
  149. <table class="table table-sm table-borderless mb-0">
  150. <tr>
  151. <th class="text-muted pe-2 text-nowrap">Block ID</th>
  152. <td><?= $h($block['block_id']) ?></td>
  153. </tr>
  154. <tr>
  155. <th class="text-muted pe-2 text-nowrap">Location</th>
  156. <td><?= $h($block['location']) ?: '—' ?></td>
  157. </tr>
  158. <tr>
  159. <th class="text-muted pe-2 text-nowrap">Area</th>
  160. <td><?= $areaHa ?> ha &nbsp;/&nbsp; <?= $areaAc ?> ac</td>
  161. </tr>
  162. <tr>
  163. <th class="text-muted pe-2 text-nowrap">GPS</th>
  164. <td>
  165. <?php if (!empty($block['gps'])): ?>
  166. <a href="https://maps.google.com/?q=<?= urlencode($block['gps']) ?>"
  167. target="_blank" rel="noopener"
  168. class="text-decoration-none">
  169. <i class="fas fa-map-pin text-danger me-1"></i><?= $h($block['gps']) ?>
  170. </a>
  171. <?php else: ?>
  172. <span class="text-muted">—</span>
  173. <?php endif; ?>
  174. </td>
  175. </tr>
  176. <tr>
  177. <th class="text-muted pe-2 text-nowrap">Crops</th>
  178. <td>
  179. <?php if ($crops): ?>
  180. <?= $h(implode(', ', $crops)) ?>
  181. <?php else: ?>
  182. <span class="text-muted">None recorded</span>
  183. <?php endif; ?>
  184. </td>
  185. </tr>
  186. <tr>
  187. <th class="text-muted pe-2 text-nowrap">Soil Type</th>
  188. <td><?= !empty($block['analysis_type'] ?? '') ? ucfirst($h($block['analysis_type'])) : '<span class="text-muted">—</span>' ?></td>
  189. </tr>
  190. <tr>
  191. <th class="text-muted pe-2 text-nowrap">Added</th>
  192. <td><?= $h($block['date_added']) ?: '—' ?></td>
  193. </tr>
  194. </table>
  195. </div>
  196. </div>
  197. </div>
  198. <!-- Weather card -->
  199. <div class="col-md-8 col-lg-5">
  200. <div class="card h-100">
  201. <div class="card-header py-2 fw-bold small">
  202. <i class="fas fa-cloud-sun me-1 text-warning"></i>Current Weather
  203. <span class="text-muted fw-normal ms-1 small" id="wx-pd-location"></span>
  204. </div>
  205. <div class="card-body py-2">
  206. <!-- Skeleton -->
  207. <div id="wx-pd-loading" class="d-flex align-items-center justify-content-center py-3">
  208. <span class="spinner-border spinner-border-sm me-2 text-secondary"></span>
  209. <span class="text-muted small">Loading weather…</span>
  210. </div>
  211. <!-- Content -->
  212. <div id="wx-pd-content" style="display:none;">
  213. <div class="d-flex align-items-center gap-3 mb-2">
  214. <canvas id="wx-pd-icon" width="64" height="64"></canvas>
  215. <div>
  216. <div class="display-6 lh-1 fw-bold" id="wx-pd-temp">—</div>
  217. <div class="text-muted" id="wx-pd-condition">—</div>
  218. </div>
  219. <div class="ms-auto text-end small text-muted">
  220. <div><i class="fa fa-tint"></i> <span id="wx-pd-humidity">—</span>% humidity</div>
  221. <div><i class="fa fa-wind"></i> <span id="wx-pd-wind">—</span> km/h wind</div>
  222. <div><i class="fa fa-cloud-rain"></i> <span id="wx-pd-rain">—</span> mm now</div>
  223. <div><i class="fa fa-thermometer-half"></i> Feels <span id="wx-pd-feels">—</span>°</div>
  224. </div>
  225. </div>
  226. <!-- 5-day mini forecast strip -->
  227. <div class="d-flex gap-2 overflow-auto pb-1" id="wx-pd-forecast">
  228. <!-- filled by JS -->
  229. </div>
  230. </div>
  231. <div id="wx-pd-error" class="alert alert-warning small py-1 mb-0" style="display:none;"></div>
  232. </div>
  233. </div>
  234. </div>
  235. <!-- Quick actions card -->
  236. <div class="col-md-12 col-lg-4">
  237. <div class="card h-100">
  238. <div class="card-header py-2 fw-bold small">
  239. <i class="fas fa-bolt me-1 text-warning"></i>Quick Actions
  240. </div>
  241. <div class="card-body py-2">
  242. <div class="d-grid gap-2">
  243. <a href="/dashboard/crop-analysis/soil-test-data/soil-test-data.php?block_id=<?= urlencode($block['block_id']) ?>"
  244. class="btn btn-outline-success btn-sm text-start">
  245. <i class="fas fa-vial me-2 text-success"></i>New Soil Test
  246. </a>
  247. <a href="/dashboard/crop-analysis/plant-test-data/plant-test-data.php?block_id=<?= urlencode($block['block_id']) ?>"
  248. class="btn btn-outline-primary btn-sm text-start">
  249. <i class="fas fa-leaf me-2 text-primary"></i>New Plant Test
  250. </a>
  251. <a href="/dashboard/crop-analysis/water-test-data/water-test-data.php?block_id=<?= urlencode($block['block_id']) ?>"
  252. class="btn btn-outline-info btn-sm text-start">
  253. <i class="fas fa-tint me-2 text-info"></i>New Water Test
  254. </a>
  255. <a href="/dashboard/planning-calendar.php?paddock=<?= urlencode($block['block_id']) ?>"
  256. class="btn btn-outline-secondary btn-sm text-start">
  257. <i class="fas fa-calendar me-2 text-secondary"></i>View Calendar
  258. </a>
  259. <a href="/dashboard/inbox.php"
  260. class="btn btn-outline-dark btn-sm text-start">
  261. <i class="fas fa-inbox me-2"></i>Inbox / Reports
  262. </a>
  263. </div>
  264. </div>
  265. </div>
  266. </div>
  267. </div><!-- /top row -->
  268. <!-- ── Rainfall history chart ──────────────────────────────── -->
  269. <div class="row g-3 mb-4">
  270. <div class="col-12">
  271. <div class="card" id="wx-pd-rainfall-card" style="display:none;">
  272. <div class="card-header py-2 fw-bold small">
  273. <i class="fas fa-chart-bar me-1 text-info"></i>Past 7 Days Rainfall (mm)
  274. </div>
  275. <div class="card-body py-2">
  276. <canvas id="wx-pd-rainfall-chart" height="60"></canvas>
  277. </div>
  278. </div>
  279. </div>
  280. </div>
  281. <!-- ── Sensor readings ─────────────────────────────────────── -->
  282. <?php if ($sensors): ?>
  283. <div class="row g-3 mb-4">
  284. <div class="col-12">
  285. <h5 class="mb-2"><i class="fas fa-satellite-dish me-2 text-secondary"></i>Live Sensor Readings</h5>
  286. </div>
  287. <?php foreach ($sensors as $sensor): ?>
  288. <div class="col-6 col-sm-4 col-md-3 col-xl-2">
  289. <div class="card border-0 shadow-sm text-center">
  290. <div class="card-body py-2 px-2">
  291. <div class="small text-muted text-truncate" title="<?= $h($sensor['sensor_name']) ?>">
  292. <?= $h($sensor['sensor_name'] ?: $sensor['sensor_id']) ?>
  293. </div>
  294. <div class="h4 mb-0 fw-bold"><?= number_format((float)$sensor['value'], 1) ?></div>
  295. <div class="text-muted" style="font-size:0.68rem;">
  296. <?= $h(date('j M H:i', strtotime($sensor['DATEUTC']))) ?>
  297. </div>
  298. </div>
  299. </div>
  300. </div>
  301. <?php endforeach; ?>
  302. </div>
  303. <?php endif; ?>
  304. <!-- ── Analysis tabs ──────────────────────────────────────── -->
  305. <div class="row mb-4">
  306. <div class="col-12">
  307. <h5 class="mb-2"><i class="fas fa-flask me-2 text-success"></i>Analysis Records</h5>
  308. <ul class="nav nav-tabs mb-0" id="analysisTabs">
  309. <li class="nav-item">
  310. <a class="nav-link active" href="#tab-soil" data-bs-toggle="list">
  311. <i class="fas fa-vial me-1 text-success"></i>Soil
  312. <span class="badge bg-success ms-1"><?= count($soilTests) ?></span>
  313. </a>
  314. </li>
  315. <li class="nav-item">
  316. <a class="nav-link" href="#tab-plant" data-bs-toggle="list">
  317. <i class="fas fa-leaf me-1 text-primary"></i>Plant
  318. <span class="badge bg-primary ms-1"><?= count($plantTests) ?></span>
  319. </a>
  320. </li>
  321. <li class="nav-item">
  322. <a class="nav-link" href="#tab-water" data-bs-toggle="list">
  323. <i class="fas fa-tint me-1 text-info"></i>Water
  324. <span class="badge bg-info ms-1"><?= count($waterTests) ?></span>
  325. </a>
  326. </li>
  327. </ul>
  328. <div class="tab-content border border-top-0 rounded-bottom p-3">
  329. <!-- Soil tests tab -->
  330. <div class="tab-pane fade show active" id="tab-soil">
  331. <?php if (empty($soilTests)): ?>
  332. <p class="text-muted mb-0 small">No soil tests recorded for block ID <strong><?= $h($block['block_id']) ?></strong>.</p>
  333. <?php else: ?>
  334. <div class="table-responsive">
  335. <table class="table table-sm table-hover align-middle mb-0">
  336. <thead class="table-light">
  337. <tr>
  338. <th>Date</th>
  339. <th>Site ID</th>
  340. <th>Client</th>
  341. <th>Soil Type</th>
  342. <th>Crop</th>
  343. <th class="text-end">Actions</th>
  344. </tr>
  345. </thead>
  346. <tbody>
  347. <?php foreach ($soilTests as $t): ?>
  348. <tr>
  349. <td><?= $h($t['date'] ? date('j M Y', strtotime($t['date'])) : '—') ?></td>
  350. <td><?= $h($t['site_id']) ?></td>
  351. <td><?= $h($t['client_name']) ?></td>
  352. <td><?= $h($t['analysis_type']) ?></td>
  353. <td><?= $h($t['crop_type']) ?></td>
  354. <td class="text-end text-nowrap">
  355. <a href="/dashboard/crop-analysis/soil-test-data/soil-analysis.php?rid=<?= $t['id'] ?>&rand=<?= urlencode($t['rand']) ?>"
  356. class="btn btn-outline-success btn-sm py-0 px-2" title="View Analysis">
  357. <i class="fas fa-chart-bar"></i>
  358. </a>
  359. <a href="/dashboard/crop-analysis/soil-test-data/soil-report.php?rid=<?= $t['id'] ?>&rand=<?= urlencode($t['rand']) ?>"
  360. class="btn btn-outline-primary btn-sm py-0 px-2" title="View Report">
  361. <i class="fas fa-file-alt"></i>
  362. </a>
  363. <a href="/pdf-files/headlessChrome_pdf.php?type=soil&rid=<?= $t['id'] ?>&rand=<?= urlencode($t['rand']) ?>"
  364. class="btn btn-outline-secondary btn-sm py-0 px-2" title="Download PDF">
  365. <i class="fas fa-file-pdf"></i>
  366. </a>
  367. </td>
  368. </tr>
  369. <?php endforeach; ?>
  370. </tbody>
  371. </table>
  372. </div>
  373. <?php endif; ?>
  374. </div>
  375. <!-- Plant tests tab -->
  376. <div class="tab-pane fade" id="tab-plant">
  377. <?php if (empty($plantTests)): ?>
  378. <p class="text-muted mb-0 small">No plant tissue tests recorded for block ID <strong><?= $h($block['block_id']) ?></strong>.</p>
  379. <?php else: ?>
  380. <div class="table-responsive">
  381. <table class="table table-sm table-hover align-middle mb-0">
  382. <thead class="table-light">
  383. <tr>
  384. <th>Date</th>
  385. <th>Site ID</th>
  386. <th>Client</th>
  387. <th>Crop</th>
  388. <th class="text-end">Actions</th>
  389. </tr>
  390. </thead>
  391. <tbody>
  392. <?php foreach ($plantTests as $t): ?>
  393. <tr>
  394. <td><?= $h($t['date'] ? date('j M Y', strtotime($t['date'])) : '—') ?></td>
  395. <td><?= $h($t['site_id']) ?></td>
  396. <td><?= $h($t['client_name']) ?></td>
  397. <td><?= $h($t['crop_type']) ?></td>
  398. <td class="text-end text-nowrap">
  399. <a href="/dashboard/crop-analysis/plant-test-data/plant-analysis.php?rid=<?= $t['id'] ?>&rand=<?= urlencode($t['rand']) ?>"
  400. class="btn btn-outline-primary btn-sm py-0 px-2" title="View Analysis">
  401. <i class="fas fa-chart-bar"></i>
  402. </a>
  403. <a href="/dashboard/crop-analysis/plant-test-data/plant-report.php?rid=<?= $t['id'] ?>&rand=<?= urlencode($t['rand']) ?>"
  404. class="btn btn-outline-success btn-sm py-0 px-2" title="View Report">
  405. <i class="fas fa-file-alt"></i>
  406. </a>
  407. <a href="/pdf-files/headlessChrome_pdf.php?type=plant&rid=<?= $t['id'] ?>&rand=<?= urlencode($t['rand']) ?>"
  408. class="btn btn-outline-secondary btn-sm py-0 px-2" title="Download PDF">
  409. <i class="fas fa-file-pdf"></i>
  410. </a>
  411. </td>
  412. </tr>
  413. <?php endforeach; ?>
  414. </tbody>
  415. </table>
  416. </div>
  417. <?php endif; ?>
  418. </div>
  419. <!-- Water tests tab -->
  420. <div class="tab-pane fade" id="tab-water">
  421. <?php if (empty($waterTests)): ?>
  422. <p class="text-muted mb-0 small">No water quality tests recorded for block ID <strong><?= $h($block['block_id']) ?></strong>.</p>
  423. <?php else: ?>
  424. <div class="table-responsive">
  425. <table class="table table-sm table-hover align-middle mb-0">
  426. <thead class="table-light">
  427. <tr>
  428. <th>Date</th>
  429. <th>Site ID</th>
  430. <th>Client</th>
  431. <th>Analysis Type</th>
  432. <th class="text-end">Actions</th>
  433. </tr>
  434. </thead>
  435. <tbody>
  436. <?php foreach ($waterTests as $t): ?>
  437. <tr>
  438. <td><?= $h($t['date'] ? date('j M Y', strtotime($t['date'])) : '—') ?></td>
  439. <td><?= $h($t['site_id']) ?></td>
  440. <td><?= $h($t['client_name']) ?></td>
  441. <td><?= $h($t['analysis_type']) ?></td>
  442. <td class="text-end text-nowrap">
  443. <a href="/dashboard/crop-analysis/water-test-data/water-analysis.php?rid=<?= $t['id'] ?>&rand=<?= urlencode($t['rand']) ?>"
  444. class="btn btn-outline-info btn-sm py-0 px-2" title="View Analysis">
  445. <i class="fas fa-chart-bar"></i>
  446. </a>
  447. <a href="/pdf-files/headlessChrome_pdf.php?type=water&rid=<?= $t['id'] ?>&rand=<?= urlencode($t['rand']) ?>"
  448. class="btn btn-outline-secondary btn-sm py-0 px-2" title="Download PDF">
  449. <i class="fas fa-file-pdf"></i>
  450. </a>
  451. </td>
  452. </tr>
  453. <?php endforeach; ?>
  454. </tbody>
  455. </table>
  456. </div>
  457. <?php endif; ?>
  458. </div>
  459. </div><!-- /tab-content -->
  460. </div>
  461. </div><!-- /analysis tabs row -->
  462. </div><!-- /container-fluid -->
  463. </main>
  464. <!-- ── Edit Paddock modal ──────────────────────────────────────────── -->
  465. <div class="modal fade" id="editPaddockModal" tabindex="-1" aria-hidden="true">
  466. <div class="modal-dialog modal-lg modal-dialog-centered">
  467. <div class="modal-content">
  468. <div class="modal-header">
  469. <h5 class="modal-title">Edit Paddock — <?= $h($block['name']) ?></h5>
  470. <button type="button" class="btn-close" data-bs-dismiss="modal"></button>
  471. </div>
  472. <form method="post" action="/controllers/blockSubmit.php">
  473. <div class="modal-body">
  474. <input type="hidden" name="csrf_token"
  475. value="<?= $h(generateCsrfToken()) ?>">
  476. <input type="hidden" name="action" value="edit">
  477. <input type="hidden" name="record_id" value="<?= $rid ?>">
  478. <input type="hidden" name="_referer" value="block-detail.php">
  479. <div class="row mb-3">
  480. <div class="col">
  481. <label class="form-label">Block ID</label>
  482. <input type="text" class="form-control form-control-sm"
  483. name="block_id" value="<?= $h($block['block_id']) ?>" required>
  484. </div>
  485. <div class="col">
  486. <label class="form-label">Block Name</label>
  487. <input type="text" class="form-control form-control-sm"
  488. name="name" value="<?= $h($block['name']) ?>" required>
  489. </div>
  490. </div>
  491. <div class="mb-3">
  492. <label class="form-label">Location / Address</label>
  493. <input type="text" class="form-control form-control-sm"
  494. name="location" value="<?= $h($block['location']) ?>">
  495. </div>
  496. <div class="row mb-3">
  497. <div class="col">
  498. <label class="form-label">Area (hectares)</label>
  499. <input type="number" step="0.01" class="form-control form-control-sm"
  500. id="ep_area_ha" name="area_ha"
  501. value="<?= $h(number_format((float)$block['area'], 2)) ?>"
  502. oninput="epAreaConvert('ha', this.value)">
  503. </div>
  504. <div class="col">
  505. <label class="form-label">Area (acres)</label>
  506. <input type="number" step="0.01" class="form-control form-control-sm"
  507. id="ep_area_ac"
  508. value="<?= $h(number_format((float)$block['area'] * 2.47105, 2)) ?>"
  509. oninput="epAreaConvert('ac', this.value)">
  510. </div>
  511. <div class="col">
  512. <label class="form-label">GPS Coordinates</label>
  513. <input type="text" class="form-control form-control-sm"
  514. name="gps" value="<?= $h($block['gps']) ?>"
  515. placeholder="e.g. -33.8688, 151.2093">
  516. </div>
  517. </div>
  518. <div>
  519. <label class="form-label">Soil Type</label>
  520. <select class="form-select form-select-sm" name="analysis_type">
  521. <option value="">Select soil type...</option>
  522. <?php foreach (['sandy','light','medium','heavy'] as $st): ?>
  523. <option value="<?= $st ?>"><?= ucfirst($st) ?></option>
  524. <?php endforeach; ?>
  525. </select>
  526. </div>
  527. </div>
  528. <div class="modal-footer">
  529. <button type="button" class="btn btn-secondary" data-bs-dismiss="modal">Close</button>
  530. <button type="submit" class="btn btn-success">Save Changes</button>
  531. </div>
  532. </form>
  533. </div>
  534. </div>
  535. </div>
  536. <script>
  537. (function () {
  538. 'use strict';
  539. // ── Edit modal area converter ──────────────────────────────────────────── //
  540. window.epAreaConvert = function (source, val) {
  541. val = parseFloat(val) || 0;
  542. var haEl = document.getElementById('ep_area_ha');
  543. var acEl = document.getElementById('ep_area_ac');
  544. if (source === 'ha') { acEl.value = (val * 2.47105).toFixed(2); }
  545. else { haEl.value = (val / 2.47105).toFixed(2); }
  546. };
  547. // ── Weather ────────────────────────────────────────────────────────────── //
  548. var wxSkycons = null;
  549. var wxFcSkycons = null;
  550. var rainfallChart = null;
  551. function renderWeather(data) {
  552. document.getElementById('wx-pd-temp').textContent = data.current.temp + '°';
  553. document.getElementById('wx-pd-condition').textContent = data.current.label;
  554. document.getElementById('wx-pd-humidity').textContent = data.current.humidity;
  555. document.getElementById('wx-pd-wind').textContent = data.current.wind;
  556. document.getElementById('wx-pd-rain').textContent = data.current.rain;
  557. document.getElementById('wx-pd-feels').textContent = data.current.feels_like;
  558. document.getElementById('wx-pd-location').textContent = '— ' + data.location;
  559. // Hero icon
  560. if (!wxSkycons) { wxSkycons = new Skycons({ color: '#1ABC9C' }); }
  561. wxSkycons.set(document.getElementById('wx-pd-icon'), data.current.icon);
  562. wxSkycons.play();
  563. // 5-day forecast strip
  564. var futureDays = data.days.filter(function (d) { return !d.is_past && !d.is_today; }).slice(0, 5);
  565. var forecastEl = document.getElementById('wx-pd-forecast');
  566. var html = '';
  567. if (!wxFcSkycons) { wxFcSkycons = new Skycons({ color: '#888' }); }
  568. futureDays.forEach(function (d, i) {
  569. var cid = 'wx-pd-fc-' + i;
  570. html +=
  571. '<div class="text-center border rounded px-2 py-1" style="min-width:70px;">' +
  572. '<div class="small fw-bold">' + d.day_name + '</div>' +
  573. '<canvas id="' + cid + '" width="40" height="40"></canvas>' +
  574. '<div class="small">' + d.temp_max + '°</div>' +
  575. '<div class="text-muted" style="font-size:0.65rem;">' + d.rain + ' mm</div>' +
  576. '</div>';
  577. });
  578. forecastEl.innerHTML = html;
  579. futureDays.forEach(function (d, i) {
  580. var el = document.getElementById('wx-pd-fc-' + i);
  581. if (el) { wxFcSkycons.set(el, d.icon); }
  582. });
  583. wxFcSkycons.play();
  584. // Rainfall history chart
  585. var pastDays = data.days.filter(function (d) { return d.is_past; }).slice(-7);
  586. if (pastDays.length > 0) {
  587. var labels = pastDays.map(function (d) { return d.day_name; });
  588. var values = pastDays.map(function (d) { return d.rain; });
  589. var ctx = document.getElementById('wx-pd-rainfall-chart').getContext('2d');
  590. if (rainfallChart) { rainfallChart.destroy(); }
  591. rainfallChart = new Chart(ctx, {
  592. type: 'bar',
  593. data: {
  594. labels: labels,
  595. datasets: [{
  596. label: 'mm',
  597. data: values,
  598. backgroundColor: 'rgba(54,162,235,0.6)',
  599. borderColor: 'rgba(54,162,235,1)',
  600. borderWidth: 1,
  601. }],
  602. },
  603. options: {
  604. responsive: true,
  605. plugins: { legend: { display: false } },
  606. scales: {
  607. y: { beginAtZero: true, ticks: { font: { size: 10 } } },
  608. x: { ticks: { font: { size: 10 } } },
  609. },
  610. },
  611. });
  612. document.getElementById('wx-pd-rainfall-card').style.display = '';
  613. }
  614. document.getElementById('wx-pd-loading').style.display = 'none';
  615. document.getElementById('wx-pd-content').style.display = '';
  616. }
  617. fetch(<?= json_encode($weatherUrl) ?>)
  618. .then(function (r) {
  619. if (!r.ok) { throw new Error('HTTP ' + r.status); }
  620. return r.json();
  621. })
  622. .then(function (data) {
  623. if (data.error) { throw new Error(data.error); }
  624. renderWeather(data);
  625. })
  626. .catch(function (err) {
  627. document.getElementById('wx-pd-loading').style.display = 'none';
  628. var el = document.getElementById('wx-pd-error');
  629. el.textContent = 'Weather unavailable: ' + err.message;
  630. el.style.display = '';
  631. });
  632. // ── Bootstrap tab fix: nav-tabs need data-bs-toggle="tab" ─────────────── //
  633. document.querySelectorAll('#analysisTabs .nav-link').forEach(function (el) {
  634. el.addEventListener('click', function (e) {
  635. e.preventDefault();
  636. document.querySelectorAll('#analysisTabs .nav-link').forEach(function (x) {
  637. x.classList.remove('active');
  638. });
  639. document.querySelectorAll('.tab-pane').forEach(function (x) {
  640. x.classList.remove('show', 'active');
  641. });
  642. el.classList.add('active');
  643. var target = document.querySelector(el.getAttribute('href'));
  644. if (target) { target.classList.add('show', 'active'); }
  645. });
  646. });
  647. })();
  648. </script>
  649. <?php include __DIR__ . '/../../layouts/footer.php'; ?>