block-detail.php 40 KB

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