Compare commits

...

2 Commits

Author SHA1 Message Date
8ed8d3bb42 Telemetria: Raporty 2026-06-18 08:41:49 +02:00
a9327760d0 Telemetria: INIT i wstępna makieta 2026-06-18 07:20:14 +02:00
10 changed files with 1895 additions and 0 deletions

80
header-telemetria.php Normal file
View File

@@ -0,0 +1,80 @@
<?php
$basePath = '';
if ($_SERVER['HTTP_HOST'] === 'localhost' || $_SERVER['HTTP_HOST'] === '127.0.0.1') {
$basePath = '/magico-prototype';
}
?>
<!DOCTYPE html>
<html lang="en" class="light-style layout-navbar-fixed layout-menu-fixed layout-compact" dir="ltr"
data-theme="theme-default" data-assets-path="<?= $basePath ?>/assets/"
data-template="vertical-menu-template-no-customizer-starter">
<head>
<meta charset="utf-8" />
<meta name="viewport"
content="width=device-width, initial-scale=1.0, user-scalable=no, minimum-scale=1.0, maximum-scale=1.0" />
<title>telemetria.magico</title>
<meta name="description" content="" />
<!-- Favicon -->
<link rel="icon" type="image/x-icon" href="<?= $basePath ?>/assets/img/favicon/favicon.ico" />
<!-- Fonts -->
<link rel="preconnect" href="https://fonts.googleapis.com" />
<link rel="preconnect" href="https://fonts.gstatic.com" crossorigin />
<link
href="https://fonts.googleapis.com/css2?family=Public+Sans:ital,wght@0,300;0,400;0,500;0,600;0,700;1,300;1,400;1,500;1,600;1,700&display=swap"
rel="stylesheet" />
<link rel="stylesheet" href="<?= $basePath ?>/assets/vendor/fonts/boxicons.css" />
<!-- <link rel="stylesheet" href="<?= $basePath ?>/assets/vendor/fonts/fontawesome.css" /> -->
<!-- <link rel="stylesheet" href="<?= $basePath ?>/assets/vendor/fonts/flag-icons.css" /> -->
<!-- Core CSS -->
<link rel="stylesheet" href="<?= $basePath ?>/assets/vendor/css/rtl/core.css" />
<link rel="stylesheet" href="<?= $basePath ?>/assets/vendor/css/rtl/theme-default.css" />
<link rel="stylesheet" href="<?= $basePath ?>/assets/css/demo.css" />
<!-- Vendors CSS -->
<link rel="stylesheet" href="<?= $basePath ?>/assets/vendor/libs/perfect-scrollbar/perfect-scrollbar.css" />
<!-- Page CSS -->
<!-- Helpers -->
<script src="<?= $basePath ?>/assets/vendor/js/helpers.js"></script>
<!--! Template customizer & Theme config files MUST be included after core stylesheets and helpers.js in the <head> section -->
<!--? Config: Mandatory theme config file contain global vars & default theme options, Set your preferred theme option in this file. -->
<script src="<?= $basePath ?>/assets/js/config.js"></script>
<?php if (!empty($enablePrototypeComments) && $enablePrototypeComments): ?>
<link rel="stylesheet" href="<?= $basePath ?>/assets/css/comments.css">
<?php endif; ?>
<style>
@media (min-width: 1200px) {
.layout-menu-fixed .layout-menu,
.layout-menu-fixed-offcanvas .layout-menu {
padding-top: 50px;
}
}
</style>
</head>
<body>
<!-- Layout wrapper -->
<div class="layout-wrapper layout-content-navbar">
<div class="layout-container">
<!-- Menu -->
<?php include 'menu.php'; ?>
<!-- / Menu -->
<!-- Layout container -->
<div class="layout-page pt-0" style="padding-top: 0 !important;">
<!-- Content wrapper -->
<div class="content-wrapper">

View File

@@ -0,0 +1,196 @@
Opis Biznesowy Aplikacji: telemetry.magico (Telemetry Manager)
1. Wstęp i Cel Projektu
telemetry.magico to nowoczesna, przemysłowa platforma klasy IoT (Internet of Things) oraz czasu rzeczywistego (Real-Time Monitoring), zintegrowana z ekosystemem biznesowym magico.pro.
Głównym celem aplikacji jest agregacja, wizualizacja oraz analiza danych telemetrycznych pochodzących z rozproszonych urządzeń pomiarowych, sensorów i sterowników przemysłowych. Kluczowym założeniem biznesowym jest pełna agnostyczność danych (generyczność) system projektowany jest w taki sposób, aby rodzaj monitorowanej infrastruktury (np. poziom gazu w zbiornikach, mikroklimat w kurnikach, parametry pracy maszyn HVAC) nie wpływał na strukturę rdzenia aplikacji, a jedynie na warstwę prezentacji danych i konfiguracji urządzeń.
2. Grupa Docelowa i Zastosowanie Biznesowe
Aplikacja odpowiada na potrzeby przedsiębiorstw zarządzających rozproszoną infrastrukturą techniczną. Przykładowe scenariusze wdrożeniowe obejmują:
Sektor Energetyczny / Gazowy (Pierwsze wdrożenie): Monitoring poziomu napełnienia zbiorników LPG/CNG, ciśnienia oraz detekcja anomalii i wycieków. Optimale planowanie logistyki dostaw surowca.
Agrotechnika (Smart Farming): Monitorowanie temperatury, wilgotności, poziomu CO2 czy zużycia paszy w obiektach inwentarskich (kurniki, chłodnie).
Smart Building & Industry: Monitorowanie parametrów pracy maszyn, zużycia energii elektrycznej, statusów online/offline urządzeń produkcyjnych.
3. Kluczowe Korzyści Biznesowe (Value Proposition)
Redukcja kosztów operacyjnych: Automatyczna kontrola stanu urządzeń eliminuje konieczność fizycznych i rutynowych inspekcji obiektów przez pracowników.
Zapobieganie awariom i przestojom: System wczesnego ostrzegania informuje o stanach krytycznych (np. drastyczny spadek ciśnienia, odcięcie zasilania), zanim wpłyną one na ciągłość biznesową.
Optymalizacja logistyczna: Na podstawie trendów zużycia (np. gazu) system pozwala precyzyjnie planować terminy kolejnych dostaw lub serwisów.
Skalowalność środowiska magico.pro: Moduł telemetryczny wymienia dane z innymi aplikacjami w ekosystemie (np. generowanie automatycznych ticketów w helpdesk.magico przy awarii sensora lub fakturowanie zużycia przez invoice.magico).
4. Architektura Funkcjonalna Systemu (Struktura Modułowa)
Aplikacja logicznie dzieli się na 5 głównych obszarów funkcjonalnych, które stanowią podstawę do zaprojektowania interfejsu użytkownika (UI):
4.1. Dashboard (Pulpit Zarządczy)
Centrum dowodzenia menedżera operacyjnego. Zapewnia natychmiastowy wgląd w kondycję całego parku maszynowego/sieci czujników bez konieczności przeklikiwania się przez poszczególne obiekty.
KPI Statusów: Zagregowane liczniki urządzeń (Wszystkie, Online, Offline, W trakcie serwisu).
Sekcja Alertów: Lista obiektów, które przekroczyły zdefiniowane normy bezpieczeństwa (kolorystyka czerwona/pomarańczowa).
Geolokalizacja: Mapa z oznaczonymi punktami pomiarowymi (opcjonalnie, na podstawie koordynatów GPS wysyłanych przez minikomputery).
4.2. Moduł: Urządzenia (Infrastruktura i Integracje)
Miejsce zarządzania fizycznymi punktami pomiarowymi. Moduł opiera się na sztywnych definicjach typów urządzeń, pisanych przez deweloperów.
Katalog urządzeń: Przeglądanie, filtrowanie po lokalizacji i typie.
Kreator urządzeń (Provisioning): 1. Wybór predefiniowanego typu (np. Sensor-LPG-v2).
2. Dynamiczny formularz konfiguracyjny (system podpowiada pola unikalne dla typu: np. maksymalna pojemność w litrach, adres IP, częstotliwość raportowania).
3. Wygenerowanie bezpiecznego tokenu dostępowego API (Bearer Token) dla minikomputera brzegowego.
4.3. Moduł: Aktualny Stan (Live Monitor)
Narzędzie dedykowane dla operatorów i techników. Pozwala na podgląd zachowania wybranego obiektu w czasie rzeczywistym.
Dynamiczny Grid Interfejsu: Layout dopasowuje się do typu urządzenia. Jeśli urządzenie mierzy 2 parametry, wyświetlane są 2 kafelki/wskaźniki zegarowe (gauges). Jeśli mierzy 10 parametrów interfejs generuje adekwatną liczbę kontenerów.
Wykresy Live: Wizualizacja trendu z ostatnich minut/godzin z automatycznym odświeżaniem danych.
4.4. Moduł: Raporty i Historia (Analityka)
Narzędzie służące do retrospektywnej analizy danych i wyciągania wniosków biznesowych.
Agregacja danych: Filtrowanie według ram czasowych (dni, tygodnie, miesiące) oraz konkretnych parametrów.
Eksport danych: Generowanie zestawień do formatów analitycznych (CSV, Excel) w celu dalszej obróbki lub audytów.
Analiza trendów: Narzędzia wykresów liniowych umożliwiające nałożenie na siebie różnych metryk w celu znalezienia korelacji (np. wpływ temperatury otoczenia na ciśnienie w zbiorniku).
4.5. Moduł: Ustawienia (Administracja)
Konfiguracja zachowania systemu oraz progów bezpieczeństwa.
Zarządzanie alertami (Progi krytyczne): Definiowanie reguł biznesowych typu: „Jeśli parametr gas_level na dowolnym urządzeniu typu gas_tank spadnie poniżej 15%, wyślij powiadomienie”.
Zarządzanie uprawnieniami: Integracja z systemem uprawnień magico.pro (kto może tylko przeglądać wykresy, a kto ma prawo dodawać nowe urządzenia).
5. Model Operacyjny (Jak to działa w praktyce?)
Warstwa Fizyczna (Edge): Przy zbiorniku lub w kurniku znajduje się minikomputer (np. sterownik przemysłowy, Raspberry Pi, brama Teltonika). Odpytuje on lokalne czujniki fizyczne (poprzez Modbus, OneWire itp.).
Warstwa Transmisyjna (API): Minikomputer formatuje uzyskane dane do ujednoliconego formatu JSON i za pomocą zapytania HTTP POST (zabezpieczonego tokenem) wypycha je do chmury telemetry.magico.
Warstwa Przetwarzania (Core): Aplikacja identyfikuje urządzenie po tokenie, sprawdza jakie metryki zostały przesłane, zapisuje je w bazie danych serii czasowych (Time-Series format) i ewentualnie uruchamia procesy sprawdzania alertów.
Warstwa Prezentacji (UI): Użytkownik końcowy widzi przetworzone, czytelne dane na wykresach i widgetach w panelu HTML.
# Dokumentacja Funkcjonalna Modułów: telemetry.magico
Ten dokument zawiera szczegółowy opis wymagań, logiki biznesowej oraz struktury interfejsu (UI) dla poszczególnych modułów aplikacji. Służy jako bezpośredni brief do przygotowania makiety HTML (Bootstrap) w ramach środowiska magico.pro.
---
## 1. Moduł: Dashboard (Pulpit Zarządczy)
### 1.1. Cel biznesowy
Zapewnienie użytkownikowi (zarządcy, dyspozytorowi) błyskawicznej oceny stanu technicznego i operacyjnego całej infrastruktury urządzeń bez konieczności wchodzenia w szczegóły każdego z nich.
### 1.2. Struktura i Układ UI (Makieta HTML)
* **Górny pasek statystyk (KPI Cards):** Cztery małe, sąsiadujące kafelki (`.card`) w jednym rzędzie:
* **Wszystkie urządzenia:** Łączna liczba zarejestrowanych maszyn (kolor neutralny / podstawowy magico).
* **Online:** Urządzenia, od których odebrano dane w zdefiniowanym oknie czasowym (zielony badge/tekst, ikona sygnału).
* **Offline:** Urządzenia, które utraciły łączność (czerwony badge/tekst, ikona wykrzyknika).
* **Aktywne Alerty:** Liczba urządzeń, na których aktualnie przekroczone są parametry krytyczne (pomarańczowy/czerwony puls).
* **Główna sekcja (Układ dwukolumnowy - `row` z podziałem `col-md-8` i `col-md-4`):**
* **Kolumna Lewa (Szeroka): Tabela "Krytyczne stany i alerty".** Lista urządzeń wymagających natychmiastowej uwagi. Kolumny: Nazwa urządzenia, Typ, Lokalizacja, Parametr przekroczony (np. *Poziom gazu: 8%*), Czas wystąpienia, Akcja (Przejdź do Live Monitora).
* **Kolumna Prawa (Wąska): Widget "Ostatnie zdarzenia" (Activity Log).** Lista typu timeline (oś czasu) pokazująca ostatnie zmiany statusów (np. *10:15 - Urządzenie 'Zbiornik 3' przeszło w tryb OFFLINE*, *09:42 - Urządzenie 'Kurnik B' zarejestrowało spadek temperatury*).
### 1.3. Logika i Zachowanie (JS / Backend)
* **Definicja Offline:** Urządzenie zmienia status na Offline automatycznie, jeśli czas od ostatniego odczytu (`last_seen_at`) jest większy niż dwukrotność jego zadeklarowanego interwału wysyłki danych.
* **Interakcja:** Kliknięcie w dowolny wiersz na liście alertów przenosi użytkownika bezpośrednio do widoku "Aktualny stan" (Live Monitor) z wybranym już tym konkretnym urządzeniem.
---
## 2. Moduł: Urządzenia (Device Management & Provisioning)
### 2.1. Cel biznesowy
Zarządzanie bazą fizycznych urządzeń oraz procesem ich bezpiecznego uwierzytelniania w chmurze magico.pro.
### 2.2. Struktura i Układ UI (Makieta HTML)
Moduł składa się z dwóch głównych widoków: **Listy urządzeń** oraz **Kreatora dodawania**.
#### Widok: Lista Urządzeń (Tabela główna)
* **Filtry nad tabelą:** Wyszukiwarka tekstowa, dropdown z filtrem po "Typie urządzenia" (Wszystkie, Zbiornik Gazu, Kurnik itp.) oraz filtr statusu (Online/Offline).
* **Tabela danych:**
* Nazwa (np. *Zbiornik LPG #1 - Stalowa Wola, ul. Przemysłowa*).
* Typ urządzenia (Badge z ikoną dedykowaną dla typu, np. ikona płomienia dla gazu, ikona termometru dla kurnika).
* Ostatni kontakt (Względny format daty, np. *Przed 2 min*, *3 dni temu*).
* Status (`.badge-success` dla Online, `.badge-danger` dla Offline).
* Akcje: Przycisk "Podgląd Live" (ikona oka), "Edycja" (ikona ołówka).
#### Widok: Kreator dodawania (Step-by-Step Wizard)
Zrealizowany za pomocą zakładek (`nav-tabs` lub niestandardowy stepper komponentu Bootstrap).
* **Krok 1: Wybór typu urządzenia.** Kafelki z ikonami i opisami predefiniowanych integracji (np. `Karta 1: Zbiornik Gazu standard (Modbus/LPG)`, `Karta 2: Sterownik Mikroklimatu Kurnika v1`). Użytkownik klika jeden z nich i przechodzi dalej.
* **Krok 2: Konfiguracja parametrów (Dynamiczny formularz).** Pola ładują się w zależności od wyboru w Kroku 1:
* *Dla Zbiornika Gazu:* Pole tekstowe: Pojemność nominalna (litry), Maksymalne ciśnienie bezpieczne (bar), Stały adres IP minikomputera (opcjonalnie).
* *Dla Kurnika:* Pole liczbowe: Liczba stref grzewczych, Liczba wentylatorów, Interwał próbkowania (w sekundach).
* **Krok 3: Generowanie tokenu.** Ekran podsumowania, na którym system wyświetla wygenerowany unikalny klucz API (`Device API Token`). Zawiera przycisk "Kopiuj do schowka".
### 2.3. Logika i Zachowanie (JS / Backend)
* **Bezpieczeństwo tokenów:** Token API (`api_token`) jest widoczny w pełnej krasie tylko raz podczas tworzenia urządzenia (Krok 3). Potem w bazie jest haszowany lub maskowany (np. `mag_tlm_•••••••••1a2b`).
* **Dynamiczne renderowanie:** W makiety HTML warto wbudować mechanizm (np. proste przełączanie klasami `d-none`), który zasymuluje, jak zmienia się formularz w Kroku 2 w zależności od wybranego w Kroku 1 typu urządzenia.
---
## 3. Moduł: Aktualny Stan (Live Monitor)
### 3.1. Cel biznesowy
Umożliwienie technikowi lub operatorowi zdalnego podglądu parametrów pracy wybranego obiektu w trybie "tu i teraz" w celu weryfikacji awarii lub kontroli procesów.
### 3.2. Struktura i Układ UI (Makieta HTML)
* **Górny pasek kontekstowy:** Duży Dropdown (`<select>` z obsługą wyszukiwania, np. Select2 w Bootstrapie) do wyboru urządzenia. Obok dropdownu wyświetla się duży badge aktualnego statusu wybranego urządzenia (Online/Offline) oraz data i godzina ostatniego odebranego pakietu danych.
* **Siatka wskaźników (Dynamic Grid):** Rząd kafelków (`row-cols-1 row-cols-md-3 g-4`), gdzie każdy kafelek prezentuje **jedną metrykę/kanał danych**.
* *Przykład komponentu wskaźnika:* Duża wartość numeryczna w centrum (np. **22.4 °C** lub **74.5 %**), pod nią nazwa metryki ("Temperatura strefa A", "Poziom napełnienia"), a na dole mały badge informujący o trendzie w ciągu ostatniej godziny (np. zielona strzałka w górę "Wzrost o 1.2%").
* **Dolna sekcja: Mini-wykres Live.** Wykres liniowy (zajmujący pełną szerokość `col-12`), pokazujący dane z wybranej kluczowej metryki z ostatnich 2 godzin, odświeżany automatycznie bez przeładowania strony.
### 3.3. Logika i Zachowanie (JS / Backend)
* **Agnostyczność interfejsu:** Layout HTML musi być przygotowany na to, że kafelków wskaźników może być 2 (dla prostego zbiornika), a może być 12 (dla rozbudowanego kurnika). Kafelki powinny płynnie zawijać się do kolejnych rzędów dzięki klasom flexbox Bootstrapa.
* **Odświeżanie danych:** Widok symuluje połączenie WebSocket lub regularny odpytywacz (Polling AJAX) co 10-30 sekund, który aktualizuje wartości liczbowe w kafelkach oraz dopisuje nowy punkt na mini-wykresie.
---
## 4. Moduł: Raporty i Historia (Analityka Historyczna)
### 4.1. Cel biznesowy
Dostarczanie twardych danych analitycznych do celów optymalizacyjnych (planowanie dostaw gazu, audyty warunków chowu drobiu) oraz eksport danych do zewnętrznych systemów lub arkuszy kalkulacyjnych.
### 4.2. Struktura i Układ UI (Makieta HTML)
* **Pasek filtrów (Górny horyzontalny formularz card):**
* Wybór urządzenia (Dropdown).
* Wybór metryk (Multiselect checkbox / dropdown np. użytkownik chce zobaczyć tylko poziom gazu, ignorując napięcie baterii minikomputera).
* Zakres dat (Komponent Date-Range Picker: Od, Do, wraz z szybkimi filtrami: *Dzisiaj, Ostatnie 7 dni, Ten miesiąc*).
* Przycisk akcji: "Generuj raport" (niebieski) oraz "Eksportuj do CSV" (zielony).
* **Sekcja Główna - Wykres Analityczny:** Duży kontener wykresu liniowego (np. przy użyciu biblioteki Chart.js wbudowanej w layout Bootstrap). Wykres musi obsługiwać wiele osi Y, jeśli użytkownik porównuje różne jednostki (np. lewa oś dla temperatury w `°C`, prawa dla wilgotności w `%`).
* **Sekcja Dolna - Tabela Surowych Danych:** Tabela z paginacją (np. stylizowana pod DataTables). Kolumny: Data i godzina (Timestamp), Nazwa metryki, Wartość, Jednostka.
### 4.3. Logika i Zachowanie (JS / Backend)
* **Agregacja danych (Downsampling):** Ponieważ dane telemetryczne przyrastają w ogromnym tempie, backend przy raportach z długich okresów (np. 3 miesiące) nie może przesłać milionów rekordów do JS. System musi agregować dane w locie (np. średnia godzinowa lub średnia dobowa) w zależności od wybranego zakresu czasu.
---
## 5. Moduł: Ustawienia (Settings & Alert Rules)
### 5.1. Cel biznesowy
Konfiguracja zachowań automatycznych systemu, w tym definiowanie progów bezpieczeństwa dla zbieranych danych.
### 5.2. Struktura i Układ UI (Makieta HTML)
Zorganizowany w formie pionowych zakładek (List-group tabs po lewej stronie, zawartość po prawej).
#### Zakładka 1: Reguły Alertów (Alerts Configuration)
* **Lista aktywnych reguł:** Tabela pokazująca zdefiniowane progi.
* **Przycisk "Dodaj nową regułę"**, który otwiera modal (okno pop-up):
* Dropdown: "Jeśli urządzenie typu..." (Wybór typu, np. *Zbiornik Gazu*).
* Dropdown: "...zarejestruje parametr..." (Dynamiczna lista metryk dla typu, np. *gas_level*).
* Dropdown warunku: (Mniejsze niż, Większe niż, Równe).
* Pole numeryczne: Wartość progu (np. *15*).
* Sekcja akcji: Checkboxy "Wyślij e-mail do administratora", "Załóż ticket w helpdesk.magico".
#### Zakładka 2: Globalna Konfiguracja Telemetrii
* Formularze konfiguracyjne:
* Czas retencji danych (Dropdown: *Przechowuj surowe dane przez: 3 miesiące / 6 miesięcy / rok*).
* Domyślny interwał sprawdzania statusów offline (np. *Co ile minut system ma weryfikować brak kontaktu z minikomputerami*).
### 5.3. Logika i Zachowanie (JS / Backend)
* **Integracja wewnątrzmagico:** Powiązanie akcji alertu z modułem `helpdesk.magico` musi opierać się na wewnętrznym API ekosystemu magico.pro. W makiecie należy przewidzieć elementy UI (np. selecty grup użytkowników, kategorie ticketów), które odzwierciedlają tę integrację.

View File

@@ -0,0 +1,30 @@
<?php
$enablePrototypeComments = true;
include '../../header-telemetria.php';
?>
<div class="container-xxl flex-grow-1 container-p-y">
<h4 class="fw-bold py-3 mb-4">
Telemetria <span class="text-muted fw-light">/ Dashboard</span>
</h4>
<div class="row">
<div class="col-12">
<div class="card">
<div class="card-header border-bottom mb-3">
<h5 class="card-title mb-0">Podsumowanie Telemetrii</h5>
</div>
<div class="card-body">
<p>Witaj w nowym module Telemetria. Ta makieta jest gotowa do dalszego rozwoju.</p>
</div>
</div>
</div>
</div>
</div>
<div class="content-backdrop fade"></div>
<?php include '../../footer.php'; ?>
</body>
</html>

View File

@@ -0,0 +1,394 @@
<?php
$enablePrototypeComments = true;
include '../../header-telemetria.php';
?>
<div class="container-xxl flex-grow-1 container-p-y">
<h4 class="fw-bold py-3 mb-4">
Telemetria <span class="text-muted fw-light">/ Dodaj instalację</span>
</h4>
<div class="card">
<div class="card-body">
<!-- Stepper / Wizard nawigacja -->
<div class="d-flex justify-content-between mb-4 border-bottom pb-3">
<div class="text-center w-100 wizard-step active-step" id="wizard-tab-1">
<div class="badge bg-primary rounded-pill mb-2 p-2"><i class="bx bx-category fs-4"></i>
</div>
<h6 class="mb-0 text-primary fw-bold">1. Typ instalacji</h6>
</div>
<div class="text-center w-100 wizard-step opacity-50" id="wizard-tab-2">
<div class="badge bg-secondary rounded-pill mb-2 p-2"><i
class="bx bx-slider-alt fs-4"></i></div>
<h6 class="mb-0 text-muted">2. Konfiguracja</h6>
</div>
<div class="text-center w-100 wizard-step opacity-50" id="wizard-tab-3">
<div class="badge bg-secondary rounded-pill mb-2 p-2"><i class="bx bx-check-shield fs-4"></i>
</div>
<h6 class="mb-0 text-muted">3. Zakończenie</h6>
</div>
</div>
<!-- KROK 1 -->
<div id="wizard-step-1" class="wizard-content">
<h5 class="mb-4 text-center">Wybierz typ integracji dla nowej instalacji</h5>
<div class="row g-4 justify-content-center">
<div class="col-md-5">
<div class="card border border-2 border-primary shadow-sm h-100 cursor-pointer"
onclick="goToStep2('gas')">
<div class="card-body text-center">
<div
class="avatar avatar-xl bg-label-warning mx-auto mb-3 rounded-circle d-flex align-items-center justify-content-center">
<i class="bx bxs-flame fs-1"></i>
</div>
<h5 class="card-title mb-2">Zbiornik Gazu (LPG/CNG)</h5>
<p class="text-muted small">Standardowy czujnik Modbus dla zbiorników paliw
i gazu z pomiarem poziomu napełnienia i ciśnienia.</p>
</div>
</div>
</div>
<div class="col-md-5">
<div class="card border border-2 border-transparent hover-border-primary shadow-sm h-100 cursor-pointer"
onclick="goToStep2('farm')">
<div class="card-body text-center">
<div
class="avatar avatar-xl bg-label-info mx-auto mb-3 rounded-circle d-flex align-items-center justify-content-center">
<i class="bx bxs-thermometer fs-1"></i>
</div>
<h5 class="card-title mb-2">Sterownik Mikroklimatu (Kurnik)</h5>
<p class="text-muted small">Integracja z kontrolerami wentylacji i
ogrzewania. Obsługa wielu stref temperaturowych i CO2.</p>
</div>
</div>
</div>
</div>
</div>
<!-- KROK 2 -->
<div id="wizard-step-2" class="wizard-content d-none">
<h5 class="mb-4">Konfiguracja specyficzna dla instalacji</h5>
<!-- Formularz wspólny -->
<div class="row mb-3">
<div class="col-md-6 mb-3">
<label class="form-label">Nazwa instalacji (Własna)</label>
<input type="text" class="form-control" placeholder="np. Zbiornik Główny Północ">
</div>
<div class="col-md-6 mb-3">
<label class="form-label">Lokalizacja / Adres</label>
<input type="text" class="form-control" placeholder="np. ul. Fabryczna 1, Warszawa">
</div>
</div>
<hr>
<h6 class="text-muted mb-3">Moduły i Czujniki (<span id="type-label-name">Zbiornik
Gazu</span>)</h6>
<p class="small text-muted mb-4">Włącz czujniki i moduły, które fizycznie obsługuje ta instalacja.</p>
<!-- Parametry Zbiornik (pokazywane dynamicznie) -->
<div id="form-gas" class="dynamic-form-part">
<div class="row mb-4">
<div class="col-md-3">
<div class="form-check form-switch mb-2">
<input class="form-check-input" type="checkbox" id="sensorGasLevel" checked onchange="toggleSensorConfig('gasLevel')">
<label class="form-check-label fw-bold" for="sensorGasLevel">Poziom napełnienia</label>
</div>
</div>
<div class="col-md-3">
<div class="form-check form-switch mb-2">
<input class="form-check-input" type="checkbox" id="sensorGasPressure" checked onchange="toggleSensorConfig('gasPressure')">
<label class="form-check-label fw-bold" for="sensorGasPressure">Ciśnienie</label>
</div>
</div>
<div class="col-md-3">
<div class="form-check form-switch mb-2">
<input class="form-check-input" type="checkbox" id="sensorGasTemp" onchange="toggleSensorConfig('gasTemp')">
<label class="form-check-label fw-bold" for="sensorGasTemp">Temperatura</label>
</div>
</div>
<div class="col-md-3">
<div class="form-check form-switch mb-2">
<input class="form-check-input" type="checkbox" id="sensorGasGps" onchange="toggleSensorConfig('gasGps')">
<label class="form-check-label fw-bold" for="sensorGasGps">Moduł GPS</label>
</div>
</div>
</div>
<div class="row bg-lighter p-3 rounded mb-3" id="config-gasLevel">
<h6 class="mb-3 text-primary"><i class="bx bx-slider-alt"></i> Konfiguracja: Poziom napełnienia</h6>
<div class="col-md-6 mb-3">
<label class="form-label">Pojemność nominalna (litry)</label>
<input type="number" class="form-control" value="2700">
</div>
<div class="col-md-6 mb-3">
<label class="form-label">Offset sondy (mm)</label>
<input type="number" class="form-control" value="0">
</div>
</div>
<div class="row bg-lighter p-3 rounded mb-3" id="config-gasPressure">
<h6 class="mb-3 text-primary"><i class="bx bx-tachometer"></i> Konfiguracja: Ciśnienie</h6>
<div class="col-md-6 mb-3">
<label class="form-label">Maks. ciśnienie bezpieczne (bar)</label>
<input type="number" step="0.1" class="form-control" value="15.5">
</div>
</div>
<div class="row bg-lighter p-3 rounded mb-3 d-none" id="config-gasTemp">
<h6 class="mb-3 text-primary"><i class="bx bxs-thermometer"></i> Konfiguracja: Temperatura</h6>
<div class="col-md-6 mb-3">
<label class="form-label">Korekta wskazań (°C)</label>
<input type="number" step="0.1" class="form-control" value="0.0">
</div>
</div>
<div class="row bg-lighter p-3 rounded mb-3 d-none" id="config-gasGps">
<h6 class="mb-3 text-primary"><i class="bx bx-map"></i> Konfiguracja: GPS</h6>
<div class="col-md-12">
<p class="small text-muted mb-0">Ten moduł nie wymaga dodatkowej konfiguracji. Raportowanie pozycji odbywa się automatycznie.</p>
</div>
</div>
</div>
<!-- Parametry Kurnik (pokazywane dynamicznie) -->
<div id="form-farm" class="dynamic-form-part d-none">
<div class="row mb-4">
<div class="col-md-4">
<div class="form-check form-switch mb-2">
<input class="form-check-input" type="checkbox" id="sensorFarmTemp" checked>
<label class="form-check-label fw-bold" for="sensorFarmTemp">Sensory Temp.</label>
</div>
</div>
<div class="col-md-4">
<div class="form-check form-switch mb-2">
<input class="form-check-input" type="checkbox" id="sensorFarmVent" checked>
<label class="form-check-label fw-bold" for="sensorFarmVent">Sterowanie Wentylacją</label>
</div>
</div>
</div>
<div class="row bg-lighter p-3 rounded mb-3">
<div class="col-md-6 mb-3">
<label class="form-label">Liczba stref grzewczych</label>
<input type="number" class="form-control" value="2">
</div>
<div class="col-md-6 mb-3">
<label class="form-label">Interwał próbkowania (sekundy)</label>
<input type="number" class="form-control" value="60">
</div>
</div>
</div>
<!-- Konfigurator Powiadomień -->
<hr class="my-4">
<div class="d-flex justify-content-between align-items-center mb-3">
<h6 class="text-muted mb-0">Konfigurator powiadomień (Alerty)</h6>
<button type="button" class="btn btn-sm btn-outline-primary" onclick="addAlertRow()">
<i class="bx bx-plus"></i> Dodaj alert
</button>
</div>
<div id="alerts-container">
<!-- Dynamiczne wiersze alertów -->
<div class="row align-items-end mb-3 alert-row">
<div class="col-md-3">
<label class="form-label">Parametr</label>
<select class="form-select">
<option>Poziom gazu (%)</option>
<option>Ciśnienie (bar)</option>
<option>Temperatura (°C)</option>
<option>Brak sygnału</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Warunek</label>
<select class="form-select">
<option value="<">Mniejszy niż (&lt;)</option>
<option value=">">Większy niż (&gt;)</option>
<option value="=">Równy (=)</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Wartość</label>
<input type="number" step="0.1" class="form-control" placeholder="np. 15">
</div>
<div class="col-md-2">
<label class="form-label">Kanał</label>
<select class="form-select">
<option>E-mail</option>
<option>SMS</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Odbiorca</label>
<input type="text" class="form-control" placeholder="Adres / Numer">
</div>
<div class="col-md-1">
<button type="button" class="btn btn-icon btn-outline-danger w-100" onclick="this.closest('.alert-row').remove()">
<i class="bx bx-trash"></i>
</button>
</div>
</div>
</div>
<div class="d-flex justify-content-between mt-4 border-top pt-4">
<button type="button" class="btn btn-outline-secondary"
onclick="goToStep1()">Wróć</button>
<button type="button" class="btn btn-primary" onclick="goToStep3()">Zapisz i Zakończ</button>
</div>
</div>
<!-- KROK 3 -->
<div id="wizard-step-3" class="wizard-content d-none text-center py-4">
<div
class="avatar avatar-xl bg-label-success mx-auto mb-4 rounded-circle d-flex align-items-center justify-content-center">
<i class="bx bx-check fs-1"></i>
</div>
<h4 class="mb-2">Instalacja zarejestrowana!</h4>
<p class="text-muted mb-4">Instalacja została pomyślnie dodana do ekosystemu.</p>
<div class="bg-lighter rounded p-5 mx-auto mb-4 border border-dashed text-center" style="max-width: 600px;">
<i class="bx bx-cog fs-1 text-muted mb-3"></i>
<h6 class="text-muted text-uppercase mb-2">Miejsce na dane uwierzytelniające</h6>
<p class="small text-muted mb-0">W zależności od ostatecznych ustaleń architektonicznych, w tym miejscu system wygeneruje dla Ciebie <strong>Token API</strong> lub <strong>Gotowy plik konfiguracyjny</strong> do załadowania na sprzęt.</p>
</div>
<a href="installations.php" class="btn btn-primary mt-2">Przejdź do listy instalacji</a>
</div>
</div>
</div>
</div>
<div class="content-backdrop fade"></div>
<?php include '../../footer.php'; ?>
<script>
// Prosta logika wizarda
function goToStep1() {
document.querySelectorAll('.wizard-content').forEach(el => el.classList.add('d-none'));
document.getElementById('wizard-step-1').classList.remove('d-none');
updateTabsUI(1);
}
function goToStep2(type) {
document.querySelectorAll('.wizard-content').forEach(el => el.classList.add('d-none'));
document.getElementById('wizard-step-2').classList.remove('d-none');
// Pokaż odpowiednie pola
document.querySelectorAll('.dynamic-form-part').forEach(el => el.classList.add('d-none'));
if (type === 'gas') {
document.getElementById('form-gas').classList.remove('d-none');
document.getElementById('type-label-name').innerText = 'Zbiornik Gazu';
} else {
document.getElementById('form-farm').classList.remove('d-none');
document.getElementById('type-label-name').innerText = 'Sterownik Mikroklimatu';
}
updateTabsUI(2);
}
function goToStep3() {
document.querySelectorAll('.wizard-content').forEach(el => el.classList.add('d-none'));
document.getElementById('wizard-step-3').classList.remove('d-none');
updateTabsUI(3);
}
function updateTabsUI(step) {
// Reset
document.querySelectorAll('.wizard-step').forEach(el => {
el.classList.remove('active-step');
el.classList.add('opacity-50');
el.querySelector('.badge').classList.replace('bg-primary', 'bg-secondary');
el.querySelector('h6').classList.remove('text-primary', 'fw-bold');
el.querySelector('h6').classList.add('text-muted');
});
// Set active
let activeTab = document.getElementById('wizard-tab-' + step);
activeTab.classList.remove('opacity-50');
activeTab.classList.add('active-step');
activeTab.querySelector('.badge').classList.replace('bg-secondary', 'bg-primary');
activeTab.querySelector('h6').classList.remove('text-muted');
activeTab.querySelector('h6').classList.add('text-primary', 'fw-bold');
}
function copyToken() {
var copyText = document.getElementById("apiTokenInput");
copyText.select();
copyText.setSelectionRange(0, 99999);
navigator.clipboard.writeText(copyText.value);
alert("Token skopiowany do schowka!");
}
function toggleSensorConfig(sensorId) {
const checkbox = document.getElementById('sensor' + sensorId.charAt(0).toUpperCase() + sensorId.slice(1));
const configDiv = document.getElementById('config-' + sensorId);
if (configDiv) {
if (checkbox.checked) {
configDiv.classList.remove('d-none');
} else {
configDiv.classList.add('d-none');
}
}
}
function addAlertRow() {
const container = document.getElementById('alerts-container');
const rowHTML = `
<div class="row align-items-end mb-3 alert-row">
<div class="col-md-3">
<label class="form-label">Parametr</label>
<select class="form-select">
<option>Poziom gazu (%)</option>
<option>Ciśnienie (bar)</option>
<option>Temperatura (°C)</option>
<option>Brak sygnału</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Warunek</label>
<select class="form-select">
<option value="<">Mniejszy niż (&lt;)</option>
<option value=">">Większy niż (&gt;)</option>
<option value="=">Równy (=)</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Wartość</label>
<input type="number" step="0.1" class="form-control" placeholder="np. 15">
</div>
<div class="col-md-2">
<label class="form-label">Kanał</label>
<select class="form-select">
<option>E-mail</option>
<option>SMS</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Odbiorca</label>
<input type="text" class="form-control" placeholder="Adres / Numer">
</div>
<div class="col-md-1">
<button type="button" class="btn btn-icon btn-outline-danger w-100" onclick="this.closest('.alert-row').remove()">
<i class="bx bx-trash"></i>
</button>
</div>
</div>`;
container.insertAdjacentHTML('beforeend', rowHTML);
}
</script>
<style>
.hover-border-primary:hover {
border-color: #696cff !important;
transition: 0.3s;
}
</style>
</body>
</html>

View File

@@ -0,0 +1,270 @@
<?php
$enablePrototypeComments = true;
include '../../header-telemetria.php';
?>
<div class="container-xxl flex-grow-1 container-p-y">
<h4 class="fw-bold py-3 mb-4">
Telemetria <span class="text-muted fw-light">/ Edycja instalacji</span>
</h4>
<div class="card">
<div class="card-body">
<div class="d-flex align-items-center mb-4">
<div class="avatar avatar-lg bg-label-warning me-3">
<i class="bx bxs-flame fs-3"></i>
</div>
<div>
<h5 class="mb-0">Edytujesz: Zbiornik LPG #1</h5>
<small class="text-muted">Typ: Zbiornik Gazu (LPG/CNG) | ID: mag_tlm_837492</small>
</div>
</div>
<!-- Formularz wspólny -->
<div class="row mb-3">
<div class="col-md-6 mb-3">
<label class="form-label">Nazwa instalacji (Własna)</label>
<input type="text" class="form-control" value="Zbiornik LPG #1 (Stalowa Wola)">
</div>
<div class="col-md-6 mb-3">
<label class="form-label">Lokalizacja / Adres</label>
<input type="text" class="form-control" value="Stalowa Wola, ul. Przemysłowa">
</div>
</div>
<hr>
<h6 class="text-muted mb-3">Moduły i Czujniki</h6>
<p class="small text-muted mb-4">Zarządzaj aktywnymi czujnikami, które fizycznie obsługuje ta instalacja.</p>
<div class="row mb-4">
<div class="col-md-3">
<div class="form-check form-switch mb-2">
<input class="form-check-input" type="checkbox" id="sensorGasLevel" checked onchange="toggleSensorConfig('gasLevel')">
<label class="form-check-label fw-bold" for="sensorGasLevel">Poziom napełnienia</label>
</div>
</div>
<div class="col-md-3">
<div class="form-check form-switch mb-2">
<input class="form-check-input" type="checkbox" id="sensorGasPressure" checked onchange="toggleSensorConfig('gasPressure')">
<label class="form-check-label fw-bold" for="sensorGasPressure">Ciśnienie</label>
</div>
</div>
<div class="col-md-3">
<div class="form-check form-switch mb-2">
<input class="form-check-input" type="checkbox" id="sensorGasTemp" onchange="toggleSensorConfig('gasTemp')">
<label class="form-check-label fw-bold" for="sensorGasTemp">Temperatura</label>
</div>
</div>
<div class="col-md-3">
<div class="form-check form-switch mb-2">
<input class="form-check-input" type="checkbox" id="sensorGasGps" checked onchange="toggleSensorConfig('gasGps')">
<label class="form-check-label fw-bold" for="sensorGasGps">Moduł GPS</label>
</div>
</div>
</div>
<!-- Parametry Zbiornik -->
<div class="row bg-lighter p-3 rounded mb-3" id="config-gasLevel">
<h6 class="mb-3 text-primary"><i class="bx bx-slider-alt"></i> Konfiguracja: Poziom napełnienia</h6>
<div class="col-md-6 mb-3">
<label class="form-label">Pojemność nominalna (litry)</label>
<input type="number" class="form-control" value="2700">
</div>
<div class="col-md-6 mb-3">
<label class="form-label">Offset sondy (mm)</label>
<input type="number" class="form-control" value="0">
</div>
</div>
<div class="row bg-lighter p-3 rounded mb-3" id="config-gasPressure">
<h6 class="mb-3 text-primary"><i class="bx bx-tachometer"></i> Konfiguracja: Ciśnienie</h6>
<div class="col-md-6 mb-3">
<label class="form-label">Maks. ciśnienie bezpieczne (bar)</label>
<input type="number" step="0.1" class="form-control" value="15.5">
</div>
</div>
<div class="row bg-lighter p-3 rounded mb-3 d-none" id="config-gasTemp">
<h6 class="mb-3 text-primary"><i class="bx bxs-thermometer"></i> Konfiguracja: Temperatura</h6>
<div class="col-md-6 mb-3">
<label class="form-label">Korekta wskazań (°C)</label>
<input type="number" step="0.1" class="form-control" value="0.0">
</div>
</div>
<div class="row bg-lighter p-3 rounded mb-3" id="config-gasGps">
<h6 class="mb-3 text-primary"><i class="bx bx-map"></i> Konfiguracja: GPS</h6>
<div class="col-md-12">
<p class="small text-muted mb-0">Ten moduł nie wymaga dodatkowej konfiguracji. Raportowanie pozycji odbywa się automatycznie na podstawie satelitów.</p>
</div>
</div>
<!-- Konfigurator Powiadomień -->
<hr class="my-4">
<div class="d-flex justify-content-between align-items-center mb-3">
<h6 class="text-muted mb-0">Konfigurator powiadomień (Alerty)</h6>
<button type="button" class="btn btn-sm btn-outline-primary" onclick="addAlertRow()">
<i class="bx bx-plus"></i> Dodaj alert
</button>
</div>
<div id="alerts-container">
<!-- Istniejące alerty (zapisane w systemie) -->
<div class="row align-items-end mb-3 alert-row">
<div class="col-md-3">
<label class="form-label">Parametr</label>
<select class="form-select">
<option selected>Poziom gazu (%)</option>
<option>Ciśnienie (bar)</option>
<option>Temperatura (°C)</option>
<option>Brak sygnału</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Warunek</label>
<select class="form-select">
<option value="<" selected>Mniejszy niż (&lt;)</option>
<option value=">">Większy niż (&gt;)</option>
<option value="=">Równy (=)</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Wartość</label>
<input type="number" step="0.1" class="form-control" value="15">
</div>
<div class="col-md-2">
<label class="form-label">Kanał</label>
<select class="form-select">
<option>E-mail</option>
<option selected>SMS</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Odbiorca</label>
<input type="text" class="form-control" value="+48 123 456 789">
</div>
<div class="col-md-1">
<button type="button" class="btn btn-icon btn-outline-danger w-100" onclick="this.closest('.alert-row').remove()">
<i class="bx bx-trash"></i>
</button>
</div>
</div>
<div class="row align-items-end mb-3 alert-row">
<div class="col-md-3">
<label class="form-label">Parametr</label>
<select class="form-select">
<option>Poziom gazu (%)</option>
<option selected>Ciśnienie (bar)</option>
<option>Temperatura (°C)</option>
<option>Brak sygnału</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Warunek</label>
<select class="form-select">
<option value="<">Mniejszy niż (&lt;)</option>
<option value=">" selected>Większy niż (&gt;)</option>
<option value="=">Równy (=)</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Wartość</label>
<input type="number" step="0.1" class="form-control" value="7.5">
</div>
<div class="col-md-2">
<label class="form-label">Kanał</label>
<select class="form-select">
<option>E-mail</option>
<option selected>SMS</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Odbiorca</label>
<input type="text" class="form-control" value="+48 123 456 789">
</div>
<div class="col-md-1">
<button type="button" class="btn btn-icon btn-outline-danger w-100" onclick="this.closest('.alert-row').remove()">
<i class="bx bx-trash"></i>
</button>
</div>
</div>
</div>
<div class="d-flex justify-content-between mt-5 border-top pt-4">
<a href="installations.php" class="btn btn-outline-secondary">Anuluj</a>
<a href="installations.php" class="btn btn-primary">Zapisz zmiany</a>
</div>
</div>
</div>
</div>
<div class="content-backdrop fade"></div>
<?php include '../../footer.php'; ?>
<script>
function addAlertRow() {
const container = document.getElementById('alerts-container');
const rowHTML = `
<div class="row align-items-end mb-3 alert-row">
<div class="col-md-3">
<label class="form-label">Parametr</label>
<select class="form-select">
<option>Poziom gazu (%)</option>
<option>Ciśnienie (bar)</option>
<option>Temperatura (°C)</option>
<option>Brak sygnału</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Warunek</label>
<select class="form-select">
<option value="<">Mniejszy niż (&lt;)</option>
<option value=">">Większy niż (&gt;)</option>
<option value="=">Równy (=)</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Wartość</label>
<input type="number" step="0.1" class="form-control" placeholder="np. 15">
</div>
<div class="col-md-2">
<label class="form-label">Kanał</label>
<select class="form-select">
<option>E-mail</option>
<option>SMS</option>
</select>
</div>
<div class="col-md-2">
<label class="form-label">Odbiorca</label>
<input type="text" class="form-control" placeholder="Adres / Numer">
</div>
<div class="col-md-1">
<button type="button" class="btn btn-icon btn-outline-danger w-100" onclick="this.closest('.alert-row').remove()">
<i class="bx bx-trash"></i>
</button>
</div>
</div>`;
container.insertAdjacentHTML('beforeend', rowHTML);
}
</script>
<script>
function toggleSensorConfig(sensorId) {
const checkbox = document.getElementById('sensor' + sensorId.charAt(0).toUpperCase() + sensorId.slice(1));
const configDiv = document.getElementById('config-' + sensorId);
if (configDiv) {
if (checkbox.checked) {
configDiv.classList.remove('d-none');
} else {
configDiv.classList.add('d-none');
}
}
}
</script>
</body>
</html>

View File

@@ -0,0 +1,362 @@
<?php
$enablePrototypeComments = true;
include '../../header-telemetria.php';
?>
<div class="container-xxl flex-grow-1 container-p-y">
<div class="d-flex justify-content-between align-items-center py-3 mb-4">
<h4 class="fw-bold mb-0">
Telemetria <span class="text-muted fw-light">/ Podgląd Live</span>
</h4>
<div class="d-flex align-items-center">
<span class="badge bg-success me-2 px-3 py-2 fs-6"><i class="bx bx-wifi"></i> ONLINE</span>
<small class="text-muted d-none d-md-block">Ostatni kontakt: Dzisiaj, 14:52:10</small>
</div>
</div>
<!-- Header z wyborem instalacji -->
<div class="card mb-4">
<div class="card-body d-flex flex-column flex-md-row justify-content-between align-items-md-center">
<div class="d-flex align-items-center mb-3 mb-md-0">
<div class="avatar avatar-lg bg-label-warning me-3">
<i class="bx bxs-flame fs-3"></i>
</div>
<div>
<h5 class="mb-0">Zbiornik LPG #1 (Stalowa Wola)</h5>
<small class="text-muted">ID: mag_tlm_837492</small>
</div>
</div>
<div>
<a href="installation-edit.php" class="btn btn-outline-secondary">
<i class="bx bx-pencil me-1"></i> Edytuj instalację
</a>
</div>
</div>
</div>
<div class="row g-4 mb-4">
<!-- SVG Zbiornika -->
<div class="col-xl-4 col-lg-5 col-md-12">
<div class="card border-0 shadow-sm mb-4">
<div class="card-header border-bottom">
<h5 class="card-title m-0">Wizualizacja stanu napełnienia</h5>
</div>
<div class="card-body d-flex flex-column align-items-center justify-content-center p-5">
<!-- SVG Silos -->
<div class="tank-container position-relative mb-3 mx-auto" style="width: 140px; height: 350px;">
<svg viewBox="0 0 100 300" width="100%" height="100%">
<!-- Nogi silosu -->
<path d="M 35 260 L 30 295 M 65 260 L 70 295" stroke="#a1acb8" stroke-width="6"
stroke-linecap="round" />
<line x1="20" y1="295" x2="80" y2="295" stroke="#a1acb8" stroke-width="4"
stroke-linecap="round" />
<!-- Zewnętrzna krawędź obudowy (Silos stojący) -->
<path id="siloOutline" d="M 20 40 A 30 20 0 0 1 80 40 L 80 250 A 30 20 0 0 1 20 250 Z"
fill="#f8f9fa" stroke="#d9dee3" stroke-width="4" />
<!-- Animowane wypełnienie (Mask/Clip) -->
<clipPath id="fillClip">
<path d="M 22 42 A 28 18 0 0 1 78 42 L 78 248 A 28 18 0 0 1 22 248 Z" />
</clipPath>
<!-- Płyn - gaz -->
<defs>
<linearGradient id="gasGradient" x1="0%" y1="0%" x2="0%" y2="100%">
<stop offset="0%" stop-color="#ffab00" />
<stop offset="100%" stop-color="#e69a00" />
</linearGradient>
</defs>
<!-- Ten prostokąt porusza się w górę i w dół, maskowany przez kształt silosu -->
<g clip-path="url(#fillClip)">
<rect id="liquidFill" x="0" y="120" width="100" height="250" fill="url(#gasGradient)" />
</g>
<!-- Pierścienie wzmacniające silosu -->
<path d="M 20 90 A 30 10 0 0 0 80 90" stroke="#d9dee3" stroke-width="2" fill="none" />
<path d="M 20 150 A 30 10 0 0 0 80 150" stroke="#d9dee3" stroke-width="2" fill="none" />
<path d="M 20 210 A 30 10 0 0 0 80 210" stroke="#d9dee3" stroke-width="2" fill="none" />
<!-- Efekt szkła/odblasku dla głębi 3D -->
<path d="M 26 50 Q 35 150 26 240" stroke="#ffffff" stroke-width="4" fill="none"
opacity="0.6" stroke-linecap="round" />
<path d="M 74 50 Q 65 150 74 240" stroke="#000000" stroke-width="3" fill="none"
opacity="0.05" stroke-linecap="round" />
<!-- Podziałka na zbiorniku -->
<line x1="85" y1="90" x2="95" y2="90" stroke="#a1acb8" stroke-width="2" />
<text x="80" y="93" font-size="8" fill="#a1acb8" text-anchor="end">75%</text>
<line x1="85" y1="150" x2="95" y2="150" stroke="#a1acb8" stroke-width="2" />
<text x="80" y="153" font-size="8" fill="#a1acb8" text-anchor="end">50%</text>
<line x1="85" y1="210" x2="95" y2="210" stroke="#a1acb8" stroke-width="2" />
<text x="80" y="213" font-size="8" fill="#a1acb8" text-anchor="end">25%</text>
</svg>
<!-- Główna etykieta procentowa nałożona na grafikę -->
<div class="position-absolute top-50 start-50 translate-middle text-center"
style="margin-top: -15px;">
<h1 class="mb-0 fw-bolder text-dark"
style="text-shadow: 0px 0px 10px rgba(255,255,255,0.8);" id="mainPercentage">60%</h1>
</div>
</div>
</div>
</div>
<div class="card shadow-sm border-0">
<div class="card-header border-bottom d-flex justify-content-between align-items-center">
<h5 class="card-title m-0">Skonfigurowane alerty</h5>
<button class="btn btn-sm btn-icon btn-text-secondary"><i class="bx bx-cog"></i></button>
</div>
<div class="card-body pt-4">
<ul class="list-unstyled mb-0">
<li class="mb-3 d-flex align-items-center">
<span class="badge bg-label-danger me-3 p-2"><i class="bx bx-down-arrow-alt"></i></span>
<div>
<h6 class="mb-0">Niski poziom gazu</h6>
<small class="text-muted">Gdy poziom &lt; 15%</small>
</div>
</li>
<li class="mb-3 d-flex align-items-center">
<span class="badge bg-label-danger me-3 p-2"><i class="bx bx-up-arrow-alt"></i></span>
<div>
<h6 class="mb-0">Nadciśnienie</h6>
<small class="text-muted">Gdy ciśnienie &gt; 7.5 bar</small>
</div>
</li>
<li class="d-flex align-items-center">
<span class="badge bg-label-warning me-3 p-2"><i class="bx bx-wifi-off"></i></span>
<div>
<h6 class="mb-0">Brak łączności</h6>
<small class="text-muted">Gdy brak sygnału &gt; 10 min</small>
</div>
</li>
</ul>
</div>
</div>
</div>
<!-- Parametry pomiarowe - Kafeleki -->
<div class="col-xl-8 col-lg-7 col-md-12">
<div class="row row-cols-1 row-cols-sm-2 g-4 align-content-start mb-4">
<!-- Poziom gazu w litrach -->
<div class="col">
<div class="card shadow-sm border-0 border-start border-warning border-3">
<div class="card-body">
<div class="d-flex align-items-start justify-content-between">
<div class="content-left">
<span class="text-muted text-uppercase small fw-bold"
style="letter-spacing: 0.5px;">Szacowana objętość</span>
<div class="d-flex align-items-end mt-2">
<h2 class="mb-0 me-2 text-dark" id="volumeValue">1 620</h2>
<span class="fs-6 fw-normal text-muted mb-1">litrów</span>
</div>
<small class="text-success fw-semibold"><i class="bx bx-up-arrow-alt"></i> +25 l w
ciągu 5 min</small>
</div>
<span class="badge bg-label-warning rounded p-2">
<i class="bx bxs-flame fs-3"></i>
</span>
</div>
</div>
</div>
</div>
<!-- Ciśnienie -->
<div class="col">
<div class="card shadow-sm border-0">
<div class="card-body">
<div class="d-flex align-items-start justify-content-between">
<div class="content-left">
<span class="text-muted text-uppercase small fw-bold"
style="letter-spacing: 0.5px;">Ciśnienie operacyjne</span>
<div class="d-flex align-items-end mt-2">
<h2 class="mb-0 me-2 text-dark" id="pressureValue">5.24</h2>
<span class="fs-6 fw-normal text-muted mb-1">bar</span>
</div>
<small class="text-muted">Norma: 3.0 - 7.0 bar</small>
</div>
<span class="badge bg-label-info rounded p-2">
<i class="bx bx-tachometer fs-3"></i>
</span>
</div>
</div>
</div>
</div>
<!-- Temperatura -->
<div class="col">
<div class="card shadow-sm border-0">
<div class="card-body">
<div class="d-flex align-items-start justify-content-between">
<div class="content-left">
<span class="text-muted text-uppercase small fw-bold"
style="letter-spacing: 0.5px;">Temperatura otoczenia</span>
<div class="d-flex align-items-end mt-2">
<h2 class="mb-0 me-2 text-dark">14.5</h2>
<span class="fs-6 fw-normal text-muted mb-1">°C</span>
</div>
<small class="text-muted">Trend stabilny</small>
</div>
<span class="badge bg-label-primary rounded p-2">
<i class="bx bxs-thermometer fs-3"></i>
</span>
</div>
</div>
</div>
</div>
<!-- Zasilanie i sieć -->
<div class="col">
<div class="card shadow-sm border-0">
<div class="card-body">
<div class="d-flex align-items-start justify-content-between mb-3">
<span class="text-muted text-uppercase small fw-bold"
style="letter-spacing: 0.5px;">Hardware & Sieć</span>
<span class="badge bg-label-secondary rounded p-2"><i
class="bx bx-chip fs-4"></i></span>
</div>
<div class="d-flex justify-content-between align-items-center mb-2">
<span class="text-muted"><i class="bx bx-battery me-1"></i> Bateria
zew.</span>
<span class="fw-semibold">12.4 V</span>
</div>
<div class="d-flex justify-content-between align-items-center mb-2">
<span class="text-muted"><i class="bx bx-signal-4 me-1"></i> Sygnał
GSM</span>
<span class="fw-semibold">82%</span>
</div>
<div class="d-flex justify-content-between align-items-center">
<span class="text-muted"><i class="bx bx-time-five me-1"></i> Ostatni
kontakt</span>
<span class="fw-semibold">14:52:10</span>
</div>
</div>
</div>
</div>
</div>
<div class="card shadow-sm border-0">
<div class="card-header border-bottom">
<h5 class="card-title m-0">Historia ostatnich odczytów</h5>
</div>
<div class="card-body pt-4 pb-0" style="max-height: 400px; overflow-y: auto;">
<ul class="timeline mb-0">
<li class="timeline-item timeline-item-transparent">
<span class="timeline-point timeline-point-primary"></span>
<div class="timeline-event">
<div class="timeline-header mb-1">
<h6 class="mb-0">Regularny odczyt</h6>
<small class="text-muted">Dzisiaj, 14:52:10</small>
</div>
<p class="mb-0 small">Poziom: 60% (1620 l) | Ciśnienie: 5.24 bar | Bateria: 12.4V</p>
</div>
</li>
<li class="timeline-item timeline-item-transparent">
<span class="timeline-point timeline-point-danger"></span>
<div class="timeline-event">
<div class="timeline-header mb-1">
<h6 class="mb-0 text-danger">ALERT: Zbyt wysokie ciśnienie!</h6>
<small class="text-muted">Dzisiaj, 13:10:05</small>
</div>
<p class="mb-0 small">Zarejestrowano skok ciśnienia: <strong>7.8 bar</strong> (Próg: 7.5
bar). Wysłano powiadomienie SMS do serwisu.</p>
</div>
</li>
<li class="timeline-item timeline-item-transparent">
<span class="timeline-point timeline-point-primary"></span>
<div class="timeline-event">
<div class="timeline-header mb-1">
<h6 class="mb-0">Regularny odczyt</h6>
<small class="text-muted">Dzisiaj, 12:52:10</small>
</div>
<p class="mb-0 small">Poziom: 61% (1640 l) | Ciśnienie: 6.8 bar | Bateria: 12.4V</p>
</div>
</li>
<li class="timeline-item timeline-item-transparent">
<span class="timeline-point timeline-point-primary"></span>
<div class="timeline-event">
<div class="timeline-header mb-1">
<h6 class="mb-0">Regularny odczyt</h6>
<small class="text-muted">Dzisiaj, 10:52:10</small>
</div>
<p class="mb-0 small">Poziom: 62% (1665 l) | Ciśnienie: 6.5 bar | Bateria: 12.5V</p>
</div>
</li>
<li class="timeline-item timeline-item-transparent">
<span class="timeline-point timeline-point-info"></span>
<div class="timeline-event">
<div class="timeline-header mb-1">
<h6 class="mb-0 text-info">Zakończono tankowanie</h6>
<small class="text-muted">Wczoraj, 18:30:00</small>
</div>
<p class="mb-0 small">Przyrost poziomu z 15% na 65%.</p>
</div>
</li>
<li class="timeline-item timeline-item-transparent border-transparent">
<span class="timeline-point timeline-point-primary"></span>
<div class="timeline-event">
<div class="timeline-header mb-1">
<h6 class="mb-0">Regularny odczyt</h6>
<small class="text-muted">Wczoraj, 16:52:10</small>
</div>
<p class="mb-0 small">Poziom: 14% (370 l) | Ciśnienie: 4.5 bar | Bateria: 12.5V</p>
</div>
</li>
</ul>
</div>
</div>
</div>
</div>
</div>
<style>
/* CSS dla animacji zbiornika */
#liquidFill {
/* Symulacja falowania i wznoszenia płynu */
animation: fillTank 5s ease-in-out infinite alternate;
transform-origin: bottom;
}
@keyframes fillTank {
0% {
y: 135;
}
100% {
y: 125;
}
}
</style>
<script>
// Prosty skrypt symulujący ruch danych (Live)
setInterval(() => {
// Losuj delikatnie ciśnienie
let currentPressure = parseFloat(document.getElementById('pressureValue').innerText);
let diff = (Math.random() * 0.04 - 0.02);
document.getElementById('pressureValue').innerText = (currentPressure + diff).toFixed(2);
// Czasami zaktualizuj litry
if (Math.random() > 0.5) {
let currentVol = parseInt(document.getElementById('volumeValue').innerText.replace(/\s/g, ''));
currentVol += Math.floor(Math.random() * 3);
document.getElementById('volumeValue').innerText = currentVol.toLocaleString('pl-PL');
}
}, 2000);
</script>
<div class="content-backdrop fade"></div>
<?php include '../../footer.php'; ?>
</body>
</html>

View File

@@ -0,0 +1,107 @@
<?php
$enablePrototypeComments = true;
include '../../header-telemetria.php';
?>
<div class="container-xxl flex-grow-1 container-p-y">
<h4 class="fw-bold py-3 mb-4">
Telemetria <span class="text-muted fw-light">/ Instalacje</span>
</h4>
<div class="card">
<div class="card-header border-bottom d-flex flex-column flex-md-row justify-content-between align-items-md-center">
<input type="text" class="form-control form-control" placeholder="Szukaj instalacji..." style="width: 200px;">
<div class="d-flex flex-column flex-sm-row gap-2 mt-3 mt-md-0">
<select class="form-select form-select" style="width: 150px;">
<option value="">Wszystkie typy</option>
<option value="gas">Zbiornik Gazu</option>
<option value="farm">Kurnik / Farma</option>
</select>
<select class="form-select form-select" style="width: 150px;">
<option value="">Każdy status</option>
<option value="online">Online</option>
<option value="offline">Offline</option>
</select>
<a href="installation-add.php" class="btn btn-primary">
<i class="bx bx-plus-circle me-1"></i> Dodaj nową instalację
</a>
</div>
</div>
<div class="table-responsive text-nowrap">
<table class="table table-hover">
<thead>
<tr>
<th>Instalacja</th>
<th>Typ</th>
<th>Ostatni kontakt</th>
<th>Status</th>
<th>Akcje</th>
</tr>
</thead>
<tbody class="table-border-bottom-0">
<!-- Element 1 -->
<tr>
<td>
<strong>Zbiornik LPG #1</strong><br>
<small class="text-muted">Stalowa Wola, ul. Przemysłowa</small>
</td>
<td><span class="badge bg-label-warning"><i class="bx bxs-flame me-1"></i> Zbiornik Gazu</span></td>
<td>Przed 2 min</td>
<td><span class="badge bg-success">Online</span></td>
<td>
<a href="installation-live.php" class="btn btn-sm btn-icon btn-outline-primary" title="Podgląd Live">
<i class="bx bx-show"></i>
</a>
<a href="installation-edit.php" class="btn btn-sm btn-icon btn-outline-secondary" title="Edycja">
<i class="bx bx-pencil"></i>
</a>
</td>
</tr>
<!-- Element 2 -->
<tr>
<td>
<strong>Kurnik B (Sektor 2)</strong><br>
<small class="text-muted">Farma Wola, Obiekt Główny</small>
</td>
<td><span class="badge bg-label-info"><i class="bx bxs-thermometer me-1"></i> Sterownik Kurnika</span></td>
<td>Przed 1 min</td>
<td><span class="badge bg-success">Online</span></td>
<td>
<a href="installation-live.php" class="btn btn-sm btn-icon btn-outline-primary" title="Podgląd Live">
<i class="bx bx-show"></i>
</a>
<a href="installation-edit.php" class="btn btn-sm btn-icon btn-outline-secondary" title="Edycja">
<i class="bx bx-pencil"></i>
</a>
</td>
</tr>
<!-- Element 3 -->
<tr>
<td>
<strong>Zbiornik LPG #2</strong><br>
<small class="text-muted">Rzeszów, Baza Północ</small>
</td>
<td><span class="badge bg-label-warning"><i class="bx bxs-flame me-1"></i> Zbiornik Gazu</span></td>
<td>3 dni temu</td>
<td><span class="badge bg-danger">Offline</span></td>
<td>
<a href="installation-live.php" class="btn btn-sm btn-icon btn-outline-primary" title="Podgląd Live">
<i class="bx bx-show"></i>
</a>
<a href="installation-edit.php" class="btn btn-sm btn-icon btn-outline-secondary" title="Edycja">
<i class="bx bx-pencil"></i>
</a>
</td>
</tr>
</tbody>
</table>
</div>
</div>
</div>
<div class="content-backdrop fade"></div>
<?php include '../../footer.php'; ?>
</body>
</html>

View File

@@ -0,0 +1,41 @@
<?php
$currentPage = basename($_SERVER['PHP_SELF']);
?>
<aside id="layout-menu" class="layout-menu menu-vertical menu bg-menu-theme">
<div class="app-brand demo">
<a href="index.php" class="app-brand-link">
<span class="app-brand-text demo menu-text fw-bold ms-2">telemetria.magico</span>
</a>
<a href="javascript:void(0);" class="layout-menu-toggle menu-link text-large ms-auto">
<i class="bx bx-chevron-left bx-sm align-middle"></i>
</a>
</div>
<div class="menu-inner-shadow"></div>
<ul class="menu-inner py-1">
<li class="menu-header small text-uppercase">
<span class="menu-header-text" data-i18n="Telemetria">Telemetria</span>
</li>
<li class="menu-item <?php echo ($currentPage == 'index.php') ? 'active' : ''; ?>">
<a href="index.php" class="menu-link">
<i class="menu-icon tf-icons bx bx-home-circle"></i>
<div class="text-truncate" data-i18n="Dashboard">Dashboard</div>
</a>
</li>
<li class="menu-item <?php echo ($currentPage == 'installations.php') ? 'active' : ''; ?>">
<a href="installations.php" class="menu-link">
<i class="menu-icon tf-icons bx bx-buildings"></i>
<div class="text-truncate" data-i18n="Instalacje">Instalacje</div>
</a>
</li>
<li class="menu-item <?php echo ($currentPage == 'reports.php') ? 'active' : ''; ?>">
<a href="reports.php" class="menu-link">
<i class="menu-icon tf-icons bx bx-bar-chart-alt-2"></i>
<div class="text-truncate" data-i18n="Raporty">Raporty</div>
</a>
</li>
</ul>
</aside>

View File

@@ -0,0 +1,281 @@
<?php
$enablePrototypeComments = true;
include '../../header-telemetria.php';
?>
<div class="container-xxl flex-grow-1 container-p-y">
<!-- Vendors CSS dla Select2 i Flatpickr -->
<link rel="stylesheet" href="../../assets/vendor/libs/select2/select2.css" />
<link rel="stylesheet" href="../../assets/vendor/libs/flatpickr/flatpickr.css" />
<!-- Nagłówek i przyciski akcji -->
<div
class="d-flex flex-column flex-md-row justify-content-between align-items-start align-items-md-center mb-4 gap-3">
<div>
<h4 class="fw-bold mb-1">
<span class="text-muted fw-light">Raporty /</span> Historia zużycia medium
</h4>
<p class="text-muted mb-0">Zestawienie konsumpcji gazu / ubytków poziomu dla wskazanej instalacji we wskazanym okresie.</p>
</div>
<div class="d-flex gap-2">
<button class="btn btn-label-secondary">
<i class="bx bx-export me-1"></i> Eksportuj CSV
</button>
<button class="btn btn-primary">
<i class="bx bx-printer me-1"></i> Drukuj raport
</button>
</div>
</div>
<!-- Karta filtrów -->
<div class="card mb-4">
<div class="card-body">
<form class="row g-3">
<div class="col-12 col-md-3">
<label class="form-label" for="reportInstallation">Instalacja</label>
<select id="reportInstallation" class="form-select select2">
<option value="all">Wszystkie zebrane razem</option>
<option value="1" selected>Zbiornik LPG #1 (Stalowa Wola)</option>
<option value="2">Zbiornik LPG #2 (Rzeszów)</option>
<option value="3">Kurnik Główny</option>
</select>
</div>
<div class="col-12 col-md-3">
<label class="form-label" for="reportDateRange">Zakres dat</label>
<div class="input-group input-group-merge">
<span class="input-group-text"><i class="bx bx-calendar"></i></span>
<input type="text" id="reportDateRange" class="form-control flatpickr-range"
placeholder="Wybierz okres" value="2026-01-01 do 2026-06-30" />
</div>
</div>
<div class="col-12 col-md-3">
<label class="form-label" for="reportGranularity">Szczegółowość</label>
<select id="reportGranularity" class="form-select select2">
<option value="days">Dni</option>
<option value="weeks">Tygodnie</option>
<option value="months" selected>Miesiące</option>
</select>
<small id="daysLimitInfo" class="text-muted d-none mt-1 d-block">
<i class="bx bx-info-circle fs-6 align-middle me-1"></i>Maksymalny okres wynosi 366 dni.
</small>
</div>
<div class="col-12 col-md-3 d-flex align-items-end">
<button type="button" class="btn btn-primary w-100 w-md-auto">
<i class="bx bx-refresh me-1"></i> Generuj
</button>
</div>
</form>
</div>
</div>
<!-- Karta z wykresem -->
<div class="card mb-4">
<div class="card-header d-flex justify-content-between align-items-center">
<h5 class="card-title m-0">Zużycie medium w czasie (litry)</h5>
</div>
<div class="card-body">
<div id="consumptionReportChart" style="min-height: 350px;"></div>
</div>
</div>
<!-- Karta z tabelą danych -->
<div class="card">
<div class="card-header border-bottom">
<h5 class="card-title m-0">Szczegółowe zestawienie</h5>
</div>
<div class="table-responsive text-nowrap">
<table class="table table-hover">
<thead class="table-light">
<tr>
<th>Okres</th>
<th class="text-center">Liczba odczytów</th>
<th class="text-end">Stan początkowy</th>
<th class="text-end">Stan końcowy</th>
<th class="text-end">Całkowite zużycie</th>
</tr>
</thead>
<tbody>
<tr>
<td class="fw-medium">Styczeń 2026</td>
<td class="text-center">744</td>
<td class="text-end">2100 l (77%)</td>
<td class="text-end">1550 l (57%)</td>
<td class="text-end fw-bold text-danger">-550 l</td>
</tr>
<tr>
<td class="fw-medium">Luty 2026</td>
<td class="text-center">672</td>
<td class="text-end">1550 l (57%)</td>
<td class="text-end">850 l (31%)</td>
<td class="text-end fw-bold text-danger">-700 l</td>
</tr>
<tr class="table-primary bg-label-primary">
<td class="fw-medium">Marzec 2026 <i class="bx bx-gas-pump ms-1 text-primary" title="Zarejestrowano tankowanie"></i></td>
<td class="text-center">744</td>
<td class="text-end">850 l (31%)</td>
<td class="text-end">2500 l (92%)</td>
<td class="text-end fw-bold text-success">+1650 l</td>
</tr>
<tr>
<td class="fw-medium">Kwiecień 2026</td>
<td class="text-center">720</td>
<td class="text-end">2500 l (92%)</td>
<td class="text-end">2100 l (77%)</td>
<td class="text-end fw-bold text-danger">-400 l</td>
</tr>
<tr>
<td class="fw-medium">Maj 2026</td>
<td class="text-center">744</td>
<td class="text-end">2100 l (77%)</td>
<td class="text-end">1800 l (66%)</td>
<td class="text-end fw-bold text-danger">-300 l</td>
</tr>
<tr>
<td class="fw-medium">Czerwiec 2026</td>
<td class="text-center">360</td>
<td class="text-end">1800 l (66%)</td>
<td class="text-end">1620 l (60%)</td>
<td class="text-end fw-bold text-danger">-180 l</td>
</tr>
</tbody>
<tfoot class="table-light fw-bold">
<tr>
<td>Podsumowanie</td>
<td class="text-center">3984</td>
<td class="text-end text-muted">-</td>
<td class="text-end text-muted">-</td>
<td class="text-end text-primary fs-5">Zbilansowano: -480 l</td>
</tr>
</tfoot>
</table>
</div>
</div>
</div>
<?php include '../../footer.php'; ?>
<!-- Vendors JS -->
<script src="../../assets/vendor/libs/select2/select2.js"></script>
<script src="../../assets/vendor/libs/flatpickr/flatpickr.js"></script>
<!-- ApexCharts scripts and local chart init -->
<script src="../../assets/vendor/libs/apex-charts/apexcharts.js"></script>
<script>
document.addEventListener('DOMContentLoaded', function () {
// 1. Inicjalizacja Flatpickr dla zakresu dat
if (typeof flatpickr !== 'undefined') {
const dateRangeEl = document.querySelector('.flatpickr-range');
if (dateRangeEl) {
const fp = flatpickr(dateRangeEl, {
mode: 'range',
dateFormat: 'Y-m-d',
locale: 'pl'
});
}
}
// 2. Inicjalizacja Select2
if (typeof $ !== 'undefined' && $('.select2').length) {
$('.select2').select2({
minimumResultsForSearch: 5
});
// Obsługa pojawiania się podpowiedzi po wybraniu z selecta
$('#reportGranularity').on('change', function () {
const daysInfo = document.getElementById('daysLimitInfo');
if (this.value === 'days') {
daysInfo.classList.remove('d-none');
} else {
daysInfo.classList.add('d-none');
}
});
}
// 3. Inicjalizacja Wykresu
const chartEl = document.querySelector('#consumptionReportChart');
if (chartEl && typeof ApexCharts !== 'undefined') {
const chartConfig = {
chart: {
height: 350,
type: 'area', // Zmieniono na area, by ładnie pokazywać spadek/wzrost litrów
parentHeightOffset: 0,
toolbar: {
show: false
}
},
dataLabels: {
enabled: false
},
stroke: {
curve: 'smooth',
width: 3
},
series: [{
name: 'Stan początkowy w danym miesiącu (litry)',
data: [2100, 1550, 850, 2500, 2100, 1800]
}],
xaxis: {
categories: ['Styczeń', 'Luty', 'Marzec', 'Kwiecień', 'Maj', 'Czerwiec'],
axisBorder: {
show: false
},
axisTicks: {
show: false
},
labels: {
style: {
colors: '#a1acb8',
fontSize: '13px'
}
}
},
yaxis: {
labels: {
formatter: function (val) {
return val.toLocaleString('pl-PL') + " l";
},
style: {
colors: '#a1acb8',
fontSize: '13px'
}
}
},
colors: ['#00cfdd'], // Kolor np. cyan
fill: {
type: 'gradient',
gradient: {
shadeIntensity: 1,
opacityFrom: 0.5,
opacityTo: 0.1,
stops: [0, 90, 100]
}
},
grid: {
borderColor: '#e9ecef',
strokeDashArray: 4,
padding: {
top: -20,
bottom: -10,
left: 20,
right: 20
}
},
tooltip: {
y: {
formatter: function (val) {
return val.toLocaleString('pl-PL') + " l";
}
}
}
};
const consumptionChart = new ApexCharts(chartEl, chartConfig);
consumptionChart.render();
}
});
</script>
</body>
</html>

View File

@@ -0,0 +1,134 @@
<?php
$enablePrototypeComments = true;
include '../../header-telemetria.php';
?>
<div class="container-xxl flex-grow-1 container-p-y">
<!-- Header with description -->
<div class="card p-0 mb-4">
<div class="card-body d-flex flex-column flex-md-row justify-content-between p-0">
<!-- Left image -->
<div class="d-none d-md-flex app-academy-md-25 card-body py-0 pt-3 ps-5 align-items-center">
<img src="../../assets/img/illustrations/bulb-light.png" width="90" class="img-fluid" alt="Bulb"
data-app-light-img="illustrations/bulb-light.png" data-app-dark-img="illustrations/bulb-dark.png">
</div>
<!-- Center context -->
<div class="app-academy-md-50 card-body d-flex align-items-md-center flex-column text-md-center mb-1 py-5">
<h4 class="card-title mb-4 px-md-12 lh-base fw-bold">
Analizuj dane z instalacji<br>
w <span class="text-primary text-nowrap">przejrzystych raportach</span>.
</h4>
<p class="mb-4 col-md-10 text-muted">
Poniżej prezentujemy dostępne raporty telemetryczne. Wybierz zestawienie, aby
przeanalizować historię zużycia medium, zidentyfikować anomalie oraz zweryfikować
stabilność połączeń na Twoich obiektach.
</p>
</div>
<!-- Right image -->
<div class="d-none d-md-flex app-academy-md-25 align-items-end justify-content-end pe-4 pb-2">
<img src="../../assets/img/illustrations/pencil-rocket.png" alt="pencil rocket" height="150"
class="scaleX-n1-rtl">
</div>
</div>
</div>
<!-- Reports Gallery -->
<div class="card p-4">
<div class="row">
<!-- Raport historii zużycia -->
<div class="col-sm-6 col-lg-4 mb-4">
<div class="card p-2 h-100 shadow-none border border-primary">
<div class="card-body p-4 pt-2 d-flex flex-column">
<div class="d-flex justify-content-between align-items-center mb-4">
<span class="badge bg-label-primary">Eksploatacja</span>
</div>
<a href="report-consumption.php" class="h5 mb-2 fw-semibold text-primary">Historia zużycia medium</a>
<p class="mt-1 text-muted mb-4 flex-grow-1">
Wykres i tabela prezentująca ubytki / konsumpcję gazu (lub innego medium) dla wskazanej instalacji w danym przedziale czasowym. Oblicza dzienne i miesięczne zużycie.
</p>
<div class="d-flex flex-column flex-md-row gap-4 mt-auto">
<a class="w-100 btn btn-primary d-flex align-items-center justify-content-center"
href="report-consumption.php">
<span class="me-2">Wygeneruj</span>
<i class="bx bx-chevron-right icon-sm lh-1 scaleX-n1-rtl"></i>
</a>
</div>
</div>
</div>
</div>
<!-- Dziennik alertów i przekroczeń -->
<div class="col-sm-6 col-lg-4 mb-4">
<div class="card p-2 h-100 shadow-none border">
<div class="card-body p-4 pt-2 d-flex flex-column">
<div class="d-flex justify-content-between align-items-center mb-4">
<span class="badge bg-label-danger">Bezpieczeństwo</span>
</div>
<a href="javascript:void(0);" class="h5 mb-2 fw-semibold">Dziennik alertów</a>
<p class="mt-1 text-muted mb-4 flex-grow-1">
Ewidencja wszystkich wygenerowanych alarmów, w tym przekroczeń ciśnienia, krytycznie niskich stanów, z datami ich wystąpienia i powiadomionymi osobami.
</p>
<div class="d-flex flex-column flex-md-row gap-4 mt-auto">
<a class="w-100 btn btn-label-primary d-flex align-items-center justify-content-center"
href="javascript:void(0);">
<span class="me-2">Wygeneruj</span>
<i class="bx bx-chevron-right icon-sm lh-1 scaleX-n1-rtl"></i>
</a>
</div>
</div>
</div>
</div>
<!-- Analiza stabilności infrastruktury -->
<div class="col-sm-6 col-lg-4 mb-4">
<div class="card p-2 h-100 shadow-none border">
<div class="card-body p-4 pt-2 d-flex flex-column">
<div class="d-flex justify-content-between align-items-center mb-4">
<span class="badge bg-label-warning">Serwis i Sprzęt</span>
</div>
<a href="javascript:void(0);" class="h5 mb-2 fw-semibold">Analiza stabilności sieci</a>
<p class="mt-1 text-muted mb-4 flex-grow-1">
Monitorowanie przerw w łączności GSM, skoków napięcia i krzywej rozładowania baterii. Służy do planowania prac serwisowych (Predictive Maintenance).
</p>
<div class="d-flex flex-column flex-md-row gap-4 mt-auto">
<a class="w-100 btn btn-label-primary d-flex align-items-center justify-content-center"
href="javascript:void(0);">
<span class="me-2">Wygeneruj</span>
<i class="bx bx-chevron-right icon-sm lh-1 scaleX-n1-rtl"></i>
</a>
</div>
</div>
</div>
</div>
<!-- Sumaryczny bilans instalacji -->
<div class="col-sm-6 col-lg-4 mb-4">
<div class="card p-2 h-100 shadow-none border">
<div class="card-body p-4 pt-2 d-flex flex-column">
<div class="d-flex justify-content-between align-items-center mb-4">
<span class="badge bg-label-info">Podsumowanie całkowite</span>
</div>
<a href="javascript:void(0);" class="h5 mb-2 fw-semibold">Bilans flotowy</a>
<p class="mt-1 text-muted mb-4 flex-grow-1">
Całkowite zestawienie posiadanych zasobów na dany dzień sumuje dostępne litry gazu we wszystkich połączonych zbiornikach, ułatwiając logistykę tankowania.
</p>
<div class="d-flex flex-column flex-md-row gap-4 mt-auto">
<a class="w-100 btn btn-label-primary d-flex align-items-center justify-content-center"
href="javascript:void(0);">
<span class="me-2">Wygeneruj</span>
<i class="bx bx-chevron-right icon-sm lh-1 scaleX-n1-rtl"></i>
</a>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
<?php include '../../footer.php'; ?>