Merge branches 'master', 'master' and 'master' of git.reim.ar:ReiMerc/temperature-alarm

This commit is contained in:
LilleBRG 2025-04-03 14:37:07 +02:00
commit c1fbebe0da
15 changed files with 658 additions and 107 deletions

View File

@ -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>

View File

@ -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);
}

View File

@ -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>

View File

@ -1,4 +1,5 @@
using Microsoft.EntityFrameworkCore;
using Api.Models;
using Api.Models.Users;
using Api.Models.Devices;

View 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
}
}
}

View File

@ -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");
}
}
}

View 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
}
}
}

View File

@ -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);
}
}
}

View File

@ -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

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

View File

@ -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>&nbsp;</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>

View File

@ -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();

View File

@ -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}`);
}

View File

@ -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;
}