add timesheet unit and e2e tests; add unit test CI setup

This commit is contained in:
Gregor Vostrak
2026-05-20 15:23:22 +02:00
parent da235dfdc8
commit 54fffd07bc
16 changed files with 4914 additions and 883 deletions

27
.github/workflows/npm-test-unit.yml vendored Normal file
View File

@@ -0,0 +1,27 @@
name: NPM Test Unit
on: [push]
permissions:
contents: read
jobs:
build:
runs-on: ubuntu-latest
timeout-minutes: 10
env:
TZ: UTC
steps:
- name: "Checkout code"
uses: actions/checkout@v4
- name: "Use Node.js"
uses: actions/setup-node@v4
with:
node-version: '20.x'
- name: "Install npm dependencies"
run: npm ci
- name: "Run vitest"
run: npm run test:unit

View File

@@ -107,7 +107,7 @@ services:
- sail
- reverse-proxy
playwright:
image: mcr.microsoft.com/playwright:v1.58.1-jammy
image: mcr.microsoft.com/playwright:v1.59.1-jammy
command: ['npx', 'playwright', 'test', '--ui-port=8080', '--ui-host=0.0.0.0']
working_dir: /src
extra_hosts:

View File

@@ -0,0 +1,437 @@
/**
* E2E coverage for the timesheet overlap-prevention logic introduced
* in `useTimesheetCellMutations` (Phase 1+2+3 of the overlap fix).
*
* Each test:
* 1. Pre-creates entries via the API to set up a deterministic
* day-of-work scenario,
* 2. Triggers ONE cell edit through the UI,
* 3. Reads the resulting entries back via the API and asserts on
* the start/end placement.
*
* Pre-creating rows (rather than driving the "Add row" + project picker
* UI) keeps the tests focused on the placement logic and out of the
* project-dropdown's flake surface.
*/
import { PLAYWRIGHT_BASE_URL } from '../playwright/config';
import { test } from '../playwright/fixtures';
import { expect } from '@playwright/test';
import type { Page, Request } from '@playwright/test';
import {
createProjectViaApi,
createTimeEntryAtHourViaApi,
getTimeEntriesViaApi,
} from './utils/api';
// ──────────────────────────────────────────────────
// Helpers
// ──────────────────────────────────────────────────
async function goToTimesheet(page: Page) {
await page.addInitScript(() => {
window.localStorage.setItem('showReleaseInfo-desktop', 'false');
});
await page.goto(PLAYWRIGHT_BASE_URL + '/timesheet');
}
function getMonday(d: Date): Date {
const date = new Date(d);
const day = date.getUTCDay();
const diff = date.getUTCDate() - day + (day === 0 ? -6 : 1);
date.setUTCDate(diff);
date.setUTCHours(0, 0, 0, 0);
return date;
}
function getCurrentWeekMonday(): Date {
return getMonday(new Date());
}
async function waitForTimesheetLoad(page: Page) {
await expect(page.getByTestId('timesheet_view')).toBeVisible();
await expect(page.getByTestId('timesheet_week_display')).toBeVisible();
const timezoneMismatchModal = page
.getByRole('dialog')
.filter({ hasText: 'Timezone mismatch detected' });
if (await timezoneMismatchModal.isVisible().catch(() => false)) {
await timezoneMismatchModal.getByRole('button', { name: 'Cancel' }).click();
await expect(timezoneMismatchModal).not.toBeVisible();
}
}
const HOUR = 3600;
function utcHourOf(iso: string): number {
return new Date(iso).getUTCHours();
}
function utcMinuteOf(iso: string): number {
return new Date(iso).getUTCMinutes();
}
function sortByStart<T extends { start: string }>(entries: T[]): T[] {
return [...entries].sort((a, b) => a.start.localeCompare(b.start));
}
/**
* Returns the locator for the row whose project name matches the given
* substring. Robust against ordering changes.
*/
function rowByProject(page: Page, projectName: string) {
return page.locator('[data-testid="timesheet_row"]').filter({ hasText: projectName });
}
/**
* Returns the locator for the input in the (row, dayIndex) cell, where
* the row is identified by project name.
*/
function cellInputByProject(page: Page, projectName: string, dayIndex: number) {
return rowByProject(page, projectName)
.locator('[data-testid="timesheet_cell"]')
.nth(dayIndex)
.locator('input');
}
/** Asserts that no entries in the list overlap each other. */
function expectNoOverlaps(entries: Array<{ start: string; end: string | null }>) {
const sorted = sortByStart(entries.filter((e) => e.end !== null));
for (let i = 1; i < sorted.length; i++) {
const prev = sorted[i - 1]!;
const curr = sorted[i]!;
expect(
curr.start >= prev.end!,
`entries overlap: ${prev.start}${prev.end} vs ${curr.start}${curr.end}`
).toBe(true);
}
}
// ──────────────────────────────────────────────────
// Phase 1: createCell — overlap avoidance when cell is empty
// ──────────────────────────────────────────────────
test('extendCell on a row that has no entries on the day yet places after another row (Scenario #4)', async ({
page,
ctx,
}) => {
// Setup: project A has Monday 09:0010:00, project B has Tuesday
// 09:0010:00. The B row is therefore visible on the timesheet but
// has an EMPTY cell on Monday. Typing into B's Monday cell exercises
// the createCell path (cell empty → place a new entry).
const monday = getCurrentWeekMonday();
const tuesday = new Date(monday);
tuesday.setUTCDate(monday.getUTCDate() + 1);
const projectA = await createProjectViaApi(ctx, { name: 'OverlapAlpha' });
const projectB = await createProjectViaApi(ctx, { name: 'OverlapBravo' });
await createTimeEntryAtHourViaApi(ctx, {
date: monday,
startHour: 9,
durationSeconds: HOUR,
projectId: projectA.id,
});
await createTimeEntryAtHourViaApi(ctx, {
date: tuesday,
startHour: 9,
durationSeconds: HOUR,
projectId: projectB.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
await expect(page.locator('[data-testid="timesheet_row"]')).toHaveCount(2);
// Type 1h into project B's Monday cell. The createCell path should
// place it AFTER project A's 09:0010:00 (i.e. at 10:00 or later),
// not at 09:00.
const input = cellInputByProject(page, 'OverlapBravo', 0);
await input.click();
await input.fill('1');
await Promise.all([
page.waitForResponse(
(resp) =>
resp.url().includes('/time-entries') &&
resp.request().method() === 'POST' &&
resp.status() === 201
),
input.press('Enter'),
]);
const entries = await getTimeEntriesViaApi(ctx);
const bMondayEntry = entries.find(
(e) =>
e.project_id === projectB.id &&
new Date(e.start).getTime() >= monday.getTime() &&
new Date(e.start).getTime() < tuesday.getTime()
)!;
expect(bMondayEntry).toBeDefined();
// 09:00 is blocked → must be at 10:00 or later.
expect(utcHourOf(bMondayEntry.start)).toBeGreaterThanOrEqual(10);
expectNoOverlaps(entries);
});
test('createCell refuses to cross midnight when day is full (Scenario #3)', async ({
page,
ctx,
}) => {
// Setup: fill Monday 01:0023:00 (22 hours, leaving 1h before and
// 1h after — neither big enough for a 3h ask). Project B is on
// Tuesday so the B row exists with an empty Monday cell. Typing 3h
// into B's Monday cell should be refused.
//
// We start at 01:00 (not 00:00) because the API's time-entry
// filter excludes entries whose `start` equals the query's `start`
// bound exactly. Using 01:00 avoids that boundary condition.
const monday = getCurrentWeekMonday();
const tuesday = new Date(monday);
tuesday.setUTCDate(monday.getUTCDate() + 1);
const projectFull = await createProjectViaApi(ctx, { name: 'OverlapFull' });
const projectNew = await createProjectViaApi(ctx, { name: 'OverlapNoRoom' });
await createTimeEntryAtHourViaApi(ctx, {
date: monday,
startHour: 1,
durationSeconds: 22 * HOUR,
projectId: projectFull.id,
});
await createTimeEntryAtHourViaApi(ctx, {
date: tuesday,
startHour: 9,
durationSeconds: HOUR,
projectId: projectNew.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
await expect(page.locator('[data-testid="timesheet_row"]')).toHaveCount(2);
const input = cellInputByProject(page, 'OverlapNoRoom', 0);
const seenMutationRequests: string[] = [];
const onRequest = (request: Request) => {
if (request.url().includes('/time-entries') && request.method() !== 'GET') {
seenMutationRequests.push(request.method());
}
};
page.on('request', onRequest);
await input.click();
await input.fill('3');
await input.press('Enter');
await expect(page.getByText("This day can't fit any more work")).toBeVisible();
page.off('request', onRequest);
const entries = await getTimeEntriesViaApi(ctx);
// The new project should still only have its Tuesday entry.
const newEntries = entries.filter((e) => e.project_id === projectNew.id);
expect(seenMutationRequests).toEqual([]);
expect(newEntries).toHaveLength(1);
expect(utcHourOf(newEntries[0]!.start)).toBe(9);
// The Tuesday entry's date is unchanged (still Tuesday).
expect(new Date(newEntries[0]!.start).getUTCDay()).toBe(2);
});
// ──────────────────────────────────────────────────
// Phase 2: extendCell — collision detection + split
// ──────────────────────────────────────────────────
test('extendCell splits the extension when another row blocks the path (Scenario #5)', async ({
page,
ctx,
}) => {
// Setup:
// - project A on Monday 09:0010:00 (1h)
// - project B on Monday 10:3011:30 (1h, blocker)
// Bumping A's Monday cell from 1h to 3h (+2h) should:
// - extend A to 09:0010:30 (filling the 30min gap)
// - place a new A entry at 11:3013:00 (the remaining 90min)
const monday = getCurrentWeekMonday();
const projectA = await createProjectViaApi(ctx, { name: 'OverlapExtend' });
const projectB = await createProjectViaApi(ctx, { name: 'OverlapBlocker' });
await createTimeEntryAtHourViaApi(ctx, {
date: monday,
startHour: 9,
durationSeconds: HOUR,
projectId: projectA.id,
});
await createTimeEntryAtHourViaApi(ctx, {
date: monday,
startHour: 10,
startMinute: 30,
durationSeconds: HOUR,
projectId: projectB.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
await expect(page.locator('[data-testid="timesheet_row"]')).toHaveCount(2);
const input = cellInputByProject(page, 'OverlapExtend', 0);
await input.click();
await input.fill('3');
await Promise.all([
page.waitForResponse(
(resp) =>
resp.url().includes('/time-entries') &&
resp.request().method() === 'PUT' &&
resp.status() === 200
),
page.waitForResponse(
(resp) =>
resp.url().includes('/time-entries') &&
resp.request().method() === 'POST' &&
resp.status() === 201
),
input.press('Enter'),
]);
const entries = await getTimeEntriesViaApi(ctx);
const aEntries = entries.filter((e) => e.project_id === projectA.id);
const bEntries = entries.filter((e) => e.project_id === projectB.id);
// The blocker is unchanged.
expect(bEntries).toHaveLength(1);
expect(utcHourOf(bEntries[0]!.start)).toBe(10);
expect(utcMinuteOf(bEntries[0]!.start)).toBe(30);
// Project A should now have 2 entries.
expect(aEntries).toHaveLength(2);
const sortedA = sortByStart(aEntries);
// Extended entry: 09:00 → 10:30
expect(utcHourOf(sortedA[0]!.start)).toBe(9);
expect(utcHourOf(sortedA[0]!.end!)).toBe(10);
expect(utcMinuteOf(sortedA[0]!.end!)).toBe(30);
// Split remainder: 11:30 → 13:00
expect(utcHourOf(sortedA[1]!.start)).toBe(11);
expect(utcMinuteOf(sortedA[1]!.start)).toBe(30);
// No overlaps anywhere on the day.
expectNoOverlaps(entries);
});
test('extendCell prefers latest-end (not latest-start) when nested entries exist (Scenario #6)', async ({
page,
ctx,
}) => {
// Pre-existing nested overlap on the same project:
// - outer: 09:00 → 12:00 (3h)
// - inner: 10:00 → 11:00 (1h, contained inside outer)
// The cell total is 3h + 1h = 4h. Bumping to 5h (+1h) should grow
// the OUTER entry's end to 13:00, not the inner.
const monday = getCurrentWeekMonday();
const project = await createProjectViaApi(ctx, { name: 'OverlapNested' });
await createTimeEntryAtHourViaApi(ctx, {
date: monday,
startHour: 9,
durationSeconds: 3 * HOUR,
projectId: project.id,
description: 'outer',
});
await createTimeEntryAtHourViaApi(ctx, {
date: monday,
startHour: 10,
durationSeconds: HOUR,
projectId: project.id,
description: 'inner',
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
await expect(page.locator('[data-testid="timesheet_row"]')).toHaveCount(1);
const input = cellInputByProject(page, 'OverlapNested', 0);
await input.click();
await input.fill('5');
await Promise.all([
page.waitForResponse(
(resp) =>
resp.url().includes('/time-entries') &&
resp.request().method() === 'PUT' &&
resp.status() === 200
),
input.press('Enter'),
]);
const entries = await getTimeEntriesViaApi(ctx);
const outer = entries.find((e) => e.description === 'outer')!;
const inner = entries.find((e) => e.description === 'inner')!;
expect(utcHourOf(outer.start)).toBe(9);
expect(utcHourOf(outer.end!)).toBe(13); // extended from 12:00 → 13:00
expect(utcHourOf(inner.start)).toBe(10);
expect(utcHourOf(inner.end!)).toBe(11); // unchanged
});
// ──────────────────────────────────────────────────
// Phase 1+2 spillover from previous day
// ──────────────────────────────────────────────────
test('createCell handles intra-week spillover from previous day (Scenario #2)', async ({
page,
ctx,
}) => {
// Setup: an entry that starts on Monday 22:00 and ends Tuesday 03:00
// (5h, crosses midnight INTO Tuesday). This spillover starts inside
// the loaded week, so the timesheet query loads it.
//
// Then we try to place 1h on Tuesday for a different project. The
// expected behavior: the new entry must NOT overlap the spillover.
// Tuesday 09:00 is well clear of the [00:00, 03:00) spillover, so
// 09:00 is the correct placement.
const monday = getCurrentWeekMonday();
const tuesday = new Date(monday);
tuesday.setUTCDate(monday.getUTCDate() + 1);
const wednesday = new Date(monday);
wednesday.setUTCDate(monday.getUTCDate() + 2);
const projectSpill = await createProjectViaApi(ctx, { name: 'OverlapSpill' });
const projectNew = await createProjectViaApi(ctx, { name: 'OverlapToday' });
// Monday 22:00 → Tuesday 03:00 (5h spillover into Tuesday).
await createTimeEntryAtHourViaApi(ctx, {
date: monday,
startHour: 22,
durationSeconds: 5 * HOUR,
projectId: projectSpill.id,
});
// Stub Wednesday entry on the new project so its row is visible
// even before we type anything in Tuesday's cell.
await createTimeEntryAtHourViaApi(ctx, {
date: wednesday,
startHour: 9,
durationSeconds: HOUR,
projectId: projectNew.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
await expect(page.locator('[data-testid="timesheet_row"]')).toHaveCount(2);
// Type 1h into the new project's Tuesday cell (day index 1).
const input = cellInputByProject(page, 'OverlapToday', 1);
await input.click();
await input.fill('1');
await Promise.all([
page.waitForResponse(
(resp) =>
resp.url().includes('/time-entries') &&
resp.request().method() === 'POST' &&
resp.status() === 201
),
input.press('Enter'),
]);
const entries = await getTimeEntriesViaApi(ctx);
const newTuesdayEntry = entries.find(
(e) =>
e.project_id === projectNew.id &&
new Date(e.start).getTime() >= tuesday.getTime() &&
new Date(e.start).getTime() < wednesday.getTime()
)!;
expect(newTuesdayEntry).toBeDefined();
// 09:00 is well past the spillover end (03:00) → should land at 09:00.
expect(utcHourOf(newTuesdayEntry.start)).toBe(9);
expectNoOverlaps(entries);
});

641
e2e/timesheet.spec.ts Normal file
View File

@@ -0,0 +1,641 @@
import { PLAYWRIGHT_BASE_URL } from '../playwright/config';
import { test } from '../playwright/fixtures';
import { expect } from '@playwright/test';
import type { Page } from '@playwright/test';
import { createProjectViaApi, createTaskViaApi, createTimeEntryOnDateViaApi } from './utils/api';
// ──────────────────────────────────────────────────
// Helpers
// ──────────────────────────────────────────────────
async function goToTimesheet(page: Page) {
await page.addInitScript(() => {
window.localStorage.setItem('showReleaseInfo-desktop', 'false');
});
await page.goto(PLAYWRIGHT_BASE_URL + '/timesheet');
}
function getMonday(d: Date): Date {
const date = new Date(d);
const day = date.getUTCDay();
const diff = date.getUTCDate() - day + (day === 0 ? -6 : 1);
date.setUTCDate(diff);
date.setUTCHours(0, 0, 0, 0);
return date;
}
function getCurrentWeekMonday(): Date {
return getMonday(new Date());
}
function getLastWeekMonday(): Date {
const monday = getCurrentWeekMonday();
monday.setUTCDate(monday.getUTCDate() - 7);
return monday;
}
function getDayOfWeek(weekStart: Date, dayOffset: number): Date {
const date = new Date(weekStart);
date.setUTCDate(date.getUTCDate() + dayOffset);
return date;
}
async function waitForTimesheetLoad(page: Page) {
await page.waitForURL(/\/timesheet(?:$|\?)/);
await expect(page.getByTestId('timesheet_view')).toBeVisible();
await expect(page.getByTestId('timesheet_week_display')).toBeVisible();
const timezoneMismatchModal = page
.getByRole('dialog')
.filter({ hasText: 'Timezone mismatch detected' });
if (await timezoneMismatchModal.isVisible().catch(() => false)) {
await timezoneMismatchModal.getByRole('button', { name: 'Cancel' }).click();
await expect(timezoneMismatchModal).not.toBeVisible();
}
}
function addRowButton(page: Page) {
return page.getByRole('button', { name: /Add row/i }).first();
}
async function chooseRowIdentity(page: Page, optionName: string) {
await addRowButton(page).click();
const dialog = page.getByRole('dialog', { name: /Add row/i });
const dialogVisible = await dialog
.waitFor({ state: 'visible', timeout: 1000 })
.then(() => true)
.catch(() => false);
if (dialogVisible) {
await dialog.getByRole('option', { name: optionName }).click();
return;
}
if (optionName === 'No Project') return;
const row = page.locator('[data-testid="timesheet_row"]').first();
await row.getByText('No Project').click();
await page.getByText(optionName).click();
}
// ──────────────────────────────────────────────────
// Navigation & Page Load
// ──────────────────────────────────────────────────
test('timesheet renders empty with add row + copy last week actions', async ({ page }) => {
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
await expect(page.locator('[data-testid="timesheet_row"]')).toHaveCount(0);
await expect(addRowButton(page)).toBeVisible();
await expect(page.getByRole('button', { name: /Copy last week/i })).toBeVisible();
});
// ──────────────────────────────────────────────────
// Display Existing Time Entries
// ──────────────────────────────────────────────────
test('timesheet displays existing time entries grouped by project', async ({ page, ctx }) => {
const monday = getCurrentWeekMonday();
const tuesday = getDayOfWeek(monday, 1);
const wednesday = getDayOfWeek(monday, 2);
const projectA = await createProjectViaApi(ctx, { name: 'Project Alpha' });
const projectB = await createProjectViaApi(ctx, { name: 'Project Beta' });
await createTimeEntryOnDateViaApi(ctx, {
date: monday,
duration: '2h',
projectId: projectA.id,
});
await createTimeEntryOnDateViaApi(ctx, {
date: wednesday,
duration: '1h',
projectId: projectA.id,
});
await createTimeEntryOnDateViaApi(ctx, {
date: tuesday,
duration: '3h',
projectId: projectB.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
const rows = page.locator('[data-testid="timesheet_row"]');
await expect(rows).toHaveCount(2);
// Check that the grand total is shown
await expect(page.getByTestId('timesheet_grand_total')).toBeVisible();
});
test('timesheet groups entries by project and task combination', async ({ page, ctx }) => {
const monday = getCurrentWeekMonday();
const project = await createProjectViaApi(ctx, { name: 'Task Project' });
const taskA = await createTaskViaApi(ctx, { name: 'Task A', project_id: project.id });
const taskB = await createTaskViaApi(ctx, { name: 'Task B', project_id: project.id });
await createTimeEntryOnDateViaApi(ctx, {
date: monday,
duration: '1h',
projectId: project.id,
taskId: taskA.id,
});
await createTimeEntryOnDateViaApi(ctx, {
date: monday,
duration: '2h',
projectId: project.id,
taskId: taskB.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
const rows = page.locator('[data-testid="timesheet_row"]');
await expect(rows).toHaveCount(2);
});
// ──────────────────────────────────────────────────
// Enter Duration in Cell
// ──────────────────────────────────────────────────
test('entering duration in empty cell creates a time entry', async ({ page, ctx }) => {
await createProjectViaApi(ctx, { name: 'Duration Test' });
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
await chooseRowIdentity(page, 'Duration Test');
const row = page.locator('[data-testid="timesheet_row"]').first();
// Click the first day cell and enter duration
const cells = row.locator('[data-testid="timesheet_cell"]');
const mondayCell = cells.first();
const mondayInput = mondayCell.locator('input');
await mondayInput.click();
await mondayInput.fill('2');
// Submit and wait for create response
const [response] = await Promise.all([
page.waitForResponse(
(resp) =>
resp.url().includes('/time-entries') &&
resp.request().method() === 'POST' &&
resp.status() === 201
),
mondayInput.press('Enter'),
]);
expect(response.status()).toBe(201);
// Verify the cell shows the duration
await expect(mondayInput).not.toHaveValue('');
});
// ──────────────────────────────────────────────────
// Edit Duration (Increase)
// ──────────────────────────────────────────────────
test('increasing duration in cell extends the last time entry', async ({ page, ctx }) => {
const monday = getCurrentWeekMonday();
const project = await createProjectViaApi(ctx, { name: 'Increase Test' });
await createTimeEntryOnDateViaApi(ctx, {
date: monday,
duration: '1h',
projectId: project.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
const row = page.locator('[data-testid="timesheet_row"]').first();
const cells = row.locator('[data-testid="timesheet_cell"]');
const mondayInput = cells.first().locator('input');
// Click and change to 3 hours
await mondayInput.click();
await mondayInput.fill('3');
const [response] = await Promise.all([
page.waitForResponse(
(resp) =>
resp.url().includes('/time-entries') &&
resp.request().method() === 'PUT' &&
resp.status() === 200
),
mondayInput.press('Enter'),
]);
expect(response.status()).toBe(200);
});
// ──────────────────────────────────────────────────
// Edit Duration (Decrease)
// ──────────────────────────────────────────────────
test('decreasing duration in cell shortens the last time entry', async ({ page, ctx }) => {
const monday = getCurrentWeekMonday();
const project = await createProjectViaApi(ctx, { name: 'Decrease Test' });
await createTimeEntryOnDateViaApi(ctx, {
date: monday,
duration: '3h',
projectId: project.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
const row = page.locator('[data-testid="timesheet_row"]').first();
const cells = row.locator('[data-testid="timesheet_cell"]');
const mondayInput = cells.first().locator('input');
await mondayInput.click();
await mondayInput.fill('1');
const [response] = await Promise.all([
page.waitForResponse(
(resp) =>
resp.url().includes('/time-entries') &&
resp.request().method() === 'PUT' &&
resp.status() === 200
),
mondayInput.press('Enter'),
]);
expect(response.status()).toBe(200);
});
// ──────────────────────────────────────────────────
// Clear Cell
// ──────────────────────────────────────────────────
test('clearing a cell deletes all time entries for that project+day', async ({ page, ctx }) => {
const monday = getCurrentWeekMonday();
const project = await createProjectViaApi(ctx, { name: 'Clear Test' });
await createTimeEntryOnDateViaApi(ctx, {
date: monday,
duration: '2h',
projectId: project.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
const row = page.locator('[data-testid="timesheet_row"]').first();
const cells = row.locator('[data-testid="timesheet_cell"]');
const mondayInput = cells.first().locator('input');
await mondayInput.click();
await mondayInput.fill('0');
const [response] = await Promise.all([
page.waitForResponse(
(resp) =>
resp.url().includes('/time-entries') &&
resp.request().method() === 'DELETE' &&
resp.status() === 200
),
mondayInput.press('Enter'),
]);
expect(response.status()).toBe(200);
});
test('Escape during cell edit reverts the displayed value without an API call', async ({
page,
ctx,
}) => {
const monday = getCurrentWeekMonday();
const project = await createProjectViaApi(ctx, { name: 'Escape Cancel Test' });
await createTimeEntryOnDateViaApi(ctx, {
date: monday,
duration: '2h',
projectId: project.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
const row = page.locator('[data-testid="timesheet_row"]').first();
const cells = row.locator('[data-testid="timesheet_cell"]');
const mondayInput = cells.first().locator('input');
// Capture the formatted display value before editing.
const originalValue = await mondayInput.inputValue();
expect(originalValue).toMatch(/2/);
let mutationFired = false;
page.on('request', (req) => {
if (req.url().includes('/time-entries') && req.method() !== 'GET') {
mutationFired = true;
}
});
await mondayInput.click();
await mondayInput.fill('5');
await mondayInput.press('Escape');
// The Escape handler reverts the displayed value synchronously, so
// once this assertion passes we know the handler ran. Any mutation
// request would have been queued by then.
await expect(mondayInput).toHaveValue(originalValue);
expect(mutationFired).toBe(false);
});
// ──────────────────────────────────────────────────
// Week Navigation
// ──────────────────────────────────────────────────
test('navigating to previous week shows entries from that week', async ({ page, ctx }) => {
const lastMonday = getLastWeekMonday();
const project = await createProjectViaApi(ctx, { name: 'Last Week Project' });
await createTimeEntryOnDateViaApi(ctx, {
date: lastMonday,
duration: '2h',
projectId: project.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
// Current week should have no entries
await expect(page.locator('[data-testid="timesheet_row"]')).toHaveCount(0);
// Go to previous week — the row-count assertion below auto-retries
// until the new week's data arrives.
await page.getByTestId('timesheet_prev_week').click();
// Should now see the entry
const rows = page.locator('[data-testid="timesheet_row"]');
await expect(rows).toHaveCount(1);
});
test('can navigate forward and return to current week', async ({ page }) => {
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
// Should show "This week"
await expect(page.getByTestId('timesheet_week_display')).toContainText('This week');
// Go to next week — the text assertions below auto-retry until the
// header label flips.
await page.getByTestId('timesheet_next_week').click();
// Should no longer show "This week"
await expect(page.getByTestId('timesheet_week_display')).not.toContainText('This week');
// Go back to this week
await page.getByTestId('timesheet_week_display').click();
await expect(page.getByTestId('timesheet_week_display')).toContainText('This week');
});
// ──────────────────────────────────────────────────
// Copy Last Week
// ──────────────────────────────────────────────────
test('copy last week adds project rows from previous week without hours', async ({ page, ctx }) => {
const lastMonday = getLastWeekMonday();
const lastWednesday = getDayOfWeek(lastMonday, 2);
const projectA = await createProjectViaApi(ctx, { name: 'Copy Project A' });
const projectB = await createProjectViaApi(ctx, { name: 'Copy Project B' });
await createTimeEntryOnDateViaApi(ctx, {
date: lastMonday,
duration: '2h',
projectId: projectA.id,
});
await createTimeEntryOnDateViaApi(ctx, {
date: lastWednesday,
duration: '3h',
projectId: projectB.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
// Current week should have no populated rows yet.
await expect(page.locator('[data-testid="timesheet_row"]')).toHaveCount(0);
// Open copy last week dropdown and click "Copy rows only"
await page.getByRole('button', { name: /Copy last week/i }).click();
await page.getByText('Copy rows only').click();
// Should now show 2 rows (one per project)
const rows = page.locator('[data-testid="timesheet_row"]');
await expect(rows).toHaveCount(2);
// All row totals should be 0
const rowTotals = page.locator('[data-testid="timesheet_row_total"]');
const count = await rowTotals.count();
for (let i = 0; i < count; i++) {
await expect(rowTotals.nth(i)).toContainText('-');
}
});
test('copy last week does not duplicate rows that already exist', async ({ page, ctx }) => {
const lastMonday = getLastWeekMonday();
const thisMonday = getCurrentWeekMonday();
const thisTuesday = getDayOfWeek(thisMonday, 1);
const project = await createProjectViaApi(ctx, { name: 'No Dup Project' });
// Create entry for last week
await createTimeEntryOnDateViaApi(ctx, {
date: lastMonday,
duration: '2h',
projectId: project.id,
});
// Create entry for current week
await createTimeEntryOnDateViaApi(ctx, {
date: thisTuesday,
duration: '1h',
projectId: project.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
// Should have 1 row (from current week entry)
const rows = page.locator('[data-testid="timesheet_row"]');
await expect(rows).toHaveCount(1);
// Open copy last week dropdown and click "Copy rows only"
await page.getByRole('button', { name: /Copy last week/i }).click();
await page.getByText('Copy rows only').click();
// Should still have only 1 row (not duplicated)
await expect(rows).toHaveCount(1);
});
test('copy last week with time entries creates rows and entries', async ({ page, ctx }) => {
const lastMonday = getLastWeekMonday();
const project = await createProjectViaApi(ctx, { name: 'Copy Time Project' });
await createTimeEntryOnDateViaApi(ctx, {
date: lastMonday,
duration: '2h',
projectId: project.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
// Current week should have no populated rows yet.
await expect(page.locator('[data-testid="timesheet_row"]')).toHaveCount(0);
// Open copy last week dropdown and click "Copy rows and time entries"
await page.getByRole('button', { name: /Copy last week/i }).click();
await Promise.all([
page.waitForResponse(
(resp) =>
resp.url().includes('/time-entries') &&
resp.request().method() === 'POST' &&
resp.status() === 201
),
page.getByText('Copy rows and time entries').click(),
]);
// Should now show 1 row with time entries
const rows = page.locator('[data-testid="timesheet_row"]');
await expect(rows).toHaveCount(1);
// Row total should not be 0 (entries were copied)
const rowTotal = page.locator('[data-testid="timesheet_row_total"]').first();
await expect(rowTotal).not.toContainText('0 h');
});
// ──────────────────────────────────────────────────
// Row Removal
// ──────────────────────────────────────────────────
test('can remove an empty project row without confirmation', async ({ page, ctx }) => {
const project = await createProjectViaApi(ctx, { name: 'Empty Remove Project' });
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
await chooseRowIdentity(page, project.name);
const rows = page.locator('[data-testid="timesheet_row"]');
await expect(rows).toHaveCount(1);
// Hover the row to reveal the X button, then click it
await rows.first().hover();
await rows.first().getByRole('button', { name: 'Remove row' }).click();
// Row should be removed immediately (no dialog)
await expect(rows).toHaveCount(0);
});
test('removing a row with entries shows confirmation dialog and deletes entries', async ({
page,
ctx,
}) => {
const monday = getCurrentWeekMonday();
const project = await createProjectViaApi(ctx, { name: 'Delete Row Project' });
await createTimeEntryOnDateViaApi(ctx, {
date: monday,
duration: '2h',
projectId: project.id,
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
const rows = page.locator('[data-testid="timesheet_row"]');
await expect(rows).toHaveCount(1);
// Hover and click X
await rows.first().hover();
await rows.first().getByRole('button', { name: 'Remove row' }).click();
// Confirmation dialog should appear
await expect(page.getByRole('alertdialog')).toBeVisible();
await expect(page.getByText('Remove timesheet row?')).toBeVisible();
// Click Delete
await Promise.all([
page.waitForResponse(
(resp) =>
resp.url().includes('/time-entries') &&
resp.request().method() === 'DELETE' &&
resp.status() === 200
),
page
.getByRole('alertdialog')
.getByRole('button', { name: /Delete/i })
.click(),
]);
// Row should be gone
await expect(rows).toHaveCount(0);
});
// ──────────────────────────────────────────────────
// Multiple Entries Same Cell
// ──────────────────────────────────────────────────
test('cell correctly sums multiple entries for same project+day', async ({ page, ctx }) => {
const monday = getCurrentWeekMonday();
const project = await createProjectViaApi(ctx, { name: 'Sum Test' });
// Create 2 entries for the same project on Monday
await createTimeEntryOnDateViaApi(ctx, {
date: monday,
duration: '1h',
projectId: project.id,
description: 'Entry 1',
});
await createTimeEntryOnDateViaApi(ctx, {
date: monday,
duration: '2h',
projectId: project.id,
description: 'Entry 2',
});
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
// Should be 1 row (both entries grouped)
const rows = page.locator('[data-testid="timesheet_row"]');
await expect(rows).toHaveCount(1);
// The Monday cell should show 3h total
const cells = rows.first().locator('[data-testid="timesheet_cell"]');
const mondayInput = cells.first().locator('input');
// The value should contain "3" (for 3h in some format)
await expect(mondayInput).toHaveValue(/3/);
});
// ──────────────────────────────────────────────────
// Duration Input Formats
// ──────────────────────────────────────────────────
test('cell accepts various duration input formats', async ({ page, ctx }) => {
await createProjectViaApi(ctx, { name: 'Format Test' });
await Promise.all([goToTimesheet(page), waitForTimesheetLoad(page)]);
await chooseRowIdentity(page, 'Format Test');
const row = page.locator('[data-testid="timesheet_row"]').first();
// Test entering "1.5" (should be 1h 30min)
const cells = row.locator('[data-testid="timesheet_cell"]');
const mondayInput = cells.first().locator('input');
await mondayInput.click();
await mondayInput.fill('1.5');
await Promise.all([
page.waitForResponse(
(resp) =>
resp.url().includes('/time-entries') &&
resp.request().method() === 'POST' &&
resp.status() === 201
),
mondayInput.press('Enter'),
]);
// 1.5 hours = 1h 30min
await expect(mondayInput).toHaveValue('1h 30min');
});

View File

@@ -170,10 +170,24 @@ function parseDurationToSeconds(duration: string): number {
return totalSeconds;
}
/**
* Builds a start/end pair anchored to 09:00 UTC on today's UTC date.
*
* Intentionally pinned to UTC (rather than the runner's local time) so
* the produced timestamps are identical regardless of where the suite
* runs. Playwright test users default to UTC, so this matches what the
* app will see and keeps day-of-week / "this week" assertions stable
* for developers running the suite locally in non-UTC timezones.
*/
function createTimestamps(duration: string): { start: string; end: string } {
const durationSeconds = parseDurationToSeconds(duration);
const now = new Date();
const start = new Date(now.getFullYear(), now.getMonth(), now.getDate(), 9, 0, 0);
const start = createUtcTimestampFromDateParts(
now.getUTCFullYear(),
now.getUTCMonth(),
now.getUTCDate(),
9
);
const end = new Date(start.getTime() + durationSeconds * 1000);
return {
@@ -186,6 +200,32 @@ function formatTimestamp(date: Date): string {
return date.toISOString().replace(/\.\d{3}Z$/, 'Z');
}
function createUtcTimestampFromDateParts(
year: number,
month: number,
date: number,
hours: number,
minutes: number = 0,
seconds: number = 0
): Date {
return new Date(Date.UTC(year, month, date, hours, minutes, seconds));
}
function createTimestampsOnDate(date: Date, duration: string): { start: string; end: string } {
const durationSeconds = parseDurationToSeconds(duration);
const start = createUtcTimestampFromDateParts(
date.getUTCFullYear(),
date.getUTCMonth(),
date.getUTCDate(),
9
);
const end = new Date(start.getTime() + durationSeconds * 1000);
return {
start: formatTimestamp(start),
end: formatTimestamp(end),
};
}
function randomColor(): string {
const colors = [
'#ef5350',
@@ -375,6 +415,39 @@ export async function createTimeEntryViaApi(
return body.data as { id: string; start: string; end: string; description: string };
}
export async function createTimeEntryOnDateViaApi(
ctx: TestContext,
data: {
date: Date;
duration: string;
description?: string;
projectId?: string | null;
taskId?: string | null;
tags?: string[];
billable?: boolean;
}
) {
const { start, end } = createTimestampsOnDate(data.date, data.duration);
const response = await ctx.request.post(
`${PLAYWRIGHT_BASE_URL}/api/v1/organizations/${ctx.orgId}/time-entries`,
{
data: {
member_id: ctx.memberId,
start,
end,
description: data.description ?? '',
project_id: data.projectId ?? null,
task_id: data.taskId ?? null,
tags: data.tags ?? [],
billable: data.billable ?? false,
},
}
);
expect(response.status()).toBe(201);
const body = await response.json();
return body.data as { id: string; start: string; end: string; description: string };
}
export async function createProjectMemberViaApi(
ctx: TestContext,
projectId: string,
@@ -613,6 +686,72 @@ export async function getInvitationsViaApi(ctx: TestContext) {
// Timestamp-based time entry helpers
// ──────────────────────────────────────────────────
/**
* Creates a time entry on `date` at a specific UTC hour with a duration
* in seconds. Playwright test users default to the UTC timezone, so this
* keeps time-placement scenarios stable across runner locales.
*/
export async function createTimeEntryAtHourViaApi(
ctx: TestContext,
data: {
date: Date;
startHour: number;
startMinute?: number;
durationSeconds: number;
projectId?: string | null;
taskId?: string | null;
description?: string;
}
) {
const start = createUtcTimestampFromDateParts(
data.date.getUTCFullYear(),
data.date.getUTCMonth(),
data.date.getUTCDate(),
data.startHour,
data.startMinute ?? 0
);
const end = new Date(start.getTime() + data.durationSeconds * 1000);
return createTimeEntryWithTimestampsViaApi(ctx, {
start: formatTimestamp(start),
end: formatTimestamp(end),
projectId: data.projectId ?? null,
taskId: data.taskId ?? null,
description: data.description ?? '',
});
}
/**
* Reads time entries for the current member, optionally filtered to a
* date range. Returns the raw API objects (id, start, end, project_id,
* etc.) so tests can assert on the database state after a UI action.
*/
export async function getTimeEntriesViaApi(
ctx: TestContext,
filters: { start?: string; end?: string } = {}
): Promise<
Array<{
id: string;
start: string;
end: string | null;
duration: number | null;
project_id: string | null;
task_id: string | null;
description: string;
}>
> {
const params = new URLSearchParams();
params.set('member_id', ctx.memberId);
if (filters.start) params.set('start', filters.start);
if (filters.end) params.set('end', filters.end);
const response = await ctx.request.get(
`${PLAYWRIGHT_BASE_URL}/api/v1/organizations/${ctx.orgId}/time-entries?${params.toString()}`
);
expect(response.status()).toBe(200);
const body = await response.json();
return body.data;
}
export async function createTimeEntryWithTimestampsViaApi(
ctx: TestContext,
data: {

2851
package-lock.json generated

File diff suppressed because it is too large Load Diff

View File

@@ -12,10 +12,17 @@
"lint": "eslint resources/js",
"lint:fix": "eslint --fix resources/js",
"type-check": "vue-tsc --noEmit",
"test:unit": "vitest run",
"test:unit:watch": "vitest",
"test:e2e": "rm -rf test-results/.auth && npx playwright test",
"zod:generate": "npx openapi-zod-client http://localhost:80/docs/api.json --output resources/js/packages/api/src/openapi.json.client.ts --base-url /api",
"format": "prettier --write './**/*.{js,jsx,cjs,mjs,ts,tsx,cts,mts,vue}'",
"format:check": "prettier --check './**/*.{js,jsx,cjs,mjs,ts,tsx,cts,mts,vue}'"
"format:check": "prettier --check './**/*.{js,jsx,cjs,mjs,ts,tsx,cts,mts,vue}'",
"build:ui": "npm run build --workspace=@solidtime/ui",
"build:api": "npm run build --workspace=@solidtime/api",
"build:packages": "npm run build:api && npm run build:ui",
"watch:ui": "npm run watch --workspace=@solidtime/ui",
"watch:api": "npm run watch --workspace=@solidtime/api"
},
"devDependencies": {
"@eslint/eslintrc": "^3.2.0",
@@ -27,10 +34,12 @@
"@types/chroma-js": "^3.1.0",
"@types/node": "^22.10.10",
"@vitejs/plugin-vue": "^6.0.3",
"@vue/test-utils": "^2.4.6",
"@vue/tsconfig": "^0.8.0",
"autoprefixer": "^10.4.20",
"axios": "^1.6.4",
"eslint-plugin-unused-imports": "^4.1.4",
"happy-dom": "^20.8.9",
"laravel-vite-plugin": "^2.1.0",
"openapi-zod-client": "^1.16.2",
"postcss": "^8.4.47",
@@ -40,6 +49,7 @@
"typescript": "^5.7.3",
"vite": "^7.0.0",
"vite-plugin-checker": "^0.12.0",
"vitest": "^4.1.4",
"vue": "^3.5.0",
"vue-tsc": "^3.0.0"
},

View File

@@ -0,0 +1,65 @@
import { describe, expect, it } from 'vitest';
import { mount } from '@vue/test-utils';
import { nextTick } from 'vue';
import TimesheetCell from './TimesheetCell.vue';
import { formatHumanReadableDuration } from '@/packages/ui/src/utils/time';
import type { TimesheetCell as TimesheetCellType } from '@/utils/useTimesheetGrid';
function buildCell(totalSeconds: number): TimesheetCellType {
return {
dayIndex: 0,
date: '2026-04-13',
entries: [],
totalSeconds,
};
}
function mountTimesheetCell(totalSeconds = 2 * 3600) {
return mount(TimesheetCell, {
props: {
cell: buildCell(totalSeconds),
dayIndex: 0,
date: '2026-04-13',
isToday: false,
hasRunningEntry: false,
},
});
}
describe('TimesheetCell', () => {
it('emits 0 when the cleared value is committed on blur', async () => {
const wrapper = mountTimesheetCell();
const input = wrapper.get('input');
await input.trigger('focus');
await input.setValue('');
await input.trigger('blur');
expect(wrapper.emitted('update')).toEqual([[0]]);
});
it('emits 0 when the cleared value is committed with Enter', async () => {
const wrapper = mountTimesheetCell();
const input = wrapper.get('input');
await input.trigger('focus');
await input.setValue('');
await input.trigger('keydown', { key: 'Enter' });
expect(wrapper.emitted('update')).toEqual([[0]]);
});
it('restores the previous value and emits nothing on Escape', async () => {
const wrapper = mountTimesheetCell();
const input = wrapper.get('input');
const previousValue = formatHumanReadableDuration(2 * 3600, 'hours-minutes', 'point');
await input.trigger('focus');
await input.setValue('');
await input.trigger('keydown', { key: 'Escape' });
await nextTick();
expect(wrapper.emitted('update')).toBeUndefined();
expect((input.element as HTMLInputElement).value).toBe(previousValue);
});
});

View File

@@ -0,0 +1,90 @@
/* eslint-disable vue/one-component-per-file */
import { mount } from '@vue/test-utils';
import { describe, expect, it, vi } from 'vitest';
import { defineComponent, h, nextTick, onMounted } from 'vue';
import TimeTrackerProjectTaskDropdown from './TimeTrackerProjectTaskDropdown.vue';
import type { Client, Project, Task } from '@/packages/api/src';
const DropdownStub = defineComponent({
props: {
modelValue: {
type: Boolean,
default: false,
},
},
emits: ['update:modelValue'],
setup(_, { emit, slots }) {
onMounted(() => emit('update:modelValue', true));
return () => h('div', [slots.trigger?.(), slots.content?.()]);
},
});
const FocusTrapStub = defineComponent({
setup(_, { slots }) {
return () => h('div', slots.default?.());
},
});
function mountDropdown(props: Record<string, unknown> = {}) {
return mount(TimeTrackerProjectTaskDropdown, {
props: {
project: null,
task: null,
projects: [] as Project[],
tasks: [] as Task[],
clients: [] as Client[],
createProject: vi.fn(),
createClient: vi.fn(),
currency: 'EUR',
enableEstimatedTime: false,
organizationBillableRate: null,
canCreateProject: false,
...props,
},
global: {
stubs: {
Dropdown: DropdownStub,
UseFocusTrap: FocusTrapStub,
},
},
});
}
async function openDropdown() {
const wrapper = mountDropdown();
await nextTick();
await nextTick();
return wrapper;
}
describe('TimeTrackerProjectTaskDropdown', () => {
it('keeps the existing empty-string no-project value by default', async () => {
const wrapper = await openDropdown();
await wrapper.find('[data-project-id=""]').trigger('click');
expect(wrapper.emitted('update:project')?.at(-1)).toEqual(['']);
expect(wrapper.emitted('changed')?.at(-1)).toEqual(['', null]);
});
it('can emit null for no-project consumers that use null as the domain value', async () => {
const wrapper = mountDropdown({ project: 'p-1', noProjectValue: null });
await nextTick();
await nextTick();
await wrapper.find('[data-project-id=""]').trigger('click');
expect(wrapper.emitted('update:project')?.at(-1)).toEqual([null]);
expect(wrapper.emitted('changed')?.at(-1)).toEqual([null, null]);
});
it('still exposes "No Project" when projects are empty and project creation is allowed', async () => {
const wrapper = mountDropdown({ canCreateProject: true });
await nextTick();
await nextTick();
await wrapper.find('[data-project-id=""]').trigger('click');
expect(wrapper.emitted('changed')?.at(-1)).toEqual(['', null]);
});
});

View File

@@ -0,0 +1,44 @@
import { describe, expect, test } from 'vitest';
import { formatHumanReadableDuration, formatReportingDuration } from './time';
const seconds = 14 * 3600 + 45 * 60 + 6; // 14h 45m 06s
describe('formatHumanReadableDuration', () => {
test('decimal', () => {
expect(formatHumanReadableDuration(seconds, 'decimal', 'comma-point')).toBe('14.75 h');
});
test('hours-minutes', () => {
expect(formatHumanReadableDuration(seconds, 'hours-minutes')).toBe('14h 45min');
});
test('hours-minutes-colon-separated', () => {
expect(formatHumanReadableDuration(seconds, 'hours-minutes-colon-separated')).toBe('14:45');
});
test('hours-minutes-seconds-colon-separated', () => {
expect(formatHumanReadableDuration(seconds, 'hours-minutes-seconds-colon-separated')).toBe(
'14:45:06'
);
});
});
describe('formatReportingDuration', () => {
test('decimal', () => {
expect(formatReportingDuration(seconds, 'decimal', 'comma-point')).toBe('14.75 h');
});
test('hours-minutes', () => {
expect(formatReportingDuration(seconds, 'hours-minutes')).toBe('14:45:06');
});
test('hours-minutes-colon-separated', () => {
expect(formatReportingDuration(seconds, 'hours-minutes-colon-separated')).toBe('14:45:06');
});
test('hours-minutes-seconds-colon-separated', () => {
expect(formatReportingDuration(seconds, 'hours-minutes-seconds-colon-separated')).toBe(
'14:45:06'
);
});
});

View File

@@ -0,0 +1,20 @@
// Vitest setup file. Wires up the globals that the production code reads
// off `window` (`getTimezoneSetting`, `getWeekStartSetting`, `getNumberFormat`,
// `getIntervalFormat`) so that helpers under test don't crash when imported
// outside the running app.
import { vi } from 'vitest';
declare global {
interface Window {
getTimezoneSetting: () => string;
getWeekStartSetting: () => string;
getNumberFormat: () => string;
getIntervalFormat: () => string;
}
}
window.getTimezoneSetting = vi.fn(() => 'UTC');
window.getWeekStartSetting = vi.fn(() => 'monday');
window.getNumberFormat = vi.fn(() => 'point');
window.getIntervalFormat = vi.fn(() => 'hours-minutes');

View File

@@ -0,0 +1,348 @@
import { describe, it, expect } from 'vitest';
import dayjs from 'dayjs';
import utc from 'dayjs/plugin/utc';
import timezone from 'dayjs/plugin/timezone';
import { findFreeWindowOnDay, freeGapSecondsAfter, NoFreeWindowError } from './cellMath';
import type { TimeEntry } from '@/packages/api/src';
dayjs.extend(utc);
dayjs.extend(timezone);
// All times in the tests are in UTC for clarity. The "day" we search is
// 2026-04-10 in UTC (so we use tz='UTC' to avoid local-machine surprises).
const TZ = 'UTC';
const DATE = '2026-04-10';
/** Build a fake TimeEntry from UTC ISO timestamps. */
function entry(start: string, end: string | null, id = `e-${start}-${end}`): TimeEntry {
const startMs = dayjs.utc(start).valueOf();
const endMs = end ? dayjs.utc(end).valueOf() : startMs;
return {
id,
start,
end,
duration: end ? Math.floor((endMs - startMs) / 1000) : null,
description: '',
member_id: 'm-1',
project_id: null,
task_id: null,
billable: false,
tags: [],
// The grid only reads the fields above; the rest are placeholders
// to satisfy the TimeEntry type without pulling in real fixtures.
user_id: 'u-1',
organization_id: 'o-1',
} as unknown as TimeEntry;
}
const HOUR = 3600;
describe('findFreeWindowOnDay', () => {
// ── Empty / trivial cases ──────────────────────────────────────
it('returns the start of the day for a totally empty day', () => {
const result = findFreeWindowOnDay([], DATE, HOUR, TZ);
expect(result).toEqual({
start: '2026-04-10T00:00:00Z',
end: '2026-04-10T01:00:00Z',
});
});
it('returns null for zero or negative required seconds', () => {
expect(findFreeWindowOnDay([], DATE, 0, TZ)).toBeNull();
expect(findFreeWindowOnDay([], DATE, -1, TZ)).toBeNull();
});
it('refuses any duration > 24h on principle (single-day constraint)', () => {
const result = findFreeWindowOnDay([], DATE, 25 * HOUR, TZ);
expect(result).toBeNull();
});
// ── Single obstacle, basic gap finding ────────────────────────
it('finds the gap before a single obstacle if it fits', () => {
const obs = [entry('2026-04-10T10:00:00Z', '2026-04-10T11:00:00Z')];
const result = findFreeWindowOnDay(obs, DATE, HOUR, TZ);
expect(result?.start).toBe('2026-04-10T00:00:00Z');
expect(result?.end).toBe('2026-04-10T01:00:00Z');
});
it('finds the gap after a single obstacle when preferredStart skips earlier gaps', () => {
const obs = [entry('2026-04-10T10:00:00Z', '2026-04-10T11:00:00Z')];
const result = findFreeWindowOnDay(obs, DATE, HOUR, TZ, '2026-04-10T11:00:00Z');
expect(result?.start).toBe('2026-04-10T11:00:00Z');
expect(result?.end).toBe('2026-04-10T12:00:00Z');
});
// ── Multi-obstacle gap walking ────────────────────────────────
it('finds a gap between two obstacles', () => {
const obs = [
entry('2026-04-10T08:00:00Z', '2026-04-10T10:00:00Z'),
entry('2026-04-10T12:00:00Z', '2026-04-10T14:00:00Z'),
];
// Search for 2h, expecting the [00:00, 08:00) gap (8h available)
const result = findFreeWindowOnDay(obs, DATE, 2 * HOUR, TZ);
expect(result?.start).toBe('2026-04-10T00:00:00Z');
});
it('walks past gaps that are too small', () => {
const obs = [
entry('2026-04-10T00:30:00Z', '2026-04-10T10:00:00Z'),
entry('2026-04-10T11:00:00Z', '2026-04-10T12:00:00Z'),
];
// First gap is 30min, second gap is 1h, third is 12h.
// Asking for 90min → first two gaps are too small, third fits.
const result = findFreeWindowOnDay(obs, DATE, 90 * 60, TZ);
expect(result?.start).toBe('2026-04-10T12:00:00Z');
});
it('uses preferredStart even when it lands inside a gap', () => {
const obs = [
entry('2026-04-10T00:00:00Z', '2026-04-10T08:00:00Z'),
entry('2026-04-10T12:00:00Z', '2026-04-10T14:00:00Z'),
];
// preferredStart 09:00 → gap is [09:00, 12:00) = 3h
const result = findFreeWindowOnDay(obs, DATE, 2 * HOUR, TZ, '2026-04-10T09:00:00Z');
expect(result?.start).toBe('2026-04-10T09:00:00Z');
expect(result?.end).toBe('2026-04-10T11:00:00Z');
});
it('skips ahead when preferredStart lands inside an obstacle', () => {
const obs = [entry('2026-04-10T08:00:00Z', '2026-04-10T12:00:00Z')];
// preferredStart 10:00 lands inside [08:00, 12:00). We must
// advance to the next free position (12:00).
const result = findFreeWindowOnDay(obs, DATE, HOUR, TZ, '2026-04-10T10:00:00Z');
expect(result?.start).toBe('2026-04-10T12:00:00Z');
});
// ── Spillover from previous day ───────────────────────────────
it('treats an entry that started yesterday but ends today as an obstacle', () => {
// Yesterday 23:00 → today 02:00 → blocks the first 2h of today.
const obs = [entry('2026-04-09T23:00:00Z', '2026-04-10T02:00:00Z')];
const result = findFreeWindowOnDay(obs, DATE, HOUR, TZ);
expect(result?.start).toBe('2026-04-10T02:00:00Z');
});
it('ignores an entry that ended exactly at the start of the day', () => {
// Yesterday 22:00 → today 00:00 (exclusive) → does NOT block today.
const obs = [entry('2026-04-09T22:00:00Z', '2026-04-10T00:00:00Z')];
const result = findFreeWindowOnDay(obs, DATE, HOUR, TZ);
expect(result?.start).toBe('2026-04-10T00:00:00Z');
});
// ── Running entries ────────────────────────────────────────────
it('treats a running entry as blocking up to "now"', () => {
const obs = [entry('2026-04-10T08:00:00Z', null)];
const now = '2026-04-10T10:30:00Z';
// The running entry blocks 08:0010:30 → first free window is
// either before 08:00 (8h available, fits a 1h request).
const result = findFreeWindowOnDay(obs, DATE, HOUR, TZ, null, now);
expect(result?.start).toBe('2026-04-10T00:00:00Z');
});
it('places after a running entry when preferredStart pushes past it', () => {
const obs = [entry('2026-04-10T08:00:00Z', null)];
const now = '2026-04-10T10:30:00Z';
const result = findFreeWindowOnDay(obs, DATE, HOUR, TZ, '2026-04-10T09:00:00Z', now);
// preferredStart 09:00 lands inside the running entry's blocked
// range [08:00, 10:30) → must skip to 10:30.
expect(result?.start).toBe('2026-04-10T10:30:00Z');
});
// ── Midnight refusal ──────────────────────────────────────────
it('refuses to return a window that would cross midnight', () => {
const obs = [entry('2026-04-10T00:00:00Z', '2026-04-10T22:30:00Z')];
// Only 90min remain in the day. Asking for 2h → null.
const result = findFreeWindowOnDay(obs, DATE, 2 * HOUR, TZ);
expect(result).toBeNull();
});
it('accepts a window that ends exactly at midnight', () => {
const obs = [entry('2026-04-10T00:00:00Z', '2026-04-10T22:00:00Z')];
// Exactly 2h remain → 22:0000:00.
const result = findFreeWindowOnDay(obs, DATE, 2 * HOUR, TZ);
expect(result?.start).toBe('2026-04-10T22:00:00Z');
expect(result?.end).toBe('2026-04-11T00:00:00Z');
});
// ── Pre-existing overlapping obstacles ────────────────────────
it('merges overlapping obstacles before computing gaps', () => {
const obs = [
entry('2026-04-10T09:00:00Z', '2026-04-10T11:00:00Z'),
entry('2026-04-10T10:00:00Z', '2026-04-10T13:00:00Z'),
entry('2026-04-10T15:00:00Z', '2026-04-10T16:00:00Z'),
];
// Effective obstacles: [09:00, 13:00) and [15:00, 16:00)
// First gap is [00:00, 09:00) = 9h. Asking for 2h → 00:00.
const result = findFreeWindowOnDay(obs, DATE, 2 * HOUR, TZ);
expect(result?.start).toBe('2026-04-10T00:00:00Z');
});
// ── Day full ──────────────────────────────────────────────────
it('returns null when no gap is large enough', () => {
const obs = [
entry('2026-04-10T00:00:00Z', '2026-04-10T11:00:00Z'),
entry('2026-04-10T11:30:00Z', '2026-04-10T22:00:00Z'),
entry('2026-04-10T22:30:00Z', '2026-04-10T23:30:00Z'),
];
// Gaps: 30min, 30min, 30min. Asking for 1h → null.
const result = findFreeWindowOnDay(obs, DATE, HOUR, TZ);
expect(result).toBeNull();
});
// ── Timezone awareness ────────────────────────────────────────
it('respects the user timezone for day boundaries', () => {
// In Pacific/Auckland (+13 in April 2026), 2026-04-10 starts at
// 2026-04-09T11:00:00Z (NZDT was UTC+13 until April 5 2026, then
// NZST UTC+12 — let's pick a date in NZST so the offset is +12).
// 2026-04-10 NZST = 2026-04-09T12:00:00Z to 2026-04-10T12:00:00Z
const tz = 'Pacific/Auckland';
const obs = [
// This entry is at 2026-04-10T08:00:00Z = 2026-04-10T20:00 NZ
// → falls in the NZ day for 2026-04-10. So it blocks from
// 20:00 to 21:00 NZ time.
entry('2026-04-10T08:00:00Z', '2026-04-10T09:00:00Z'),
];
const result = findFreeWindowOnDay(obs, '2026-04-10', HOUR, tz);
// Day starts at 2026-04-09T12:00:00Z in NZST. First gap is the
// 20h before the obstacle starts.
expect(result?.start).toBe('2026-04-09T12:00:00Z');
});
it('does not place work past local midnight on spring-forward days', () => {
const tz = 'Europe/Vienna';
const obs = [entry('2026-03-28T23:00:00Z', '2026-03-29T22:00:00Z')];
const result = findFreeWindowOnDay(obs, '2026-03-29', HOUR, tz);
expect(result).toBeNull();
});
it('can use the final local hour on fall-back days', () => {
const tz = 'Europe/Vienna';
const result = findFreeWindowOnDay([], '2026-10-25', HOUR, tz, '2026-10-25T22:00:00Z');
expect(result).toEqual({
start: '2026-10-25T22:00:00Z',
end: '2026-10-25T23:00:00Z',
});
});
});
describe('freeGapSecondsAfter', () => {
it('returns the rest of the day for an empty day', () => {
const result = freeGapSecondsAfter([], DATE, TZ, '2026-04-10T09:00:00Z');
// 09:00 → 24:00 = 15 hours
expect(result).toBe(15 * HOUR);
});
it('returns 0 when cursor is at end-of-day', () => {
const result = freeGapSecondsAfter([], DATE, TZ, '2026-04-11T00:00:00Z');
expect(result).toBe(0);
});
it('returns 0 when cursor is after end-of-day', () => {
const result = freeGapSecondsAfter([], DATE, TZ, '2026-04-11T05:00:00Z');
expect(result).toBe(0);
});
it('returns 0 when cursor is before the day starts', () => {
const result = freeGapSecondsAfter([], DATE, TZ, '2026-04-09T23:00:00Z');
expect(result).toBe(0);
});
it('returns the gap to the next obstacle', () => {
const obs = [entry('2026-04-10T11:00:00Z', '2026-04-10T12:00:00Z')];
// cursor 09:00 → next obstacle 11:00 → 2h gap
const result = freeGapSecondsAfter(obs, DATE, TZ, '2026-04-10T09:00:00Z');
expect(result).toBe(2 * HOUR);
});
it('returns 0 when cursor sits inside an obstacle', () => {
const obs = [entry('2026-04-10T08:00:00Z', '2026-04-10T12:00:00Z')];
const result = freeGapSecondsAfter(obs, DATE, TZ, '2026-04-10T10:00:00Z');
expect(result).toBe(0);
});
it('returns the rest of the day when no obstacle is ahead', () => {
const obs = [entry('2026-04-10T08:00:00Z', '2026-04-10T09:00:00Z')];
const result = freeGapSecondsAfter(obs, DATE, TZ, '2026-04-10T09:00:00Z');
// 09:00 → 24:00 = 15h
expect(result).toBe(15 * HOUR);
});
it('skips obstacles strictly before the cursor', () => {
const obs = [
entry('2026-04-10T08:00:00Z', '2026-04-10T09:00:00Z'),
entry('2026-04-10T15:00:00Z', '2026-04-10T16:00:00Z'),
];
// cursor 09:00 → first obstacle (08-09) is behind, next is 15:00 → 6h
const result = freeGapSecondsAfter(obs, DATE, TZ, '2026-04-10T09:00:00Z');
expect(result).toBe(6 * HOUR);
});
it('treats a running entry as blocking up to "now"', () => {
const obs = [entry('2026-04-10T08:00:00Z', null)];
const now = '2026-04-10T10:00:00Z';
// cursor 11:00 is after the running entry's effective end (10:00),
// so the gap is the rest of the day = 13h
const result = freeGapSecondsAfter(obs, DATE, TZ, '2026-04-10T11:00:00Z', now);
expect(result).toBe(13 * HOUR);
});
it('returns 0 when cursor is inside a running entry (cursor < now)', () => {
const obs = [entry('2026-04-10T08:00:00Z', null)];
const now = '2026-04-10T12:00:00Z';
// cursor 10:00 falls inside [08:00, 12:00) → blocked
const result = freeGapSecondsAfter(obs, DATE, TZ, '2026-04-10T10:00:00Z', now);
expect(result).toBe(0);
});
it('clips the gap to midnight, never beyond', () => {
const obs = [entry('2026-04-10T08:00:00Z', '2026-04-10T09:00:00Z')];
// cursor 23:00 → 1h until midnight
const result = freeGapSecondsAfter(obs, DATE, TZ, '2026-04-10T23:00:00Z');
expect(result).toBe(HOUR);
});
it('clips spring-forward days at the next local midnight', () => {
const result = freeGapSecondsAfter(
[],
'2026-03-29',
'Europe/Vienna',
'2026-03-29T22:00:00Z'
);
expect(result).toBe(0);
});
it('includes the final local hour on fall-back days', () => {
const result = freeGapSecondsAfter(
[],
'2026-10-25',
'Europe/Vienna',
'2026-10-25T22:00:00Z'
);
expect(result).toBe(HOUR);
});
});
describe('NoFreeWindowError', () => {
it('carries the date and required seconds', () => {
const err = new NoFreeWindowError('2026-04-10', 7200);
expect(err.code).toBe('no_free_window');
expect(err.date).toBe('2026-04-10');
expect(err.requiredSeconds).toBe(7200);
expect(err.message).toContain('2026-04-10');
expect(err instanceof Error).toBe(true);
});
});

View File

@@ -0,0 +1,551 @@
import { describe, it, expect, vi, beforeEach, afterEach } from 'vitest';
import { ref } from 'vue';
import { createPinia, setActivePinia } from 'pinia';
import { useTimesheetCellMutations } from './useTimesheetCellMutations';
import { api } from '@/packages/api/src';
import type { TimesheetRow, TimesheetCell } from '@/utils/useTimesheetGrid';
import type { TimeEntry } from '@/packages/api/src';
const addNotification = vi.fn();
vi.mock('@/utils/useUser', () => ({
getCurrentOrganizationId: vi.fn(() => 'org-1'),
getCurrentMembershipId: vi.fn(() => 'mem-1'),
}));
vi.mock('@tanstack/vue-query', () => ({
useQueryClient: () => ({
invalidateQueries: vi.fn(),
}),
}));
vi.mock('@/utils/notification', () => ({
useNotificationsStore: () => ({
addNotification,
}),
}));
vi.mock('@/packages/api/src', () => ({
api: {
createTimeEntry: vi.fn(async () => ({ data: { id: 'new-id' } })),
updateTimeEntry: vi.fn(async () => undefined),
deleteTimeEntry: vi.fn(async () => undefined),
deleteTimeEntries: vi.fn(async () => undefined),
},
}));
// All scenarios use UTC so the local "day" matches the UTC day exactly
// (the test-setup mocks getTimezoneSetting to return 'UTC').
const DATE = '2026-04-10';
const HOUR = 3600;
function entry(start: string, end: string | null, overrides: Partial<TimeEntry> = {}): TimeEntry {
const startMs = new Date(start).valueOf();
const endMs = end ? new Date(end).valueOf() : startMs;
return {
id: overrides.id ?? `e-${start}-${end ?? 'running'}`,
start,
end,
duration: end ? Math.floor((endMs - startMs) / 1000) : null,
description: '',
member_id: 'm-1',
project_id: 'p-1',
task_id: null,
billable: false,
tags: [],
...overrides,
} as unknown as TimeEntry;
}
function buildCell(entries: TimeEntry[]): TimesheetCell {
return {
dayIndex: 0,
date: DATE,
entries,
totalSeconds: entries.reduce((sum, e) => sum + (e.duration ?? 0), 0),
};
}
function buildRow(
projectId: string | null,
entries: TimeEntry[],
key = `${projectId}:null`
): TimesheetRow {
const cell = buildCell(entries);
return {
key,
projectId,
taskId: null,
billable: false,
tags: [],
cells: new Map([[0, cell]]),
totalSeconds: cell.totalSeconds,
};
}
function buildEmptyRow(projectId: string | null, key = `${projectId}:null`): TimesheetRow {
return {
key,
projectId,
taskId: null,
billable: false,
tags: [],
cells: new Map(),
totalSeconds: 0,
};
}
/** Shape of the body the cell-mutation logic passes to api.createTimeEntry. */
interface CapturedEntry {
id?: string;
start: string;
end: string | null;
project_id?: string | null;
task_id?: string | null;
description?: string | null;
}
const apiMocks = vi.mocked(api);
function firstArg(
mock: typeof apiMocks.createTimeEntry | typeof apiMocks.updateTimeEntry
): CapturedEntry {
return mock.mock.calls[0]?.[0] as unknown as CapturedEntry;
}
function setup(
allEntries: TimeEntry[],
rowsValue: TimesheetRow[] = [],
removeSlot: (key: string) => void = () => undefined
) {
const cellMutations = useTimesheetCellMutations(
ref([
DATE,
'2026-04-11',
'2026-04-12',
'2026-04-13',
'2026-04-14',
'2026-04-15',
'2026-04-16',
]),
ref(allEntries),
ref(rowsValue),
removeSlot
);
return { cellMutations };
}
beforeEach(() => {
setActivePinia(createPinia());
apiMocks.createTimeEntry.mockClear();
apiMocks.updateTimeEntry.mockClear();
apiMocks.deleteTimeEntry.mockClear();
apiMocks.deleteTimeEntries.mockClear();
addNotification.mockClear();
// Lock the clock to mid-afternoon on the test day so running-entry
// tests have a deterministic "now". Past 12:00 to make spillover
// scenarios meaningful.
vi.useFakeTimers();
vi.setSystemTime(new Date('2026-04-10T14:00:00Z'));
});
afterEach(() => {
vi.useRealTimers();
});
describe('useTimesheetCellMutations.handleCellUpdate', () => {
// ── No-op ─────────────────────────────────────────────────────
it('does nothing when the diff is zero', async () => {
const { cellMutations } = setup([]);
const row = buildRow('p-1', [entry('2026-04-10T09:00:00Z', '2026-04-10T10:00:00Z')]);
await cellMutations.handleCellUpdate(row, 0, HOUR);
expect(apiMocks.createTimeEntry).not.toHaveBeenCalled();
expect(apiMocks.updateTimeEntry).not.toHaveBeenCalled();
expect(apiMocks.deleteTimeEntry).not.toHaveBeenCalled();
expect(apiMocks.deleteTimeEntries).not.toHaveBeenCalled();
});
// ── Delete cell ───────────────────────────────────────────────
it('clearing a cell deletes all entries in it', async () => {
const cellEntry = entry('2026-04-10T09:00:00Z', '2026-04-10T10:00:00Z');
const { cellMutations } = setup([cellEntry]);
const row = buildRow('p-1', [cellEntry]);
await cellMutations.handleCellUpdate(row, 0, 0);
expect(apiMocks.deleteTimeEntries).toHaveBeenCalledTimes(1);
const [, options] = apiMocks.deleteTimeEntries.mock.calls[0]!;
expect(options?.queries?.ids).toEqual([cellEntry.id]);
expect(options?.params?.organization).toBe('org-1');
});
// ── Create cell (Phase 1) ──────────────────────────────────────
describe('createCell', () => {
it('places a new entry at 09:00 on an empty day', async () => {
const { cellMutations } = setup([]);
const row = buildEmptyRow('p-1');
await cellMutations.handleCellUpdate(row, 0, HOUR);
expect(apiMocks.createTimeEntry).toHaveBeenCalledTimes(1);
const arg = firstArg(apiMocks.createTimeEntry);
expect(arg.start).toBe('2026-04-10T09:00:00Z');
expect(arg.end).toBe('2026-04-10T10:00:00Z');
expect(arg.project_id).toBe('p-1');
});
it('passes no-project rows to the API as null', async () => {
const { cellMutations } = setup([]);
const row = buildEmptyRow(null);
await cellMutations.handleCellUpdate(row, 0, HOUR);
expect(apiMocks.createTimeEntry).toHaveBeenCalledTimes(1);
expect(firstArg(apiMocks.createTimeEntry).project_id).toBeNull();
});
it('collapses an empty duplicate row after its first entry is created', async () => {
const removeSlot = vi.fn();
const existingRow = buildEmptyRow('p-1', 'existing-slot');
const duplicateRow = buildEmptyRow('p-1', 'duplicate-slot');
const { cellMutations } = setup([], [existingRow, duplicateRow], removeSlot);
await cellMutations.handleCellUpdate(duplicateRow, 0, HOUR);
expect(apiMocks.createTimeEntry).toHaveBeenCalledTimes(1);
expect(removeSlot).toHaveBeenCalledWith('duplicate-slot');
expect(addNotification).toHaveBeenCalledWith(
'success',
'Merged into matching row',
'Another row with the same project, task, billable status and tags already exists.'
);
});
it("falls back to the start of the day when 09:00 wouldn't fit", async () => {
// Block 09:00 → 23:30 with another row's entry. The only
// gap big enough for 1h is 00:00 → 09:00.
const blocker = entry('2026-04-10T09:00:00Z', '2026-04-10T23:30:00Z', {
id: 'blocker',
});
const { cellMutations } = setup([blocker]);
const row = buildEmptyRow('p-1');
await cellMutations.handleCellUpdate(row, 0, HOUR);
expect(apiMocks.createTimeEntry).toHaveBeenCalledTimes(1);
const arg = firstArg(apiMocks.createTimeEntry);
expect(arg.start).toBe('2026-04-10T00:00:00Z');
expect(arg.end).toBe('2026-04-10T01:00:00Z');
});
it('avoids overlapping with another row on the same day (Scenario #4)', async () => {
// Another row has an entry 09:00 → 10:00. The new entry must
// not overlap it.
const blocker = entry('2026-04-10T09:00:00Z', '2026-04-10T10:00:00Z', {
id: 'blocker',
});
const { cellMutations } = setup([blocker]);
const row = buildEmptyRow('p-1');
await cellMutations.handleCellUpdate(row, 0, HOUR);
const arg = firstArg(apiMocks.createTimeEntry);
// 09:00 is blocked → must place after, at 10:00
expect(arg.start).toBe('2026-04-10T10:00:00Z');
expect(arg.end).toBe('2026-04-10T11:00:00Z');
});
it('treats a running entry as a blocker (Scenario #1)', async () => {
const running = entry('2026-04-10T08:00:00Z', null, { id: 'running' });
const { cellMutations } = setup([running]);
const row = buildEmptyRow('p-1');
await cellMutations.handleCellUpdate(row, 0, HOUR);
const arg = firstArg(apiMocks.createTimeEntry);
// The running entry blocks [08:00, now=14:00). The 09:00
// work-hours default lands inside that block, so the
// search advances the cursor to 14:00 and places the new
// entry at 14:00 → 15:00. Critically, this does NOT
// overlap the running timer.
expect(arg.start).toBe('2026-04-10T14:00:00Z');
expect(arg.end).toBe('2026-04-10T15:00:00Z');
});
it('avoids spillover from the previous day (Scenario #2)', async () => {
// An entry from yesterday spills into today's first 4h.
const spillover = entry('2026-04-09T22:00:00Z', '2026-04-10T04:00:00Z', {
id: 'spillover',
});
const { cellMutations } = setup([spillover]);
const row = buildEmptyRow('p-1');
await cellMutations.handleCellUpdate(row, 0, HOUR);
const arg = firstArg(apiMocks.createTimeEntry);
// 09:00 is free (the spillover ends at 04:00).
expect(arg.start).toBe('2026-04-10T09:00:00Z');
});
it('refuses to cross midnight (Scenario #3)', async () => {
// Block all of the day except the last 2h. Asking for 3h
// → no single-day window fits → notification, no API call.
const blocker = entry('2026-04-10T00:00:00Z', '2026-04-10T22:00:00Z', {
id: 'blocker',
});
const { cellMutations } = setup([blocker]);
const row = buildEmptyRow('p-1');
await cellMutations.handleCellUpdate(row, 0, 3 * HOUR);
expect(apiMocks.createTimeEntry).not.toHaveBeenCalled();
});
});
// ── Extend cell (Phase 2) ──────────────────────────────────────
describe('extendCell', () => {
it("extends the cell's latest-ended entry forward when there is room", async () => {
const cellEntry = entry('2026-04-10T09:00:00Z', '2026-04-10T10:00:00Z', {
id: 'extend-me',
});
const { cellMutations } = setup([cellEntry]);
const row = buildRow('p-1', [cellEntry]);
// Cell is 1h, request 2h total → +1h
await cellMutations.handleCellUpdate(row, 0, 2 * HOUR);
expect(apiMocks.updateTimeEntry).toHaveBeenCalledTimes(1);
const updated = firstArg(apiMocks.updateTimeEntry);
expect(updated.id).toBe('extend-me');
expect(updated.end).toBe('2026-04-10T11:00:00Z');
expect(apiMocks.createTimeEntry).not.toHaveBeenCalled();
});
it('picks the latest-END entry when nested entries exist (Scenario #6)', async () => {
// Outer entry 09:00 → 12:00, inner entry 10:00 → 11:00.
// The latest START is "inner" but the latest END is "outer".
// Extending should grow the OUTER entry, not the inner.
const outer = entry('2026-04-10T09:00:00Z', '2026-04-10T12:00:00Z', { id: 'outer' });
const inner = entry('2026-04-10T10:00:00Z', '2026-04-10T11:00:00Z', { id: 'inner' });
const { cellMutations } = setup([outer, inner]);
const row = buildRow('p-1', [outer, inner]);
// Cell total = 3h + 1h = 4h. Bump to 5h → +1h.
await cellMutations.handleCellUpdate(row, 0, 5 * HOUR);
expect(apiMocks.updateTimeEntry).toHaveBeenCalledTimes(1);
const updated = firstArg(apiMocks.updateTimeEntry);
expect(updated.id).toBe('outer');
expect(updated.end).toBe('2026-04-10T13:00:00Z');
});
it('splits the extension when another row blocks the path (Scenario #5)', async () => {
// Cell entry: 09:00 → 10:00 (1h). Blocker on another row:
// 10:30 → 11:30. Bump cell to 3h (+2h):
// - 30 minutes fit in the gap [10:00, 10:30) → extend to 10:30
// - 90 minutes remain → place a NEW entry in the next free
// window (11:30 → 13:00)
const cellEntry = entry('2026-04-10T09:00:00Z', '2026-04-10T10:00:00Z', {
id: 'cell-entry',
});
const blocker = entry('2026-04-10T10:30:00Z', '2026-04-10T11:30:00Z', {
id: 'blocker',
project_id: 'other-project',
});
const { cellMutations } = setup([cellEntry, blocker]);
const row = buildRow('p-1', [cellEntry]);
await cellMutations.handleCellUpdate(row, 0, 3 * HOUR);
expect(apiMocks.updateTimeEntry).toHaveBeenCalledTimes(1);
const updated = firstArg(apiMocks.updateTimeEntry);
expect(updated.id).toBe('cell-entry');
expect(updated.end).toBe('2026-04-10T10:30:00Z');
expect(apiMocks.createTimeEntry).toHaveBeenCalledTimes(1);
const created = firstArg(apiMocks.createTimeEntry);
// Remainder = 2h - 30min = 90min, placed in next free window
// (11:30 → 13:00)
expect(created.start).toBe('2026-04-10T11:30:00Z');
expect(created.end).toBe('2026-04-10T13:00:00Z');
});
it("places everything as a new entry when the cell's tail abuts another row immediately (Scenario #5 zero-gap)", async () => {
// Cell entry: 09:00 → 10:00. Another row starts EXACTLY at 10:00.
// Gap after cell entry = 0. The whole add becomes a new entry
// somewhere else.
const cellEntry = entry('2026-04-10T09:00:00Z', '2026-04-10T10:00:00Z', {
id: 'cell-entry',
});
const blocker = entry('2026-04-10T10:00:00Z', '2026-04-10T11:00:00Z', {
id: 'blocker',
project_id: 'other-project',
});
const { cellMutations } = setup([cellEntry, blocker]);
const row = buildRow('p-1', [cellEntry]);
// Bump cell from 1h to 2h → +1h
await cellMutations.handleCellUpdate(row, 0, 2 * HOUR);
// No update to cell entry (gap is zero)
expect(apiMocks.updateTimeEntry).not.toHaveBeenCalled();
// New entry placed in next free window: 11:00 → 12:00
expect(apiMocks.createTimeEntry).toHaveBeenCalledTimes(1);
const created = firstArg(apiMocks.createTimeEntry);
expect(created.start).toBe('2026-04-10T11:00:00Z');
expect(created.end).toBe('2026-04-10T12:00:00Z');
});
it('falls back to creating a new entry when the latest cell entry is running (Scenario #8)', async () => {
// Cell contains only a running timer.
const running = entry('2026-04-10T08:00:00Z', null, {
id: 'running',
duration: null,
});
// The grid would treat the cell total as 0 because duration is
// null, so we fake a non-zero totalSeconds via a manual cell.
const cell: TimesheetCell = {
dayIndex: 0,
date: DATE,
entries: [running],
// Pretend we computed 1h so that handleCellUpdate sees a
// diff > 0 and routes through extendCell rather than
// createCell.
totalSeconds: HOUR,
};
const row: TimesheetRow = {
key: 'p-1:null',
projectId: 'p-1',
taskId: null,
billable: false,
tags: [],
cells: new Map([[0, cell]]),
totalSeconds: HOUR,
};
const { cellMutations } = setup([running]);
// Bump from 1h to 2h → +1h. extendCell sees the running entry
// as the latest-end and falls through to createCell.
await cellMutations.handleCellUpdate(row, 0, 2 * HOUR);
expect(apiMocks.updateTimeEntry).not.toHaveBeenCalled();
expect(apiMocks.createTimeEntry).toHaveBeenCalledTimes(1);
// Running entry blocks [08:00, now=14:00). 09:00 lands inside
// → cursor advances to 14:00. Crucially the running entry is
// never modified.
const created = firstArg(apiMocks.createTimeEntry);
expect(created.start).toBe('2026-04-10T14:00:00Z');
expect(created.end).toBe('2026-04-10T15:00:00Z');
});
it('places the remainder strictly after the just-extended end (no stale-overlap)', async () => {
// Regression: timeEntries.value is stale right after the
// updateEntry call (still shows candidate.end at the old
// value). Without an explicit floor, findFreeWindowOnDay
// would propose a window inside the gap we just filled —
// overlapping the now-extended candidate.
//
// Cell entry: 09:00 → 10:00 (1h). Blocker on another row at
// 11:00 → 12:00. Gap is 1h. Bump cell to 2.5h (+1.5h):
// - extend by 1h → newEnd = 11:00
// - remainder = 30 min must land AFTER 11:00, not in the
// stale-looking [10:00, 11:00] window.
const cellEntry = entry('2026-04-10T09:00:00Z', '2026-04-10T10:00:00Z', {
id: 'cell-entry',
});
const blocker = entry('2026-04-10T11:00:00Z', '2026-04-10T12:00:00Z', {
id: 'blocker',
project_id: 'other-project',
});
const { cellMutations } = setup([cellEntry, blocker]);
const row = buildRow('p-1', [cellEntry]);
await cellMutations.handleCellUpdate(row, 0, Math.round(2.5 * HOUR));
expect(apiMocks.updateTimeEntry).toHaveBeenCalledTimes(1);
const updated = firstArg(apiMocks.updateTimeEntry);
expect(updated.id).toBe('cell-entry');
expect(updated.end).toBe('2026-04-10T11:00:00Z');
expect(apiMocks.createTimeEntry).toHaveBeenCalledTimes(1);
const created = firstArg(apiMocks.createTimeEntry);
// Next free window after blocker, at 12:00.
expect(created.start).toBe('2026-04-10T12:00:00Z');
expect(created.end).toBe('2026-04-10T12:30:00Z');
});
it('refuses extension that would cross midnight before patching the server (Scenario #7)', async () => {
// Cell entry 22:00 → 23:00. Bump to 4h (+3h). Only 1h is
// available before midnight. Because the remaining 2h don't
// fit anywhere else on the day, the mutation now aborts
// before issuing the PATCH, leaving the entry untouched.
const cellEntry = entry('2026-04-10T22:00:00Z', '2026-04-10T23:00:00Z', {
id: 'cell-entry',
});
const fillEarly = entry('2026-04-10T00:00:00Z', '2026-04-10T22:00:00Z', {
id: 'fill',
project_id: 'other-project',
});
const { cellMutations } = setup([cellEntry, fillEarly]);
const row = buildRow('p-1', [cellEntry]);
// Bump from 1h to 4h → +3h
await cellMutations.handleCellUpdate(row, 0, 4 * HOUR);
// Nothing should be patched or created because the preflight
// fit-check rejects the whole edit.
expect(apiMocks.updateTimeEntry).not.toHaveBeenCalled();
expect(apiMocks.createTimeEntry).not.toHaveBeenCalled();
});
});
// ── Shrink (unchanged behavior, still correct) ─────────────────
describe('shrinkFromEnd', () => {
it('shortens the latest entry by the requested amount', async () => {
const cellEntry = entry('2026-04-10T09:00:00Z', '2026-04-10T12:00:00Z', {
id: 'shrink-me',
});
const { cellMutations } = setup([cellEntry]);
const row = buildRow('p-1', [cellEntry]);
// 3h → 2h: shrink by 1h
await cellMutations.handleCellUpdate(row, 0, 2 * HOUR);
expect(apiMocks.updateTimeEntry).toHaveBeenCalledTimes(1);
const updated = firstArg(apiMocks.updateTimeEntry);
expect(updated.end).toBe('2026-04-10T11:00:00Z');
});
it('deletes entries that are entirely consumed', async () => {
const a = entry('2026-04-10T09:00:00Z', '2026-04-10T10:00:00Z', {
id: 'a',
});
const b = entry('2026-04-10T11:00:00Z', '2026-04-10T12:00:00Z', {
id: 'b',
});
const { cellMutations } = setup([a, b]);
const row = buildRow('p-1', [a, b]);
// Cell = 2h, shrink to 0.5h: must delete b entirely (1h) and
// shorten a from 1h to 30min.
await cellMutations.handleCellUpdate(row, 0, 1800);
expect(apiMocks.deleteTimeEntry).toHaveBeenCalledTimes(1);
const [, deleteOptions] = apiMocks.deleteTimeEntry.mock.calls[0]!;
expect(deleteOptions?.params?.timeEntry).toBe('b');
expect(apiMocks.updateTimeEntry).toHaveBeenCalledTimes(1);
const updated = firstArg(apiMocks.updateTimeEntry);
expect(updated.id).toBe('a');
expect(updated.end).toBe('2026-04-10T09:30:00Z');
});
});
});

View File

@@ -0,0 +1,272 @@
import { beforeEach, describe, expect, it, vi } from 'vitest';
import { ref } from 'vue';
import type { Project, TimeEntry } from '@/packages/api/src';
import type { TimesheetCell, TimesheetRow } from '@/utils/useTimesheetGrid';
import { useTimesheetRowMutations } from './useTimesheetRowMutations';
const addNotification = vi.fn();
vi.mock('@/utils/notification', () => ({
useNotificationsStore: () => ({
addNotification,
}),
}));
function project(id: string, isBillable: boolean): Project {
return {
id,
name: id,
is_billable: isBillable,
} as unknown as Project;
}
function entry(id: string): TimeEntry {
return {
id,
start: '2026-04-10T09:00:00Z',
end: '2026-04-10T10:00:00Z',
duration: 3600,
description: '',
member_id: 'm-1',
project_id: 'p-1',
task_id: null,
billable: false,
tags: [],
} as unknown as TimeEntry;
}
function buildCell(entries: TimeEntry[]): TimesheetCell {
return {
dayIndex: 0,
date: '2026-04-10',
entries,
totalSeconds: entries.reduce((sum, item) => sum + (item.duration ?? 0), 0),
};
}
function buildRow(key: string, projectId: string | null, entries: TimeEntry[]): TimesheetRow {
const cells = entries.length > 0 ? new Map([[0, buildCell(entries)]]) : new Map();
const totalSeconds = entries.reduce((sum, item) => sum + (item.duration ?? 0), 0);
return {
key,
projectId,
taskId: null,
billable: false,
tags: [],
cells,
totalSeconds,
};
}
describe('useTimesheetRowMutations', () => {
beforeEach(() => {
addNotification.mockClear();
});
it('collapses a populated row into an existing matching row after identity change', async () => {
const existingRow = buildRow('row-a', 'p-1', [entry('e-a')]);
const editedRow = buildRow('row-b', 'p-2', [entry('e-b')]);
const rows = ref([existingRow, editedRow]);
const updateTimeEntries = vi.fn().mockResolvedValue(undefined);
const addSlot = vi.fn();
const updateSlot = vi.fn();
const removeSlot = vi.fn();
const { handleRowIdentityChange } = useTimesheetRowMutations(
{ updateTimeEntries },
ref<Project[]>([]),
rows,
addSlot,
updateSlot,
removeSlot
);
await handleRowIdentityChange(editedRow, { projectId: 'p-1' });
expect(updateTimeEntries).toHaveBeenCalledWith({
ids: ['e-b'],
changes: { project_id: 'p-1' },
});
expect(removeSlot).toHaveBeenCalledWith('row-b');
expect(updateSlot).not.toHaveBeenCalled();
expect(addNotification).toHaveBeenCalledWith(
'success',
'Merged into matching row',
'Another row with the same project, task, billable status and tags already exists.'
);
});
it('translates identity fields to snake_case API field names', async () => {
const row = buildRow('row-a', 'p-1', [entry('e-a')]);
const rows = ref([row]);
const updateTimeEntries = vi.fn().mockResolvedValue(undefined);
const { handleRowIdentityChange } = useTimesheetRowMutations(
{ updateTimeEntries },
ref<Project[]>([]),
rows,
vi.fn(),
vi.fn(),
vi.fn()
);
await handleRowIdentityChange(row, {
projectId: 'p-2',
taskId: 't-1',
billable: true,
tags: ['tag-1'],
});
expect(updateTimeEntries).toHaveBeenCalledWith({
ids: ['e-a'],
changes: {
project_id: 'p-2',
task_id: 't-1',
billable: true,
tags: ['tag-1'],
},
});
});
it('only includes touched fields in the API changeset', async () => {
const row = buildRow('row-a', 'p-1', [entry('e-a')]);
const rows = ref([row]);
const updateTimeEntries = vi.fn().mockResolvedValue(undefined);
const { handleRowIdentityChange } = useTimesheetRowMutations(
{ updateTimeEntries },
ref<Project[]>([]),
rows,
vi.fn(),
vi.fn(),
vi.fn()
);
await handleRowIdentityChange(row, { tags: ['tag-1'] });
expect(updateTimeEntries).toHaveBeenCalledWith({
ids: ['e-a'],
changes: { tags: ['tag-1'] },
});
});
it('keeps an empty duplicate row until it receives time', async () => {
const existingRow = buildRow('row-a', 'p-1', [entry('e-a')]);
const emptyRow = buildRow('row-b', 'p-2', []);
const rows = ref([existingRow, emptyRow]);
const updateTimeEntries = vi.fn().mockResolvedValue(undefined);
const addSlot = vi.fn();
const updateSlot = vi.fn();
const removeSlot = vi.fn();
const { handleRowIdentityChange } = useTimesheetRowMutations(
{ updateTimeEntries },
ref<Project[]>([]),
rows,
addSlot,
updateSlot,
removeSlot
);
await handleRowIdentityChange(emptyRow, { projectId: 'p-1' });
expect(updateTimeEntries).not.toHaveBeenCalled();
expect(updateSlot).toHaveBeenCalledWith('row-b', {
projectId: 'p-1',
taskId: null,
billable: false,
tags: [],
});
expect(removeSlot).not.toHaveBeenCalled();
});
it('sends null project updates when an existing row changes to no project', async () => {
const row = buildRow('row-a', 'p-1', [entry('e-a')]);
const rows = ref([row]);
const updateTimeEntries = vi.fn().mockResolvedValue(undefined);
const updateSlot = vi.fn();
const { handleRowIdentityChange } = useTimesheetRowMutations(
{ updateTimeEntries },
ref<Project[]>([]),
rows,
vi.fn(),
updateSlot,
vi.fn()
);
await handleRowIdentityChange(row, { projectId: null });
expect(updateTimeEntries).toHaveBeenCalledWith({
ids: ['e-a'],
changes: { project_id: null },
});
expect(updateSlot).toHaveBeenCalledWith('row-a', {
projectId: null,
taskId: null,
billable: false,
tags: [],
});
});
it('defaults billable from the selected project when an empty row picks its first project', async () => {
const emptyRow = buildRow('row-a', null, []);
const rows = ref([emptyRow]);
const updateTimeEntries = vi.fn().mockResolvedValue(undefined);
const updateSlot = vi.fn();
const { handleRowIdentityChange } = useTimesheetRowMutations(
{ updateTimeEntries },
ref([project('p-billable', true)]),
rows,
vi.fn(),
updateSlot,
vi.fn()
);
await handleRowIdentityChange(emptyRow, { projectId: 'p-billable' });
expect(updateTimeEntries).not.toHaveBeenCalled();
expect(updateSlot).toHaveBeenCalledWith('row-a', {
projectId: 'p-billable',
taskId: null,
billable: true,
tags: [],
});
});
it('handleAddRow defaults billable from the selected project metadata', () => {
const addSlot = vi.fn();
const { handleAddRow } = useTimesheetRowMutations(
{ updateTimeEntries: vi.fn() },
ref([project('p-billable', true)]),
ref<TimesheetRow[]>([]),
addSlot,
vi.fn(),
vi.fn()
);
handleAddRow('p-billable', 't-1');
expect(addSlot).toHaveBeenCalledWith('p-billable', 't-1', true, []);
});
it('can add a no-project row', () => {
const addSlot = vi.fn();
const { handleAddRow } = useTimesheetRowMutations(
{ updateTimeEntries: vi.fn() },
ref([project('p-billable', true)]),
ref<TimesheetRow[]>([]),
addSlot,
vi.fn(),
vi.fn()
);
handleAddRow(null, null);
expect(addSlot).toHaveBeenCalledWith(null, null, false, []);
});
});

View File

@@ -0,0 +1,277 @@
import { describe, expect, it } from 'vitest';
import { nextTick, ref } from 'vue';
import dayjs from 'dayjs';
import utc from 'dayjs/plugin/utc';
import type { Dayjs } from 'dayjs';
import type { Project, Task, TimeEntry } from '@/packages/api/src';
import { useTimesheetGrid } from './useTimesheetGrid';
dayjs.extend(utc);
const WEEK_DAYS = [
'2026-04-06',
'2026-04-07',
'2026-04-08',
'2026-04-09',
'2026-04-10',
'2026-04-11',
'2026-04-12',
];
function entry(start: string, end: string | null, overrides: Partial<TimeEntry> = {}): TimeEntry {
const startMs = new Date(start).valueOf();
const endMs = end ? new Date(end).valueOf() : startMs;
return {
id: overrides.id ?? `e-${start}-${end ?? 'running'}`,
start,
end,
duration: end ? Math.floor((endMs - startMs) / 1000) : null,
description: '',
member_id: 'm-1',
project_id: 'p-1',
task_id: null,
billable: false,
tags: [],
...overrides,
} as unknown as TimeEntry;
}
function project(id: string, name: string, isBillable = false): Project {
return {
id,
name,
is_billable: isBillable,
} as unknown as Project;
}
function task(id: string, name: string, projectId: string): Task {
return {
id,
name,
project_id: projectId,
} as unknown as Task;
}
describe('useTimesheetGrid', () => {
it('seeds unseen identities and re-sorts seeded rows when project metadata changes', async () => {
const timeEntries = ref([
entry('2026-04-10T09:00:00Z', '2026-04-10T10:00:00Z', {
id: 'seed-b',
project_id: 'p-b',
task_id: null,
}),
entry('2026-04-10T10:00:00Z', '2026-04-10T11:00:00Z', {
id: 'seed-a-z',
project_id: 'p-a',
task_id: 't-z',
}),
entry('2026-04-10T11:00:00Z', '2026-04-10T12:00:00Z', {
id: 'seed-a-a',
project_id: 'p-a',
task_id: 't-a',
}),
]);
const projects = ref([project('p-a', 'Bravo'), project('p-b', 'Alpha')]);
const tasks = ref([task('t-z', 'Zulu Task', 'p-a'), task('t-a', 'Alpha Task', 'p-a')]);
const { rows } = useTimesheetGrid(
timeEntries,
ref(WEEK_DAYS),
projects,
tasks,
ref<Dayjs | null>(null)
);
expect(rows.value.map((row) => `${row.projectId}:${row.taskId}`)).toEqual([
'p-b:null',
'p-a:t-a',
'p-a:t-z',
]);
projects.value = [project('p-a', 'Aardvark'), project('p-b', 'Zulu')];
await nextTick();
expect(rows.value.map((row) => `${row.projectId}:${row.taskId}`)).toEqual([
'p-a:t-a',
'p-a:t-z',
'p-b:null',
]);
});
it('keeps user-added slots below seeded rows in insertion order', async () => {
const timeEntries = ref([
entry('2026-04-10T09:00:00Z', '2026-04-10T10:00:00Z', {
id: 'seed-b',
project_id: 'p-seed-b',
}),
entry('2026-04-10T10:00:00Z', '2026-04-10T11:00:00Z', {
id: 'seed-a',
project_id: 'p-seed-a',
}),
]);
const projects = ref([
project('p-seed-a', 'Bravo Seed'),
project('p-seed-b', 'Alpha Seed'),
project('p-user-1', 'Alpha User'),
project('p-user-2', 'Zulu User'),
]);
const { rows, addSlot } = useTimesheetGrid(
timeEntries,
ref(WEEK_DAYS),
projects,
ref<Task[]>([]),
ref<Dayjs | null>(null)
);
addSlot('p-user-2', null, false, []);
addSlot('p-user-1', null, false, []);
await nextTick();
expect(rows.value.map((row) => row.projectId)).toEqual([
'p-seed-b',
'p-seed-a',
'p-user-2',
'p-user-1',
]);
projects.value = [
project('p-seed-a', 'Zulu Seed'),
project('p-seed-b', 'Alpha Seed'),
project('p-user-1', 'Aardvark User'),
project('p-user-2', 'Bravo User'),
];
await nextTick();
expect(rows.value.map((row) => row.projectId)).toEqual([
'p-seed-b',
'p-seed-a',
'p-user-2',
'p-user-1',
]);
});
it('assigns entries to the first duplicate-identity slot and leaves later duplicates empty', async () => {
const timeEntries = ref<TimeEntry[]>([]);
const { rows, addSlot } = useTimesheetGrid(
timeEntries,
ref(WEEK_DAYS),
ref([project('p-1', 'Project One')]),
ref<Task[]>([]),
ref<Dayjs | null>(null)
);
const firstKey = addSlot('p-1', null, false, []);
const secondKey = addSlot('p-1', null, false, []);
await nextTick();
timeEntries.value = [
entry('2026-04-10T09:00:00Z', '2026-04-10T10:00:00Z', { project_id: 'p-1' }),
];
await nextTick();
expect(rows.value).toHaveLength(2);
expect(rows.value[0]?.key).toBe(firstKey);
expect(rows.value[0]?.totalSeconds).toBe(3600);
expect(rows.value[1]?.key).toBe(secondKey);
expect(rows.value[1]?.totalSeconds).toBe(0);
expect(rows.value[1]?.cells.size).toBe(0);
});
it('keeps no-project user slots aligned with refetched no-project entries', async () => {
const timeEntries = ref<TimeEntry[]>([]);
const { rows, addSlot } = useTimesheetGrid(
timeEntries,
ref(WEEK_DAYS),
ref<Project[]>([]),
ref<Task[]>([]),
ref<Dayjs | null>(null)
);
const rowKey = addSlot(null, null, false, []);
await nextTick();
expect(rows.value).toHaveLength(1);
expect(rows.value[0]).toMatchObject({ key: rowKey, projectId: null });
timeEntries.value = [
entry('2026-04-10T09:00:00Z', '2026-04-10T10:00:00Z', {
id: 'no-project',
project_id: null,
}),
];
await nextTick();
expect(rows.value).toHaveLength(1);
expect(rows.value[0]).toMatchObject({
key: rowKey,
projectId: null,
totalSeconds: 3600,
});
});
it('updates a slot identity in place and clearSlots removes all rows', async () => {
const { rows, slots, addSlot, updateSlot, clearSlots } = useTimesheetGrid(
ref<TimeEntry[]>([]),
ref(WEEK_DAYS),
ref([project('p-next', 'Next Project')]),
ref([task('t-1', 'Task One', 'p-next')]),
ref<Dayjs | null>(null)
);
const key = addSlot(null, null, false, []);
await nextTick();
updateSlot(key, {
projectId: 'p-next',
taskId: 't-1',
billable: true,
tags: ['b-tag', 'a-tag'],
});
await nextTick();
expect(rows.value).toHaveLength(1);
expect(rows.value[0]).toMatchObject({
key,
projectId: 'p-next',
taskId: 't-1',
billable: true,
tags: ['a-tag', 'b-tag'],
});
clearSlots();
await nextTick();
expect(rows.value).toHaveLength(0);
expect(slots.value).toHaveLength(0);
});
it('includes running entries in row and week totals using the live timer clock', () => {
const currentTime = ref(dayjs.utc('2026-04-10T10:00:00Z'));
const runningEntry = entry('2026-04-10T09:00:00Z', null, { id: 'running' });
const { rows, dayTotals, grandTotal } = useTimesheetGrid(
ref([runningEntry]),
ref(WEEK_DAYS),
ref<Project[]>([]),
ref<Task[]>([]),
currentTime
);
expect(rows.value).toHaveLength(1);
expect(rows.value[0]?.cells.get(4)?.totalSeconds).toBe(3600);
expect(rows.value[0]?.totalSeconds).toBe(3600);
expect(dayTotals.value[4]).toBe(3600);
expect(grandTotal.value).toBe(3600);
currentTime.value = dayjs.utc('2026-04-10T11:30:00Z');
expect(rows.value[0]?.cells.get(4)?.totalSeconds).toBe(9000);
expect(rows.value[0]?.totalSeconds).toBe(9000);
expect(dayTotals.value[4]).toBe(9000);
expect(grandTotal.value).toBe(9000);
});
});

19
vitest.config.ts Normal file
View File

@@ -0,0 +1,19 @@
import { defineConfig } from 'vitest/config';
import vue from '@vitejs/plugin-vue';
import { fileURLToPath } from 'node:url';
export default defineConfig({
plugins: [vue()],
resolve: {
alias: {
'@': fileURLToPath(new URL('./resources/js', import.meta.url)),
},
},
test: {
environment: 'happy-dom',
globals: false,
setupFiles: ['./resources/js/test-setup.ts'],
include: ['resources/js/**/*.{test,spec}.{ts,tsx}'],
exclude: ['**/node_modules/**', '**/e2e/**', '**/dist/**'],
},
});