block-detail.php 40 KB

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