Merge branches 'master', 'master' and 'master' of git.reim.ar:ReiMerc/temperature-alarm
This commit is contained in:
commit
c1fbebe0da
@ -79,19 +79,9 @@ namespace Api.BusinessLogic
|
||||
|
||||
if (device == null) { return new ConflictObjectResult(new { message = "Could not find device" }); }
|
||||
|
||||
var logs = await _dbAccess.ReadLogs(deviceId);
|
||||
var logs = await _dbAccess.ReadLogs(deviceId, dateTimeRange);
|
||||
|
||||
if (logs.Count == 0) { return new ConflictObjectResult(new { message = "Could not find any logs connected to the device" }); }
|
||||
|
||||
if (dateTimeRange.DateTimeStart == dateTimeRange.DateTimeEnd) { return new OkObjectResult(logs); }
|
||||
|
||||
List<TemperatureLogs> rangedLogs = new List<TemperatureLogs>();
|
||||
foreach (var log in logs)
|
||||
{
|
||||
if (log.Date <= dateTimeRange.DateTimeStart && log.Date >= dateTimeRange.DateTimeEnd) { rangedLogs.Add(log); }
|
||||
}
|
||||
|
||||
return new OkObjectResult(rangedLogs);
|
||||
return new OkObjectResult(logs);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
|
@ -56,8 +56,8 @@ namespace Api.Controllers
|
||||
}
|
||||
else
|
||||
{
|
||||
dateTimeRange.DateTimeStart = DateTime.Now;
|
||||
dateTimeRange.DateTimeEnd = dateTimeRange.DateTimeStart;
|
||||
dateTimeRange.DateTimeStart = DateTime.UnixEpoch;
|
||||
dateTimeRange.DateTimeEnd = DateTime.Now;
|
||||
}
|
||||
return await _deviceLogic.GetLogs(dateTimeRange, deviceId);
|
||||
}
|
||||
|
@ -313,16 +313,11 @@ namespace Api.DBAccess
|
||||
/// Returns the logs from the device
|
||||
/// </summary>
|
||||
/// <param name="deviceId">Has the id for the device that the los belong too</param>
|
||||
/// <param name="range">Return only logs within the specified datetime range</param>
|
||||
/// <returns></returns>
|
||||
public async Task<List<TemperatureLogs>> ReadLogs(int deviceId)
|
||||
public async Task<List<TemperatureLogs>> ReadLogs(int deviceId, DateTimeRange range)
|
||||
{
|
||||
var device = await _context.Devices.Include(d => d.Logs).FirstOrDefaultAsync(d => d.Id == deviceId);
|
||||
|
||||
if (device == null || device.Logs == null) { return new List<TemperatureLogs>(); }
|
||||
|
||||
var logs = device.Logs;
|
||||
|
||||
return logs;
|
||||
return _context.Devices.Include(d => d.Logs.Where(l => l.Date > range.DateTimeStart && l.Date < range.DateTimeEnd)).Where(d => d.Id == deviceId).FirstOrDefault().Logs;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
|
@ -1,4 +1,5 @@
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Api.Models;
|
||||
using Api.Models.Users;
|
||||
using Api.Models.Devices;
|
||||
|
||||
|
140
backend/Api/Migrations/20250402113522_ChangeModelsNamespace.Designer.cs
generated
Normal file
140
backend/Api/Migrations/20250402113522_ChangeModelsNamespace.Designer.cs
generated
Normal file
@ -0,0 +1,140 @@
|
||||
// <auto-generated />
|
||||
using System;
|
||||
using Api;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Microsoft.EntityFrameworkCore.Infrastructure;
|
||||
using Microsoft.EntityFrameworkCore.Migrations;
|
||||
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
|
||||
|
||||
#nullable disable
|
||||
|
||||
namespace Api.Migrations
|
||||
{
|
||||
[DbContext(typeof(DBContext))]
|
||||
[Migration("20250402113522_ChangeModelsNamespace")]
|
||||
partial class ChangeModelsNamespace
|
||||
{
|
||||
/// <inheritdoc />
|
||||
protected override void BuildTargetModel(ModelBuilder modelBuilder)
|
||||
{
|
||||
#pragma warning disable 612, 618
|
||||
modelBuilder.HasAnnotation("ProductVersion", "9.0.3");
|
||||
|
||||
modelBuilder.Entity("Api.Models.Devices.Device", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("INTEGER");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.IsRequired()
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<string>("ReferenceId")
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<double>("TempHigh")
|
||||
.HasColumnType("REAL");
|
||||
|
||||
b.Property<double>("TempLow")
|
||||
.HasColumnType("REAL");
|
||||
|
||||
b.Property<int?>("UserId")
|
||||
.HasColumnType("INTEGER");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("UserId");
|
||||
|
||||
b.ToTable("Devices");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.TemperatureLogs", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("INTEGER");
|
||||
|
||||
b.Property<DateTime>("Date")
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<int>("DeviceId")
|
||||
.HasColumnType("INTEGER");
|
||||
|
||||
b.Property<double>("TempHigh")
|
||||
.HasColumnType("REAL");
|
||||
|
||||
b.Property<double>("TempLow")
|
||||
.HasColumnType("REAL");
|
||||
|
||||
b.Property<double>("Temperature")
|
||||
.HasColumnType("REAL");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("DeviceId");
|
||||
|
||||
b.ToTable("TemperatureLogs");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.Users.User", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("INTEGER");
|
||||
|
||||
b.Property<string>("Email")
|
||||
.IsRequired()
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<string>("Password")
|
||||
.IsRequired()
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<string>("RefreshToken")
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<DateTime>("RefreshTokenExpiresAt")
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<string>("Salt")
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<string>("UserName")
|
||||
.IsRequired()
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.ToTable("Users");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.Devices.Device", b =>
|
||||
{
|
||||
b.HasOne("Api.Models.Users.User", null)
|
||||
.WithMany("Devices")
|
||||
.HasForeignKey("UserId");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.TemperatureLogs", b =>
|
||||
{
|
||||
b.HasOne("Api.Models.Devices.Device", null)
|
||||
.WithMany("Logs")
|
||||
.HasForeignKey("DeviceId")
|
||||
.OnDelete(DeleteBehavior.Cascade)
|
||||
.IsRequired();
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.Devices.Device", b =>
|
||||
{
|
||||
b.Navigation("Logs");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.Users.User", b =>
|
||||
{
|
||||
b.Navigation("Devices");
|
||||
});
|
||||
#pragma warning restore 612, 618
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,59 @@
|
||||
using Microsoft.EntityFrameworkCore.Migrations;
|
||||
|
||||
#nullable disable
|
||||
|
||||
namespace Api.Migrations
|
||||
{
|
||||
/// <inheritdoc />
|
||||
public partial class ChangeModelsNamespace : Migration
|
||||
{
|
||||
/// <inheritdoc />
|
||||
protected override void Up(MigrationBuilder migrationBuilder)
|
||||
{
|
||||
migrationBuilder.DropForeignKey(
|
||||
name: "FK_TemperatureLogs_Devices_DeviceId",
|
||||
table: "TemperatureLogs");
|
||||
|
||||
migrationBuilder.AlterColumn<int>(
|
||||
name: "DeviceId",
|
||||
table: "TemperatureLogs",
|
||||
type: "INTEGER",
|
||||
nullable: false,
|
||||
defaultValue: 0,
|
||||
oldClrType: typeof(int),
|
||||
oldType: "INTEGER",
|
||||
oldNullable: true);
|
||||
|
||||
migrationBuilder.AddForeignKey(
|
||||
name: "FK_TemperatureLogs_Devices_DeviceId",
|
||||
table: "TemperatureLogs",
|
||||
column: "DeviceId",
|
||||
principalTable: "Devices",
|
||||
principalColumn: "Id",
|
||||
onDelete: ReferentialAction.Cascade);
|
||||
}
|
||||
|
||||
/// <inheritdoc />
|
||||
protected override void Down(MigrationBuilder migrationBuilder)
|
||||
{
|
||||
migrationBuilder.DropForeignKey(
|
||||
name: "FK_TemperatureLogs_Devices_DeviceId",
|
||||
table: "TemperatureLogs");
|
||||
|
||||
migrationBuilder.AlterColumn<int>(
|
||||
name: "DeviceId",
|
||||
table: "TemperatureLogs",
|
||||
type: "INTEGER",
|
||||
nullable: true,
|
||||
oldClrType: typeof(int),
|
||||
oldType: "INTEGER");
|
||||
|
||||
migrationBuilder.AddForeignKey(
|
||||
name: "FK_TemperatureLogs_Devices_DeviceId",
|
||||
table: "TemperatureLogs",
|
||||
column: "DeviceId",
|
||||
principalTable: "Devices",
|
||||
principalColumn: "Id");
|
||||
}
|
||||
}
|
||||
}
|
138
backend/Api/Migrations/20250403070932_RemovedLogsFromDbset.Designer.cs
generated
Normal file
138
backend/Api/Migrations/20250403070932_RemovedLogsFromDbset.Designer.cs
generated
Normal file
@ -0,0 +1,138 @@
|
||||
// <auto-generated />
|
||||
using System;
|
||||
using Api;
|
||||
using Microsoft.EntityFrameworkCore;
|
||||
using Microsoft.EntityFrameworkCore.Infrastructure;
|
||||
using Microsoft.EntityFrameworkCore.Migrations;
|
||||
using Microsoft.EntityFrameworkCore.Storage.ValueConversion;
|
||||
|
||||
#nullable disable
|
||||
|
||||
namespace Api.Migrations
|
||||
{
|
||||
[DbContext(typeof(DBContext))]
|
||||
[Migration("20250403070932_RemovedLogsFromDbset")]
|
||||
partial class RemovedLogsFromDbset
|
||||
{
|
||||
/// <inheritdoc />
|
||||
protected override void BuildTargetModel(ModelBuilder modelBuilder)
|
||||
{
|
||||
#pragma warning disable 612, 618
|
||||
modelBuilder.HasAnnotation("ProductVersion", "9.0.3");
|
||||
|
||||
modelBuilder.Entity("Api.Models.Devices.Device", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("INTEGER");
|
||||
|
||||
b.Property<string>("Name")
|
||||
.IsRequired()
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<string>("ReferenceId")
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<double>("TempHigh")
|
||||
.HasColumnType("REAL");
|
||||
|
||||
b.Property<double>("TempLow")
|
||||
.HasColumnType("REAL");
|
||||
|
||||
b.Property<int?>("UserId")
|
||||
.HasColumnType("INTEGER");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("UserId");
|
||||
|
||||
b.ToTable("Devices");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.TemperatureLogs", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("INTEGER");
|
||||
|
||||
b.Property<DateTime>("Date")
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<int?>("DeviceId")
|
||||
.HasColumnType("INTEGER");
|
||||
|
||||
b.Property<double>("TempHigh")
|
||||
.HasColumnType("REAL");
|
||||
|
||||
b.Property<double>("TempLow")
|
||||
.HasColumnType("REAL");
|
||||
|
||||
b.Property<double>("Temperature")
|
||||
.HasColumnType("REAL");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.HasIndex("DeviceId");
|
||||
|
||||
b.ToTable("TemperatureLogs");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.Users.User", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
.HasColumnType("INTEGER");
|
||||
|
||||
b.Property<string>("Email")
|
||||
.IsRequired()
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<string>("Password")
|
||||
.IsRequired()
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<string>("RefreshToken")
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<DateTime>("RefreshTokenExpiresAt")
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<string>("Salt")
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.Property<string>("UserName")
|
||||
.IsRequired()
|
||||
.HasColumnType("TEXT");
|
||||
|
||||
b.HasKey("Id");
|
||||
|
||||
b.ToTable("Users");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.Devices.Device", b =>
|
||||
{
|
||||
b.HasOne("Api.Models.Users.User", null)
|
||||
.WithMany("Devices")
|
||||
.HasForeignKey("UserId");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.TemperatureLogs", b =>
|
||||
{
|
||||
b.HasOne("Api.Models.Devices.Device", null)
|
||||
.WithMany("Logs")
|
||||
.HasForeignKey("DeviceId");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.Devices.Device", b =>
|
||||
{
|
||||
b.Navigation("Logs");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.Users.User", b =>
|
||||
{
|
||||
b.Navigation("Devices");
|
||||
});
|
||||
#pragma warning restore 612, 618
|
||||
}
|
||||
}
|
||||
}
|
@ -0,0 +1,59 @@
|
||||
using Microsoft.EntityFrameworkCore.Migrations;
|
||||
|
||||
#nullable disable
|
||||
|
||||
namespace Api.Migrations
|
||||
{
|
||||
/// <inheritdoc />
|
||||
public partial class RemovedLogsFromDbset : Migration
|
||||
{
|
||||
/// <inheritdoc />
|
||||
protected override void Up(MigrationBuilder migrationBuilder)
|
||||
{
|
||||
migrationBuilder.DropForeignKey(
|
||||
name: "FK_TemperatureLogs_Devices_DeviceId",
|
||||
table: "TemperatureLogs");
|
||||
|
||||
migrationBuilder.AlterColumn<int>(
|
||||
name: "DeviceId",
|
||||
table: "TemperatureLogs",
|
||||
type: "INTEGER",
|
||||
nullable: true,
|
||||
oldClrType: typeof(int),
|
||||
oldType: "INTEGER");
|
||||
|
||||
migrationBuilder.AddForeignKey(
|
||||
name: "FK_TemperatureLogs_Devices_DeviceId",
|
||||
table: "TemperatureLogs",
|
||||
column: "DeviceId",
|
||||
principalTable: "Devices",
|
||||
principalColumn: "Id");
|
||||
}
|
||||
|
||||
/// <inheritdoc />
|
||||
protected override void Down(MigrationBuilder migrationBuilder)
|
||||
{
|
||||
migrationBuilder.DropForeignKey(
|
||||
name: "FK_TemperatureLogs_Devices_DeviceId",
|
||||
table: "TemperatureLogs");
|
||||
|
||||
migrationBuilder.AlterColumn<int>(
|
||||
name: "DeviceId",
|
||||
table: "TemperatureLogs",
|
||||
type: "INTEGER",
|
||||
nullable: false,
|
||||
defaultValue: 0,
|
||||
oldClrType: typeof(int),
|
||||
oldType: "INTEGER",
|
||||
oldNullable: true);
|
||||
|
||||
migrationBuilder.AddForeignKey(
|
||||
name: "FK_TemperatureLogs_Devices_DeviceId",
|
||||
table: "TemperatureLogs",
|
||||
column: "DeviceId",
|
||||
principalTable: "Devices",
|
||||
principalColumn: "Id",
|
||||
onDelete: ReferentialAction.Cascade);
|
||||
}
|
||||
}
|
||||
}
|
@ -17,7 +17,7 @@ namespace Api.Migrations
|
||||
#pragma warning disable 612, 618
|
||||
modelBuilder.HasAnnotation("ProductVersion", "9.0.3");
|
||||
|
||||
modelBuilder.Entity("Api.Models.Device", b =>
|
||||
modelBuilder.Entity("Api.Models.Devices.Device", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
@ -74,7 +74,7 @@ namespace Api.Migrations
|
||||
b.ToTable("TemperatureLogs");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.User", b =>
|
||||
modelBuilder.Entity("Api.Models.Users.User", b =>
|
||||
{
|
||||
b.Property<int>("Id")
|
||||
.ValueGeneratedOnAdd()
|
||||
@ -106,26 +106,26 @@ namespace Api.Migrations
|
||||
b.ToTable("Users");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.Device", b =>
|
||||
modelBuilder.Entity("Api.Models.Devices.Device", b =>
|
||||
{
|
||||
b.HasOne("Api.Models.User", null)
|
||||
b.HasOne("Api.Models.Users.User", null)
|
||||
.WithMany("Devices")
|
||||
.HasForeignKey("UserId");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.TemperatureLogs", b =>
|
||||
{
|
||||
b.HasOne("Api.Models.Device", null)
|
||||
b.HasOne("Api.Models.Devices.Device", null)
|
||||
.WithMany("Logs")
|
||||
.HasForeignKey("DeviceId");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.Device", b =>
|
||||
modelBuilder.Entity("Api.Models.Devices.Device", b =>
|
||||
{
|
||||
b.Navigation("Logs");
|
||||
});
|
||||
|
||||
modelBuilder.Entity("Api.Models.User", b =>
|
||||
modelBuilder.Entity("Api.Models.Users.User", b =>
|
||||
{
|
||||
b.Navigation("Devices");
|
||||
});
|
||||
|
BIN
docs/Domainmodel.png
Normal file
BIN
docs/Domainmodel.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 176 KiB |
Binary file not shown.
Before Width: | Height: | Size: 58 KiB After Width: | Height: | Size: 136 KiB |
@ -5,8 +5,8 @@
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
|
||||
<title>Temperature-Alarm-Web</title>
|
||||
<script defer src="https://cdn.jsdelivr.net/npm/chart.js"></script>
|
||||
<script defer src="https://cdn.jsdelivr.net/npm/moment"></script>
|
||||
<script defer src="https://cdn.jsdelivr.net/npm/chartjs-adapter-moment"></script>
|
||||
<script defer src="https://cdn.jsdelivr.net/npm/luxon"></script>
|
||||
<script defer src="https://cdn.jsdelivr.net/npm/chartjs-adapter-luxon"></script>
|
||||
<script defer type="module" src="/scripts/home.js"></script>
|
||||
<link rel="stylesheet" href="/styles/common.css">
|
||||
<link rel="stylesheet" href="/styles/home.css" />
|
||||
@ -26,8 +26,40 @@
|
||||
</div>
|
||||
<div id="container">
|
||||
<div class="chart-container">
|
||||
<canvas id="myChart" style="width: 49%; height: 49%;"></canvas>
|
||||
<canvas id="chart" height="400"></canvas>
|
||||
</div>
|
||||
|
||||
<div id="filters">
|
||||
<div>
|
||||
<div class="form-control">
|
||||
<label for="device-selector">Select device</label>
|
||||
<select id="device-selector"></select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div>
|
||||
<div class="form-control">
|
||||
<label> </label>
|
||||
<div id="period-templates">
|
||||
<div class="period-template last-x-days" data-days="0">Today</div>
|
||||
<div class="period-template last-x-days" data-days="3">Last 3 days</div>
|
||||
<div class="period-template last-x-days" data-days="7">Last week</div>
|
||||
<div class="period-template" id="all-time">All time</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="form-control">
|
||||
<label for="period-start">Start</label>
|
||||
<input id="period-start" type="datetime-local">
|
||||
</div>
|
||||
|
||||
<div class="form-control">
|
||||
<label for="period-end">End</label>
|
||||
<input id="period-end" type="datetime-local">
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div id="table-wrapper">
|
||||
<table>
|
||||
<thead>
|
||||
|
@ -2,40 +2,7 @@ import { logout } from "../shared/utils.js";
|
||||
import { getUser } from "../shared/utils.js";
|
||||
import { getDevices, getLogsOnDeviceId } from "./services/devices.service.js";
|
||||
|
||||
async function buildChart(data) {
|
||||
const xValues = data.map((log) =>
|
||||
new Date(log.date).toLocaleString()
|
||||
); // Full Date labels
|
||||
const yValues = data.map((log) => log.temperature); // Temperature values
|
||||
new Chart("myChart", {
|
||||
type: "line",
|
||||
data: {
|
||||
labels: xValues,
|
||||
datasets: [
|
||||
{
|
||||
label: "Temperature",
|
||||
fill: false,
|
||||
lineTension: 0.4,
|
||||
backgroundColor: "rgba(0,0,255,1.0)",
|
||||
borderColor: "rgba(0,0,255,0.1)",
|
||||
data: yValues,
|
||||
},
|
||||
],
|
||||
},
|
||||
options: {
|
||||
tooltips: {
|
||||
callbacks: {
|
||||
title: function (tooltipItem) {
|
||||
return `Date: ${tooltipItem[0].label}`;
|
||||
},
|
||||
label: function (tooltipItem) {
|
||||
return `Temperature: ${tooltipItem.value}°C`;
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
});
|
||||
}
|
||||
let chart;
|
||||
|
||||
const TABLE_PAGINATION_SIZE = 30;
|
||||
|
||||
@ -62,8 +29,9 @@ function buildTable(data, offset = 0) {
|
||||
color = "tempNormal";
|
||||
}
|
||||
|
||||
const date = new Date(log.date).toLocaleDateString();
|
||||
const time = new Date(log.date).toLocaleTimeString();
|
||||
const parsedDate = luxon.DateTime.fromISO(log.date).setZone("Europe/Copenhagen").setLocale("gb");
|
||||
const date = parsedDate.toLocaleString(luxon.DateTime.DATE_SHORT);
|
||||
const time = parsedDate.toLocaleString(luxon.DateTime.TIME_WITH_SECONDS);
|
||||
|
||||
document.getElementById("table-body").innerHTML += `
|
||||
<tr>
|
||||
@ -97,14 +65,39 @@ function handleError(err) {
|
||||
document.getElementById("container").style.display = "none";
|
||||
}
|
||||
|
||||
async function fetchData(startDate = null, endDate = null) {
|
||||
const devices = await getDevices()
|
||||
.catch(handleError);
|
||||
function addDeviceToDropdown(device) {
|
||||
const opt = document.createElement("option");
|
||||
opt.innerText = `${device.name} (${device.referenceId})`;
|
||||
opt.value = device.id;
|
||||
document.getElementById("device-selector").appendChild(opt);
|
||||
}
|
||||
|
||||
function randomColorChannelValue() {
|
||||
return Math.floor(Math.random() * 256);
|
||||
}
|
||||
|
||||
function isSameDay(a, b) {
|
||||
return a.getFullYear() === b.getFullYear() &&
|
||||
a.getMonth() === b.getMonth() &&
|
||||
a.getDate() === b.getDate();
|
||||
}
|
||||
|
||||
function localToUTC(date) {
|
||||
if (!date) return null;
|
||||
|
||||
return luxon.DateTime.fromISO(date, { zone: "Europe/Copenhagen" }).setZone("UTC");
|
||||
}
|
||||
|
||||
async function fetchData() {
|
||||
document.body.classList.add("loading");
|
||||
|
||||
const startDate = localToUTC(document.getElementById("period-start").value);
|
||||
const endDate = localToUTC(document.getElementById("period-end").value);
|
||||
|
||||
const deviceData = [];
|
||||
|
||||
for (const device of devices) {
|
||||
const data = await getLogsOnDeviceId(device.id)
|
||||
const data = await getLogsOnDeviceId(device.id, startDate, endDate)
|
||||
.catch(handleError);
|
||||
|
||||
deviceData.push(data);
|
||||
@ -114,47 +107,114 @@ async function fetchData(startDate = null, endDate = null) {
|
||||
return;
|
||||
}
|
||||
|
||||
document.getElementById("table-body").innerHTML = "";
|
||||
buildTable(deviceData[0]);
|
||||
|
||||
new Chart("myChart", {
|
||||
type: "line",
|
||||
data: {
|
||||
datasets: deviceData.map(dataset => ({
|
||||
label: "Temperature",
|
||||
fill: false,
|
||||
lineTension: 0.4,
|
||||
backgroundColor: "rgba(0,0,255,1.0)",
|
||||
borderColor: "rgba(0,0,255,0.1)",
|
||||
data: dataset.map(log => ({
|
||||
x: new Date(log.date).getTime(),
|
||||
y: log.temperature,
|
||||
})),
|
||||
})),
|
||||
},
|
||||
options: {
|
||||
parsing: false,
|
||||
plugins: {
|
||||
tooltip: {
|
||||
callbacks: {
|
||||
label: item => `Temperature: ${item.formattedValue}°C`,
|
||||
if (!chart) {
|
||||
chart = new Chart("chart", {
|
||||
type: "line",
|
||||
data: {
|
||||
datasets: [],
|
||||
},
|
||||
options: {
|
||||
responsive: false,
|
||||
parsing: false,
|
||||
plugins: {
|
||||
tooltip: {
|
||||
callbacks: {
|
||||
label: item => `Temperature: ${item.formattedValue}°C`,
|
||||
},
|
||||
},
|
||||
decimation: {
|
||||
enabled: true,
|
||||
algorithm: "lttb",
|
||||
samples: window.innerWidth / 2,
|
||||
},
|
||||
},
|
||||
decimation: {
|
||||
enabled: true,
|
||||
algorithm: "lttb",
|
||||
samples: window.innerWidth / 2,
|
||||
scales: {
|
||||
x: {
|
||||
type: "time",
|
||||
time: {
|
||||
displayFormats: {
|
||||
hour: "HH:mm",
|
||||
},
|
||||
},
|
||||
adapters: {
|
||||
date: {
|
||||
zone: "Europe/Copenhagen",
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
},
|
||||
scales: {
|
||||
x: {
|
||||
type: "time",
|
||||
},
|
||||
},
|
||||
},
|
||||
});
|
||||
}
|
||||
|
||||
chart.options.scales.x.time.unit = isSameDay(new Date(startDate), new Date(endDate))
|
||||
? "hour"
|
||||
: "day";
|
||||
|
||||
chart.data.datasets = deviceData.map((dataset, i) => {
|
||||
const color = new Array(3)
|
||||
.fill(null)
|
||||
.map(randomColorChannelValue)
|
||||
.join(",");
|
||||
|
||||
return {
|
||||
label: devices[i].name,
|
||||
fill: false,
|
||||
lineTension: 0.4,
|
||||
backgroundColor: `rgba(${color}, 1.0)`,
|
||||
borderColor: `rgba(${color}, 0.1)`,
|
||||
data: dataset.map(log => ({
|
||||
x: new Date(log.date).getTime(),
|
||||
y: log.temperature,
|
||||
})),
|
||||
};
|
||||
});
|
||||
|
||||
chart.update();
|
||||
|
||||
document.body.classList.remove("loading");
|
||||
}
|
||||
|
||||
fetchData();
|
||||
function setPeriod(start, end) {
|
||||
const startDate = start && new Date(start);
|
||||
startDate?.setMinutes(startDate.getMinutes() - startDate.getTimezoneOffset());
|
||||
|
||||
const endDate = start && new Date(end);
|
||||
endDate?.setMinutes(endDate.getMinutes() - endDate.getTimezoneOffset());
|
||||
|
||||
document.getElementById("period-start").value = startDate?.toISOString().slice(0, 16);
|
||||
document.getElementById("period-end").value = endDate?.toISOString().slice(0, 16);
|
||||
|
||||
fetchData();
|
||||
}
|
||||
|
||||
function setPeriodLastDays(days) {
|
||||
const start = new Date()
|
||||
start.setDate(new Date().getDate() - days);
|
||||
start.setHours(0, 0, 0, 0);
|
||||
setPeriod(start, new Date().setHours(23, 59, 0, 0));
|
||||
}
|
||||
|
||||
for (const elem of document.getElementsByClassName("last-x-days")) {
|
||||
elem.onclick = event => setPeriodLastDays(event.target.dataset.days);
|
||||
}
|
||||
|
||||
for (const elem of document.querySelectorAll("#period-start, #period-end")) {
|
||||
elem.onchange = fetchData;
|
||||
}
|
||||
|
||||
document.getElementById("all-time").onclick = () => setPeriod(null, null);
|
||||
|
||||
document.querySelector(".logout-container").addEventListener("click", logout);
|
||||
|
||||
const devices = await getDevices().catch(handleError);
|
||||
for (const device of devices) {
|
||||
addDeviceToDropdown(device);
|
||||
}
|
||||
|
||||
setPeriodLastDays(3);
|
||||
fetchData();
|
||||
|
||||
|
@ -20,6 +20,7 @@ export function update(deviceId, name, temphigh, tempLow) {
|
||||
});
|
||||
}
|
||||
|
||||
export function getLogsOnDeviceId(id) {
|
||||
return request("GET", `/device/logs/${id}`);
|
||||
export function getLogsOnDeviceId(id, startDate = null, endDate = null) {
|
||||
const query = startDate && endDate ? `?dateTimeStart=${startDate}&dateTimeEnd=${endDate}` : "";
|
||||
return request("GET", `/device/logs/${id}${query}`);
|
||||
}
|
||||
|
@ -4,6 +4,18 @@ body {
|
||||
background-color: #F9F9F9;
|
||||
}
|
||||
|
||||
.loading * {
|
||||
pointer-events: none;
|
||||
}
|
||||
|
||||
.loading #chart, .loading table {
|
||||
opacity: 50%;
|
||||
}
|
||||
|
||||
.loading select, .loading input {
|
||||
color: rgba(0, 0, 0, 0.3);
|
||||
}
|
||||
|
||||
#container {
|
||||
margin: 0 2rem;
|
||||
}
|
||||
@ -13,16 +25,30 @@ body {
|
||||
border-radius: 8px;
|
||||
box-shadow: 2px 2px 2px rgba(0, 0, 0, 0.1);
|
||||
border: 1px solid #DDD;
|
||||
margin-bottom: 2rem;
|
||||
background-color: white;
|
||||
}
|
||||
|
||||
#filters {
|
||||
margin: 2rem auto;
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
}
|
||||
|
||||
#filters > * {
|
||||
display: flex;
|
||||
gap: 1rem;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
table {
|
||||
font-family: arial, sans-serif;
|
||||
border-collapse: collapse;
|
||||
width: 100%;
|
||||
background-color: white;
|
||||
color: #616161;
|
||||
font-family: arial, sans-serif;
|
||||
border-collapse: collapse;
|
||||
transition: opacity ease-in 100ms;
|
||||
}
|
||||
|
||||
td,
|
||||
@ -50,6 +76,9 @@ table .temperature {
|
||||
table tr:not(:last-child) .temperature {
|
||||
border-bottom-color: white;
|
||||
}
|
||||
td {
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.tempHigh {
|
||||
background-color: #ff0000;
|
||||
@ -82,6 +111,12 @@ table tr:not(:last-child) .temperature {
|
||||
box-shadow: 2px 2px 2px rgba(0, 0, 0, 0.1);
|
||||
}
|
||||
|
||||
#chart {
|
||||
width: 100%;
|
||||
height: 400px;
|
||||
transition: opacity ease-in 100ms;
|
||||
}
|
||||
|
||||
#error {
|
||||
margin: 2rem;
|
||||
}
|
||||
@ -92,6 +127,7 @@ table tr:not(:last-child) .temperature {
|
||||
padding: 0.5rem;
|
||||
color: #616161;
|
||||
background-color: white;
|
||||
font-size: 0.85rem;
|
||||
cursor: pointer;
|
||||
transition-duration: 100ms;
|
||||
}
|
||||
@ -104,3 +140,43 @@ table tr:not(:last-child) .temperature {
|
||||
background-color: #FCFCFC;
|
||||
}
|
||||
|
||||
.form-control {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
}
|
||||
|
||||
label {
|
||||
font-size: 0.85rem;
|
||||
padding-left: 0.2rem;
|
||||
color: #616161;
|
||||
}
|
||||
|
||||
select, input {
|
||||
background-color: white;
|
||||
color: #424242;
|
||||
border: 1px solid #DDD;
|
||||
border-radius: 8px;
|
||||
padding: 0.5rem 1rem;
|
||||
transition: color ease-in 100ms;
|
||||
}
|
||||
|
||||
select:focus, input:focus {
|
||||
border-color: #1E88E5;
|
||||
}
|
||||
|
||||
#period-templates {
|
||||
display: flex;
|
||||
gap: 1rem;
|
||||
}
|
||||
|
||||
.period-template {
|
||||
font-size: 0.85rem;
|
||||
color: #616161;
|
||||
cursor: pointer;
|
||||
transition-duration: 100ms;
|
||||
}
|
||||
|
||||
.period-template:hover {
|
||||
color: #212121;
|
||||
}
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user