Compare commits
57 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| e8279bd521 | |||
| 36d50c0736 | |||
| 5b6c6abb69 | |||
| 30c97e2f0d | |||
| de1a508e52 | |||
| 54d26e0241 | |||
| ac47fc001d | |||
| 4936929849 | |||
| 6672fb78c9 | |||
| b349e26833 | |||
| 56d244aa61 | |||
| 1a348c62f5 | |||
| 067a8c166b | |||
| 8fdd79ff33 | |||
| cd8263e607 | |||
| e6a097d81d | |||
| f4723c6f99 | |||
| aad6b143ef | |||
| da004b5c3e | |||
| cd18581bdd | |||
| 508bc764d5 | |||
| 9e8a6315e7 | |||
| 8efd99d738 | |||
| dc98dfda44 | |||
| 8aaeca6b26 | |||
| 7accb2aad6 | |||
| 2f2edfa479 | |||
| b009d9e158 | |||
| 8e4cb5dcd4 | |||
| 7f26dca7a7 | |||
| 46d768dd4e | |||
| c62b6d7893 | |||
| 1668eb935c | |||
| 1ea4919323 | |||
| ba0ab672b9 | |||
| 57c998ba09 | |||
| cc22f80209 | |||
| 6b27d234d9 | |||
| 19ba4bb7d2 | |||
| 8b3901c1e1 | |||
| fd7cc56bb7 | |||
| aabe58d05f | |||
| b35101d339 | |||
| 8420c74a55 | |||
| 872b63f665 | |||
| f599ac45ab | |||
| f36d56c523 | |||
| f0496e8ca5 | |||
| de300ad919 | |||
| 06bf608913 | |||
| a47bde0956 | |||
| d02f16af3a | |||
| dbdf3b61cb | |||
| aa29d1c699 | |||
| bfc9aaaa6d | |||
| 2a9ca39c24 | |||
| 691550fb33 |
+12
-1
@@ -11,10 +11,18 @@ PGID=1000
|
|||||||
|
|
||||||
PORT=3000
|
PORT=3000
|
||||||
CORS_ORIGINS=http://localhost:4174
|
CORS_ORIGINS=http://localhost:4174
|
||||||
LOG_LEVEL=info
|
LOG_LEVEL=warn
|
||||||
# Levels: debug, info, warn, error, silent
|
# Levels: debug, info, warn, error, silent
|
||||||
# Controls: backend Fastify logging, frontend nginx access logs (Docker),
|
# Controls: backend Fastify logging, frontend nginx access logs (Docker),
|
||||||
# and frontend browser console (via build-time injection)
|
# and frontend browser console (via build-time injection)
|
||||||
|
#
|
||||||
|
# Behavior per level:
|
||||||
|
# debug — all app logs + all HTTP request logs (including polling endpoints)
|
||||||
|
# info — all app logs + HTTP request logs, EXCEPT high-frequency polling
|
||||||
|
# (GET /doses/taken, GET /share/:token/doses, GET /health are hidden)
|
||||||
|
# warn — only warnings and errors
|
||||||
|
# error — only errors
|
||||||
|
# silent — no logs
|
||||||
|
|
||||||
# Rate limit: max requests per minute per IP (default: 100)
|
# Rate limit: max requests per minute per IP (default: 100)
|
||||||
# Increase for development/testing environments
|
# Increase for development/testing environments
|
||||||
@@ -32,6 +40,9 @@ AUTH_ENABLED=false
|
|||||||
# Allow new user registrations (auto-enabled when no users exist)
|
# Allow new user registrations (auto-enabled when no users exist)
|
||||||
# REGISTRATION_ENABLED=false
|
# REGISTRATION_ENABLED=false
|
||||||
|
|
||||||
|
# Disable username/password form login (useful for OIDC-only setups)
|
||||||
|
# FORM_LOGIN_ENABLED=true
|
||||||
|
|
||||||
# JWT Secrets - REQUIRED when AUTH_ENABLED=true
|
# JWT Secrets - REQUIRED when AUTH_ENABLED=true
|
||||||
# Generate with: openssl rand -hex 32
|
# Generate with: openssl rand -hex 32
|
||||||
# JWT_SECRET=
|
# JWT_SECRET=
|
||||||
|
|||||||
@@ -0,0 +1,42 @@
|
|||||||
|
---
|
||||||
|
description: 'Provide principal-level software engineering guidance with focus on engineering excellence, technical leadership, and pragmatic implementation.'
|
||||||
|
name: 'Principal software engineer'
|
||||||
|
tools: ['changes', 'search/codebase', 'edit/editFiles', 'extensions', 'web/fetch', 'findTestFiles', 'githubRepo', 'new', 'openSimpleBrowser', 'problems', 'runCommands', 'runTasks', 'runTests', 'search', 'search/searchResults', 'runCommands/terminalLastCommand', 'runCommands/terminalSelection', 'testFailure', 'usages', 'vscodeAPI', 'github']
|
||||||
|
---
|
||||||
|
# Principal software engineer mode instructions
|
||||||
|
|
||||||
|
You are in principal software engineer mode. Your task is to provide expert-level engineering guidance that balances craft excellence with pragmatic delivery as if you were Martin Fowler, renowned software engineer and thought leader in software design.
|
||||||
|
|
||||||
|
## Core Engineering Principles
|
||||||
|
|
||||||
|
You will provide guidance on:
|
||||||
|
|
||||||
|
- **Engineering Fundamentals**: Gang of Four design patterns, SOLID principles, DRY, YAGNI, and KISS - applied pragmatically based on context
|
||||||
|
- **Clean Code Practices**: Readable, maintainable code that tells a story and minimizes cognitive load
|
||||||
|
- **Test Automation**: Comprehensive testing strategy including unit, integration, and end-to-end tests with clear test pyramid implementation
|
||||||
|
- **Quality Attributes**: Balancing testability, maintainability, scalability, performance, security, and understandability
|
||||||
|
- **Technical Leadership**: Clear feedback, improvement recommendations, and mentoring through code reviews
|
||||||
|
|
||||||
|
## Implementation Focus
|
||||||
|
|
||||||
|
- **Requirements Analysis**: Carefully review requirements, document assumptions explicitly, identify edge cases and assess risks
|
||||||
|
- **Implementation Excellence**: Implement the best design that meets architectural requirements without over-engineering
|
||||||
|
- **Pragmatic Craft**: Balance engineering excellence with delivery needs - good over perfect, but never compromising on fundamentals
|
||||||
|
- **Forward Thinking**: Anticipate future needs, identify improvement opportunities, and proactively address technical debt
|
||||||
|
|
||||||
|
## Technical Debt Management
|
||||||
|
|
||||||
|
When technical debt is incurred or identified:
|
||||||
|
|
||||||
|
- **MUST** offer to create GitHub Issues using the `create_issue` tool to track remediation
|
||||||
|
- Clearly document consequences and remediation plans
|
||||||
|
- Regularly recommend GitHub Issues for requirements gaps, quality issues, or design improvements
|
||||||
|
- Assess long-term impact of untended technical debt
|
||||||
|
|
||||||
|
## Deliverables
|
||||||
|
|
||||||
|
- Clear, actionable feedback with specific improvement recommendations
|
||||||
|
- Risk assessments with mitigation strategies
|
||||||
|
- Edge case identification and testing strategies
|
||||||
|
- Explicit documentation of assumptions and decisions
|
||||||
|
- Technical debt remediation plans with GitHub Issue creation
|
||||||
@@ -12,10 +12,14 @@ You are the release manager for **MedAssist-ng**. Your job is to guide code from
|
|||||||
|
|
||||||
## Critical Safety Rules
|
## Critical Safety Rules
|
||||||
|
|
||||||
|
- **Do EXACTLY what the user asks — nothing more.** If the user says "create a PR and merge to main", do only that. Do NOT also start a release. If the user says "do a release", do only the release. Never chain additional steps the user did not request.
|
||||||
- **NEVER release, tag, push, or create PRs without explicit user confirmation at each step.** Always present your plan and wait for approval.
|
- **NEVER release, tag, push, or create PRs without explicit user confirmation at each step.** Always present your plan and wait for approval.
|
||||||
|
- **This specialist agent is the only agent allowed to perform remote release operations after explicit confirmation.**
|
||||||
- **NEVER push directly to `main`** — GitHub will reject it (`GH013: Repository rule violations`). All changes go through Pull Requests.
|
- **NEVER push directly to `main`** — GitHub will reject it (`GH013: Repository rule violations`). All changes go through Pull Requests.
|
||||||
- **NEVER skip CI checks.** Wait for all status checks to pass before merging.
|
- **NEVER skip CI checks.** Wait for all status checks to pass before merging.
|
||||||
- **Testing ownership belongs to `@testing-manager`**. Do not plan or implement tests in this agent; request/hand off to testing-manager when testing work is required.
|
- **Testing ownership belongs to `@testing-manager`**. Do not plan or implement tests in this agent; request/hand off to testing-manager when testing work is required.
|
||||||
|
- **Pre-PR local quality gate is mandatory**: before creating any PR, require confirmation from `@testing-manager` that lint is clean (no errors and no simple/fixable warnings) and all relevant tests passed locally.
|
||||||
|
- **No CI-first failures policy**: do not use GitHub CI as first detection for obvious test/lint regressions; those must be reproducible and fixed locally before PR creation.
|
||||||
- **Track all work in the GitHub Project board.** Every PR should reference an issue. Move issues through the board as work progresses.
|
- **Track all work in the GitHub Project board.** Every PR should reference an issue. Move issues through the board as work progresses.
|
||||||
- **ALWAYS verify Project board status after merge.** The `project-auto-done.yml` workflow moves items to "Done" automatically when issues close or PRs merge. Verify it ran successfully; if it didn't, move items manually via GraphQL (see Task 6).
|
- **ALWAYS verify Project board status after merge.** The `project-auto-done.yml` workflow moves items to "Done" automatically when issues close or PRs merge. Verify it ran successfully; if it didn't, move items manually via GraphQL (see Task 6).
|
||||||
|
|
||||||
@@ -48,12 +52,11 @@ This repository intentionally uses only two operational agents for CI/CD handoff
|
|||||||
|
|
||||||
- Never use `gh` commands that can open an interactive pager and block execution (requiring `q`).
|
- Never use `gh` commands that can open an interactive pager and block execution (requiring `q`).
|
||||||
- Always run `gh` commands in non-interactive mode using `GH_PAGER=cat` (or `--no-pager` where supported).
|
- Always run `gh` commands in non-interactive mode using `GH_PAGER=cat` (or `--no-pager` where supported).
|
||||||
- Do not use these commands in agent flows:
|
- Avoid hardcoded PR/repo examples in instructions; always use parameterized placeholders.
|
||||||
- `gh pr view 155 --json statusCheckRollup --jq '.statusCheckRollup[] | {name:.name,conclusion:.conclusion,detailsUrl:.detailsUrl,workflowName:.workflowName}'`
|
- Use safe command patterns:
|
||||||
- `SHA=$(gh pr view 155 --json headRefOid --jq .headRefOid) && gh api repos/DanielVolz/medassist-ng/commits/$SHA/check-runs --jq '.check_runs[] | {name,conclusion,details_url,html_url,app:.app.name}'`
|
|
||||||
- Use safe variants instead:
|
|
||||||
- `GH_PAGER=cat gh pr view <PR_NUMBER> --json statusCheckRollup --jq '<jq-filter>'`
|
- `GH_PAGER=cat gh pr view <PR_NUMBER> --json statusCheckRollup --jq '<jq-filter>'`
|
||||||
- `GH_PAGER=cat gh api repos/<owner>/<repo>/commits/<sha>/check-runs --jq '<jq-filter>'`
|
- `SHA=$(GH_PAGER=cat gh pr view <PR_NUMBER> --json headRefOid --jq .headRefOid)`
|
||||||
|
- `GH_PAGER=cat gh api repos/<owner>/<repo>/commits/$SHA/check-runs --jq '<jq-filter>'`
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
@@ -119,7 +122,9 @@ When code changes (features or bug fixes) are complete:
|
|||||||
### Step 1: Verify Readiness
|
### Step 1: Verify Readiness
|
||||||
|
|
||||||
1. Check for uncommitted changes: `git status`
|
1. Check for uncommitted changes: `git status`
|
||||||
2. Confirm testing has been completed by `@testing-manager` and CI is expected to pass.
|
2. Confirm testing has been completed by `@testing-manager`.
|
||||||
|
3. Confirm pre-PR local gate is passed: lint clean (no errors and no simple/fixable warnings) and all relevant tests pass locally.
|
||||||
|
4. Only after local gate is confirmed, proceed to push/create PR and then monitor CI.
|
||||||
|
|
||||||
### Step 2: Create Feature Branch
|
### Step 2: Create Feature Branch
|
||||||
|
|
||||||
@@ -140,11 +145,12 @@ When code changes (features or bug fixes) are complete:
|
|||||||
|
|
||||||
### Step 3: Push and Create PR
|
### Step 3: Push and Create PR
|
||||||
|
|
||||||
1. Push the branch:
|
1. Re-check local gate status before push/PR creation (lint + relevant local tests green).
|
||||||
|
2. Push the branch:
|
||||||
```bash
|
```bash
|
||||||
git push -u origin feat/short-description
|
git push -u origin feat/short-description
|
||||||
```
|
```
|
||||||
2. Create a Pull Request via GitHub CLI with **all metadata fields populated**:
|
3. Create a Pull Request via GitHub CLI with **all metadata fields populated**:
|
||||||
```bash
|
```bash
|
||||||
gh pr create \
|
gh pr create \
|
||||||
--title "fix: short description" \
|
--title "fix: short description" \
|
||||||
@@ -157,8 +163,9 @@ When code changes (features or bug fixes) are complete:
|
|||||||
```
|
```
|
||||||
- Use `--label enhancement` for `feat/` branches, `--label bug` for `fix/` branches, `--label documentation` for `docs/` branches.
|
- Use `--label enhancement` for `feat/` branches, `--label bug` for `fix/` branches, `--label documentation` for `docs/` branches.
|
||||||
- Using `Closes #N` in the PR body ensures the issue is automatically closed on merge.
|
- Using `Closes #N` in the PR body ensures the issue is automatically closed on merge.
|
||||||
|
- Always add an explicit issue comment with the PR link and short fix summary (do not rely on auto-close event only).
|
||||||
- The `--project` flag links the PR to the Project board.
|
- The `--project` flag links the PR to the Project board.
|
||||||
3. **Present the PR URL to the user and wait for confirmation.**
|
4. **Present the PR URL to the user and wait for confirmation.**
|
||||||
|
|
||||||
### Step 4: Wait for CI and Merge
|
### Step 4: Wait for CI and Merge
|
||||||
|
|
||||||
@@ -445,6 +452,7 @@ All work is tracked in the [GitHub Project board](https://github.com/users/Danie
|
|||||||
Issues with `enhancement`, `bug`, or `triage` labels are **automatically added** to the board.
|
Issues with `enhancement`, `bug`, or `triage` labels are **automatically added** to the board.
|
||||||
|
|
||||||
2. **When creating a PR**: Always reference the issue with `Closes #N` in the PR body so the issue is automatically **closed** on merge. Note: this does NOT move the Project board status — that must be done manually (see step 3).
|
2. **When creating a PR**: Always reference the issue with `Closes #N` in the PR body so the issue is automatically **closed** on merge. Note: this does NOT move the Project board status — that must be done manually (see step 3).
|
||||||
|
Also add a direct issue comment with the PR link and a one-line summary for clear issue-thread traceability.
|
||||||
|
|
||||||
3. **After merge — verify automation**: The `project-auto-done.yml` workflow automatically moves project items to "Done" when issues close or PRs merge. After merge, verify it ran:
|
3. **After merge — verify automation**: The `project-auto-done.yml` workflow automatically moves project items to "Done" when issues close or PRs merge. After merge, verify it ran:
|
||||||
```bash
|
```bash
|
||||||
|
|||||||
@@ -14,10 +14,17 @@ You are the testing manager for **MedAssist-ng**. Your job is to ensure every fe
|
|||||||
|
|
||||||
- **Tests are mandatory**: Every new feature and every bug fix MUST have corresponding tests.
|
- **Tests are mandatory**: Every new feature and every bug fix MUST have corresponding tests.
|
||||||
- **Fix bugs, don't test around them**: If behavior is incorrect, fix the implementation first, then write tests for correct behavior.
|
- **Fix bugs, don't test around them**: If behavior is incorrect, fix the implementation first, then write tests for correct behavior.
|
||||||
|
- **Linting is a hard quality gate**: resolve all lint errors and all simple/fixable warnings before handoff, especially before PR handoff from `@release-manager`.
|
||||||
|
- **Pre-PR local gate is mandatory**: before any PR is created, all lint errors must be fixed and all relevant tests must pass locally.
|
||||||
|
- **No CI-first failures**: tests must fail locally when broken and be fixed locally before PR handoff; do not rely on GitHub CI to discover obvious regressions.
|
||||||
- **Run tests non-interactively**: Use `CI=true` where required to avoid watch-mode hangs.
|
- **Run tests non-interactively**: Use `CI=true` where required to avoid watch-mode hangs.
|
||||||
|
- **Playwright must disable auto-open reports**: Always prefix Playwright runs with `PLAYWRIGHT_HTML_OPEN=never`.
|
||||||
|
- **Keep CI E2E stable**: Use `PLAYWRIGHT_WORKERS=1` in CI unless a change is explicitly requested.
|
||||||
- **Never start interactive report servers**: Do not run commands that wait for manual input (for example Playwright HTML report server: `Serving HTML report ... Press Ctrl+C to quit`). Always use finite, non-interactive commands and reporters.
|
- **Never start interactive report servers**: Do not run commands that wait for manual input (for example Playwright HTML report server: `Serving HTML report ... Press Ctrl+C to quit`). Always use finite, non-interactive commands and reporters.
|
||||||
- **No remote git operations**: Do not push, merge, create PRs, tags, or releases. Hand over to `@release-manager` when ready.
|
- **No remote git operations**: Do not push, merge, create PRs, tags, or releases. Hand over to `@release-manager` when ready.
|
||||||
- **Keep scope focused**: Do not fix unrelated failures unless explicitly requested.
|
- **Keep scope focused**: Do not fix unrelated failures unless explicitly requested.
|
||||||
|
- **Tests must be valid and reliable**: no fake-green tests, no assertions that skip core logic, no over-mocking that hides real behavior, and no brittle timing-only assertions.
|
||||||
|
- **Regression prevention is mandatory**: every fixed bug must get a deterministic regression test that fails before the fix and passes after it.
|
||||||
|
|
||||||
## CI/CD Ownership Boundary
|
## CI/CD Ownership Boundary
|
||||||
|
|
||||||
@@ -27,9 +34,9 @@ You are the testing manager for **MedAssist-ng**. Your job is to ensure every fe
|
|||||||
|
|
||||||
## Test Stack & Locations
|
## Test Stack & Locations
|
||||||
|
|
||||||
- **Backend**: Vitest 2.1 + v8 coverage
|
- **Backend unit/integration**: Vitest 4 + v8 coverage (`backend/src/test/*.test.ts`)
|
||||||
- **Frontend unit/integration**: Vitest
|
- **Frontend unit/integration**: Vitest 4 + Testing Library (`frontend/src/test/**`)
|
||||||
- **E2E**: Playwright
|
- **Frontend E2E**: Playwright (`frontend/e2e/**`) using stable config for CI-like runs
|
||||||
|
|
||||||
Primary locations:
|
Primary locations:
|
||||||
|
|
||||||
@@ -43,22 +50,41 @@ Primary locations:
|
|||||||
2. Add/update tests near the affected feature.
|
2. Add/update tests near the affected feature.
|
||||||
3. Run the smallest relevant subset first.
|
3. Run the smallest relevant subset first.
|
||||||
4. Expand to broader suites if subset passes.
|
4. Expand to broader suites if subset passes.
|
||||||
5. Report what was run, what passed, and any remaining known failures.
|
5. Run lint + required local test/build gates before PR handoff.
|
||||||
|
6. Report what was run, what passed, and any remaining known failures.
|
||||||
|
|
||||||
|
## Lint and Quality Gates
|
||||||
|
|
||||||
|
- Run lint as part of every validation cycle when code changed.
|
||||||
|
- Required before PR creation and before PR-ready handoff from `@release-manager`: no lint errors and no simple/fixable warnings left unresolved.
|
||||||
|
- If lint fails, fix root causes first, then re-run affected tests.
|
||||||
|
- Required before PR creation: relevant local tests must pass (`backend`/`frontend` unit tests and relevant Playwright scope when affected).
|
||||||
|
- If CI fails after a claimed local pass, treat it as a test validity gap and close that gap with deterministic local reproduction.
|
||||||
|
|
||||||
|
Recommended commands:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
npm run lint
|
||||||
|
cd backend && npm run check
|
||||||
|
cd frontend && npm run check
|
||||||
|
```
|
||||||
|
|
||||||
## Commands
|
## Commands
|
||||||
|
|
||||||
### Backend
|
### Backend
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd backend && CI=true npm test
|
cd backend && CI=true npm run test:run
|
||||||
cd backend && CI=true npm run test:coverage
|
cd backend && CI=true npm run test:coverage
|
||||||
cd backend && CI=true npm test -- -t "test name"
|
cd backend && CI=true npm run test:run -- -t "test name"
|
||||||
```
|
```
|
||||||
|
|
||||||
### Frontend
|
### Frontend
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd frontend && CI=true npm test
|
cd frontend && CI=true npm run test:run
|
||||||
|
cd frontend && CI=true npm run test:coverage
|
||||||
|
cd frontend && CI=true npm run test:run -- -t "test name"
|
||||||
cd frontend && npm run lint
|
cd frontend && npm run lint
|
||||||
cd frontend && npm run build
|
cd frontend && npm run build
|
||||||
```
|
```
|
||||||
@@ -66,8 +92,10 @@ cd frontend && npm run build
|
|||||||
### Playwright E2E
|
### Playwright E2E
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
cd frontend && npm run test:e2e
|
cd frontend && PLAYWRIGHT_HTML_OPEN=never npm run test:e2e
|
||||||
cd frontend && npm run test:e2e -- --project=chromium
|
cd frontend && PLAYWRIGHT_HTML_OPEN=never PLAYWRIGHT_WORKERS=1 npm run test:e2e -- --workers=1
|
||||||
|
cd frontend && PLAYWRIGHT_HTML_OPEN=never PLAYWRIGHT_WORKERS=4 npm run test:e2e:local
|
||||||
|
cd frontend && PLAYWRIGHT_HTML_OPEN=never npm run test:e2e -- --project=chromium
|
||||||
# Never use interactive UI/headed/report-server commands in agent runs.
|
# Never use interactive UI/headed/report-server commands in agent runs.
|
||||||
# Do not use: npm run test:e2e:ui, npm run test:e2e:headed, npx playwright show-report
|
# Do not use: npm run test:e2e:ui, npm run test:e2e:headed, npx playwright show-report
|
||||||
```
|
```
|
||||||
@@ -78,6 +106,7 @@ cd frontend && npm run test:e2e -- --project=chromium
|
|||||||
- Validate both status codes and response payloads.
|
- Validate both status codes and response payloads.
|
||||||
- Add regression tests for every fixed bug.
|
- Add regression tests for every fixed bug.
|
||||||
- Keep tests deterministic and isolated.
|
- Keep tests deterministic and isolated.
|
||||||
|
- Validate observable behavior, not implementation details.
|
||||||
|
|
||||||
## E2E Test Patterns
|
## E2E Test Patterns
|
||||||
|
|
||||||
@@ -85,6 +114,15 @@ cd frontend && npm run test:e2e -- --project=chromium
|
|||||||
- Avoid flaky timing assumptions; prefer waiting for concrete UI states.
|
- Avoid flaky timing assumptions; prefer waiting for concrete UI states.
|
||||||
- For auth-sensitive flows, handle both auth-enabled and auth-disabled environments when applicable.
|
- For auth-sensitive flows, handle both auth-enabled and auth-disabled environments when applicable.
|
||||||
- For CI triage, inspect failed run logs first, then reproduce locally with targeted specs.
|
- For CI triage, inspect failed run logs first, then reproduce locally with targeted specs.
|
||||||
|
- Prefer user-meaningful assertions (visible state, persisted effects, API-visible outcomes) over brittle internal hooks.
|
||||||
|
|
||||||
|
## Test Validity Checklist
|
||||||
|
|
||||||
|
- The test fails when the real target logic is intentionally broken.
|
||||||
|
- The assertion verifies functional behavior, not just mocked calls.
|
||||||
|
- Mocks/stubs are minimal and do not replace the unit under test.
|
||||||
|
- The test is deterministic across repeated local and CI runs.
|
||||||
|
- The test protects against the specific regression that was fixed.
|
||||||
|
|
||||||
## CI Failure Triage
|
## CI Failure Triage
|
||||||
|
|
||||||
@@ -115,6 +153,9 @@ When test checks fail:
|
|||||||
Testing work is complete when:
|
Testing work is complete when:
|
||||||
|
|
||||||
- Required tests exist and validate intended behavior.
|
- Required tests exist and validate intended behavior.
|
||||||
|
- Tests are proven valid (not fake-green) and reliable.
|
||||||
|
- Lint is clean: no errors and no simple/fixable warnings left.
|
||||||
|
- Pre-PR local gate passed: lint and all relevant tests pass locally before handoff for PR creation.
|
||||||
- Relevant local test commands pass.
|
- Relevant local test commands pass.
|
||||||
- CI test failures are resolved or clearly documented with rationale.
|
- CI test failures are resolved or clearly documented with rationale.
|
||||||
- No temporary debugging files remain in the workspace.
|
- No temporary debugging files remain in the workspace.
|
||||||
|
|||||||
@@ -1,77 +1,19 @@
|
|||||||
# MedAssist-ng - AI Coding Instructions
|
# MedAssist-ng - Copilot Entry Point
|
||||||
|
|
||||||
## Purpose
|
## VERY IMPORTANT
|
||||||
|
|
||||||
Use `AGENTS.md` as the canonical governance source. Read the referenced skill files before starting any task.
|
- Always keep agent work memory updated in `doku/memory_notes.md` so progress and decisions remain recoverable across context loss.
|
||||||
|
- Always keep a user-facing work report updated in `doku/report.md` so completed work is easy to review.
|
||||||
|
- This memory/report rule replaces the previous `doku/APP_BEHAVIOR.md` persistence requirement.
|
||||||
|
|
||||||
## Project Orientation (Read First)
|
Use `AGENTS.md` as the single source of truth for all governance, workflow, and skill rules.
|
||||||
|
|
||||||
- **Product**: MedAssist-ng is a medication planner with stock tracking, reminders (email/push), refill history, and schedule sharing.
|
## Required Startup Steps
|
||||||
- **Tech stack**: React + TypeScript + Vite (`frontend/`), Fastify + TypeScript + Drizzle + SQLite (`backend/`).
|
|
||||||
- **Request path**: Frontend uses `/api/*` only; backend route handlers live in `backend/src/routes/`.
|
|
||||||
- **Primary backend modules**:
|
|
||||||
- Auth/SSO: `backend/src/routes/auth.ts`, `backend/src/routes/oidc.ts`, `backend/src/plugins/auth.ts`
|
|
||||||
- Medications/data: `backend/src/routes/medications.ts`, `backend/src/db/schema.ts`
|
|
||||||
- Reminders: `backend/src/services/reminder-scheduler.ts`, `backend/src/routes/planner.ts`, `backend/src/routes/settings.ts`
|
|
||||||
- **Primary frontend modules**:
|
|
||||||
- Pages: `frontend/src/pages/`
|
|
||||||
- Shared app state: `frontend/src/context/AppContext.tsx`
|
|
||||||
- Domain hooks: `frontend/src/hooks/`
|
|
||||||
- Translations: `frontend/src/i18n/en.json`, `frontend/src/i18n/de.json`
|
|
||||||
|
|
||||||
Use this orientation for quick navigation before applying the rules below.
|
1. Read `AGENTS.md` first.
|
||||||
|
2. Identify triggered skills from `AGENTS.md` and read each referenced `SKILL.md` before making changes.
|
||||||
|
3. Follow delegation boundaries exactly (`@testing-manager` for testing, `@release-manager` for release orchestration).
|
||||||
|
|
||||||
## Always-On Rules
|
## Scope
|
||||||
|
|
||||||
- English only for project artifacts.
|
This file intentionally stays minimal to prevent duplicated or conflicting instructions.
|
||||||
- **NEVER run remote git commands** — no `git push`, no `gh pr create/merge`, no `gh release`, no `git tag`. Prepare locally, then hand off to `@release-manager`.
|
|
||||||
- Testing work belongs to `@testing-manager`.
|
|
||||||
- PR/release/CI orchestration belongs to `@release-manager`.
|
|
||||||
- Keep changes local, focused, and consistent with existing UI/API patterns.
|
|
||||||
- **Hard PR scope + size rule**: one cohesive objective per PR; if scope drifts or diff becomes large (target <= 500 changed lines, hard split at ~800+), split into logical follow-up PRs instead of bundling.
|
|
||||||
- Remove obsolete code when re-implementing — never leave dead code behind.
|
|
||||||
- **Document behavioral discoveries**: When you discover or clarify how a feature works (e.g., what triggers notifications, how thresholds interact, which code paths exist), **always** add or update the relevant section in `doku/APP_BEHAVIOR.md`. This is mandatory — do not rely on conversation context alone.
|
|
||||||
|
|
||||||
## MedAssist Essentials
|
|
||||||
|
|
||||||
- Frontend calls backend through `/api/*`.
|
|
||||||
- DB changes must stay backward-compatible (schema default + alter migration + null-safe reads).
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Skills (MANDATORY — read before every task)
|
|
||||||
|
|
||||||
Before starting any task, identify which skills apply and **read their full SKILL.md file** for detailed rules.
|
|
||||||
|
|
||||||
| Skill | Trigger | File |
|
|
||||||
|---|---|---|
|
|
||||||
| **Architecture Guard** | API endpoints, frontend API calls, routing, code placement | `.github/skills/medassist-architecture-guard/SKILL.md` |
|
|
||||||
| **DB Compatibility** | Persisted data, schema changes, migrations | `.github/skills/medassist-db-compat-check/SKILL.md` |
|
|
||||||
| **i18n Enforcer** ⚠️ | Any user-facing text in frontend or backend | `.github/skills/medassist-i18n-enforcer/SKILL.md` |
|
|
||||||
| **UI Consistency** | UI flows, modals, buttons, forms, settings | `.github/skills/medassist-ui-consistency/SKILL.md` |
|
|
||||||
| **Frontend Polish** | Visual quality improvements | `.github/skills/medassist-frontend-polish/SKILL.md` |
|
|
||||||
| **Security Sanity** | Backend routes, auth, file handling, external input | `.github/skills/medassist-security-sanity/SKILL.md` |
|
|
||||||
| **Observability Guard** | Services, schedulers, startup, failure handling | `.github/skills/medassist-observability-guard/SKILL.md` |
|
|
||||||
| **Config Change Guard** | `.env`, Docker, Vite proxy, runtime defaults | `.github/skills/medassist-config-change-guard/SKILL.md` |
|
|
||||||
| **Doc Sync Guard** | Behavior changes, setup, env vars, workflows | `.github/skills/medassist-doc-sync-guard/SKILL.md` |
|
|
||||||
| **Testing Handoff** | Writing/running tests, CI test failures | `.github/skills/medassist-testing-handoff/SKILL.md` |
|
|
||||||
| **Release Handoff** | Branch push, PR, merge, tagging, release | `.github/skills/medassist-release-handoff/SKILL.md` |
|
|
||||||
| **Skill Quality Review** | Creating/modifying skills | `.github/skills/medassist-skill-quality-review/SKILL.md` |
|
|
||||||
|
|
||||||
### Non-negotiable parity rules (always apply)
|
|
||||||
|
|
||||||
1. **Desktop + Mobile Parity**: Medication edit has two paths — `MedicationsPage.tsx` (desktop) and `MobileEditModal` (mobile). **Always update BOTH**.
|
|
||||||
2. **Notification Dual Code Paths**: Notifications have two code paths — `backend/src/services/reminder-scheduler.ts` (scheduler) and `backend/src/routes/planner.ts` (manual). **Always update BOTH**.
|
|
||||||
|
|
||||||
---
|
|
||||||
|
|
||||||
## Delegation
|
|
||||||
|
|
||||||
- **Testing handoff → `@testing-manager`**: test planning, writing, execution, CI test triage.
|
|
||||||
- **Release handoff → `@release-manager`**: PR/release orchestration, merge flow, workflow monitoring.
|
|
||||||
|
|
||||||
## Key References
|
|
||||||
|
|
||||||
- Canonical governance: `AGENTS.md`
|
|
||||||
- Skill files: `.github/skills/*/SKILL.md`
|
|
||||||
- Specialist agents: `.github/agents/testing-manager.agent.md`, `.github/agents/release-manager.agent.md`
|
|
||||||
|
|||||||
@@ -7,9 +7,11 @@ updates:
|
|||||||
schedule:
|
schedule:
|
||||||
interval: "weekly"
|
interval: "weekly"
|
||||||
day: "monday"
|
day: "monday"
|
||||||
|
time: "06:20"
|
||||||
open-pull-requests-limit: 10
|
open-pull-requests-limit: 10
|
||||||
labels:
|
labels:
|
||||||
- "dependencies"
|
- "dependencies"
|
||||||
|
- "backend"
|
||||||
groups:
|
groups:
|
||||||
minor-and-patch:
|
minor-and-patch:
|
||||||
update-types:
|
update-types:
|
||||||
@@ -22,9 +24,11 @@ updates:
|
|||||||
schedule:
|
schedule:
|
||||||
interval: "weekly"
|
interval: "weekly"
|
||||||
day: "monday"
|
day: "monday"
|
||||||
|
time: "06:10"
|
||||||
open-pull-requests-limit: 10
|
open-pull-requests-limit: 10
|
||||||
labels:
|
labels:
|
||||||
- "dependencies"
|
- "dependencies"
|
||||||
|
- "frontend"
|
||||||
groups:
|
groups:
|
||||||
minor-and-patch:
|
minor-and-patch:
|
||||||
update-types:
|
update-types:
|
||||||
@@ -37,9 +41,16 @@ updates:
|
|||||||
schedule:
|
schedule:
|
||||||
interval: "weekly"
|
interval: "weekly"
|
||||||
day: "monday"
|
day: "monday"
|
||||||
|
time: "06:00"
|
||||||
open-pull-requests-limit: 5
|
open-pull-requests-limit: 5
|
||||||
labels:
|
labels:
|
||||||
- "dependencies"
|
- "dependencies"
|
||||||
|
- "root"
|
||||||
|
groups:
|
||||||
|
minor-and-patch:
|
||||||
|
update-types:
|
||||||
|
- "minor"
|
||||||
|
- "patch"
|
||||||
|
|
||||||
# GitHub Actions
|
# GitHub Actions
|
||||||
- package-ecosystem: "github-actions"
|
- package-ecosystem: "github-actions"
|
||||||
@@ -47,7 +58,13 @@ updates:
|
|||||||
schedule:
|
schedule:
|
||||||
interval: "weekly"
|
interval: "weekly"
|
||||||
day: "monday"
|
day: "monday"
|
||||||
|
time: "06:30"
|
||||||
open-pull-requests-limit: 5
|
open-pull-requests-limit: 5
|
||||||
labels:
|
labels:
|
||||||
- "dependencies"
|
- "dependencies"
|
||||||
- "ci"
|
- "ci"
|
||||||
|
groups:
|
||||||
|
minor-and-patch:
|
||||||
|
update-types:
|
||||||
|
- "minor"
|
||||||
|
- "patch"
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ Use one governance source to avoid duplicated or conflicting policy text.
|
|||||||
|
|
||||||
## Skills
|
## Skills
|
||||||
|
|
||||||
- `medassist-karpathy-core` — enforce assumption clarity, simplicity, surgical diffs, and verifiable execution.
|
- `medassist-karpathy-core` — enforce think-before-coding, simplicity-first changes, surgical diffs, and goal-driven verification.
|
||||||
- `medassist-architecture-guard` — enforce frontend/backend boundary and `/api/*` data-flow conventions.
|
- `medassist-architecture-guard` — enforce frontend/backend boundary and `/api/*` data-flow conventions.
|
||||||
- `medassist-db-compat-check` — enforce backward-compatible SQLite/Drizzle schema changes.
|
- `medassist-db-compat-check` — enforce backward-compatible SQLite/Drizzle schema changes.
|
||||||
- `medassist-i18n-enforcer` — enforce translation-key-only UI copy with EN/DE parity.
|
- `medassist-i18n-enforcer` — enforce translation-key-only UI copy with EN/DE parity.
|
||||||
|
|||||||
@@ -0,0 +1,69 @@
|
|||||||
|
---
|
||||||
|
name: medassist-karpathy-core
|
||||||
|
description: Apply assumption clarity, simplicity-first implementation, surgical diffs, and goal-driven verification for non-trivial coding tasks.
|
||||||
|
---
|
||||||
|
|
||||||
|
# Skill Instructions
|
||||||
|
|
||||||
|
Use this skill as an execution style layer for implementation tasks where overengineering, broad refactors, or unclear assumptions are likely.
|
||||||
|
|
||||||
|
## Use When
|
||||||
|
|
||||||
|
- The request is ambiguous and assumptions must be made explicit.
|
||||||
|
- The change can easily balloon in scope.
|
||||||
|
- A bug fix or feature needs explicit success criteria and verification.
|
||||||
|
- You need to keep diffs minimal and directly tied to the request.
|
||||||
|
|
||||||
|
## Do Not Use When
|
||||||
|
|
||||||
|
- The task is trivial and can be completed safely without extra process overhead.
|
||||||
|
- The task is only about ownership routing (use `medassist-testing-handoff` / `medassist-release-handoff`).
|
||||||
|
- The task is only about domain guardrails already covered by specialized skills (architecture, DB, i18n, UI, security, config, observability).
|
||||||
|
|
||||||
|
## Core Principles
|
||||||
|
|
||||||
|
### 1. Think Before Coding
|
||||||
|
|
||||||
|
- Do not assume silently.
|
||||||
|
- State assumptions explicitly.
|
||||||
|
- If multiple interpretations exist, present them instead of picking one invisibly.
|
||||||
|
- If uncertain or blocked by ambiguity, stop and ask.
|
||||||
|
- If a simpler approach exists, call it out.
|
||||||
|
|
||||||
|
### 2. Simplicity First
|
||||||
|
|
||||||
|
- Implement the minimum code required to solve the asked problem.
|
||||||
|
- Do not add speculative features, abstractions, or configurability.
|
||||||
|
- Avoid defensive handling for impossible scenarios.
|
||||||
|
- If the solution feels overcomplicated, simplify before finalizing.
|
||||||
|
|
||||||
|
### 3. Surgical Changes
|
||||||
|
|
||||||
|
- Touch only lines required for the request.
|
||||||
|
- Do not refactor unrelated areas.
|
||||||
|
- Match existing local style and patterns.
|
||||||
|
- Remove only unused code introduced by your own change.
|
||||||
|
- If unrelated dead code is discovered, mention it but do not remove it unless requested.
|
||||||
|
|
||||||
|
### 4. Goal-Driven Execution
|
||||||
|
|
||||||
|
- Translate requests into verifiable outcomes before implementation.
|
||||||
|
- For multi-step tasks, define short steps with checks.
|
||||||
|
- Verify the requested behavior explicitly before declaring done.
|
||||||
|
|
||||||
|
Example execution frame:
|
||||||
|
|
||||||
|
```text
|
||||||
|
1. [Step] -> verify: [check]
|
||||||
|
2. [Step] -> verify: [check]
|
||||||
|
3. [Step] -> verify: [check]
|
||||||
|
```
|
||||||
|
|
||||||
|
## Response Format
|
||||||
|
|
||||||
|
When this skill is used, report briefly:
|
||||||
|
|
||||||
|
- Assumptions made (or clarifications requested)
|
||||||
|
- Why the chosen approach is the simplest viable one
|
||||||
|
- What was changed (and what was intentionally not changed)
|
||||||
|
- Verification performed and result
|
||||||
|
|||||||
@@ -0,0 +1,37 @@
|
|||||||
|
name: Dependabot Automerge
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request_target:
|
||||||
|
types:
|
||||||
|
- opened
|
||||||
|
- reopened
|
||||||
|
- synchronize
|
||||||
|
- ready_for_review
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
pull-requests: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
enable-automerge:
|
||||||
|
if: github.actor == 'dependabot[bot]'
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Read Dependabot metadata
|
||||||
|
id: metadata
|
||||||
|
uses: dependabot/fetch-metadata@v2
|
||||||
|
with:
|
||||||
|
github-token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: Enable auto-merge for safe updates
|
||||||
|
if: >-
|
||||||
|
(steps.metadata.outputs.package-ecosystem == 'npm' ||
|
||||||
|
steps.metadata.outputs.package-ecosystem == 'github_actions') &&
|
||||||
|
(steps.metadata.outputs.update-type == 'version-update:semver-minor' ||
|
||||||
|
steps.metadata.outputs.update-type == 'version-update:semver-patch')
|
||||||
|
uses: peter-evans/enable-pull-request-automerge@v3
|
||||||
|
with:
|
||||||
|
token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
pull-request-number: ${{ github.event.pull_request.number }}
|
||||||
|
merge-method: squash
|
||||||
@@ -4,6 +4,12 @@ on:
|
|||||||
push:
|
push:
|
||||||
branches: [main]
|
branches: [main]
|
||||||
tags: ['v*']
|
tags: ['v*']
|
||||||
|
paths:
|
||||||
|
- 'backend/**'
|
||||||
|
- 'frontend/**'
|
||||||
|
- 'docker-compose.yml'
|
||||||
|
- 'docker-compose.dev.yml'
|
||||||
|
- '.github/workflows/docker-build.yml'
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
inputs:
|
inputs:
|
||||||
tag:
|
tag:
|
||||||
|
|||||||
@@ -50,11 +50,13 @@ jobs:
|
|||||||
run: npx playwright test --project=chromium
|
run: npx playwright test --project=chromium
|
||||||
env:
|
env:
|
||||||
CI: true
|
CI: true
|
||||||
|
PLAYWRIGHT_WORKERS: 1
|
||||||
|
PLAYWRIGHT_HTML_OPEN: never
|
||||||
JWT_SECRET: e2e-test-secret-that-is-long-enough
|
JWT_SECRET: e2e-test-secret-that-is-long-enough
|
||||||
SESSION_SECRET: e2e-test-session-secret-long-enough
|
SESSION_SECRET: e2e-test-session-secret-long-enough
|
||||||
|
|
||||||
- name: Upload Playwright report
|
- name: Upload Playwright report
|
||||||
uses: actions/upload-artifact@v6
|
uses: actions/upload-artifact@v7
|
||||||
if: always()
|
if: always()
|
||||||
with:
|
with:
|
||||||
name: playwright-report
|
name: playwright-report
|
||||||
@@ -62,7 +64,7 @@ jobs:
|
|||||||
retention-days: 7
|
retention-days: 7
|
||||||
|
|
||||||
- name: Upload test results
|
- name: Upload test results
|
||||||
uses: actions/upload-artifact@v6
|
uses: actions/upload-artifact@v7
|
||||||
if: always()
|
if: always()
|
||||||
with:
|
with:
|
||||||
name: playwright-results
|
name: playwright-results
|
||||||
|
|||||||
@@ -73,7 +73,7 @@ jobs:
|
|||||||
run: npm run test:coverage
|
run: npm run test:coverage
|
||||||
|
|
||||||
- name: Upload coverage report
|
- name: Upload coverage report
|
||||||
uses: actions/upload-artifact@v6
|
uses: actions/upload-artifact@v7
|
||||||
if: always()
|
if: always()
|
||||||
with:
|
with:
|
||||||
name: backend-coverage
|
name: backend-coverage
|
||||||
@@ -118,7 +118,7 @@ jobs:
|
|||||||
run: npm run build
|
run: npm run build
|
||||||
|
|
||||||
- name: Upload coverage report
|
- name: Upload coverage report
|
||||||
uses: actions/upload-artifact@v6
|
uses: actions/upload-artifact@v7
|
||||||
if: always()
|
if: always()
|
||||||
with:
|
with:
|
||||||
name: frontend-coverage
|
name: frontend-coverage
|
||||||
|
|||||||
+5
-1
@@ -82,4 +82,8 @@ Thumbs.db
|
|||||||
.claude/
|
.claude/
|
||||||
AGENTS.md
|
AGENTS.md
|
||||||
docs/TECH_STACK.md
|
docs/TECH_STACK.md
|
||||||
doku
|
doku/
|
||||||
|
doku/memory_notes.md
|
||||||
|
doku/report.md
|
||||||
|
plan/
|
||||||
|
.copilot-tracking
|
||||||
@@ -10,7 +10,7 @@
|
|||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
<img src="https://img.shields.io/badge/React-18-61DAFB?logo=react" alt="React 18" />
|
<img src="https://img.shields.io/badge/React-19-61DAFB?logo=react" alt="React 19" />
|
||||||
<img src="https://img.shields.io/badge/TypeScript-5-3178C6?logo=typescript" alt="TypeScript" />
|
<img src="https://img.shields.io/badge/TypeScript-5-3178C6?logo=typescript" alt="TypeScript" />
|
||||||
<img src="https://img.shields.io/badge/Fastify-5-000000?logo=fastify" alt="Fastify" />
|
<img src="https://img.shields.io/badge/Fastify-5-000000?logo=fastify" alt="Fastify" />
|
||||||
<img src="https://img.shields.io/badge/SQLite-Database-003B57?logo=sqlite" alt="SQLite" />
|
<img src="https://img.shields.io/badge/SQLite-Database-003B57?logo=sqlite" alt="SQLite" />
|
||||||
@@ -18,13 +18,13 @@
|
|||||||
</p>
|
</p>
|
||||||
|
|
||||||
<p align="center">
|
<p align="center">
|
||||||
<img src="https://img.shields.io/badge/Backend_Tests-569%2F569-brightgreen?logo=vitest" alt="Backend Tests 454/454" />
|
<img src="https://img.shields.io/badge/Backend_Tests-577%2F577-brightgreen?logo=vitest" alt="Backend Tests 454/454" />
|
||||||
<img src="https://img.shields.io/badge/Frontend_Tests-771%2F771-brightgreen?logo=vitest" alt="Frontend Tests 611/611" />
|
<img src="https://img.shields.io/badge/Frontend_Tests-777%2F777-brightgreen?logo=vitest" alt="Frontend Tests 611/611" />
|
||||||
</p>
|
</p>
|
||||||
|
|
||||||
### 🤖 AI-Generated Code
|
### 🤖 AI-Generated Code
|
||||||
|
|
||||||
> This app was 100% coded with Claude Opus 4.5. Use at your own risk.
|
> This app was 100% coded with [Claude Opus 4.6](https://www.anthropic.com/claude) and [GPT-5.3 Codex](https://openai.com/index/gpt-5/). Use at your own risk.
|
||||||
|
|
||||||
### ⚠️ Disclaimer
|
### ⚠️ Disclaimer
|
||||||
|
|
||||||
@@ -120,10 +120,10 @@ Share your medication schedule with others via a public link.
|
|||||||
</details>
|
</details>
|
||||||
|
|
||||||
### Smart Inventory
|
### Smart Inventory
|
||||||
- Track exact stock: packs, blisters, bottles, and loose pills
|
- Track exact stock with package profiles (blister, bottle, tube, liquid container)
|
||||||
- Display remaining days of supply
|
- Display remaining days of supply
|
||||||
- Automatic calculation based on intake schedule
|
- Automatic calculation based on intake schedule
|
||||||
- Manual stock correction supports partial blisters and loose pills
|
- Manual stock correction supports profile-specific stock semantics (sealed units + loose stock for blister, amount-based stock for bottle/tube/liquid)
|
||||||
|
|
||||||
### Medication Refill
|
### Medication Refill
|
||||||
- One-click refill with pack or loose pill options
|
- One-click refill with pack or loose pill options
|
||||||
@@ -141,7 +141,7 @@ Share your medication schedule with others via a public link.
|
|||||||
- Intake reminders via push notifications
|
- Intake reminders via push notifications
|
||||||
|
|
||||||
### Trip Planner
|
### Trip Planner
|
||||||
- Calculate how many pills you need for a trip or date range
|
- Calculate medication demand for a trip or date range with package-aware units
|
||||||
- Plan ahead for vacations, business trips, or hospital stays
|
- Plan ahead for vacations, business trips, or hospital stays
|
||||||
- Send demand reports via email or push notification
|
- Send demand reports via email or push notification
|
||||||
|
|
||||||
@@ -194,7 +194,7 @@ All configuration is done via environment variables in `.env`. Copy `.env.exampl
|
|||||||
| `PGID` | `1000` | Group ID for container file permissions |
|
| `PGID` | `1000` | Group ID for container file permissions |
|
||||||
| `PORT` | `3000` | Backend API port |
|
| `PORT` | `3000` | Backend API port |
|
||||||
| `CORS_ORIGINS` | `http://localhost:4174` | Allowed origins for CORS |
|
| `CORS_ORIGINS` | `http://localhost:4174` | Allowed origins for CORS |
|
||||||
| `LOG_LEVEL` | `info` | Log verbosity (`debug`, `info`, `warn`, `error`) |
|
| `LOG_LEVEL` | `info` | Log verbosity (`debug`, `info`, `warn`, `error`, `silent`). At `info` (default), high-frequency polling endpoints are suppressed. Set `debug` to see all requests. |
|
||||||
| `TZ` | `Europe/Berlin` | Timezone for scheduled reminders |
|
| `TZ` | `Europe/Berlin` | Timezone for scheduled reminders |
|
||||||
|
|
||||||
### Authentication
|
### Authentication
|
||||||
@@ -250,7 +250,9 @@ Generate secrets with: `openssl rand -hex 32`
|
|||||||
|
|
||||||
MedAssist uses [Shoutrrr](https://containrrr.dev/shoutrrr/) for push notifications, supporting many services with a single URL format.
|
MedAssist uses [Shoutrrr](https://containrrr.dev/shoutrrr/) for push notifications, supporting many services with a single URL format.
|
||||||
|
|
||||||
**Supported services:** ntfy, Pushover, Gotify, Discord, Telegram, Slack, Matrix, and [many more](https://containrrr.dev/shoutrrr/v0.8/services/overview/).
|
**Implemented URL schemes in MedAssist:** `ntfy://`, `discord://`, `pushover://`, `gotify://`, `telegram://`, plus direct `https://` webhooks.
|
||||||
|
|
||||||
|
This covers common providers like ntfy, Discord, Pushover, Gotify, Telegram, Slack webhooks, and many others via webhook URLs.
|
||||||
|
|
||||||
Configure push notifications in Settings → Push, or set defaults via environment variables:
|
Configure push notifications in Settings → Push, or set defaults via environment variables:
|
||||||
|
|
||||||
@@ -288,6 +290,7 @@ Get your keys at [pushover.net](https://pushover.net/):
|
|||||||
**Gotify** (self-hosted):
|
**Gotify** (self-hosted):
|
||||||
```
|
```
|
||||||
gotify://your-server.com/TOKEN
|
gotify://your-server.com/TOKEN
|
||||||
|
gotify://your-server.com:443/path/to/gotify/TOKEN?priority=1
|
||||||
```
|
```
|
||||||
|
|
||||||
**Discord**:
|
**Discord**:
|
||||||
@@ -298,6 +301,7 @@ discord://TOKEN@WEBHOOK_ID
|
|||||||
**Telegram**:
|
**Telegram**:
|
||||||
```
|
```
|
||||||
telegram://TOKEN@telegram?chats=CHAT_ID
|
telegram://TOKEN@telegram?chats=CHAT_ID
|
||||||
|
telegram://TOKEN@telegram?chats=@your_channel,-1001234567890
|
||||||
```
|
```
|
||||||
|
|
||||||
For all services and options, see the [Shoutrrr documentation](https://containrrr.dev/shoutrrr/v0.8/services/overview/).
|
For all services and options, see the [Shoutrrr documentation](https://containrrr.dev/shoutrrr/v0.8/services/overview/).
|
||||||
@@ -311,6 +315,24 @@ docker compose -f docker-compose.dev.yml up
|
|||||||
- Frontend: `http://localhost:5173` (hot reload)
|
- Frontend: `http://localhost:5173` (hot reload)
|
||||||
- Backend: `http://localhost:3000`
|
- Backend: `http://localhost:3000`
|
||||||
|
|
||||||
|
Playwright E2E recommendations:
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd frontend
|
||||||
|
npm run test:e2e:local # local run with PLAYWRIGHT_WORKERS=4
|
||||||
|
npm run test:e2e:all:local # local all-browser run with PLAYWRIGHT_WORKERS=4
|
||||||
|
```
|
||||||
|
|
||||||
|
- CI stays at `PLAYWRIGHT_WORKERS=1` for stability.
|
||||||
|
- Data-heavy specs remain sequential via the `chromium-data` project config.
|
||||||
|
|
||||||
|
# Dependency Updates
|
||||||
|
|
||||||
|
- Dependabot checks dependencies weekly for `frontend`, `backend`, repository root tooling, and GitHub Actions.
|
||||||
|
- Minor and patch updates are grouped to reduce PR noise.
|
||||||
|
- Dependabot minor/patch PRs are configured for auto-merge after required CI checks pass.
|
||||||
|
- Major updates still require manual review before merge.
|
||||||
|
|
||||||
# Acknowledgements
|
# Acknowledgements
|
||||||
|
|
||||||
This project was inspired by [MedAssist](https://github.com/njic/medassist) by njic.
|
This project was inspired by [MedAssist](https://github.com/njic/medassist) by njic.
|
||||||
|
|||||||
@@ -0,0 +1,5 @@
|
|||||||
|
ALTER TABLE `medications` ADD `medication_form` text(20) DEFAULT 'tablet' NOT NULL;--> statement-breakpoint
|
||||||
|
ALTER TABLE `medications` ADD `pill_form` text(20);--> statement-breakpoint
|
||||||
|
ALTER TABLE `medications` ADD `lifecycle_category` text(30) DEFAULT 'refill_when_empty' NOT NULL;--> statement-breakpoint
|
||||||
|
ALTER TABLE `medications` ADD `medication_end_date` text;--> statement-breakpoint
|
||||||
|
ALTER TABLE `medications` ADD `auto_mark_obsolete_after_end_date` integer DEFAULT true NOT NULL;
|
||||||
File diff suppressed because it is too large
Load Diff
@@ -78,6 +78,13 @@
|
|||||||
"when": 1771694832866,
|
"when": 1771694832866,
|
||||||
"tag": "0010_mean_spot",
|
"tag": "0010_mean_spot",
|
||||||
"breakpoints": true
|
"breakpoints": true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"idx": 11,
|
||||||
|
"version": "6",
|
||||||
|
"when": 1772219947541,
|
||||||
|
"tag": "0011_stiff_randall_flagg",
|
||||||
|
"breakpoints": true
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
Generated
+245
-124
@@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "medassist-ng-backend",
|
"name": "medassist-ng-backend",
|
||||||
"version": "1.15.1",
|
"version": "1.18.2",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "medassist-ng-backend",
|
"name": "medassist-ng-backend",
|
||||||
"version": "1.15.1",
|
"version": "1.18.2",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@fastify/cookie": "^11.0.2",
|
"@fastify/cookie": "^11.0.2",
|
||||||
"@fastify/cors": "^11.2.0",
|
"@fastify/cors": "^11.2.0",
|
||||||
@@ -20,7 +20,7 @@
|
|||||||
"argon2": "^0.44.0",
|
"argon2": "^0.44.0",
|
||||||
"dotenv": "^17.3.1",
|
"dotenv": "^17.3.1",
|
||||||
"drizzle-orm": "^0.45.1",
|
"drizzle-orm": "^0.45.1",
|
||||||
"fastify": "^5.7.4",
|
"fastify": "^5.8.1",
|
||||||
"nodemailer": "^8.0.1",
|
"nodemailer": "^8.0.1",
|
||||||
"openid-client": "^6.8.2",
|
"openid-client": "^6.8.2",
|
||||||
"sharp": "^0.34.5",
|
"sharp": "^0.34.5",
|
||||||
@@ -28,11 +28,12 @@
|
|||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@biomejs/biome": "^2.4.4",
|
"@biomejs/biome": "^2.4.4",
|
||||||
"@types/node": "^25.3.0",
|
"@types/node": "^25.3.3",
|
||||||
"@types/nodemailer": "^7.0.11",
|
"@types/nodemailer": "^7.0.11",
|
||||||
"@types/supertest": "^6.0.2",
|
"@types/supertest": "^7.2.0",
|
||||||
"@vitest/coverage-v8": "^4.0.18",
|
"@vitest/coverage-v8": "^4.0.18",
|
||||||
"drizzle-kit": "^0.31.9",
|
"drizzle-kit": "^0.31.9",
|
||||||
|
"pino-pretty": "^13.1.3",
|
||||||
"supertest": "^7.2.2",
|
"supertest": "^7.2.2",
|
||||||
"tsx": "^4.19.0",
|
"tsx": "^4.19.0",
|
||||||
"typescript": "^5.5.4",
|
"typescript": "^5.5.4",
|
||||||
@@ -2228,9 +2229,9 @@
|
|||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-android-arm-eabi": {
|
"node_modules/@rollup/rollup-android-arm-eabi": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.59.0.tgz",
|
||||||
"integrity": "sha512-A6ehUVSiSaaliTxai040ZpZ2zTevHYbvu/lDoeAteHI8QnaosIzm4qwtezfRg1jOYaUmnzLX1AOD6Z+UJjtifg==",
|
"integrity": "sha512-upnNBkA6ZH2VKGcBj9Fyl9IGNPULcjXRlg0LLeaioQWueH30p6IXtJEbKAgvyv+mJaMxSm1l6xwDXYjpEMiLMg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm"
|
"arm"
|
||||||
],
|
],
|
||||||
@@ -2242,9 +2243,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-android-arm64": {
|
"node_modules/@rollup/rollup-android-arm64": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.59.0.tgz",
|
||||||
"integrity": "sha512-dQaAddCY9YgkFHZcFNS/606Exo8vcLHwArFZ7vxXq4rigo2bb494/xKMMwRRQW6ug7Js6yXmBZhSBRuBvCCQ3w==",
|
"integrity": "sha512-hZ+Zxj3SySm4A/DylsDKZAeVg0mvi++0PYVceVyX7hemkw7OreKdCvW2oQ3T1FMZvCaQXqOTHb8qmBShoqk69Q==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -2256,9 +2257,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-darwin-arm64": {
|
"node_modules/@rollup/rollup-darwin-arm64": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.59.0.tgz",
|
||||||
"integrity": "sha512-crNPrwJOrRxagUYeMn/DZwqN88SDmwaJ8Cvi/TN1HnWBU7GwknckyosC2gd0IqYRsHDEnXf328o9/HC6OkPgOg==",
|
"integrity": "sha512-W2Psnbh1J8ZJw0xKAd8zdNgF9HRLkdWwwdWqubSVk0pUuQkoHnv7rx4GiF9rT4t5DIZGAsConRE3AxCdJ4m8rg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -2270,9 +2271,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-darwin-x64": {
|
"node_modules/@rollup/rollup-darwin-x64": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.59.0.tgz",
|
||||||
"integrity": "sha512-Ji8g8ChVbKrhFtig5QBV7iMaJrGtpHelkB3lsaKzadFBe58gmjfGXAOfI5FV0lYMH8wiqsxKQ1C9B0YTRXVy4w==",
|
"integrity": "sha512-ZW2KkwlS4lwTv7ZVsYDiARfFCnSGhzYPdiOU4IM2fDbL+QGlyAbjgSFuqNRbSthybLbIJ915UtZBtmuLrQAT/w==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -2284,9 +2285,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-freebsd-arm64": {
|
"node_modules/@rollup/rollup-freebsd-arm64": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.59.0.tgz",
|
||||||
"integrity": "sha512-R+/WwhsjmwodAcz65guCGFRkMb4gKWTcIeLy60JJQbXrJ97BOXHxnkPFrP+YwFlaS0m+uWJTstrUA9o+UchFug==",
|
"integrity": "sha512-EsKaJ5ytAu9jI3lonzn3BgG8iRBjV4LxZexygcQbpiU0wU0ATxhNVEpXKfUa0pS05gTcSDMKpn3Sx+QB9RlTTA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -2298,9 +2299,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-freebsd-x64": {
|
"node_modules/@rollup/rollup-freebsd-x64": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.59.0.tgz",
|
||||||
"integrity": "sha512-IEQTCHeiTOnAUC3IDQdzRAGj3jOAYNr9kBguI7MQAAZK3caezRrg0GxAb6Hchg4lxdZEI5Oq3iov/w/hnFWY9Q==",
|
"integrity": "sha512-d3DuZi2KzTMjImrxoHIAODUZYoUUMsuUiY4SRRcJy6NJoZ6iIqWnJu9IScV9jXysyGMVuW+KNzZvBLOcpdl3Vg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -2312,9 +2313,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-arm-gnueabihf": {
|
"node_modules/@rollup/rollup-linux-arm-gnueabihf": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.59.0.tgz",
|
||||||
"integrity": "sha512-F8sWbhZ7tyuEfsmOxwc2giKDQzN3+kuBLPwwZGyVkLlKGdV1nvnNwYD0fKQ8+XS6hp9nY7B+ZeK01EBUE7aHaw==",
|
"integrity": "sha512-t4ONHboXi/3E0rT6OZl1pKbl2Vgxf9vJfWgmUoCEVQVxhW6Cw/c8I6hbbu7DAvgp82RKiH7TpLwxnJeKv2pbsw==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm"
|
"arm"
|
||||||
],
|
],
|
||||||
@@ -2326,9 +2327,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-arm-musleabihf": {
|
"node_modules/@rollup/rollup-linux-arm-musleabihf": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.59.0.tgz",
|
||||||
"integrity": "sha512-rGfNUfn0GIeXtBP1wL5MnzSj98+PZe/AXaGBCRmT0ts80lU5CATYGxXukeTX39XBKsxzFpEeK+Mrp9faXOlmrw==",
|
"integrity": "sha512-CikFT7aYPA2ufMD086cVORBYGHffBo4K8MQ4uPS/ZnY54GKj36i196u8U+aDVT2LX4eSMbyHtyOh7D7Zvk2VvA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm"
|
"arm"
|
||||||
],
|
],
|
||||||
@@ -2340,9 +2341,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-arm64-gnu": {
|
"node_modules/@rollup/rollup-linux-arm64-gnu": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-MMtej3YHWeg/0klK2Qodf3yrNzz6CGjo2UntLvk2RSPlhzgLvYEB3frRvbEF2wRKh1Z2fDIg9KRPe1fawv7C+g==",
|
"integrity": "sha512-jYgUGk5aLd1nUb1CtQ8E+t5JhLc9x5WdBKew9ZgAXg7DBk0ZHErLHdXM24rfX+bKrFe+Xp5YuJo54I5HFjGDAA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -2354,9 +2355,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-arm64-musl": {
|
"node_modules/@rollup/rollup-linux-arm64-musl": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.59.0.tgz",
|
||||||
"integrity": "sha512-1a/qhaaOXhqXGpMFMET9VqwZakkljWHLmZOX48R0I/YLbhdxr1m4gtG1Hq7++VhVUmf+L3sTAf9op4JlhQ5u1Q==",
|
"integrity": "sha512-peZRVEdnFWZ5Bh2KeumKG9ty7aCXzzEsHShOZEFiCQlDEepP1dpUl/SrUNXNg13UmZl+gzVDPsiCwnV1uI0RUA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -2368,9 +2369,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-loong64-gnu": {
|
"node_modules/@rollup/rollup-linux-loong64-gnu": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-QWO6RQTZ/cqYtJMtxhkRkidoNGXc7ERPbZN7dVW5SdURuLeVU7lwKMpo18XdcmpWYd0qsP1bwKPf7DNSUinhvA==",
|
"integrity": "sha512-gbUSW/97f7+r4gHy3Jlup8zDG190AuodsWnNiXErp9mT90iCy9NKKU0Xwx5k8VlRAIV2uU9CsMnEFg/xXaOfXg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"loong64"
|
"loong64"
|
||||||
],
|
],
|
||||||
@@ -2382,9 +2383,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-loong64-musl": {
|
"node_modules/@rollup/rollup-linux-loong64-musl": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.59.0.tgz",
|
||||||
"integrity": "sha512-xpObYIf+8gprgWaPP32xiN5RVTi/s5FCR+XMXSKmhfoJjrpRAjCuuqQXyxUa/eJTdAE6eJ+KDKaoEqjZQxh3Gw==",
|
"integrity": "sha512-yTRONe79E+o0FWFijasoTjtzG9EBedFXJMl888NBEDCDV9I2wGbFFfJQQe63OijbFCUZqxpHz1GzpbtSFikJ4Q==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"loong64"
|
"loong64"
|
||||||
],
|
],
|
||||||
@@ -2396,9 +2397,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-ppc64-gnu": {
|
"node_modules/@rollup/rollup-linux-ppc64-gnu": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-4BrCgrpZo4hvzMDKRqEaW1zeecScDCR+2nZ86ATLhAoJ5FQ+lbHVD3ttKe74/c7tNT9c6F2viwB3ufwp01Oh2w==",
|
"integrity": "sha512-sw1o3tfyk12k3OEpRddF68a1unZ5VCN7zoTNtSn2KndUE+ea3m3ROOKRCZxEpmT9nsGnogpFP9x6mnLTCaoLkA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"ppc64"
|
"ppc64"
|
||||||
],
|
],
|
||||||
@@ -2410,9 +2411,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-ppc64-musl": {
|
"node_modules/@rollup/rollup-linux-ppc64-musl": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.59.0.tgz",
|
||||||
"integrity": "sha512-NOlUuzesGauESAyEYFSe3QTUguL+lvrN1HtwEEsU2rOwdUDeTMJdO5dUYl/2hKf9jWydJrO9OL/XSSf65R5+Xw==",
|
"integrity": "sha512-+2kLtQ4xT3AiIxkzFVFXfsmlZiG5FXYW7ZyIIvGA7Bdeuh9Z0aN4hVyXS/G1E9bTP/vqszNIN/pUKCk/BTHsKA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"ppc64"
|
"ppc64"
|
||||||
],
|
],
|
||||||
@@ -2424,9 +2425,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-riscv64-gnu": {
|
"node_modules/@rollup/rollup-linux-riscv64-gnu": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-ptA88htVp0AwUUqhVghwDIKlvJMD/fmL/wrQj99PRHFRAG6Z5nbWoWG4o81Nt9FT+IuqUQi+L31ZKAFeJ5Is+A==",
|
"integrity": "sha512-NDYMpsXYJJaj+I7UdwIuHHNxXZ/b/N2hR15NyH3m2qAtb/hHPA4g4SuuvrdxetTdndfj9b1WOmy73kcPRoERUg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"riscv64"
|
"riscv64"
|
||||||
],
|
],
|
||||||
@@ -2438,9 +2439,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-riscv64-musl": {
|
"node_modules/@rollup/rollup-linux-riscv64-musl": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.59.0.tgz",
|
||||||
"integrity": "sha512-S51t7aMMTNdmAMPpBg7OOsTdn4tySRQvklmL3RpDRyknk87+Sp3xaumlatU+ppQ+5raY7sSTcC2beGgvhENfuw==",
|
"integrity": "sha512-nLckB8WOqHIf1bhymk+oHxvM9D3tyPndZH8i8+35p/1YiVoVswPid2yLzgX7ZJP0KQvnkhM4H6QZ5m0LzbyIAg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"riscv64"
|
"riscv64"
|
||||||
],
|
],
|
||||||
@@ -2452,9 +2453,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-s390x-gnu": {
|
"node_modules/@rollup/rollup-linux-s390x-gnu": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-Bl00OFnVFkL82FHbEqy3k5CUCKH6OEJL54KCyx2oqsmZnFTR8IoNqBF+mjQVcRCT5sB6yOvK8A37LNm/kPJiZg==",
|
"integrity": "sha512-oF87Ie3uAIvORFBpwnCvUzdeYUqi2wY6jRFWJAy1qus/udHFYIkplYRW+wo+GRUP4sKzYdmE1Y3+rY5Gc4ZO+w==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"s390x"
|
"s390x"
|
||||||
],
|
],
|
||||||
@@ -2466,9 +2467,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-x64-gnu": {
|
"node_modules/@rollup/rollup-linux-x64-gnu": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-ABca4ceT4N+Tv/GtotnWAeXZUZuM/9AQyCyKYyKnpk4yoA7QIAuBt6Hkgpw8kActYlew2mvckXkvx0FfoInnLg==",
|
"integrity": "sha512-3AHmtQq/ppNuUspKAlvA8HtLybkDflkMuLK4DPo77DfthRb71V84/c4MlWJXixZz4uruIH4uaa07IqoAkG64fg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -2480,9 +2481,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-x64-musl": {
|
"node_modules/@rollup/rollup-linux-x64-musl": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.59.0.tgz",
|
||||||
"integrity": "sha512-HFps0JeGtuOR2convgRRkHCekD7j+gdAuXM+/i6kGzQtFhlCtQkpwtNzkNj6QhCDp7DRJ7+qC/1Vg2jt5iSOFw==",
|
"integrity": "sha512-2UdiwS/9cTAx7qIUZB/fWtToJwvt0Vbo0zmnYt7ED35KPg13Q0ym1g442THLC7VyI6JfYTP4PiSOWyoMdV2/xg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -2494,9 +2495,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-openbsd-x64": {
|
"node_modules/@rollup/rollup-openbsd-x64": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.59.0.tgz",
|
||||||
"integrity": "sha512-H+hXEv9gdVQuDTgnqD+SQffoWoc0Of59AStSzTEj/feWTBAnSfSD3+Dql1ZruJQxmykT/JVY0dE8Ka7z0DH1hw==",
|
"integrity": "sha512-M3bLRAVk6GOwFlPTIxVBSYKUaqfLrn8l0psKinkCFxl4lQvOSz8ZrKDz2gxcBwHFpci0B6rttydI4IpS4IS/jQ==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -2508,9 +2509,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-openharmony-arm64": {
|
"node_modules/@rollup/rollup-openharmony-arm64": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.59.0.tgz",
|
||||||
"integrity": "sha512-4wYoDpNg6o/oPximyc/NG+mYUejZrCU2q+2w6YZqrAs2UcNUChIZXjtafAiiZSUc7On8v5NyNj34Kzj/Ltk6dQ==",
|
"integrity": "sha512-tt9KBJqaqp5i5HUZzoafHZX8b5Q2Fe7UjYERADll83O4fGqJ49O1FsL6LpdzVFQcpwvnyd0i+K/VSwu/o/nWlA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -2522,9 +2523,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-win32-arm64-msvc": {
|
"node_modules/@rollup/rollup-win32-arm64-msvc": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.59.0.tgz",
|
||||||
"integrity": "sha512-O54mtsV/6LW3P8qdTcamQmuC990HDfR71lo44oZMZlXU4tzLrbvTii87Ni9opq60ds0YzuAlEr/GNwuNluZyMQ==",
|
"integrity": "sha512-V5B6mG7OrGTwnxaNUzZTDTjDS7F75PO1ae6MJYdiMu60sq0CqN5CVeVsbhPxalupvTX8gXVSU9gq+Rx1/hvu6A==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -2536,9 +2537,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-win32-ia32-msvc": {
|
"node_modules/@rollup/rollup-win32-ia32-msvc": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.59.0.tgz",
|
||||||
"integrity": "sha512-P3dLS+IerxCT/7D2q2FYcRdWRl22dNbrbBEtxdWhXrfIMPP9lQhb5h4Du04mdl5Woq05jVCDPCMF7Ub0NAjIew==",
|
"integrity": "sha512-UKFMHPuM9R0iBegwzKF4y0C4J9u8C6MEJgFuXTBerMk7EJ92GFVFYBfOZaSGLu6COf7FxpQNqhNS4c4icUPqxA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"ia32"
|
"ia32"
|
||||||
],
|
],
|
||||||
@@ -2550,9 +2551,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-win32-x64-gnu": {
|
"node_modules/@rollup/rollup-win32-x64-gnu": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-VMBH2eOOaKGtIJYleXsi2B8CPVADrh+TyNxJ4mWPnKfLB/DBUmzW+5m1xUrcwWoMfSLagIRpjUFeW5CO5hyciQ==",
|
"integrity": "sha512-laBkYlSS1n2L8fSo1thDNGrCTQMmxjYY5G0WFWjFFYZkKPjsMBsgJfGf4TLxXrF6RyhI60L8TMOjBMvXiTcxeA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -2564,9 +2565,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-win32-x64-msvc": {
|
"node_modules/@rollup/rollup-win32-x64-msvc": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.59.0.tgz",
|
||||||
"integrity": "sha512-mxRFDdHIWRxg3UfIIAwCm6NzvxG0jDX/wBN6KsQFTvKFqqg9vTrWUE68qEjHt19A5wwx5X5aUi2zuZT7YR0jrA==",
|
"integrity": "sha512-2HRCml6OztYXyJXAvdDXPKcawukWY2GpR5/nxKp4iBgiO3wcoEGkAaqctIbZcNB6KlUQBIqt8VYkNSj2397EfA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -2624,9 +2625,9 @@
|
|||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/@types/node": {
|
"node_modules/@types/node": {
|
||||||
"version": "25.3.0",
|
"version": "25.3.3",
|
||||||
"resolved": "https://registry.npmjs.org/@types/node/-/node-25.3.0.tgz",
|
"resolved": "https://registry.npmjs.org/@types/node/-/node-25.3.3.tgz",
|
||||||
"integrity": "sha512-4K3bqJpXpqfg2XKGK9bpDTc6xO/xoUP/RBWS7AtRMug6zZFaRekiLzjVtAoZMquxoAbzBvy5nxQ7veS5eYzf8A==",
|
"integrity": "sha512-DpzbrH7wIcBaJibpKo9nnSQL0MTRdnWttGyE5haGwK86xgMOkFLp7vEyfQPGLOJh5wNYiJ3V9PmUMDhV9u8kkQ==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"undici-types": "~7.18.0"
|
"undici-types": "~7.18.0"
|
||||||
@@ -2656,9 +2657,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@types/supertest": {
|
"node_modules/@types/supertest": {
|
||||||
"version": "6.0.3",
|
"version": "7.2.0",
|
||||||
"resolved": "https://registry.npmjs.org/@types/supertest/-/supertest-6.0.3.tgz",
|
"resolved": "https://registry.npmjs.org/@types/supertest/-/supertest-7.2.0.tgz",
|
||||||
"integrity": "sha512-8WzXq62EXFhJ7QsH3Ocb/iKQ/Ty9ZVWnVzoTKc9tyyFRRF3a74Tk2+TLFgaFFw364Ere+npzHKEJ6ga2LzIL7w==",
|
"integrity": "sha512-uh2Lv57xvggst6lCqNdFAmDSvoMG7M/HDtX4iUCquxQ5EGPtaPM5PL5Hmi7LCvOG8db7YaCPNJEeoI8s/WzIQw==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
@@ -2952,9 +2953,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/bn.js": {
|
"node_modules/bn.js": {
|
||||||
"version": "4.12.2",
|
"version": "4.12.3",
|
||||||
"resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.2.tgz",
|
"resolved": "https://registry.npmjs.org/bn.js/-/bn.js-4.12.3.tgz",
|
||||||
"integrity": "sha512-n4DSx829VRTRByMRGdjQ9iqsN0Bh4OolPsFnaZBLcbi8iXcB+kJ9s7EnRt4wILZNV3kPLHkRVfOc/HvhC3ovDw==",
|
"integrity": "sha512-fGTi3gxV/23FTYdAoUtLYp6qySe2KE3teyZitipKNRuVYcBkoP/bB3guXN/XVKUe9mxCHXnc9C4ocyz8OmgN0g==",
|
||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/brace-expansion": {
|
"node_modules/brace-expansion": {
|
||||||
@@ -3017,6 +3018,13 @@
|
|||||||
"node": ">=18"
|
"node": ">=18"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/colorette": {
|
||||||
|
"version": "2.0.20",
|
||||||
|
"resolved": "https://registry.npmjs.org/colorette/-/colorette-2.0.20.tgz",
|
||||||
|
"integrity": "sha512-IfEDxwoWIjkeXL1eXcDiow4UbKjhLdq6/EuSVR9GMN7KVH3r9gQ83e73hsz1Nd1T3ijd5xv1wcWRYO+D6kCI2w==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/combined-stream": {
|
"node_modules/combined-stream": {
|
||||||
"version": "1.0.8",
|
"version": "1.0.8",
|
||||||
"resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
|
"resolved": "https://registry.npmjs.org/combined-stream/-/combined-stream-1.0.8.tgz",
|
||||||
@@ -3161,6 +3169,16 @@
|
|||||||
"node": ">= 12"
|
"node": ">= 12"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/dateformat": {
|
||||||
|
"version": "4.6.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/dateformat/-/dateformat-4.6.3.tgz",
|
||||||
|
"integrity": "sha512-2P0p0pFGzHS5EMnhdxQi7aJN+iMheud0UhG4dlE1DLAlvL8JHjJJTX/CSm4JXwV0Ka5nGk3zC5mcb5bUQUxxMA==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": "*"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/debug": {
|
"node_modules/debug": {
|
||||||
"version": "4.4.3",
|
"version": "4.4.3",
|
||||||
"resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz",
|
"resolved": "https://registry.npmjs.org/debug/-/debug-4.4.3.tgz",
|
||||||
@@ -3888,6 +3906,16 @@
|
|||||||
"safe-buffer": "^5.0.1"
|
"safe-buffer": "^5.0.1"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/end-of-stream": {
|
||||||
|
"version": "1.4.5",
|
||||||
|
"resolved": "https://registry.npmjs.org/end-of-stream/-/end-of-stream-1.4.5.tgz",
|
||||||
|
"integrity": "sha512-ooEGc6HP26xXq/N+GCGOT0JKCLDGrq2bQUZrQ7gyrJiZANJ/8YDTxTpQBXGMn+WbIQXNVpyWymm7KYVICQnyOg==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"once": "^1.4.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/es-define-property": {
|
"node_modules/es-define-property": {
|
||||||
"version": "1.0.1",
|
"version": "1.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/es-define-property/-/es-define-property-1.0.1.tgz",
|
||||||
@@ -4025,6 +4053,13 @@
|
|||||||
"node": ">=12.0.0"
|
"node": ">=12.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/fast-copy": {
|
||||||
|
"version": "4.0.2",
|
||||||
|
"resolved": "https://registry.npmjs.org/fast-copy/-/fast-copy-4.0.2.tgz",
|
||||||
|
"integrity": "sha512-ybA6PDXIXOXivLJK/z9e+Otk7ve13I4ckBvGO5I2RRmBU1gMHLVDJYEuJYhGwez7YNlYji2M2DvVU+a9mSFDlw==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/fast-decode-uri-component": {
|
"node_modules/fast-decode-uri-component": {
|
||||||
"version": "1.0.1",
|
"version": "1.0.1",
|
||||||
"resolved": "https://registry.npmjs.org/fast-decode-uri-component/-/fast-decode-uri-component-1.0.1.tgz",
|
"resolved": "https://registry.npmjs.org/fast-decode-uri-component/-/fast-decode-uri-component-1.0.1.tgz",
|
||||||
@@ -4121,9 +4156,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/fastify": {
|
"node_modules/fastify": {
|
||||||
"version": "5.7.4",
|
"version": "5.8.1",
|
||||||
"resolved": "https://registry.npmjs.org/fastify/-/fastify-5.7.4.tgz",
|
"resolved": "https://registry.npmjs.org/fastify/-/fastify-5.8.1.tgz",
|
||||||
"integrity": "sha512-e6l5NsRdaEP8rdD8VR0ErJASeyaRbzXYpmkrpr2SuvuMq6Si3lvsaVy5C+7gLanEkvjpMDzBXWE5HPeb/hgTxA==",
|
"integrity": "sha512-y0kicFvvn7CYWoPOVLOcvn4YyKQz03DIY7UxmyOy21/J8eXm09R+tmb+tVDBW5h+pja30cHI5dqUcSlvY86V2A==",
|
||||||
"funding": [
|
"funding": [
|
||||||
{
|
{
|
||||||
"type": "github",
|
"type": "github",
|
||||||
@@ -4145,7 +4180,7 @@
|
|||||||
"fast-json-stringify": "^6.0.0",
|
"fast-json-stringify": "^6.0.0",
|
||||||
"find-my-way": "^9.0.0",
|
"find-my-way": "^9.0.0",
|
||||||
"light-my-request": "^6.0.0",
|
"light-my-request": "^6.0.0",
|
||||||
"pino": "^10.1.0",
|
"pino": "^9.14.0 || ^10.1.0",
|
||||||
"process-warning": "^5.0.0",
|
"process-warning": "^5.0.0",
|
||||||
"rfdc": "^1.3.1",
|
"rfdc": "^1.3.1",
|
||||||
"secure-json-parse": "^4.0.0",
|
"secure-json-parse": "^4.0.0",
|
||||||
@@ -4500,6 +4535,13 @@
|
|||||||
"node": ">=18.0.0"
|
"node": ">=18.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/help-me": {
|
||||||
|
"version": "5.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/help-me/-/help-me-5.0.0.tgz",
|
||||||
|
"integrity": "sha512-7xgomUX6ADmcYzFik0HzAxh/73YlKR9bmFzf51CZwR+b6YtzU2m0u49hQCqV6SvlqIqsaxovfwdvbnsw3b/zpg==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT"
|
||||||
|
},
|
||||||
"node_modules/html-escaper": {
|
"node_modules/html-escaper": {
|
||||||
"version": "2.0.2",
|
"version": "2.0.2",
|
||||||
"resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz",
|
"resolved": "https://registry.npmjs.org/html-escaper/-/html-escaper-2.0.2.tgz",
|
||||||
@@ -4611,6 +4653,16 @@
|
|||||||
"url": "https://github.com/sponsors/panva"
|
"url": "https://github.com/sponsors/panva"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/joycon": {
|
||||||
|
"version": "3.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/joycon/-/joycon-3.1.1.tgz",
|
||||||
|
"integrity": "sha512-34wB/Y7MW7bzjKRjUKTa46I2Z7eV62Rkhva+KkopW7Qvv/OSWBqvkSY7vusOPrNuZcUG3tApvdVgNB8POj3SPw==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=10"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/js-base64": {
|
"node_modules/js-base64": {
|
||||||
"version": "3.7.8",
|
"version": "3.7.8",
|
||||||
"resolved": "https://registry.npmjs.org/js-base64/-/js-base64-3.7.8.tgz",
|
"resolved": "https://registry.npmjs.org/js-base64/-/js-base64-3.7.8.tgz",
|
||||||
@@ -4838,9 +4890,9 @@
|
|||||||
"license": "ISC"
|
"license": "ISC"
|
||||||
},
|
},
|
||||||
"node_modules/minimatch": {
|
"node_modules/minimatch": {
|
||||||
"version": "10.2.2",
|
"version": "10.2.4",
|
||||||
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.2.tgz",
|
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.4.tgz",
|
||||||
"integrity": "sha512-+G4CpNBxa5MprY+04MbgOw1v7So6n5JY166pFi9KfYwT78fxScCeSNQSNzp6dpPSW2rONOps6Ocam1wFhCgoVw==",
|
"integrity": "sha512-oRjTw/97aTBN0RHbYCdtF1MQfvusSIBQM0IZEgzl6426+8jSC0nF1a/GmnVLpfB9yyr6g6FTqWqiZVbxrtaCIg==",
|
||||||
"license": "BlueOak-1.0.0",
|
"license": "BlueOak-1.0.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"brace-expansion": "^5.0.2"
|
"brace-expansion": "^5.0.2"
|
||||||
@@ -4852,6 +4904,16 @@
|
|||||||
"url": "https://github.com/sponsors/isaacs"
|
"url": "https://github.com/sponsors/isaacs"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/minimist": {
|
||||||
|
"version": "1.2.8",
|
||||||
|
"resolved": "https://registry.npmjs.org/minimist/-/minimist-1.2.8.tgz",
|
||||||
|
"integrity": "sha512-2yyAR8qBkN3YuheJanUpWC5U3bb5osDywNB8RzDVlDwDHbocAJveqqj1u8+SVD7jkWT4yvsHCpWqqWqAxb0zCA==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/ljharb"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/minipass": {
|
"node_modules/minipass": {
|
||||||
"version": "7.1.2",
|
"version": "7.1.2",
|
||||||
"resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz",
|
"resolved": "https://registry.npmjs.org/minipass/-/minipass-7.1.2.tgz",
|
||||||
@@ -5117,6 +5179,41 @@
|
|||||||
"split2": "^4.0.0"
|
"split2": "^4.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/pino-pretty": {
|
||||||
|
"version": "13.1.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/pino-pretty/-/pino-pretty-13.1.3.tgz",
|
||||||
|
"integrity": "sha512-ttXRkkOz6WWC95KeY9+xxWL6AtImwbyMHrL1mSwqwW9u+vLp/WIElvHvCSDg0xO/Dzrggz1zv3rN5ovTRVowKg==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"colorette": "^2.0.7",
|
||||||
|
"dateformat": "^4.6.3",
|
||||||
|
"fast-copy": "^4.0.0",
|
||||||
|
"fast-safe-stringify": "^2.1.1",
|
||||||
|
"help-me": "^5.0.0",
|
||||||
|
"joycon": "^3.1.1",
|
||||||
|
"minimist": "^1.2.6",
|
||||||
|
"on-exit-leak-free": "^2.1.0",
|
||||||
|
"pino-abstract-transport": "^3.0.0",
|
||||||
|
"pump": "^3.0.0",
|
||||||
|
"secure-json-parse": "^4.0.0",
|
||||||
|
"sonic-boom": "^4.0.1",
|
||||||
|
"strip-json-comments": "^5.0.2"
|
||||||
|
},
|
||||||
|
"bin": {
|
||||||
|
"pino-pretty": "bin.js"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/pino-pretty/node_modules/pino-abstract-transport": {
|
||||||
|
"version": "3.0.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/pino-abstract-transport/-/pino-abstract-transport-3.0.0.tgz",
|
||||||
|
"integrity": "sha512-wlfUczU+n7Hy/Ha5j9a/gZNy7We5+cXp8YL+X+PG8S0KXxw7n/JXA3c46Y0zQznIJ83URJiwy7Lh56WLokNuxg==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"split2": "^4.0.0"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/pino-std-serializers": {
|
"node_modules/pino-std-serializers": {
|
||||||
"version": "7.0.0",
|
"version": "7.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/pino-std-serializers/-/pino-std-serializers-7.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/pino-std-serializers/-/pino-std-serializers-7.0.0.tgz",
|
||||||
@@ -5174,6 +5271,17 @@
|
|||||||
"integrity": "sha512-7nJ6v5lnJsXwGprnGXga4wx6d1POjvi5Qmf1ivTRxTjH4Z/9Czja/UCMLVmB9N93GeWOU93XaFaEt6jbuoagNw==",
|
"integrity": "sha512-7nJ6v5lnJsXwGprnGXga4wx6d1POjvi5Qmf1ivTRxTjH4Z/9Czja/UCMLVmB9N93GeWOU93XaFaEt6jbuoagNw==",
|
||||||
"license": "ISC"
|
"license": "ISC"
|
||||||
},
|
},
|
||||||
|
"node_modules/pump": {
|
||||||
|
"version": "3.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/pump/-/pump-3.0.3.tgz",
|
||||||
|
"integrity": "sha512-todwxLMY7/heScKmntwQG8CXVkWUOdYxIvY2s0VWAAMh/nd8SoYiRaKjlr7+iCs984f2P8zvrfWcDDYVb73NfA==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"end-of-stream": "^1.1.0",
|
||||||
|
"once": "^1.3.1"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/qs": {
|
"node_modules/qs": {
|
||||||
"version": "6.14.2",
|
"version": "6.14.2",
|
||||||
"resolved": "https://registry.npmjs.org/qs/-/qs-6.14.2.tgz",
|
"resolved": "https://registry.npmjs.org/qs/-/qs-6.14.2.tgz",
|
||||||
@@ -5250,9 +5358,9 @@
|
|||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/rollup": {
|
"node_modules/rollup": {
|
||||||
"version": "4.57.1",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/rollup/-/rollup-4.57.1.tgz",
|
"resolved": "https://registry.npmjs.org/rollup/-/rollup-4.59.0.tgz",
|
||||||
"integrity": "sha512-oQL6lgK3e2QZeQ7gcgIkS2YZPg5slw37hYufJ3edKlfQSGGm8ICoxswK15ntSzF/a8+h7ekRy7k7oWc3BQ7y8A==",
|
"integrity": "sha512-2oMpl67a3zCH9H79LeMcbDhXW/UmWG/y2zuqnF2jQq5uq9TbM9TVyXvA4+t+ne2IIkBdrLpAaRQAvo7YI/Yyeg==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
@@ -5266,31 +5374,31 @@
|
|||||||
"npm": ">=8.0.0"
|
"npm": ">=8.0.0"
|
||||||
},
|
},
|
||||||
"optionalDependencies": {
|
"optionalDependencies": {
|
||||||
"@rollup/rollup-android-arm-eabi": "4.57.1",
|
"@rollup/rollup-android-arm-eabi": "4.59.0",
|
||||||
"@rollup/rollup-android-arm64": "4.57.1",
|
"@rollup/rollup-android-arm64": "4.59.0",
|
||||||
"@rollup/rollup-darwin-arm64": "4.57.1",
|
"@rollup/rollup-darwin-arm64": "4.59.0",
|
||||||
"@rollup/rollup-darwin-x64": "4.57.1",
|
"@rollup/rollup-darwin-x64": "4.59.0",
|
||||||
"@rollup/rollup-freebsd-arm64": "4.57.1",
|
"@rollup/rollup-freebsd-arm64": "4.59.0",
|
||||||
"@rollup/rollup-freebsd-x64": "4.57.1",
|
"@rollup/rollup-freebsd-x64": "4.59.0",
|
||||||
"@rollup/rollup-linux-arm-gnueabihf": "4.57.1",
|
"@rollup/rollup-linux-arm-gnueabihf": "4.59.0",
|
||||||
"@rollup/rollup-linux-arm-musleabihf": "4.57.1",
|
"@rollup/rollup-linux-arm-musleabihf": "4.59.0",
|
||||||
"@rollup/rollup-linux-arm64-gnu": "4.57.1",
|
"@rollup/rollup-linux-arm64-gnu": "4.59.0",
|
||||||
"@rollup/rollup-linux-arm64-musl": "4.57.1",
|
"@rollup/rollup-linux-arm64-musl": "4.59.0",
|
||||||
"@rollup/rollup-linux-loong64-gnu": "4.57.1",
|
"@rollup/rollup-linux-loong64-gnu": "4.59.0",
|
||||||
"@rollup/rollup-linux-loong64-musl": "4.57.1",
|
"@rollup/rollup-linux-loong64-musl": "4.59.0",
|
||||||
"@rollup/rollup-linux-ppc64-gnu": "4.57.1",
|
"@rollup/rollup-linux-ppc64-gnu": "4.59.0",
|
||||||
"@rollup/rollup-linux-ppc64-musl": "4.57.1",
|
"@rollup/rollup-linux-ppc64-musl": "4.59.0",
|
||||||
"@rollup/rollup-linux-riscv64-gnu": "4.57.1",
|
"@rollup/rollup-linux-riscv64-gnu": "4.59.0",
|
||||||
"@rollup/rollup-linux-riscv64-musl": "4.57.1",
|
"@rollup/rollup-linux-riscv64-musl": "4.59.0",
|
||||||
"@rollup/rollup-linux-s390x-gnu": "4.57.1",
|
"@rollup/rollup-linux-s390x-gnu": "4.59.0",
|
||||||
"@rollup/rollup-linux-x64-gnu": "4.57.1",
|
"@rollup/rollup-linux-x64-gnu": "4.59.0",
|
||||||
"@rollup/rollup-linux-x64-musl": "4.57.1",
|
"@rollup/rollup-linux-x64-musl": "4.59.0",
|
||||||
"@rollup/rollup-openbsd-x64": "4.57.1",
|
"@rollup/rollup-openbsd-x64": "4.59.0",
|
||||||
"@rollup/rollup-openharmony-arm64": "4.57.1",
|
"@rollup/rollup-openharmony-arm64": "4.59.0",
|
||||||
"@rollup/rollup-win32-arm64-msvc": "4.57.1",
|
"@rollup/rollup-win32-arm64-msvc": "4.59.0",
|
||||||
"@rollup/rollup-win32-ia32-msvc": "4.57.1",
|
"@rollup/rollup-win32-ia32-msvc": "4.59.0",
|
||||||
"@rollup/rollup-win32-x64-gnu": "4.57.1",
|
"@rollup/rollup-win32-x64-gnu": "4.59.0",
|
||||||
"@rollup/rollup-win32-x64-msvc": "4.57.1",
|
"@rollup/rollup-win32-x64-msvc": "4.59.0",
|
||||||
"fsevents": "~2.3.2"
|
"fsevents": "~2.3.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -5630,6 +5738,19 @@
|
|||||||
"reusify": "^1.0.0"
|
"reusify": "^1.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/strip-json-comments": {
|
||||||
|
"version": "5.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/strip-json-comments/-/strip-json-comments-5.0.3.tgz",
|
||||||
|
"integrity": "sha512-1tB5mhVo7U+ETBKNf92xT4hrQa3pm0MZ0PQvuDnWgAAGHDsfp4lPSpiS6psrSiet87wyGPh9ft6wmhOMQ0hDiw==",
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=14.16"
|
||||||
|
},
|
||||||
|
"funding": {
|
||||||
|
"url": "https://github.com/sponsors/sindresorhus"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/superagent": {
|
"node_modules/superagent": {
|
||||||
"version": "10.3.0",
|
"version": "10.3.0",
|
||||||
"resolved": "https://registry.npmjs.org/superagent/-/superagent-10.3.0.tgz",
|
"resolved": "https://registry.npmjs.org/superagent/-/superagent-10.3.0.tgz",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "medassist-ng-backend",
|
"name": "medassist-ng-backend",
|
||||||
"version": "1.16.0",
|
"version": "1.19.0",
|
||||||
"private": true,
|
"private": true,
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
@@ -29,7 +29,7 @@
|
|||||||
"argon2": "^0.44.0",
|
"argon2": "^0.44.0",
|
||||||
"dotenv": "^17.3.1",
|
"dotenv": "^17.3.1",
|
||||||
"drizzle-orm": "^0.45.1",
|
"drizzle-orm": "^0.45.1",
|
||||||
"fastify": "^5.7.4",
|
"fastify": "^5.8.1",
|
||||||
"nodemailer": "^8.0.1",
|
"nodemailer": "^8.0.1",
|
||||||
"openid-client": "^6.8.2",
|
"openid-client": "^6.8.2",
|
||||||
"sharp": "^0.34.5",
|
"sharp": "^0.34.5",
|
||||||
@@ -37,11 +37,12 @@
|
|||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@biomejs/biome": "^2.4.4",
|
"@biomejs/biome": "^2.4.4",
|
||||||
"@types/node": "^25.3.0",
|
"@types/node": "^25.3.3",
|
||||||
"@types/nodemailer": "^7.0.11",
|
"@types/nodemailer": "^7.0.11",
|
||||||
"@types/supertest": "^6.0.2",
|
"@types/supertest": "^7.2.0",
|
||||||
"@vitest/coverage-v8": "^4.0.18",
|
"@vitest/coverage-v8": "^4.0.18",
|
||||||
"drizzle-kit": "^0.31.9",
|
"drizzle-kit": "^0.31.9",
|
||||||
|
"pino-pretty": "^13.1.3",
|
||||||
"supertest": "^7.2.2",
|
"supertest": "^7.2.2",
|
||||||
"tsx": "^4.19.0",
|
"tsx": "^4.19.0",
|
||||||
"typescript": "^5.5.4",
|
"typescript": "^5.5.4",
|
||||||
|
|||||||
@@ -78,10 +78,6 @@ async function runMigrations() {
|
|||||||
const migrateResult = await runDrizzleMigrations(db);
|
const migrateResult = await runDrizzleMigrations(db);
|
||||||
if (!migrateResult.success) {
|
if (!migrateResult.success) {
|
||||||
log.error(`[DB] Migration error: ${migrateResult.error}`);
|
log.error(`[DB] Migration error: ${migrateResult.error}`);
|
||||||
} else if (migrateResult.warning) {
|
|
||||||
log.warn(`[DB] Migration warning: ${migrateResult.warning}`);
|
|
||||||
} else {
|
|
||||||
log.debug(`[DB] Drizzle migrations completed`);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Run ALTER TABLE migrations for backward compatibility
|
// Run ALTER TABLE migrations for backward compatibility
|
||||||
|
|||||||
@@ -88,13 +88,12 @@ export async function runDrizzleMigrations(
|
|||||||
await migrate(database, { migrationsFolder });
|
await migrate(database, { migrationsFolder });
|
||||||
return { success: true };
|
return { success: true };
|
||||||
} catch (err: unknown) {
|
} catch (err: unknown) {
|
||||||
// If the error is about existing schema objects, the DB is already up-to-date
|
const msg = (err as Error).message ?? "";
|
||||||
// This happens when ALTER migrations in client.ts have already added the columns,
|
// Duplicate column / already exists = DB is already up-to-date (expected for existing DBs)
|
||||||
// or when tables were created before drizzle migrations were introduced
|
if (msg.includes("duplicate column") || msg.includes("already exists")) {
|
||||||
if ((err as Error).message?.includes("duplicate column") || (err as Error).message?.includes("already exists")) {
|
return { success: true };
|
||||||
return { success: true, warning: `Schema already up-to-date: ${(err as Error).message}` };
|
|
||||||
}
|
}
|
||||||
return { success: false, error: (err as Error).message };
|
return { success: false, error: msg };
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -126,6 +125,14 @@ export async function runAlterMigrations(client: Client): Promise<{ success: boo
|
|||||||
`ALTER TABLE medications ADD COLUMN obsolete_at integer`,
|
`ALTER TABLE medications ADD COLUMN obsolete_at integer`,
|
||||||
// Added for explicit medication lifecycle start date
|
// Added for explicit medication lifecycle start date
|
||||||
`ALTER TABLE medications ADD COLUMN medication_start_date text NOT NULL DEFAULT ''`,
|
`ALTER TABLE medications ADD COLUMN medication_start_date text NOT NULL DEFAULT ''`,
|
||||||
|
// Added for form/lifecycle modeling (V1 medication forms)
|
||||||
|
`ALTER TABLE medications ADD COLUMN medication_form text NOT NULL DEFAULT 'tablet'`,
|
||||||
|
`ALTER TABLE medications ADD COLUMN pill_form text`,
|
||||||
|
`ALTER TABLE medications ADD COLUMN lifecycle_category text NOT NULL DEFAULT 'refill_when_empty'`,
|
||||||
|
`ALTER TABLE medications ADD COLUMN medication_end_date text`,
|
||||||
|
`ALTER TABLE medications ADD COLUMN auto_mark_obsolete_after_end_date integer NOT NULL DEFAULT 1`,
|
||||||
|
`ALTER TABLE medications ADD COLUMN package_amount_value integer NOT NULL DEFAULT 0`,
|
||||||
|
`ALTER TABLE medications ADD COLUMN package_amount_unit text NOT NULL DEFAULT 'ml'`,
|
||||||
// Added for more detailed reminder info display
|
// Added for more detailed reminder info display
|
||||||
`ALTER TABLE user_settings ADD COLUMN last_reminder_med_name text`,
|
`ALTER TABLE user_settings ADD COLUMN last_reminder_med_name text`,
|
||||||
`ALTER TABLE user_settings ADD COLUMN last_reminder_taken_by text`,
|
`ALTER TABLE user_settings ADD COLUMN last_reminder_taken_by text`,
|
||||||
|
|||||||
@@ -29,6 +29,11 @@ export const medications = sqliteTable("medications", {
|
|||||||
genericName: text("generic_name", { length: 100 }),
|
genericName: text("generic_name", { length: 100 }),
|
||||||
takenByJson: text("taken_by_json").notNull().default("[]"), // JSON array of person names
|
takenByJson: text("taken_by_json").notNull().default("[]"), // JSON array of person names
|
||||||
packageType: text("package_type", { length: 20 }).notNull().default("blister"), // 'blister' or 'bottle'
|
packageType: text("package_type", { length: 20 }).notNull().default("blister"), // 'blister' or 'bottle'
|
||||||
|
medicationForm: text("medication_form", { length: 20 }).notNull().default("tablet"), // 'capsule' | 'tablet' | 'liquid' | 'topical'
|
||||||
|
pillForm: text("pill_form", { length: 20 }), // Only for blister/bottle with pill-based medications: 'tablet' | 'capsule'
|
||||||
|
lifecycleCategory: text("lifecycle_category", { length: 30 }).notNull().default("refill_when_empty"), // 'refill_when_empty' | 'treatment_period'
|
||||||
|
packageAmountValue: integer("package_amount_value").notNull().default(0), // Informational package quantity (ml/g)
|
||||||
|
packageAmountUnit: text("package_amount_unit", { length: 10 }).notNull().default("ml"), // 'ml' | 'g'
|
||||||
packCount: integer("pack_count").notNull().default(1),
|
packCount: integer("pack_count").notNull().default(1),
|
||||||
blistersPerPack: integer("blisters_per_pack").notNull().default(1),
|
blistersPerPack: integer("blisters_per_pack").notNull().default(1),
|
||||||
pillsPerBlister: integer("pills_per_blister").notNull().default(1),
|
pillsPerBlister: integer("pills_per_blister").notNull().default(1),
|
||||||
@@ -48,6 +53,10 @@ export const medications = sqliteTable("medications", {
|
|||||||
notes: text("notes"),
|
notes: text("notes"),
|
||||||
intakeRemindersEnabled: integer("intake_reminders_enabled", { mode: "boolean" }).notNull().default(false),
|
intakeRemindersEnabled: integer("intake_reminders_enabled", { mode: "boolean" }).notNull().default(false),
|
||||||
medicationStartDate: text("medication_start_date").notNull().default(""),
|
medicationStartDate: text("medication_start_date").notNull().default(""),
|
||||||
|
medicationEndDate: text("medication_end_date"),
|
||||||
|
autoMarkObsoleteAfterEndDate: integer("auto_mark_obsolete_after_end_date", { mode: "boolean" })
|
||||||
|
.notNull()
|
||||||
|
.default(true),
|
||||||
isObsolete: integer("is_obsolete", { mode: "boolean" }).notNull().default(false),
|
isObsolete: integer("is_obsolete", { mode: "boolean" }).notNull().default(false),
|
||||||
obsoleteAt: integer("obsolete_at", { mode: "timestamp" }),
|
obsoleteAt: integer("obsolete_at", { mode: "timestamp" }),
|
||||||
prescriptionEnabled: integer("prescription_enabled", { mode: "boolean" }).notNull().default(false),
|
prescriptionEnabled: integer("prescription_enabled", { mode: "boolean" }).notNull().default(false),
|
||||||
|
|||||||
@@ -179,6 +179,8 @@ type TranslationKeys = {
|
|||||||
common: {
|
common: {
|
||||||
pill: string;
|
pill: string;
|
||||||
pills: string;
|
pills: string;
|
||||||
|
units: string;
|
||||||
|
ml: string;
|
||||||
blister: string;
|
blister: string;
|
||||||
blisters: string;
|
blisters: string;
|
||||||
day: string;
|
day: string;
|
||||||
@@ -299,6 +301,8 @@ const translations: Record<Language, TranslationKeys> = {
|
|||||||
common: {
|
common: {
|
||||||
pill: "pill",
|
pill: "pill",
|
||||||
pills: "pills",
|
pills: "pills",
|
||||||
|
units: "units",
|
||||||
|
ml: "ml",
|
||||||
blister: "blister",
|
blister: "blister",
|
||||||
blisters: "blisters",
|
blisters: "blisters",
|
||||||
day: "day",
|
day: "day",
|
||||||
@@ -420,6 +424,8 @@ const translations: Record<Language, TranslationKeys> = {
|
|||||||
common: {
|
common: {
|
||||||
pill: "Tablette",
|
pill: "Tablette",
|
||||||
pills: "Tabletten",
|
pills: "Tabletten",
|
||||||
|
units: "Einheiten",
|
||||||
|
ml: "ml",
|
||||||
blister: "Blister",
|
blister: "Blister",
|
||||||
blisters: "Blister",
|
blisters: "Blister",
|
||||||
day: "Tag",
|
day: "Tag",
|
||||||
|
|||||||
+17
-4
@@ -57,6 +57,21 @@ function sanitizeCorrelationId(headers: IncomingHttpHeaders): string | null {
|
|||||||
return trimmed;
|
return trimmed;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function buildLoggerOptions(level: string) {
|
||||||
|
const base = {
|
||||||
|
level,
|
||||||
|
timestamp: () => `,"time":"${new Date().toISOString()}"`,
|
||||||
|
};
|
||||||
|
// Human-readable logs in development, structured JSON in production/test
|
||||||
|
if (process.env.NODE_ENV !== "production" && process.env.NODE_ENV !== "test") {
|
||||||
|
return {
|
||||||
|
...base,
|
||||||
|
transport: { target: "pino-pretty", options: { translateTime: "SYS:yyyy-mm-dd HH:MM:ss.l" } },
|
||||||
|
};
|
||||||
|
}
|
||||||
|
return base;
|
||||||
|
}
|
||||||
|
|
||||||
/** Create and configure Fastify app (without starting) */
|
/** Create and configure Fastify app (without starting) */
|
||||||
export async function createApp(options?: {
|
export async function createApp(options?: {
|
||||||
logLevel?: string;
|
logLevel?: string;
|
||||||
@@ -84,7 +99,7 @@ export async function createApp(options?: {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const app = Fastify({
|
const app = Fastify({
|
||||||
logger: { level: opts.logLevel },
|
logger: buildLoggerOptions(opts.logLevel),
|
||||||
genReqId: (request) => sanitizeCorrelationId(request.headers) ?? randomUUID(),
|
genReqId: (request) => sanitizeCorrelationId(request.headers) ?? randomUUID(),
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -157,9 +172,7 @@ log.info("[DB] Migrations complete, starting server...");
|
|||||||
const imagesDir = ensureImagesDirectory();
|
const imagesDir = ensureImagesDirectory();
|
||||||
|
|
||||||
const app = Fastify({
|
const app = Fastify({
|
||||||
logger: {
|
logger: buildLoggerOptions(env.LOG_LEVEL),
|
||||||
level: env.LOG_LEVEL,
|
|
||||||
},
|
|
||||||
genReqId: (request) => sanitizeCorrelationId(request.headers) ?? randomUUID(),
|
genReqId: (request) => sanitizeCorrelationId(request.headers) ?? randomUUID(),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -47,7 +47,7 @@ export async function getAnonymousUserId(): Promise<number> {
|
|||||||
export interface AuthState {
|
export interface AuthState {
|
||||||
authEnabled: boolean;
|
authEnabled: boolean;
|
||||||
registrationEnabled: boolean;
|
registrationEnabled: boolean;
|
||||||
localAuthEnabled: boolean;
|
formLoginEnabled: boolean;
|
||||||
oidcEnabled: boolean;
|
oidcEnabled: boolean;
|
||||||
oidcProviderName: string;
|
oidcProviderName: string;
|
||||||
hasUsers: boolean;
|
hasUsers: boolean;
|
||||||
@@ -59,15 +59,18 @@ export async function getAuthState(): Promise<AuthState> {
|
|||||||
const [result] = await db.select({ count: count() }).from(users).where(sql`${users.id} != ${ANONYMOUS_USER_ID}`);
|
const [result] = await db.select({ count: count() }).from(users).where(sql`${users.id} != ${ANONYMOUS_USER_ID}`);
|
||||||
const hasUsers = result.count > 0;
|
const hasUsers = result.count > 0;
|
||||||
|
|
||||||
|
const needsSetup = env.AUTH_ENABLED && !hasUsers;
|
||||||
|
|
||||||
return {
|
return {
|
||||||
authEnabled: env.AUTH_ENABLED,
|
authEnabled: env.AUTH_ENABLED,
|
||||||
// Registration: enabled via ENV OR no users exist (first-time setup)
|
// Registration: enabled via ENV OR no users exist (first-time setup)
|
||||||
registrationEnabled: env.REGISTRATION_ENABLED || !hasUsers,
|
registrationEnabled: env.REGISTRATION_ENABLED || !hasUsers,
|
||||||
localAuthEnabled: env.AUTH_ENABLED, // Password auth available when auth is enabled
|
// Form login: enabled when auth + form login are both on, or forced on for first-user setup
|
||||||
|
formLoginEnabled: needsSetup || (env.AUTH_ENABLED && env.FORM_LOGIN_ENABLED),
|
||||||
oidcEnabled: env.OIDC_ENABLED,
|
oidcEnabled: env.OIDC_ENABLED,
|
||||||
oidcProviderName: env.OIDC_PROVIDER_NAME,
|
oidcProviderName: env.OIDC_PROVIDER_NAME,
|
||||||
hasUsers,
|
hasUsers,
|
||||||
needsSetup: env.AUTH_ENABLED && !hasUsers,
|
needsSetup,
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -28,7 +28,11 @@ const EnvSchema = z.object({
|
|||||||
.string()
|
.string()
|
||||||
.transform((v) => v === "true")
|
.transform((v) => v === "true")
|
||||||
.default("false"),
|
.default("false"),
|
||||||
// Disable local auth when using SSO only
|
// Disable username/password form login (useful for OIDC-only setups)
|
||||||
|
FORM_LOGIN_ENABLED: z
|
||||||
|
.string()
|
||||||
|
.transform((v) => v === "true")
|
||||||
|
.default("true"),
|
||||||
|
|
||||||
// JWT Secrets - only required when AUTH_ENABLED=true
|
// JWT Secrets - only required when AUTH_ENABLED=true
|
||||||
JWT_SECRET: z.string().min(10).optional(),
|
JWT_SECRET: z.string().min(10).optional(),
|
||||||
@@ -128,4 +132,26 @@ if (parsed.OIDC_ENABLED) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Validate that at least one login method is available when auth is enabled
|
||||||
|
if (parsed.AUTH_ENABLED && !parsed.FORM_LOGIN_ENABLED && !parsed.OIDC_ENABLED) {
|
||||||
|
console.error("=".repeat(60));
|
||||||
|
console.error("AUTHENTICATION CONFIGURATION ERROR");
|
||||||
|
console.error("=".repeat(60));
|
||||||
|
console.error("AUTH_ENABLED=true but no login method is available.");
|
||||||
|
console.error("FORM_LOGIN_ENABLED=false and OIDC_ENABLED=false means users cannot log in.");
|
||||||
|
console.error("");
|
||||||
|
console.error("To fix this, either:");
|
||||||
|
console.error(" 1. Set FORM_LOGIN_ENABLED=true to allow username/password login");
|
||||||
|
console.error(" 2. Set OIDC_ENABLED=true to allow SSO login");
|
||||||
|
console.error("=".repeat(60));
|
||||||
|
process.exit(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Warn about ineffective registration when form login is disabled
|
||||||
|
if (parsed.REGISTRATION_ENABLED && !parsed.FORM_LOGIN_ENABLED) {
|
||||||
|
console.warn(
|
||||||
|
"[config] REGISTRATION_ENABLED=true has no effect when FORM_LOGIN_ENABLED=false (no registration form available)"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
export const env = parsed;
|
export const env = parsed;
|
||||||
|
|||||||
@@ -123,8 +123,8 @@ export async function authRoutes(app: FastifyInstance) {
|
|||||||
return reply.status(400).send({ error: "Registration is disabled", code: "REGISTRATION_DISABLED" });
|
return reply.status(400).send({ error: "Registration is disabled", code: "REGISTRATION_DISABLED" });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!state.localAuthEnabled) {
|
if (!state.formLoginEnabled) {
|
||||||
return reply.status(400).send({ error: "Local authentication is disabled", code: "LOCAL_AUTH_DISABLED" });
|
return reply.status(400).send({ error: "Form login is disabled", code: "FORM_LOGIN_DISABLED" });
|
||||||
}
|
}
|
||||||
|
|
||||||
// Validate input
|
// Validate input
|
||||||
@@ -185,8 +185,8 @@ export async function authRoutes(app: FastifyInstance) {
|
|||||||
return reply.status(400).send({ error: "Authentication is disabled", code: "AUTH_DISABLED" });
|
return reply.status(400).send({ error: "Authentication is disabled", code: "AUTH_DISABLED" });
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!state.localAuthEnabled) {
|
if (!state.formLoginEnabled) {
|
||||||
return reply.status(400).send({ error: "Local authentication is disabled", code: "LOCAL_AUTH_DISABLED" });
|
return reply.status(400).send({ error: "Form login is disabled", code: "FORM_LOGIN_DISABLED" });
|
||||||
}
|
}
|
||||||
|
|
||||||
const parsed = loginSchema.safeParse(request.body);
|
const parsed = loginSchema.safeParse(request.body);
|
||||||
|
|||||||
@@ -137,8 +137,9 @@ async function validateShareDoseId(share: typeof shareTokens.$inferSelect, doseI
|
|||||||
export async function doseRoutes(app: FastifyInstance) {
|
export async function doseRoutes(app: FastifyInstance) {
|
||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
// GET /doses/taken - PROTECTED: Get all taken doses for the user
|
// GET /doses/taken - PROTECTED: Get all taken doses for the user
|
||||||
|
// Suppress request logs — polled every 5s by frontend
|
||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
app.get("/doses/taken", { preHandler: requireAuth }, async (request, reply) => {
|
app.get("/doses/taken", { preHandler: requireAuth, logLevel: "warn" }, async (request, reply) => {
|
||||||
const userId = await getUserId(request, reply);
|
const userId = await getUserId(request, reply);
|
||||||
|
|
||||||
// Get all taken doses for this user (no time limit)
|
// Get all taken doses for this user (no time limit)
|
||||||
@@ -304,8 +305,9 @@ export async function doseRoutes(app: FastifyInstance) {
|
|||||||
|
|
||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
// GET /share/:token/doses - PUBLIC: Get taken doses for a share link
|
// GET /share/:token/doses - PUBLIC: Get taken doses for a share link
|
||||||
|
// Suppress request logs — polled every 5s by SharedSchedule
|
||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
app.get<{ Params: { token: string } }>("/share/:token/doses", async (request, reply) => {
|
app.get<{ Params: { token: string } }>("/share/:token/doses", { logLevel: "warn" }, async (request, reply) => {
|
||||||
const { token } = request.params;
|
const { token } = request.params;
|
||||||
|
|
||||||
const { share, reason } = await getActiveShareToken(token);
|
const { share, reason } = await getActiveShareToken(token);
|
||||||
|
|||||||
@@ -10,6 +10,7 @@ import { doseTracking, medications, refillHistory, shareTokens, userSettings } f
|
|||||||
import { getAnonymousUserId, requireAuth } from "../plugins/auth.js";
|
import { getAnonymousUserId, requireAuth } from "../plugins/auth.js";
|
||||||
import { env } from "../plugins/env.js";
|
import { env } from "../plugins/env.js";
|
||||||
import type { AuthUser } from "../types/fastify.js";
|
import type { AuthUser } from "../types/fastify.js";
|
||||||
|
import { normalizePackageType, PACKAGE_TYPES } from "../utils/package-profiles.js";
|
||||||
import { parseIntakesJson, parseTakenByJson } from "../utils/scheduler-utils.js";
|
import { parseIntakesJson, parseTakenByJson } from "../utils/scheduler-utils.js";
|
||||||
|
|
||||||
const IMAGES_DIR = resolve(getDataDir(), "images");
|
const IMAGES_DIR = resolve(getDataDir(), "images");
|
||||||
@@ -17,7 +18,7 @@ const IMAGES_DIR = resolve(getDataDir(), "images");
|
|||||||
// =============================================================================
|
// =============================================================================
|
||||||
// Export Format Version (bump this when format changes)
|
// Export Format Version (bump this when format changes)
|
||||||
// =============================================================================
|
// =============================================================================
|
||||||
const EXPORT_VERSION = "1.1";
|
const EXPORT_VERSION = "1.3";
|
||||||
|
|
||||||
// =============================================================================
|
// =============================================================================
|
||||||
// Zod Schemas for Import Validation
|
// Zod Schemas for Import Validation
|
||||||
@@ -27,6 +28,7 @@ const scheduleSchema = z.object({
|
|||||||
usage: z.number().nonnegative(),
|
usage: z.number().nonnegative(),
|
||||||
every: z.number().int().min(1),
|
every: z.number().int().min(1),
|
||||||
start: z.string(), // ISO datetime string
|
start: z.string(), // ISO datetime string
|
||||||
|
intakeUnit: z.enum(["ml", "tsp", "tbsp"]).nullable().optional(),
|
||||||
remind: z.boolean().optional().default(false),
|
remind: z.boolean().optional().default(false),
|
||||||
takenBy: z.string().nullable().optional(), // Per-intake takenBy (new field)
|
takenBy: z.string().nullable().optional(), // Per-intake takenBy (new field)
|
||||||
});
|
});
|
||||||
@@ -38,7 +40,9 @@ const inventorySchema = z.object({
|
|||||||
totalPills: z.number().int().nullable().optional(), // For bottle type: total capacity
|
totalPills: z.number().int().nullable().optional(), // For bottle type: total capacity
|
||||||
looseTablets: z.number().int().min(0).default(0),
|
looseTablets: z.number().int().min(0).default(0),
|
||||||
stockAdjustment: z.number().int().default(0), // Manual stock correction
|
stockAdjustment: z.number().int().default(0), // Manual stock correction
|
||||||
packageType: z.enum(["blister", "bottle"]).default("blister"),
|
packageType: z.enum(PACKAGE_TYPES).default("blister"),
|
||||||
|
packageAmountValue: z.number().int().min(0).default(0),
|
||||||
|
packageAmountUnit: z.enum(["ml", "g"]).default("ml"),
|
||||||
});
|
});
|
||||||
|
|
||||||
const medicationExportSchema = z.object({
|
const medicationExportSchema = z.object({
|
||||||
@@ -46,11 +50,16 @@ const medicationExportSchema = z.object({
|
|||||||
name: z.string().min(1),
|
name: z.string().min(1),
|
||||||
genericName: z.string().nullable().optional(),
|
genericName: z.string().nullable().optional(),
|
||||||
takenBy: z.array(z.string()).default([]),
|
takenBy: z.array(z.string()).default([]),
|
||||||
|
medicationForm: z.enum(["capsule", "tablet", "liquid", "topical"]).default("tablet"),
|
||||||
|
pillForm: z.enum(["capsule", "tablet"]).nullable().optional(),
|
||||||
|
lifecycleCategory: z.enum(["refill_when_empty", "treatment_period"]).default("refill_when_empty"),
|
||||||
inventory: inventorySchema,
|
inventory: inventorySchema,
|
||||||
pillWeightMg: z.number().int().nullable().optional(),
|
pillWeightMg: z.number().int().nullable().optional(),
|
||||||
doseUnit: z.enum(["mg", "g", "mcg", "ml", "IU", "units", "drops", "puffs"]).default("mg"),
|
doseUnit: z.enum(["mg", "g", "mcg", "ml", "IU", "units", "drops", "puffs"]).default("mg"),
|
||||||
schedules: z.array(scheduleSchema).default([]),
|
schedules: z.array(scheduleSchema).default([]),
|
||||||
medicationStartDate: z.string().nullable().optional(),
|
medicationStartDate: z.string().nullable().optional(),
|
||||||
|
medicationEndDate: z.string().nullable().optional(),
|
||||||
|
autoMarkObsoleteAfterEndDate: z.boolean().default(true),
|
||||||
expiryDate: z.string().nullable().optional(),
|
expiryDate: z.string().nullable().optional(),
|
||||||
notes: z.string().nullable().optional(),
|
notes: z.string().nullable().optional(),
|
||||||
intakeRemindersEnabled: z.boolean().default(false),
|
intakeRemindersEnabled: z.boolean().default(false),
|
||||||
@@ -155,9 +164,14 @@ async function getUserId(request: FastifyRequest, reply: FastifyReply): Promise<
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Parse intakes from DB format to export format (with per-intake takenBy)
|
// Parse intakes from DB format to export format (with per-intake takenBy)
|
||||||
function parseIntakesForExport(
|
function parseIntakesForExport(row: typeof medications.$inferSelect): Array<{
|
||||||
row: typeof medications.$inferSelect
|
usage: number;
|
||||||
): Array<{ usage: number; every: number; start: string; remind: boolean; takenBy: string | null }> {
|
every: number;
|
||||||
|
start: string;
|
||||||
|
intakeUnit: "ml" | "tsp" | "tbsp" | null;
|
||||||
|
remind: boolean;
|
||||||
|
takenBy: string | null;
|
||||||
|
}> {
|
||||||
// Use the new parseIntakesJson which falls back to legacy format
|
// Use the new parseIntakesJson which falls back to legacy format
|
||||||
const intakes = parseIntakesJson(
|
const intakes = parseIntakesJson(
|
||||||
row.intakesJson,
|
row.intakesJson,
|
||||||
@@ -169,6 +183,7 @@ function parseIntakesForExport(
|
|||||||
usage: intake.usage,
|
usage: intake.usage,
|
||||||
every: intake.every,
|
every: intake.every,
|
||||||
start: intake.start,
|
start: intake.start,
|
||||||
|
intakeUnit: null,
|
||||||
remind: intake.intakeRemindersEnabled,
|
remind: intake.intakeRemindersEnabled,
|
||||||
takenBy: intake.takenBy, // Per-intake takenBy
|
takenBy: intake.takenBy, // Per-intake takenBy
|
||||||
}));
|
}));
|
||||||
@@ -295,6 +310,9 @@ export async function exportRoutes(app: FastifyInstance) {
|
|||||||
name: med.name,
|
name: med.name,
|
||||||
genericName: med.genericName,
|
genericName: med.genericName,
|
||||||
takenBy: parseTakenByJson(med.takenByJson),
|
takenBy: parseTakenByJson(med.takenByJson),
|
||||||
|
medicationForm: med.medicationForm ?? "tablet",
|
||||||
|
pillForm: med.pillForm ?? null,
|
||||||
|
lifecycleCategory: med.lifecycleCategory ?? "refill_when_empty",
|
||||||
inventory: {
|
inventory: {
|
||||||
packCount: med.packCount ?? 1,
|
packCount: med.packCount ?? 1,
|
||||||
blistersPerPack: med.blistersPerPack ?? 1,
|
blistersPerPack: med.blistersPerPack ?? 1,
|
||||||
@@ -302,12 +320,16 @@ export async function exportRoutes(app: FastifyInstance) {
|
|||||||
totalPills: med.totalPills ?? null,
|
totalPills: med.totalPills ?? null,
|
||||||
looseTablets: med.looseTablets ?? 0,
|
looseTablets: med.looseTablets ?? 0,
|
||||||
stockAdjustment: med.stockAdjustment ?? 0,
|
stockAdjustment: med.stockAdjustment ?? 0,
|
||||||
packageType: med.packageType ?? "blister",
|
packageType: normalizePackageType(med.packageType),
|
||||||
|
packageAmountValue: med.packageAmountValue ?? 0,
|
||||||
|
packageAmountUnit: (med.packageAmountUnit ?? "ml") as "ml" | "g",
|
||||||
},
|
},
|
||||||
pillWeightMg: med.pillWeightMg,
|
pillWeightMg: med.pillWeightMg,
|
||||||
doseUnit: med.doseUnit ?? "mg",
|
doseUnit: med.doseUnit ?? "mg",
|
||||||
schedules: parseIntakesForExport(med),
|
schedules: parseIntakesForExport(med),
|
||||||
medicationStartDate: med.medicationStartDate || null,
|
medicationStartDate: med.medicationStartDate || null,
|
||||||
|
medicationEndDate: med.medicationEndDate || null,
|
||||||
|
autoMarkObsoleteAfterEndDate: med.autoMarkObsoleteAfterEndDate ?? true,
|
||||||
expiryDate: med.expiryDate,
|
expiryDate: med.expiryDate,
|
||||||
notes: med.notes,
|
notes: med.notes,
|
||||||
intakeRemindersEnabled: med.intakeRemindersEnabled ?? false,
|
intakeRemindersEnabled: med.intakeRemindersEnabled ?? false,
|
||||||
@@ -555,6 +577,7 @@ export async function exportRoutes(app: FastifyInstance) {
|
|||||||
usage: s.usage,
|
usage: s.usage,
|
||||||
every: s.every,
|
every: s.every,
|
||||||
start: s.start,
|
start: s.start,
|
||||||
|
intakeUnit: s.intakeUnit ?? null,
|
||||||
takenBy: s.takenBy || null,
|
takenBy: s.takenBy || null,
|
||||||
intakeRemindersEnabled: s.remind ?? false,
|
intakeRemindersEnabled: s.remind ?? false,
|
||||||
}))
|
}))
|
||||||
@@ -570,7 +593,12 @@ export async function exportRoutes(app: FastifyInstance) {
|
|||||||
name: med.name,
|
name: med.name,
|
||||||
genericName: med.genericName || null,
|
genericName: med.genericName || null,
|
||||||
takenByJson,
|
takenByJson,
|
||||||
packageType: med.inventory.packageType ?? "blister",
|
medicationForm: med.medicationForm ?? "tablet",
|
||||||
|
pillForm: med.pillForm || null,
|
||||||
|
lifecycleCategory: med.lifecycleCategory ?? "refill_when_empty",
|
||||||
|
packageType: normalizePackageType(med.inventory.packageType),
|
||||||
|
packageAmountValue: med.inventory.packageAmountValue ?? 0,
|
||||||
|
packageAmountUnit: med.inventory.packageAmountUnit ?? "ml",
|
||||||
packCount: med.inventory.packCount,
|
packCount: med.inventory.packCount,
|
||||||
blistersPerPack: med.inventory.blistersPerPack,
|
blistersPerPack: med.inventory.blistersPerPack,
|
||||||
pillsPerBlister: med.inventory.pillsPerBlister,
|
pillsPerBlister: med.inventory.pillsPerBlister,
|
||||||
@@ -581,6 +609,8 @@ export async function exportRoutes(app: FastifyInstance) {
|
|||||||
pillWeightMg: med.pillWeightMg || null,
|
pillWeightMg: med.pillWeightMg || null,
|
||||||
doseUnit: med.doseUnit ?? "mg",
|
doseUnit: med.doseUnit ?? "mg",
|
||||||
medicationStartDate: med.medicationStartDate || "",
|
medicationStartDate: med.medicationStartDate || "",
|
||||||
|
medicationEndDate: med.medicationEndDate || null,
|
||||||
|
autoMarkObsoleteAfterEndDate: med.autoMarkObsoleteAfterEndDate ?? true,
|
||||||
intakesJson,
|
intakesJson,
|
||||||
usageJson,
|
usageJson,
|
||||||
everyJson,
|
everyJson,
|
||||||
|
|||||||
@@ -10,11 +10,10 @@ const packageJson = JSON.parse(readFileSync(packageJsonPath, "utf-8"));
|
|||||||
const backendVersion = packageJson.version || "unknown";
|
const backendVersion = packageJson.version || "unknown";
|
||||||
|
|
||||||
export async function healthRoutes(app: FastifyInstance) {
|
export async function healthRoutes(app: FastifyInstance) {
|
||||||
// Exempt from rate limit - lightweight health check
|
// Exempt from rate limit + suppress request logs (called every 30s by Docker healthcheck)
|
||||||
app.get("/health", { config: { rateLimit: false } }, async () => ({
|
app.get("/health", { config: { rateLimit: false }, logLevel: "warn" }, async () => ({
|
||||||
status: "ok",
|
status: "ok",
|
||||||
version: backendVersion,
|
version: backendVersion,
|
||||||
smtpConfigured: Boolean(process.env.SMTP_HOST),
|
smtpConfigured: Boolean(process.env.SMTP_HOST),
|
||||||
shoutrrrConfigured: Boolean(process.env.SHOUTRRR_URL),
|
|
||||||
}));
|
}));
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -14,15 +14,63 @@ import {
|
|||||||
streamToBuffer,
|
streamToBuffer,
|
||||||
writeOptimizedImageSet,
|
writeOptimizedImageSet,
|
||||||
} from "../utils/image-upload.js";
|
} from "../utils/image-upload.js";
|
||||||
import { type Intake, parseIntakesJson, parseLocalDateTime, parseTakenByJson } from "../utils/scheduler-utils.js";
|
import {
|
||||||
|
isAmountBasedPackageType,
|
||||||
|
isLiquidContainerPackageType,
|
||||||
|
isTubePackageType,
|
||||||
|
normalizePackageType,
|
||||||
|
PACKAGE_TYPES,
|
||||||
|
} from "../utils/package-profiles.js";
|
||||||
|
import {
|
||||||
|
type Intake,
|
||||||
|
normalizeIntakeUsageForStock,
|
||||||
|
parseIntakesJson,
|
||||||
|
parseLocalDateTime,
|
||||||
|
parseTakenByJson,
|
||||||
|
} from "../utils/scheduler-utils.js";
|
||||||
|
|
||||||
const IMAGES_DIR = resolve(getDataDir(), "images");
|
const IMAGES_DIR = resolve(getDataDir(), "images");
|
||||||
|
|
||||||
|
function isIntakeUnit(value: unknown): value is "ml" | "tsp" | "tbsp" {
|
||||||
|
return value === "ml" || value === "tsp" || value === "tbsp";
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseRawIntakeUnits(intakesJson: string | null | undefined): Array<"ml" | "tsp" | "tbsp" | null> {
|
||||||
|
if (!intakesJson) return [];
|
||||||
|
try {
|
||||||
|
const parsed = JSON.parse(intakesJson);
|
||||||
|
if (!Array.isArray(parsed)) return [];
|
||||||
|
return parsed.map((item: unknown) => {
|
||||||
|
if (!item || typeof item !== "object") return null;
|
||||||
|
const unit = (item as Record<string, unknown>).intakeUnit;
|
||||||
|
return isIntakeUnit(unit) ? unit : null;
|
||||||
|
});
|
||||||
|
} catch {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseIntakesWithUnits(
|
||||||
|
intakesJson: string | null | undefined,
|
||||||
|
legacyRow: { usageJson: string; everyJson: string; startJson: string },
|
||||||
|
medicationIntakeRemindersEnabled?: boolean
|
||||||
|
): Intake[] {
|
||||||
|
const intakes = parseIntakesJson(intakesJson, legacyRow, medicationIntakeRemindersEnabled);
|
||||||
|
const rawUnits = parseRawIntakeUnits(intakesJson);
|
||||||
|
if (rawUnits.length === 0) return intakes;
|
||||||
|
|
||||||
|
return intakes.map((intake, idx) => ({
|
||||||
|
...intake,
|
||||||
|
intakeUnit: rawUnits[idx] ?? intake.intakeUnit ?? null,
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
// New intake schema with per-intake takenBy
|
// New intake schema with per-intake takenBy
|
||||||
const intakeSchema = z.object({
|
const intakeSchema = z.object({
|
||||||
usage: z.number().nonnegative(),
|
usage: z.number().nonnegative(),
|
||||||
every: z.number().int().min(1),
|
every: z.number().int().min(1),
|
||||||
start: z.string().datetime({ local: true }),
|
start: z.string().datetime({ local: true }),
|
||||||
|
intakeUnit: z.enum(["ml", "tsp", "tbsp"]).nullable().optional(),
|
||||||
takenBy: z.string().trim().max(100).nullable().optional(), // Person for this specific intake
|
takenBy: z.string().trim().max(100).nullable().optional(), // Person for this specific intake
|
||||||
intakeRemindersEnabled: z.boolean().default(false), // Per-intake reminder setting
|
intakeRemindersEnabled: z.boolean().default(false), // Per-intake reminder setting
|
||||||
});
|
});
|
||||||
@@ -34,26 +82,37 @@ const blisterSchema = z.object({
|
|||||||
start: z.string().datetime({ local: true }),
|
start: z.string().datetime({ local: true }),
|
||||||
});
|
});
|
||||||
|
|
||||||
const packageTypeSchema = z.enum(["blister", "bottle"]).default("blister");
|
const packageTypeSchema = z.enum(PACKAGE_TYPES).default("blister");
|
||||||
|
const medicationFormSchema = z.enum(["capsule", "tablet", "liquid", "topical"]).default("tablet");
|
||||||
|
const pillFormSchema = z.enum(["capsule", "tablet"]);
|
||||||
|
const lifecycleCategorySchema = z.enum(["refill_when_empty", "treatment_period"]).default("refill_when_empty");
|
||||||
const doseUnitSchema = z.enum(["mg", "g", "mcg", "ml", "IU", "units", "drops", "puffs"]).default("mg");
|
const doseUnitSchema = z.enum(["mg", "g", "mcg", "ml", "IU", "units", "drops", "puffs"]).default("mg");
|
||||||
const medicationStartDateSchema = z
|
const medicationStartDateSchema = z
|
||||||
.union([z.string().regex(/^\d{4}-\d{2}-\d{2}$/), z.literal(""), z.null()])
|
.union([z.string().regex(/^\d{4}-\d{2}-\d{2}$/), z.literal(""), z.null()])
|
||||||
.optional();
|
.optional();
|
||||||
|
const medicationEndDateSchema = z.union([z.string().regex(/^\d{4}-\d{2}-\d{2}$/), z.literal(""), z.null()]).optional();
|
||||||
|
|
||||||
const medicationSchema = z
|
const medicationSchema = z
|
||||||
.object({
|
.object({
|
||||||
name: z.string().trim().min(1).max(100),
|
name: z.string().trim().max(100).default(""),
|
||||||
genericName: z.string().trim().max(100).nullable().optional(),
|
genericName: z.string().trim().max(100).nullable().optional(),
|
||||||
takenBy: z.array(z.string().trim().max(100)).default([]), // Medication-level takenBy (fallback)
|
takenBy: z.array(z.string().trim().max(100)).default([]), // Medication-level takenBy (fallback)
|
||||||
|
medicationForm: medicationFormSchema,
|
||||||
|
pillForm: pillFormSchema.nullable().optional(),
|
||||||
|
lifecycleCategory: lifecycleCategorySchema,
|
||||||
packageType: packageTypeSchema,
|
packageType: packageTypeSchema,
|
||||||
packCount: z.number().int().min(0).default(1),
|
packCount: z.number().int().min(0).default(1),
|
||||||
blistersPerPack: z.number().int().min(1).default(1),
|
blistersPerPack: z.number().int().min(1).default(1),
|
||||||
pillsPerBlister: z.number().int().min(1).default(1),
|
pillsPerBlister: z.number().int().min(1).default(1),
|
||||||
|
packageAmountValue: z.number().int().min(0).default(0),
|
||||||
|
packageAmountUnit: z.enum(["ml", "g"]).default("ml"),
|
||||||
totalPills: z.number().int().min(1).nullable().optional(), // For bottle type: total capacity
|
totalPills: z.number().int().min(1).nullable().optional(), // For bottle type: total capacity
|
||||||
looseTablets: z.number().int().min(0).default(0),
|
looseTablets: z.number().int().min(0).default(0),
|
||||||
pillWeightMg: z.number().nonnegative().nullable().optional(),
|
pillWeightMg: z.number().nonnegative().nullable().optional(),
|
||||||
doseUnit: doseUnitSchema,
|
doseUnit: doseUnitSchema,
|
||||||
medicationStartDate: medicationStartDateSchema,
|
medicationStartDate: medicationStartDateSchema,
|
||||||
|
medicationEndDate: medicationEndDateSchema,
|
||||||
|
autoMarkObsoleteAfterEndDate: z.boolean().default(true),
|
||||||
expiryDate: z.string().nullable().optional(),
|
expiryDate: z.string().nullable().optional(),
|
||||||
notes: z.string().max(2000).nullable().optional(),
|
notes: z.string().max(2000).nullable().optional(),
|
||||||
prescriptionEnabled: z.boolean().default(false),
|
prescriptionEnabled: z.boolean().default(false),
|
||||||
@@ -66,6 +125,10 @@ const medicationSchema = z
|
|||||||
intakes: z.array(intakeSchema).min(1).max(12).optional(),
|
intakes: z.array(intakeSchema).min(1).max(12).optional(),
|
||||||
blisters: z.array(blisterSchema).min(1).max(12).optional(), // Legacy format
|
blisters: z.array(blisterSchema).min(1).max(12).optional(), // Legacy format
|
||||||
})
|
})
|
||||||
|
.refine((data) => (data.name && data.name.length > 0) || (data.genericName && data.genericName.length > 0), {
|
||||||
|
message: "Either 'name' or 'genericName' must be provided",
|
||||||
|
path: ["name"],
|
||||||
|
})
|
||||||
.refine((data) => data.intakes || data.blisters, { message: "Either 'intakes' or 'blisters' must be provided" })
|
.refine((data) => data.intakes || data.blisters, { message: "Either 'intakes' or 'blisters' must be provided" })
|
||||||
.refine(
|
.refine(
|
||||||
(data) => {
|
(data) => {
|
||||||
@@ -80,6 +143,77 @@ const medicationSchema = z
|
|||||||
path: ["medicationStartDate"],
|
path: ["medicationStartDate"],
|
||||||
}
|
}
|
||||||
)
|
)
|
||||||
|
.refine(
|
||||||
|
(data) => {
|
||||||
|
const startDate = data.medicationStartDate ?? "";
|
||||||
|
const endDate = data.medicationEndDate ?? "";
|
||||||
|
if (!startDate || !endDate) return true;
|
||||||
|
return startDate <= endDate;
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "Medication end date must be on or after medication start date",
|
||||||
|
path: ["medicationEndDate"],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.refine(
|
||||||
|
(data) => {
|
||||||
|
if (data.medicationForm === "capsule" || data.medicationForm === "tablet") {
|
||||||
|
return data.pillForm == null || data.pillForm === "capsule" || data.pillForm === "tablet";
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "pillForm must be capsule or tablet for capsule/tablet medications",
|
||||||
|
path: ["pillForm"],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.refine(
|
||||||
|
(data) => {
|
||||||
|
if (data.medicationForm === "topical") {
|
||||||
|
return isTubePackageType(data.packageType);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "Topical medications must use tube package type",
|
||||||
|
path: ["packageType"],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.refine(
|
||||||
|
(data) => {
|
||||||
|
if (data.medicationForm === "liquid") {
|
||||||
|
return isLiquidContainerPackageType(data.packageType);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "Liquid medications must use liquid_container package type",
|
||||||
|
path: ["packageType"],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.refine(
|
||||||
|
(data) => {
|
||||||
|
if (data.medicationForm === "capsule" || data.medicationForm === "tablet") {
|
||||||
|
return !isTubePackageType(data.packageType) && !isLiquidContainerPackageType(data.packageType);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "Capsule and tablet medications cannot use tube or liquid_container package type",
|
||||||
|
path: ["packageType"],
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.refine(
|
||||||
|
(data) => {
|
||||||
|
const schedules = data.intakes ?? data.blisters ?? [];
|
||||||
|
if (data.pillForm !== "capsule") return true;
|
||||||
|
return schedules.every((entry) => Number.isInteger(entry.usage));
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "Fractional intake is not allowed for capsule",
|
||||||
|
path: ["intakes"],
|
||||||
|
}
|
||||||
|
)
|
||||||
.refine(
|
.refine(
|
||||||
(data) => {
|
(data) => {
|
||||||
if (!data.prescriptionEnabled) return true;
|
if (!data.prescriptionEnabled) return true;
|
||||||
@@ -127,13 +261,33 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
app.get<{ Querystring: { includeObsolete?: string } }>("/medications", async (request, reply) => {
|
app.get<{ Querystring: { includeObsolete?: string } }>("/medications", async (request, reply) => {
|
||||||
const userId = await getUserId(request, reply);
|
const userId = await getUserId(request, reply);
|
||||||
const includeObsolete = request.query.includeObsolete === "true";
|
const includeObsolete = request.query.includeObsolete === "true";
|
||||||
|
const initialRows = await db
|
||||||
|
.select()
|
||||||
|
.from(medications)
|
||||||
|
.where(eq(medications.userId, userId))
|
||||||
|
.orderBy(medications.id);
|
||||||
|
const todayDate = new Date().toISOString().slice(0, 10);
|
||||||
|
|
||||||
|
for (const row of initialRows) {
|
||||||
|
if (row.isObsolete) continue;
|
||||||
|
if (!(row.autoMarkObsoleteAfterEndDate ?? true)) continue;
|
||||||
|
const endDate = row.medicationEndDate?.slice(0, 10);
|
||||||
|
if (!endDate) continue;
|
||||||
|
if (endDate > todayDate) continue;
|
||||||
|
|
||||||
|
await db
|
||||||
|
.update(medications)
|
||||||
|
.set({ isObsolete: true, obsoleteAt: new Date(), updatedAt: new Date() })
|
||||||
|
.where(and(eq(medications.id, row.id), eq(medications.userId, userId)));
|
||||||
|
}
|
||||||
|
|
||||||
const whereClause = includeObsolete
|
const whereClause = includeObsolete
|
||||||
? eq(medications.userId, userId)
|
? eq(medications.userId, userId)
|
||||||
: and(eq(medications.userId, userId), eq(medications.isObsolete, false));
|
: and(eq(medications.userId, userId), eq(medications.isObsolete, false));
|
||||||
const rows = await db.select().from(medications).where(whereClause).orderBy(medications.id);
|
const rows = await db.select().from(medications).where(whereClause).orderBy(medications.id);
|
||||||
return rows.map((row) => {
|
return rows.map((row) => {
|
||||||
// Parse intakes from new format, falling back to legacy
|
// Parse intakes from new format, falling back to legacy
|
||||||
const intakes = parseIntakesJson(
|
const intakes = parseIntakesWithUnits(
|
||||||
row.intakesJson,
|
row.intakesJson,
|
||||||
{ usageJson: row.usageJson, everyJson: row.everyJson, startJson: row.startJson },
|
{ usageJson: row.usageJson, everyJson: row.everyJson, startJson: row.startJson },
|
||||||
row.intakeRemindersEnabled ?? false
|
row.intakeRemindersEnabled ?? false
|
||||||
@@ -144,10 +298,15 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
name: row.name,
|
name: row.name,
|
||||||
genericName: row.genericName,
|
genericName: row.genericName,
|
||||||
takenBy: parseTakenByJson(row.takenByJson),
|
takenBy: parseTakenByJson(row.takenByJson),
|
||||||
packageType: row.packageType ?? "blister",
|
medicationForm: row.medicationForm ?? "tablet",
|
||||||
|
pillForm: row.pillForm ?? null,
|
||||||
|
lifecycleCategory: row.lifecycleCategory ?? "refill_when_empty",
|
||||||
|
packageType: normalizePackageType(row.packageType),
|
||||||
packCount: row.packCount ?? 1,
|
packCount: row.packCount ?? 1,
|
||||||
blistersPerPack: row.blistersPerPack ?? 1,
|
blistersPerPack: row.blistersPerPack ?? 1,
|
||||||
pillsPerBlister: row.pillsPerBlister ?? 1,
|
pillsPerBlister: row.pillsPerBlister ?? 1,
|
||||||
|
packageAmountValue: row.packageAmountValue ?? 0,
|
||||||
|
packageAmountUnit: (row.packageAmountUnit ?? "ml") as "ml" | "g",
|
||||||
totalPills: row.totalPills ?? null,
|
totalPills: row.totalPills ?? null,
|
||||||
looseTablets: row.looseTablets ?? 0,
|
looseTablets: row.looseTablets ?? 0,
|
||||||
stockAdjustment: row.stockAdjustment ?? 0,
|
stockAdjustment: row.stockAdjustment ?? 0,
|
||||||
@@ -155,6 +314,8 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
pillWeightMg: row.pillWeightMg,
|
pillWeightMg: row.pillWeightMg,
|
||||||
doseUnit: row.doseUnit ?? "mg",
|
doseUnit: row.doseUnit ?? "mg",
|
||||||
medicationStartDate: row.medicationStartDate || null,
|
medicationStartDate: row.medicationStartDate || null,
|
||||||
|
medicationEndDate: row.medicationEndDate || null,
|
||||||
|
autoMarkObsoleteAfterEndDate: row.autoMarkObsoleteAfterEndDate ?? true,
|
||||||
intakes, // New unified format with per-intake takenBy
|
intakes, // New unified format with per-intake takenBy
|
||||||
// Legacy blisters format (for backward compat with frontend during transition)
|
// Legacy blisters format (for backward compat with frontend during transition)
|
||||||
blisters: intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })),
|
blisters: intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })),
|
||||||
@@ -184,15 +345,22 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
name,
|
name,
|
||||||
genericName,
|
genericName,
|
||||||
takenBy,
|
takenBy,
|
||||||
|
medicationForm,
|
||||||
|
pillForm,
|
||||||
|
lifecycleCategory,
|
||||||
packageType,
|
packageType,
|
||||||
packCount,
|
packCount,
|
||||||
blistersPerPack,
|
blistersPerPack,
|
||||||
pillsPerBlister,
|
pillsPerBlister,
|
||||||
|
packageAmountValue,
|
||||||
|
packageAmountUnit,
|
||||||
totalPills,
|
totalPills,
|
||||||
looseTablets,
|
looseTablets,
|
||||||
pillWeightMg,
|
pillWeightMg,
|
||||||
doseUnit,
|
doseUnit,
|
||||||
medicationStartDate,
|
medicationStartDate,
|
||||||
|
medicationEndDate,
|
||||||
|
autoMarkObsoleteAfterEndDate,
|
||||||
expiryDate,
|
expiryDate,
|
||||||
notes,
|
notes,
|
||||||
prescriptionEnabled,
|
prescriptionEnabled,
|
||||||
@@ -205,6 +373,9 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
blisters: inputBlisters,
|
blisters: inputBlisters,
|
||||||
} = parsed.data;
|
} = parsed.data;
|
||||||
|
|
||||||
|
const normalizedPillForm =
|
||||||
|
medicationForm === "capsule" || medicationForm === "tablet" ? (pillForm ?? medicationForm) : null;
|
||||||
|
|
||||||
// Convert to unified intakes format
|
// Convert to unified intakes format
|
||||||
let intakes: Intake[];
|
let intakes: Intake[];
|
||||||
if (inputIntakes) {
|
if (inputIntakes) {
|
||||||
@@ -213,6 +384,7 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
usage: i.usage,
|
usage: i.usage,
|
||||||
every: i.every,
|
every: i.every,
|
||||||
start: i.start,
|
start: i.start,
|
||||||
|
intakeUnit: i.intakeUnit ?? null,
|
||||||
takenBy: i.takenBy || null,
|
takenBy: i.takenBy || null,
|
||||||
intakeRemindersEnabled: i.intakeRemindersEnabled ?? false,
|
intakeRemindersEnabled: i.intakeRemindersEnabled ?? false,
|
||||||
}));
|
}));
|
||||||
@@ -222,6 +394,7 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
usage: b.usage,
|
usage: b.usage,
|
||||||
every: b.every,
|
every: b.every,
|
||||||
start: b.start,
|
start: b.start,
|
||||||
|
intakeUnit: null,
|
||||||
takenBy: null, // No per-intake takenBy from legacy
|
takenBy: null, // No per-intake takenBy from legacy
|
||||||
intakeRemindersEnabled: intakeRemindersEnabled ?? false,
|
intakeRemindersEnabled: intakeRemindersEnabled ?? false,
|
||||||
}));
|
}));
|
||||||
@@ -243,15 +416,22 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
name,
|
name,
|
||||||
genericName: genericName || null,
|
genericName: genericName || null,
|
||||||
takenByJson,
|
takenByJson,
|
||||||
packageType: packageType ?? "blister",
|
medicationForm: medicationForm ?? "tablet",
|
||||||
|
pillForm: normalizedPillForm,
|
||||||
|
lifecycleCategory: lifecycleCategory ?? "refill_when_empty",
|
||||||
|
packageType: normalizePackageType(packageType),
|
||||||
packCount,
|
packCount,
|
||||||
blistersPerPack,
|
blistersPerPack,
|
||||||
pillsPerBlister,
|
pillsPerBlister,
|
||||||
|
packageAmountValue,
|
||||||
|
packageAmountUnit,
|
||||||
totalPills: totalPills || null,
|
totalPills: totalPills || null,
|
||||||
looseTablets,
|
looseTablets,
|
||||||
pillWeightMg: pillWeightMg || null,
|
pillWeightMg: pillWeightMg || null,
|
||||||
doseUnit: doseUnit ?? "mg",
|
doseUnit: doseUnit ?? "mg",
|
||||||
medicationStartDate: medicationStartDate ?? "",
|
medicationStartDate: medicationStartDate ?? "",
|
||||||
|
medicationEndDate: medicationEndDate || null,
|
||||||
|
autoMarkObsoleteAfterEndDate: autoMarkObsoleteAfterEndDate ?? true,
|
||||||
expiryDate: expiryDate || null,
|
expiryDate: expiryDate || null,
|
||||||
notes: notes || null,
|
notes: notes || null,
|
||||||
prescriptionEnabled: prescriptionEnabled ?? false,
|
prescriptionEnabled: prescriptionEnabled ?? false,
|
||||||
@@ -272,10 +452,15 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
name: inserted.name,
|
name: inserted.name,
|
||||||
genericName: inserted.genericName,
|
genericName: inserted.genericName,
|
||||||
takenBy: parseTakenByJson(inserted.takenByJson),
|
takenBy: parseTakenByJson(inserted.takenByJson),
|
||||||
packageType: inserted.packageType ?? "blister",
|
medicationForm: inserted.medicationForm ?? "tablet",
|
||||||
|
pillForm: inserted.pillForm ?? null,
|
||||||
|
lifecycleCategory: inserted.lifecycleCategory ?? "refill_when_empty",
|
||||||
|
packageType: normalizePackageType(inserted.packageType),
|
||||||
packCount: inserted.packCount,
|
packCount: inserted.packCount,
|
||||||
blistersPerPack: inserted.blistersPerPack,
|
blistersPerPack: inserted.blistersPerPack,
|
||||||
pillsPerBlister: inserted.pillsPerBlister,
|
pillsPerBlister: inserted.pillsPerBlister,
|
||||||
|
packageAmountValue: inserted.packageAmountValue ?? 0,
|
||||||
|
packageAmountUnit: (inserted.packageAmountUnit ?? "ml") as "ml" | "g",
|
||||||
totalPills: inserted.totalPills ?? null,
|
totalPills: inserted.totalPills ?? null,
|
||||||
looseTablets: inserted.looseTablets,
|
looseTablets: inserted.looseTablets,
|
||||||
stockAdjustment: inserted.stockAdjustment ?? 0,
|
stockAdjustment: inserted.stockAdjustment ?? 0,
|
||||||
@@ -283,6 +468,8 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
pillWeightMg: inserted.pillWeightMg,
|
pillWeightMg: inserted.pillWeightMg,
|
||||||
doseUnit: inserted.doseUnit ?? "mg",
|
doseUnit: inserted.doseUnit ?? "mg",
|
||||||
medicationStartDate: inserted.medicationStartDate || null,
|
medicationStartDate: inserted.medicationStartDate || null,
|
||||||
|
medicationEndDate: inserted.medicationEndDate || null,
|
||||||
|
autoMarkObsoleteAfterEndDate: inserted.autoMarkObsoleteAfterEndDate ?? true,
|
||||||
intakes,
|
intakes,
|
||||||
blisters: intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })),
|
blisters: intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })),
|
||||||
imageUrl: inserted.imageUrl,
|
imageUrl: inserted.imageUrl,
|
||||||
@@ -319,15 +506,22 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
name,
|
name,
|
||||||
genericName,
|
genericName,
|
||||||
takenBy,
|
takenBy,
|
||||||
|
medicationForm,
|
||||||
|
pillForm,
|
||||||
|
lifecycleCategory,
|
||||||
packageType,
|
packageType,
|
||||||
packCount,
|
packCount,
|
||||||
blistersPerPack,
|
blistersPerPack,
|
||||||
pillsPerBlister,
|
pillsPerBlister,
|
||||||
|
packageAmountValue,
|
||||||
|
packageAmountUnit,
|
||||||
totalPills,
|
totalPills,
|
||||||
looseTablets,
|
looseTablets,
|
||||||
pillWeightMg,
|
pillWeightMg,
|
||||||
doseUnit,
|
doseUnit,
|
||||||
medicationStartDate,
|
medicationStartDate,
|
||||||
|
medicationEndDate,
|
||||||
|
autoMarkObsoleteAfterEndDate,
|
||||||
expiryDate,
|
expiryDate,
|
||||||
notes,
|
notes,
|
||||||
prescriptionEnabled,
|
prescriptionEnabled,
|
||||||
@@ -340,6 +534,9 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
blisters: inputBlisters,
|
blisters: inputBlisters,
|
||||||
} = parsed.data;
|
} = parsed.data;
|
||||||
|
|
||||||
|
const normalizedPillForm =
|
||||||
|
medicationForm === "capsule" || medicationForm === "tablet" ? (pillForm ?? medicationForm) : null;
|
||||||
|
|
||||||
// Convert to unified intakes format
|
// Convert to unified intakes format
|
||||||
let intakes: Intake[];
|
let intakes: Intake[];
|
||||||
if (inputIntakes) {
|
if (inputIntakes) {
|
||||||
@@ -348,6 +545,7 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
usage: i.usage,
|
usage: i.usage,
|
||||||
every: i.every,
|
every: i.every,
|
||||||
start: i.start,
|
start: i.start,
|
||||||
|
intakeUnit: i.intakeUnit ?? null,
|
||||||
takenBy: i.takenBy || null,
|
takenBy: i.takenBy || null,
|
||||||
intakeRemindersEnabled: i.intakeRemindersEnabled ?? false,
|
intakeRemindersEnabled: i.intakeRemindersEnabled ?? false,
|
||||||
}));
|
}));
|
||||||
@@ -357,6 +555,7 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
usage: b.usage,
|
usage: b.usage,
|
||||||
every: b.every,
|
every: b.every,
|
||||||
start: b.start,
|
start: b.start,
|
||||||
|
intakeUnit: null,
|
||||||
takenBy: null, // No per-intake takenBy from legacy
|
takenBy: null, // No per-intake takenBy from legacy
|
||||||
intakeRemindersEnabled: intakeRemindersEnabled ?? false,
|
intakeRemindersEnabled: intakeRemindersEnabled ?? false,
|
||||||
}));
|
}));
|
||||||
@@ -388,15 +587,22 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
name,
|
name,
|
||||||
genericName: genericName || null,
|
genericName: genericName || null,
|
||||||
takenByJson,
|
takenByJson,
|
||||||
packageType: packageType ?? "blister",
|
medicationForm: medicationForm ?? "tablet",
|
||||||
|
pillForm: normalizedPillForm,
|
||||||
|
lifecycleCategory: lifecycleCategory ?? "refill_when_empty",
|
||||||
|
packageType: normalizePackageType(packageType),
|
||||||
packCount,
|
packCount,
|
||||||
blistersPerPack,
|
blistersPerPack,
|
||||||
pillsPerBlister,
|
pillsPerBlister,
|
||||||
totalPills: totalPills || null,
|
totalPills: totalPills || null,
|
||||||
|
packageAmountValue,
|
||||||
|
packageAmountUnit,
|
||||||
looseTablets,
|
looseTablets,
|
||||||
pillWeightMg: pillWeightMg || null,
|
pillWeightMg: pillWeightMg || null,
|
||||||
doseUnit: doseUnit ?? "mg",
|
doseUnit: doseUnit ?? "mg",
|
||||||
medicationStartDate: medicationStartDate ?? "",
|
medicationStartDate: medicationStartDate ?? "",
|
||||||
|
medicationEndDate: medicationEndDate || null,
|
||||||
|
autoMarkObsoleteAfterEndDate: autoMarkObsoleteAfterEndDate ?? true,
|
||||||
expiryDate: expiryDate || null,
|
expiryDate: expiryDate || null,
|
||||||
notes: notes || null,
|
notes: notes || null,
|
||||||
prescriptionEnabled: prescriptionEnabled ?? false,
|
prescriptionEnabled: prescriptionEnabled ?? false,
|
||||||
@@ -421,7 +627,7 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
// Migrate dose tracking IDs when intake schedule changes
|
// Migrate dose tracking IDs when intake schedule changes
|
||||||
// ---------------------------------------------------------------
|
// ---------------------------------------------------------------
|
||||||
// Parse old intakes from the existing medication row
|
// Parse old intakes from the existing medication row
|
||||||
const oldIntakes = parseIntakesJson(
|
const oldIntakes = parseIntakesWithUnits(
|
||||||
existing.intakesJson,
|
existing.intakesJson,
|
||||||
{ usageJson: existing.usageJson, everyJson: existing.everyJson, startJson: existing.startJson },
|
{ usageJson: existing.usageJson, everyJson: existing.everyJson, startJson: existing.startJson },
|
||||||
existing.intakeRemindersEnabled
|
existing.intakeRemindersEnabled
|
||||||
@@ -541,10 +747,15 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
name: result[0].name,
|
name: result[0].name,
|
||||||
genericName: result[0].genericName,
|
genericName: result[0].genericName,
|
||||||
takenBy: parseTakenByJson(result[0].takenByJson),
|
takenBy: parseTakenByJson(result[0].takenByJson),
|
||||||
packageType: result[0].packageType ?? "blister",
|
medicationForm: result[0].medicationForm ?? "tablet",
|
||||||
|
pillForm: result[0].pillForm ?? null,
|
||||||
|
lifecycleCategory: result[0].lifecycleCategory ?? "refill_when_empty",
|
||||||
|
packageType: normalizePackageType(result[0].packageType),
|
||||||
packCount: result[0].packCount,
|
packCount: result[0].packCount,
|
||||||
blistersPerPack: result[0].blistersPerPack,
|
blistersPerPack: result[0].blistersPerPack,
|
||||||
pillsPerBlister: result[0].pillsPerBlister,
|
pillsPerBlister: result[0].pillsPerBlister,
|
||||||
|
packageAmountValue: result[0].packageAmountValue ?? 0,
|
||||||
|
packageAmountUnit: (result[0].packageAmountUnit ?? "ml") as "ml" | "g",
|
||||||
totalPills: result[0].totalPills ?? null,
|
totalPills: result[0].totalPills ?? null,
|
||||||
looseTablets: result[0].looseTablets,
|
looseTablets: result[0].looseTablets,
|
||||||
stockAdjustment: result[0].stockAdjustment ?? 0,
|
stockAdjustment: result[0].stockAdjustment ?? 0,
|
||||||
@@ -552,6 +763,8 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
pillWeightMg: result[0].pillWeightMg,
|
pillWeightMg: result[0].pillWeightMg,
|
||||||
doseUnit: result[0].doseUnit ?? "mg",
|
doseUnit: result[0].doseUnit ?? "mg",
|
||||||
medicationStartDate: result[0].medicationStartDate || null,
|
medicationStartDate: result[0].medicationStartDate || null,
|
||||||
|
medicationEndDate: result[0].medicationEndDate || null,
|
||||||
|
autoMarkObsoleteAfterEndDate: result[0].autoMarkObsoleteAfterEndDate ?? true,
|
||||||
intakes,
|
intakes,
|
||||||
blisters: intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })),
|
blisters: intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start })),
|
||||||
imageUrl: result[0].imageUrl,
|
imageUrl: result[0].imageUrl,
|
||||||
@@ -627,62 +840,101 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
|
|
||||||
// Stock correction endpoint - updates stockAdjustment and optionally looseTablets (for blister type)
|
// Stock correction endpoint - updates stockAdjustment and optionally base amount fields for amount-based corrections
|
||||||
// Also sets lastStockCorrectionAt so consumed doses before this point don't count
|
// Also sets lastStockCorrectionAt so consumed doses before this point don't count
|
||||||
app.patch<{ Params: { id: string }; Body: { stockAdjustment: number; looseTablets?: number } }>(
|
app.patch<{
|
||||||
"/medications/:id/stock-adjustment",
|
Params: { id: string };
|
||||||
async (req, reply) => {
|
Body: {
|
||||||
const idNum = Number(req.params.id);
|
stockAdjustment: number;
|
||||||
if (Number.isNaN(idNum)) return reply.badRequest("Invalid id");
|
looseTablets?: number;
|
||||||
|
totalPills?: number;
|
||||||
|
packageAmountValue?: number;
|
||||||
|
packCount?: number;
|
||||||
|
};
|
||||||
|
}>("/medications/:id/stock-adjustment", async (req, reply) => {
|
||||||
|
const idNum = Number(req.params.id);
|
||||||
|
if (Number.isNaN(idNum)) return reply.badRequest("Invalid id");
|
||||||
|
|
||||||
const userId = await getUserId(req, reply);
|
const userId = await getUserId(req, reply);
|
||||||
|
|
||||||
// Verify ownership
|
// Verify ownership
|
||||||
const [existing] = await db
|
const [existing] = await db
|
||||||
.select()
|
.select()
|
||||||
.from(medications)
|
.from(medications)
|
||||||
.where(and(eq(medications.id, idNum), eq(medications.userId, userId)));
|
.where(and(eq(medications.id, idNum), eq(medications.userId, userId)));
|
||||||
if (!existing) return reply.notFound();
|
if (!existing) return reply.notFound();
|
||||||
|
|
||||||
const { stockAdjustment, looseTablets } = req.body as { stockAdjustment: number; looseTablets?: number };
|
const { stockAdjustment, looseTablets, totalPills, packageAmountValue, packCount } = req.body as {
|
||||||
if (typeof stockAdjustment !== "number") return reply.badRequest("stockAdjustment must be a number");
|
stockAdjustment: number;
|
||||||
if (
|
looseTablets?: number;
|
||||||
looseTablets !== undefined &&
|
totalPills?: number;
|
||||||
(typeof looseTablets !== "number" || !Number.isInteger(looseTablets) || looseTablets < 0)
|
packageAmountValue?: number;
|
||||||
) {
|
packCount?: number;
|
||||||
return reply.badRequest("looseTablets must be a non-negative integer");
|
};
|
||||||
}
|
if (typeof stockAdjustment !== "number") return reply.badRequest("stockAdjustment must be a number");
|
||||||
|
if (
|
||||||
const updateFields: {
|
looseTablets !== undefined &&
|
||||||
stockAdjustment: number;
|
(typeof looseTablets !== "number" || !Number.isInteger(looseTablets) || looseTablets < 0)
|
||||||
lastStockCorrectionAt: Date;
|
) {
|
||||||
updatedAt: Date;
|
return reply.badRequest("looseTablets must be a non-negative integer");
|
||||||
looseTablets?: number;
|
|
||||||
} = {
|
|
||||||
stockAdjustment,
|
|
||||||
lastStockCorrectionAt: new Date(),
|
|
||||||
updatedAt: new Date(),
|
|
||||||
};
|
|
||||||
if (looseTablets !== undefined) {
|
|
||||||
updateFields.looseTablets = looseTablets;
|
|
||||||
}
|
|
||||||
|
|
||||||
const result = await db
|
|
||||||
.update(medications)
|
|
||||||
.set(updateFields)
|
|
||||||
.where(and(eq(medications.id, idNum), eq(medications.userId, userId)))
|
|
||||||
.returning();
|
|
||||||
|
|
||||||
if (!result.length) return reply.notFound();
|
|
||||||
|
|
||||||
return {
|
|
||||||
id: result[0].id,
|
|
||||||
stockAdjustment: result[0].stockAdjustment ?? 0,
|
|
||||||
lastStockCorrectionAt: result[0].lastStockCorrectionAt?.toISOString() ?? null,
|
|
||||||
updatedAt: result[0].updatedAt,
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
);
|
if (
|
||||||
|
totalPills !== undefined &&
|
||||||
|
(typeof totalPills !== "number" || !Number.isInteger(totalPills) || totalPills < 0)
|
||||||
|
) {
|
||||||
|
return reply.badRequest("totalPills must be a non-negative integer");
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
packageAmountValue !== undefined &&
|
||||||
|
(typeof packageAmountValue !== "number" || !Number.isInteger(packageAmountValue) || packageAmountValue < 0)
|
||||||
|
) {
|
||||||
|
return reply.badRequest("packageAmountValue must be a non-negative integer");
|
||||||
|
}
|
||||||
|
if (packCount !== undefined && (typeof packCount !== "number" || !Number.isInteger(packCount) || packCount < 1)) {
|
||||||
|
return reply.badRequest("packCount must be an integer >= 1");
|
||||||
|
}
|
||||||
|
|
||||||
|
const updateFields: {
|
||||||
|
stockAdjustment: number;
|
||||||
|
lastStockCorrectionAt: Date;
|
||||||
|
updatedAt: Date;
|
||||||
|
looseTablets?: number;
|
||||||
|
totalPills?: number | null;
|
||||||
|
packageAmountValue?: number;
|
||||||
|
packCount?: number;
|
||||||
|
} = {
|
||||||
|
stockAdjustment,
|
||||||
|
lastStockCorrectionAt: new Date(),
|
||||||
|
updatedAt: new Date(),
|
||||||
|
};
|
||||||
|
|
||||||
|
const packageType = normalizePackageType(existing.packageType);
|
||||||
|
const allowsAmountBaseUpdate = isTubePackageType(packageType) || isLiquidContainerPackageType(packageType);
|
||||||
|
if (allowsAmountBaseUpdate) {
|
||||||
|
if (totalPills !== undefined) updateFields.totalPills = totalPills;
|
||||||
|
if (looseTablets !== undefined) updateFields.looseTablets = looseTablets;
|
||||||
|
if (packageAmountValue !== undefined) updateFields.packageAmountValue = packageAmountValue;
|
||||||
|
if (packCount !== undefined) updateFields.packCount = packCount;
|
||||||
|
}
|
||||||
|
if (looseTablets !== undefined) {
|
||||||
|
updateFields.looseTablets = looseTablets;
|
||||||
|
}
|
||||||
|
|
||||||
|
const result = await db
|
||||||
|
.update(medications)
|
||||||
|
.set(updateFields)
|
||||||
|
.where(and(eq(medications.id, idNum), eq(medications.userId, userId)))
|
||||||
|
.returning();
|
||||||
|
|
||||||
|
if (!result.length) return reply.notFound();
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: result[0].id,
|
||||||
|
stockAdjustment: result[0].stockAdjustment ?? 0,
|
||||||
|
lastStockCorrectionAt: result[0].lastStockCorrectionAt?.toISOString() ?? null,
|
||||||
|
updatedAt: result[0].updatedAt,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
app.delete<{ Params: { id: string } }>("/medications/:id", async (req, reply) => {
|
app.delete<{ Params: { id: string } }>("/medications/:id", async (req, reply) => {
|
||||||
const idNum = Number(req.params.id);
|
const idNum = Number(req.params.id);
|
||||||
@@ -836,24 +1088,29 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
|
|
||||||
const payload = rows.map((row) => {
|
const payload = rows.map((row) => {
|
||||||
// Parse intakes from new format, falling back to legacy
|
// Parse intakes from new format, falling back to legacy
|
||||||
const intakes = parseIntakesJson(
|
const intakes = parseIntakesWithUnits(
|
||||||
row.intakesJson,
|
row.intakesJson,
|
||||||
{ usageJson: row.usageJson, everyJson: row.everyJson, startJson: row.startJson },
|
{ usageJson: row.usageJson, everyJson: row.everyJson, startJson: row.startJson },
|
||||||
row.intakeRemindersEnabled ?? false
|
row.intakeRemindersEnabled ?? false
|
||||||
);
|
);
|
||||||
const blisters = intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start }));
|
const medForm = row.medicationForm ?? "tablet";
|
||||||
|
const blisters = intakes.map((i) => ({
|
||||||
|
usage: normalizeIntakeUsageForStock(i, medForm, row.packageType),
|
||||||
|
every: i.every,
|
||||||
|
start: i.start,
|
||||||
|
}));
|
||||||
const pillsPerBlister = row.pillsPerBlister ?? 1;
|
const pillsPerBlister = row.pillsPerBlister ?? 1;
|
||||||
const packCount = row.packCount ?? 1;
|
const packCount = row.packCount ?? 1;
|
||||||
const blistersPerPack = row.blistersPerPack ?? 1;
|
const blistersPerPack = row.blistersPerPack ?? 1;
|
||||||
const looseTablets = row.looseTablets ?? 0;
|
const looseTablets = row.looseTablets ?? 0;
|
||||||
const stockAdjustment = row.stockAdjustment ?? 0;
|
const stockAdjustment = row.stockAdjustment ?? 0;
|
||||||
const packageType = row.packageType ?? "blister";
|
const packageType = normalizePackageType(row.packageType);
|
||||||
|
|
||||||
// For bottle type, looseTablets IS the current stock (no blister math)
|
// For bottle type, looseTablets IS the current stock (no blister math)
|
||||||
const originalTotalPills =
|
const isTopical = medForm === "topical" || isTubePackageType(packageType);
|
||||||
packageType === "bottle"
|
const originalTotalPills = isAmountBasedPackageType(packageType)
|
||||||
? looseTablets + stockAdjustment
|
? looseTablets + stockAdjustment
|
||||||
: packCount * blistersPerPack * pillsPerBlister + looseTablets + stockAdjustment;
|
: packCount * blistersPerPack * pillsPerBlister + looseTablets + stockAdjustment;
|
||||||
|
|
||||||
// Calculate consumption with the same automatic/manual behavior as frontend coverage.
|
// Calculate consumption with the same automatic/manual behavior as frontend coverage.
|
||||||
const stockCorrectionCutoff = row.lastStockCorrectionAt ? new Date(row.lastStockCorrectionAt).getTime() : 0;
|
const stockCorrectionCutoff = row.lastStockCorrectionAt ? new Date(row.lastStockCorrectionAt).getTime() : 0;
|
||||||
@@ -863,7 +1120,9 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
let consumedUntilNow = 0;
|
let consumedUntilNow = 0;
|
||||||
const msPerDay = 86400000;
|
const msPerDay = 86400000;
|
||||||
|
|
||||||
if (stockCalculationMode === "automatic") {
|
if (isTopical) {
|
||||||
|
consumedUntilNow = 0;
|
||||||
|
} else if (stockCalculationMode === "automatic") {
|
||||||
blisters.forEach((blister, blisterIdx) => {
|
blisters.forEach((blister, blisterIdx) => {
|
||||||
const blisterStart = parseLocalDateTime(blister.start).getTime();
|
const blisterStart = parseLocalDateTime(blister.start).getTime();
|
||||||
if (Number.isNaN(blisterStart)) return;
|
if (Number.isNaN(blisterStart)) return;
|
||||||
@@ -959,7 +1218,7 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
const currentStock = Math.max(0, originalTotalPills - consumedUntilNow);
|
const currentStock = isTopical ? originalTotalPills : Math.max(0, originalTotalPills - consumedUntilNow);
|
||||||
|
|
||||||
// Calculate usage for the planning period
|
// Calculate usage for the planning period
|
||||||
// Always use the user-selected start date for the usage calculation.
|
// Always use the user-selected start date for the usage calculation.
|
||||||
@@ -969,7 +1228,7 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
// The stock already reflects consumed doses, so no double-counting occurs.
|
// The stock already reflects consumed doses, so no double-counting occurs.
|
||||||
// When includeUntilStart is true, calculate from now to end (useful for trip planning)
|
// When includeUntilStart is true, calculate from now to end (useful for trip planning)
|
||||||
const effectivePlannerStart = includeUntilStart ? now : start;
|
const effectivePlannerStart = includeUntilStart ? now : start;
|
||||||
const usageTotal = calculateUsageInRange(blisters, effectivePlannerStart, end);
|
const usageTotal = isTopical ? 0 : calculateUsageInRange(blisters, effectivePlannerStart, end);
|
||||||
|
|
||||||
const blistersNeeded = pillsPerBlister > 0 ? Math.ceil(usageTotal / pillsPerBlister) : 0;
|
const blistersNeeded = pillsPerBlister > 0 ? Math.ceil(usageTotal / pillsPerBlister) : 0;
|
||||||
|
|
||||||
@@ -979,7 +1238,7 @@ export async function medicationRoutes(app: FastifyInstance) {
|
|||||||
let fullBlisters: number;
|
let fullBlisters: number;
|
||||||
let loosePills: number;
|
let loosePills: number;
|
||||||
|
|
||||||
if (packageType === "bottle") {
|
if (isAmountBasedPackageType(packageType)) {
|
||||||
// Bottle type: no blisters, everything is loose pills
|
// Bottle type: no blisters, everything is loose pills
|
||||||
fullBlisters = 0;
|
fullBlisters = 0;
|
||||||
loosePills = availableAfterPeriod;
|
loosePills = availableAfterPeriod;
|
||||||
|
|||||||
+234
-15
@@ -15,6 +15,12 @@ import { getAnonymousUserId, requireAuth } from "../plugins/auth.js";
|
|||||||
import { env } from "../plugins/env.js";
|
import { env } from "../plugins/env.js";
|
||||||
import { updateReminderSentTime, updateUserReminderSentTime } from "../services/reminder-scheduler.js";
|
import { updateReminderSentTime, updateUserReminderSentTime } from "../services/reminder-scheduler.js";
|
||||||
import type { AuthUser } from "../types/fastify.js";
|
import type { AuthUser } from "../types/fastify.js";
|
||||||
|
import {
|
||||||
|
getPlannerUnitKind,
|
||||||
|
isAmountBasedPackageType,
|
||||||
|
isTubePackageType,
|
||||||
|
normalizePackageType,
|
||||||
|
} from "../utils/package-profiles.js";
|
||||||
import { loadUserSettings, sendShoutrrrNotification } from "./settings.js";
|
import { loadUserSettings, sendShoutrrrNotification } from "./settings.js";
|
||||||
|
|
||||||
// Escape HTML to prevent XSS in email templates
|
// Escape HTML to prevent XSS in email templates
|
||||||
@@ -29,6 +35,43 @@ function escapeHtml(text: string): string {
|
|||||||
return text.replace(/[&<>"']/g, (char) => htmlEscapes[char] || char);
|
return text.replace(/[&<>"']/g, (char) => htmlEscapes[char] || char);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function maskEmail(email: string): string {
|
||||||
|
const [localPart, domain] = email.split("@");
|
||||||
|
if (!domain) return "invalid-email";
|
||||||
|
if (localPart.length <= 2) return `${localPart[0] ?? "*"}*@${domain}`;
|
||||||
|
return `${localPart.slice(0, 2)}***@${domain}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
type MailDeliveryInfo = {
|
||||||
|
accepted?: unknown;
|
||||||
|
rejected?: unknown;
|
||||||
|
response?: unknown;
|
||||||
|
};
|
||||||
|
|
||||||
|
function normalizeRecipients(value: unknown): string[] {
|
||||||
|
if (!Array.isArray(value)) return [];
|
||||||
|
return value
|
||||||
|
.map((entry) => (typeof entry === "string" ? entry : String(entry ?? "")))
|
||||||
|
.map((entry) => entry.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDeliveryError(info: MailDeliveryInfo): string | null {
|
||||||
|
const accepted = normalizeRecipients(info.accepted);
|
||||||
|
const rejected = normalizeRecipients(info.rejected);
|
||||||
|
|
||||||
|
if (accepted.length > 0) return null;
|
||||||
|
if (rejected.length > 0) {
|
||||||
|
return `SMTP rejected all recipients: ${rejected.join(", ")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof info.response === "string" && info.response.trim()) {
|
||||||
|
return `SMTP did not confirm accepted recipients. Response: ${info.response}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return "SMTP did not confirm accepted recipients.";
|
||||||
|
}
|
||||||
|
|
||||||
type PlannerRow = {
|
type PlannerRow = {
|
||||||
medicationId: number;
|
medicationId: number;
|
||||||
medicationName: string;
|
medicationName: string;
|
||||||
@@ -42,6 +85,17 @@ type PlannerRow = {
|
|||||||
packageType?: string;
|
packageType?: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function isContainerPackage(packageType?: string): boolean {
|
||||||
|
return isAmountBasedPackageType(packageType);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getPlannerUnit(packageType: string | undefined, tr: ReturnType<typeof getTranslations>): string {
|
||||||
|
const unitKind = getPlannerUnitKind(packageType);
|
||||||
|
if (unitKind === "units") return tr.common.units;
|
||||||
|
if (unitKind === "ml") return tr.common.ml;
|
||||||
|
return tr.common.pills;
|
||||||
|
}
|
||||||
|
|
||||||
type SendEmailBody = {
|
type SendEmailBody = {
|
||||||
email: string;
|
email: string;
|
||||||
from: string;
|
from: string;
|
||||||
@@ -96,6 +150,10 @@ export async function plannerRoutes(app: FastifyInstance) {
|
|||||||
// Demand calculator notification (supports email and push)
|
// Demand calculator notification (supports email and push)
|
||||||
app.post<{ Body: SendEmailBody }>("/planner/send-email", async (request, reply) => {
|
app.post<{ Body: SendEmailBody }>("/planner/send-email", async (request, reply) => {
|
||||||
const { email, from, until, rows, language: bodyLanguage } = request.body;
|
const { email, from, until, rows, language: bodyLanguage } = request.body;
|
||||||
|
request.log.info(
|
||||||
|
{ hasEmail: Boolean(email), rowCount: rows?.length ?? 0 },
|
||||||
|
"[Planner] Demand notification request received"
|
||||||
|
);
|
||||||
|
|
||||||
if (!rows || rows.length === 0) {
|
if (!rows || rows.length === 0) {
|
||||||
return reply.status(400).send({ error: "Missing planner data" });
|
return reply.status(400).send({ error: "Missing planner data" });
|
||||||
@@ -110,6 +168,7 @@ export async function plannerRoutes(app: FastifyInstance) {
|
|||||||
const activeMedIds = new Set(activeMeds.map((med) => med.id));
|
const activeMedIds = new Set(activeMeds.map((med) => med.id));
|
||||||
const activeRows = rows.filter((row) => activeMedIds.has(row.medicationId));
|
const activeRows = rows.filter((row) => activeMedIds.has(row.medicationId));
|
||||||
if (activeRows.length === 0) {
|
if (activeRows.length === 0) {
|
||||||
|
request.log.warn("[Planner] Demand notification skipped: no active medications in request");
|
||||||
return reply.status(400).send({ error: "No active medications to notify" });
|
return reply.status(400).send({ error: "No active medications to notify" });
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -119,6 +178,16 @@ export async function plannerRoutes(app: FastifyInstance) {
|
|||||||
shoutrrrEnabled: userSettings.shoutrrrEnabled,
|
shoutrrrEnabled: userSettings.shoutrrrEnabled,
|
||||||
shoutrrrUrl: userSettings.shoutrrrUrl || "",
|
shoutrrrUrl: userSettings.shoutrrrUrl || "",
|
||||||
};
|
};
|
||||||
|
request.log.info(
|
||||||
|
{
|
||||||
|
userId,
|
||||||
|
emailEnabled: notificationSettings.emailEnabled,
|
||||||
|
pushEnabled: notificationSettings.shoutrrrEnabled,
|
||||||
|
hasPushUrl: Boolean(notificationSettings.shoutrrrUrl),
|
||||||
|
activeRowCount: activeRows.length,
|
||||||
|
},
|
||||||
|
"[Planner] Demand notification channel state"
|
||||||
|
);
|
||||||
|
|
||||||
// Get locale from user settings or use the language passed in the body
|
// Get locale from user settings or use the language passed in the body
|
||||||
const language: Language = (userSettings.language as Language) || bodyLanguage || "en";
|
const language: Language = (userSettings.language as Language) || bodyLanguage || "en";
|
||||||
@@ -168,16 +237,18 @@ ${summaryText}
|
|||||||
|
|
||||||
${activeRows
|
${activeRows
|
||||||
.map((r) => {
|
.map((r) => {
|
||||||
const isBottle = r.packageType === "bottle";
|
const isBottle = isContainerPackage(r.packageType);
|
||||||
const usage = `${r.plannerUsage} ${tr.common.pills}`;
|
const usageUnit = getPlannerUnit(r.packageType, tr);
|
||||||
|
const usage = `${r.plannerUsage} ${usageUnit}`;
|
||||||
const needed = isBottle ? "–" : `${r.blistersNeeded} × ${r.blisterSize}`;
|
const needed = isBottle ? "–" : `${r.blistersNeeded} × ${r.blisterSize}`;
|
||||||
const medPrescription = prescriptionMap.get(r.medicationId);
|
const medPrescription = prescriptionMap.get(r.medicationId);
|
||||||
const rxRefills = medPrescription?.prescriptionEnabled
|
const rxRefills = medPrescription?.prescriptionEnabled
|
||||||
? String(medPrescription.prescriptionRemainingRefills ?? 0)
|
? String(medPrescription.prescriptionRemainingRefills ?? 0)
|
||||||
: dc.prescriptionNotApplicable;
|
: dc.prescriptionNotApplicable;
|
||||||
const loosePills = Math.round((Number(r.loosePills) || 0) * 10) / 10;
|
const loosePills = Math.round((Number(r.loosePills) || 0) * 10) / 10;
|
||||||
|
const availableUnit = getPlannerUnit(r.packageType, tr);
|
||||||
const available = isBottle
|
const available = isBottle
|
||||||
? `${loosePills} ${tr.common.pills}`
|
? `${loosePills} ${availableUnit}`
|
||||||
: `${r.fullBlisters} ${tr.common.blisters}${loosePills > 0 ? ` + ${loosePills} ${tr.common.pills}` : ""}`;
|
: `${r.fullBlisters} ${tr.common.blisters}${loosePills > 0 ? ` + ${loosePills} ${tr.common.pills}` : ""}`;
|
||||||
const status = r.enough ? dc.statusEnough : dc.statusEmpty;
|
const status = r.enough ? dc.statusEnough : dc.statusEmpty;
|
||||||
return `${r.medicationName}: ${usage}, ${needed}, ${dc.tableHeaders.prescriptionRefills}: ${rxRefills}, ${available} - ${status}`;
|
return `${r.medicationName}: ${usage}, ${needed}, ${dc.tableHeaders.prescriptionRefills}: ${rxRefills}, ${available} - ${status}`;
|
||||||
@@ -198,6 +269,19 @@ ${getFooterPlain(language)}`;
|
|||||||
const smtpSecure = process.env.SMTP_SECURE === "true";
|
const smtpSecure = process.env.SMTP_SECURE === "true";
|
||||||
const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
|
const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
|
||||||
|
|
||||||
|
request.log.info(
|
||||||
|
{
|
||||||
|
hasSmtpHost: Boolean(smtpHost),
|
||||||
|
hasSmtpUser: Boolean(smtpUser),
|
||||||
|
hasSmtpPass: Boolean(smtpPass),
|
||||||
|
smtpPort,
|
||||||
|
smtpSecure,
|
||||||
|
hasSmtpFrom: Boolean(smtpFrom),
|
||||||
|
to: maskEmail(email),
|
||||||
|
},
|
||||||
|
"[Planner] Demand email path selected"
|
||||||
|
);
|
||||||
|
|
||||||
if (smtpHost && smtpUser) {
|
if (smtpHost && smtpUser) {
|
||||||
// Build HTML table with horizontal scroll for mobile
|
// Build HTML table with horizontal scroll for mobile
|
||||||
// Escape/coerce all user-provided values to prevent XSS
|
// Escape/coerce all user-provided values to prevent XSS
|
||||||
@@ -209,7 +293,7 @@ ${getFooterPlain(language)}`;
|
|||||||
const safeBlisterSize = Number(row.blisterSize) || 0;
|
const safeBlisterSize = Number(row.blisterSize) || 0;
|
||||||
const safeFullBlisters = Number(row.fullBlisters) || 0;
|
const safeFullBlisters = Number(row.fullBlisters) || 0;
|
||||||
const safeLoosePills = Math.round((Number(row.loosePills) || 0) * 10) / 10;
|
const safeLoosePills = Math.round((Number(row.loosePills) || 0) * 10) / 10;
|
||||||
const isBottle = row.packageType === "bottle";
|
const isBottle = isContainerPackage(row.packageType);
|
||||||
|
|
||||||
// "Blisters needed" column: dash for bottles
|
// "Blisters needed" column: dash for bottles
|
||||||
const neededCell = isBottle ? "–" : `${safeBlistersNeeded} × ${safeBlisterSize}`;
|
const neededCell = isBottle ? "–" : `${safeBlistersNeeded} × ${safeBlisterSize}`;
|
||||||
@@ -223,7 +307,8 @@ ${getFooterPlain(language)}`;
|
|||||||
// "Available" column: match frontend format
|
// "Available" column: match frontend format
|
||||||
let availableCell: string;
|
let availableCell: string;
|
||||||
if (isBottle) {
|
if (isBottle) {
|
||||||
availableCell = `${safeLoosePills} ${tr.common.pills}`;
|
const availableUnit = getPlannerUnit(row.packageType, tr);
|
||||||
|
availableCell = `${safeLoosePills} ${availableUnit}`;
|
||||||
} else {
|
} else {
|
||||||
availableCell = `${safeFullBlisters} ${tr.common.blisters}`;
|
availableCell = `${safeFullBlisters} ${tr.common.blisters}`;
|
||||||
if (safeLoosePills > 0) {
|
if (safeLoosePills > 0) {
|
||||||
@@ -236,7 +321,7 @@ ${getFooterPlain(language)}`;
|
|||||||
return `
|
return `
|
||||||
<tr style="${rowBg}">
|
<tr style="${rowBg}">
|
||||||
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; white-space: nowrap;">${safeName}</td>
|
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; white-space: nowrap;">${safeName}</td>
|
||||||
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;"><strong>${safePlannerUsage}</strong> ${tr.common.pills}</td>
|
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;"><strong>${safePlannerUsage}</strong> ${getPlannerUnit(row.packageType, tr)}</td>
|
||||||
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${neededCell}</td>
|
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${neededCell}</td>
|
||||||
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${rxCell}</td>
|
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${rxCell}</td>
|
||||||
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${availableCell}</td>
|
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${availableCell}</td>
|
||||||
@@ -303,7 +388,9 @@ ${getFooterPlain(language)}`;
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
await transporter.sendMail({
|
request.log.info({ to: maskEmail(email) }, "[Planner] Sending demand email");
|
||||||
|
|
||||||
|
const mailResult = await transporter.sendMail({
|
||||||
from: smtpFrom,
|
from: smtpFrom,
|
||||||
to: email,
|
to: email,
|
||||||
subject: t(dc.subject, { from: fromDate, until: untilDate }),
|
subject: t(dc.subject, { from: fromDate, until: untilDate }),
|
||||||
@@ -311,12 +398,33 @@ ${getFooterPlain(language)}`;
|
|||||||
html,
|
html,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const deliveryError = getDeliveryError(mailResult);
|
||||||
|
if (deliveryError) {
|
||||||
|
throw new Error(deliveryError);
|
||||||
|
}
|
||||||
|
|
||||||
|
request.log.info({ to: maskEmail(email), messageId: mailResult.messageId }, "[Planner] Demand email sent");
|
||||||
results.email = true;
|
results.email = true;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
request.log.error({ error, to: maskEmail(email) }, "[Planner] Demand email failed");
|
||||||
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
||||||
results.errors.push(`Email: ${errorMessage}`);
|
results.errors.push(`Email: ${errorMessage}`);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
request.log.warn(
|
||||||
|
{
|
||||||
|
hasSmtpHost: Boolean(smtpHost),
|
||||||
|
hasSmtpUser: Boolean(smtpUser),
|
||||||
|
to: maskEmail(email),
|
||||||
|
},
|
||||||
|
"[Planner] Demand email skipped: SMTP not configured"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
request.log.info(
|
||||||
|
{ emailEnabled: notificationSettings.emailEnabled, hasRecipient: Boolean(email) },
|
||||||
|
"[Planner] Demand email channel not active"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Send push notification if enabled
|
// Send push notification if enabled
|
||||||
@@ -324,7 +432,7 @@ ${getFooterPlain(language)}`;
|
|||||||
const pushTitle = t(dc.subject, { from: fromDate, until: untilDate });
|
const pushTitle = t(dc.subject, { from: fromDate, until: untilDate });
|
||||||
const pushMessage = `${summaryText}\n\n${activeRows
|
const pushMessage = `${summaryText}\n\n${activeRows
|
||||||
.map((r) => {
|
.map((r) => {
|
||||||
const usage = `${r.plannerUsage} ${tr.common.pills}`;
|
const usage = `${r.plannerUsage} ${getPlannerUnit(r.packageType, tr)}`;
|
||||||
const status = r.enough ? dc.statusEnough : dc.statusEmpty;
|
const status = r.enough ? dc.statusEnough : dc.statusEmpty;
|
||||||
return `${r.enough ? "✓" : "✗"} ${r.medicationName}: ${usage} - ${status}`;
|
return `${r.enough ? "✓" : "✗"} ${r.medicationName}: ${usage} - ${status}`;
|
||||||
})
|
})
|
||||||
@@ -363,6 +471,10 @@ ${getFooterPlain(language)}`;
|
|||||||
// Reminder notification for low stock medications (supports email and push)
|
// Reminder notification for low stock medications (supports email and push)
|
||||||
app.post<{ Body: ReminderEmailBody }>("/reminder/send-email", async (request, reply) => {
|
app.post<{ Body: ReminderEmailBody }>("/reminder/send-email", async (request, reply) => {
|
||||||
const { email, lowStock } = request.body;
|
const { email, lowStock } = request.body;
|
||||||
|
request.log.info(
|
||||||
|
{ hasEmail: Boolean(email), lowStockCount: lowStock?.length ?? 0 },
|
||||||
|
"[ReminderManual] Stock reminder request received"
|
||||||
|
);
|
||||||
|
|
||||||
if (!lowStock || lowStock.length === 0) {
|
if (!lowStock || lowStock.length === 0) {
|
||||||
return reply.status(400).send({ error: "Missing low stock data" });
|
return reply.status(400).send({ error: "Missing low stock data" });
|
||||||
@@ -371,12 +483,22 @@ ${getFooterPlain(language)}`;
|
|||||||
// Load user settings
|
// Load user settings
|
||||||
const userId = await getUserId(request);
|
const userId = await getUserId(request);
|
||||||
const activeMeds = await db
|
const activeMeds = await db
|
||||||
.select({ name: medications.name })
|
.select({ name: medications.name, genericName: medications.genericName, packageType: medications.packageType })
|
||||||
.from(medications)
|
.from(medications)
|
||||||
.where(and(eq(medications.userId, userId), eq(medications.isObsolete, false)));
|
.where(and(eq(medications.userId, userId), eq(medications.isObsolete, false)));
|
||||||
const activeMedNames = new Set(activeMeds.map((med) => med.name));
|
const activeMedicationByName = new Map(
|
||||||
const filteredLowStock = lowStock.filter((item) => activeMedNames.has(item.name));
|
activeMeds
|
||||||
|
.map((med) => [med.name || med.genericName || "", normalizePackageType(med.packageType)] as const)
|
||||||
|
.filter(([name]) => name.length > 0)
|
||||||
|
);
|
||||||
|
const filteredLowStock = lowStock.filter((item) => {
|
||||||
|
const packageType = activeMedicationByName.get(item.name);
|
||||||
|
if (!packageType) return false;
|
||||||
|
if (isTubePackageType(packageType)) return false;
|
||||||
|
return true;
|
||||||
|
});
|
||||||
if (filteredLowStock.length === 0) {
|
if (filteredLowStock.length === 0) {
|
||||||
|
request.log.warn("[ReminderManual] Stock reminder skipped: no active medications after filtering");
|
||||||
return reply.status(400).send({ error: "No active medications to notify" });
|
return reply.status(400).send({ error: "No active medications to notify" });
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -386,6 +508,16 @@ ${getFooterPlain(language)}`;
|
|||||||
shoutrrrEnabled: userSettings.shoutrrrEnabled,
|
shoutrrrEnabled: userSettings.shoutrrrEnabled,
|
||||||
shoutrrrUrl: userSettings.shoutrrrUrl || "",
|
shoutrrrUrl: userSettings.shoutrrrUrl || "",
|
||||||
};
|
};
|
||||||
|
request.log.info(
|
||||||
|
{
|
||||||
|
userId,
|
||||||
|
emailEnabled: notificationSettings.emailEnabled,
|
||||||
|
pushEnabled: notificationSettings.shoutrrrEnabled,
|
||||||
|
hasPushUrl: Boolean(notificationSettings.shoutrrrUrl),
|
||||||
|
filteredLowStockCount: filteredLowStock.length,
|
||||||
|
},
|
||||||
|
"[ReminderManual] Stock reminder channel state"
|
||||||
|
);
|
||||||
|
|
||||||
// Get translations based on user language
|
// Get translations based on user language
|
||||||
const language = (userSettings.language as Language) || "en";
|
const language = (userSettings.language as Language) || "en";
|
||||||
@@ -457,6 +589,19 @@ ${getFooterPlain(language)}`;
|
|||||||
const smtpSecure = process.env.SMTP_SECURE === "true";
|
const smtpSecure = process.env.SMTP_SECURE === "true";
|
||||||
const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
|
const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
|
||||||
|
|
||||||
|
request.log.info(
|
||||||
|
{
|
||||||
|
hasSmtpHost: Boolean(smtpHost),
|
||||||
|
hasSmtpUser: Boolean(smtpUser),
|
||||||
|
hasSmtpPass: Boolean(smtpPass),
|
||||||
|
smtpPort,
|
||||||
|
smtpSecure,
|
||||||
|
hasSmtpFrom: Boolean(smtpFrom),
|
||||||
|
to: maskEmail(email),
|
||||||
|
},
|
||||||
|
"[ReminderManual] Stock email path selected"
|
||||||
|
);
|
||||||
|
|
||||||
if (smtpHost && smtpUser) {
|
if (smtpHost && smtpUser) {
|
||||||
// Build subject line from shared title parts
|
// Build subject line from shared title parts
|
||||||
const subjectText = titleParts.join(", ");
|
const subjectText = titleParts.join(", ");
|
||||||
@@ -570,7 +715,9 @@ ${getFooterPlain(language)}`;
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
await transporter.sendMail({
|
request.log.info({ to: maskEmail(email) }, "[ReminderManual] Sending stock reminder email");
|
||||||
|
|
||||||
|
const mailResult = await transporter.sendMail({
|
||||||
from: smtpFrom,
|
from: smtpFrom,
|
||||||
to: email,
|
to: email,
|
||||||
subject: `MedAssist-ng: ${subjectText}`,
|
subject: `MedAssist-ng: ${subjectText}`,
|
||||||
@@ -578,12 +725,36 @@ ${getFooterPlain(language)}`;
|
|||||||
html,
|
html,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const deliveryError = getDeliveryError(mailResult);
|
||||||
|
if (deliveryError) {
|
||||||
|
throw new Error(deliveryError);
|
||||||
|
}
|
||||||
|
|
||||||
|
request.log.info(
|
||||||
|
{ to: maskEmail(email), messageId: mailResult.messageId },
|
||||||
|
"[ReminderManual] Stock reminder email sent"
|
||||||
|
);
|
||||||
results.email = true;
|
results.email = true;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
request.log.error({ error, to: maskEmail(email) }, "[ReminderManual] Stock reminder email failed");
|
||||||
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
||||||
results.errors.push(`Email: ${errorMessage}`);
|
results.errors.push(`Email: ${errorMessage}`);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
request.log.warn(
|
||||||
|
{
|
||||||
|
hasSmtpHost: Boolean(smtpHost),
|
||||||
|
hasSmtpUser: Boolean(smtpUser),
|
||||||
|
to: maskEmail(email),
|
||||||
|
},
|
||||||
|
"[ReminderManual] Stock reminder email skipped: SMTP not configured"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
request.log.info(
|
||||||
|
{ emailEnabled: notificationSettings.emailEnabled, hasRecipient: Boolean(email) },
|
||||||
|
"[ReminderManual] Stock email channel not active"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Send push notification if enabled
|
// Send push notification if enabled
|
||||||
@@ -634,6 +805,10 @@ ${getFooterPlain(language)}`;
|
|||||||
// Manual prescription reminder (supports email and push)
|
// Manual prescription reminder (supports email and push)
|
||||||
app.post<{ Body: PrescriptionReminderBody }>("/reminder/send-prescription", async (request, reply) => {
|
app.post<{ Body: PrescriptionReminderBody }>("/reminder/send-prescription", async (request, reply) => {
|
||||||
const { email, prescriptionLow } = request.body;
|
const { email, prescriptionLow } = request.body;
|
||||||
|
request.log.info(
|
||||||
|
{ hasEmail: Boolean(email), prescriptionCount: prescriptionLow?.length ?? 0 },
|
||||||
|
"[ReminderManual] Prescription reminder request received"
|
||||||
|
);
|
||||||
|
|
||||||
if (!prescriptionLow || prescriptionLow.length === 0) {
|
if (!prescriptionLow || prescriptionLow.length === 0) {
|
||||||
return reply.status(400).send({ error: "Missing prescription reminder data" });
|
return reply.status(400).send({ error: "Missing prescription reminder data" });
|
||||||
@@ -641,12 +816,13 @@ ${getFooterPlain(language)}`;
|
|||||||
|
|
||||||
const userId = await getUserId(request);
|
const userId = await getUserId(request);
|
||||||
const activeMeds = await db
|
const activeMeds = await db
|
||||||
.select({ name: medications.name })
|
.select({ name: medications.name, genericName: medications.genericName })
|
||||||
.from(medications)
|
.from(medications)
|
||||||
.where(and(eq(medications.userId, userId), eq(medications.isObsolete, false)));
|
.where(and(eq(medications.userId, userId), eq(medications.isObsolete, false)));
|
||||||
const activeMedNames = new Set(activeMeds.map((med) => med.name));
|
const activeMedNames = new Set(activeMeds.map((med) => med.name || med.genericName || ""));
|
||||||
const filteredPrescriptionLow = prescriptionLow.filter((item) => activeMedNames.has(item.name));
|
const filteredPrescriptionLow = prescriptionLow.filter((item) => activeMedNames.has(item.name));
|
||||||
if (filteredPrescriptionLow.length === 0) {
|
if (filteredPrescriptionLow.length === 0) {
|
||||||
|
request.log.warn("[ReminderManual] Prescription reminder skipped: no active medications after filtering");
|
||||||
return reply.status(400).send({ error: "No active medications to notify" });
|
return reply.status(400).send({ error: "No active medications to notify" });
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -684,6 +860,19 @@ ${getFooterPlain(language)}`;
|
|||||||
const smtpSecure = process.env.SMTP_SECURE === "true";
|
const smtpSecure = process.env.SMTP_SECURE === "true";
|
||||||
const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
|
const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
|
||||||
|
|
||||||
|
request.log.info(
|
||||||
|
{
|
||||||
|
hasSmtpHost: Boolean(smtpHost),
|
||||||
|
hasSmtpUser: Boolean(smtpUser),
|
||||||
|
hasSmtpPass: Boolean(smtpPass),
|
||||||
|
smtpPort,
|
||||||
|
smtpSecure,
|
||||||
|
hasSmtpFrom: Boolean(smtpFrom),
|
||||||
|
to: maskEmail(email),
|
||||||
|
},
|
||||||
|
"[ReminderManual] Prescription email path selected"
|
||||||
|
);
|
||||||
|
|
||||||
if (smtpHost && smtpUser) {
|
if (smtpHost && smtpUser) {
|
||||||
try {
|
try {
|
||||||
const transporter = nodemailer.createTransport({
|
const transporter = nodemailer.createTransport({
|
||||||
@@ -767,7 +956,9 @@ ${getFooterPlain(language)}`;
|
|||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
|
|
||||||
await transporter.sendMail({
|
request.log.info({ to: maskEmail(email) }, "[ReminderManual] Sending prescription reminder email");
|
||||||
|
|
||||||
|
const mailResult = await transporter.sendMail({
|
||||||
from: smtpFrom,
|
from: smtpFrom,
|
||||||
to: email,
|
to: email,
|
||||||
subject,
|
subject,
|
||||||
@@ -775,12 +966,40 @@ ${getFooterPlain(language)}`;
|
|||||||
html,
|
html,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const deliveryError = getDeliveryError(mailResult);
|
||||||
|
if (deliveryError) {
|
||||||
|
throw new Error(deliveryError);
|
||||||
|
}
|
||||||
|
|
||||||
|
request.log.info(
|
||||||
|
{ to: maskEmail(email), messageId: mailResult.messageId },
|
||||||
|
"[ReminderManual] Prescription reminder email sent"
|
||||||
|
);
|
||||||
results.email = true;
|
results.email = true;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
request.log.error({ error, to: maskEmail(email) }, "[ReminderManual] Prescription reminder email failed");
|
||||||
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
||||||
results.errors.push(`Email: ${errorMessage}`);
|
results.errors.push(`Email: ${errorMessage}`);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
request.log.warn(
|
||||||
|
{
|
||||||
|
hasSmtpHost: Boolean(smtpHost),
|
||||||
|
hasSmtpUser: Boolean(smtpUser),
|
||||||
|
to: maskEmail(email),
|
||||||
|
},
|
||||||
|
"[ReminderManual] Prescription reminder email skipped: SMTP not configured"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
request.log.info(
|
||||||
|
{
|
||||||
|
emailEnabled: userSettings.emailEnabled,
|
||||||
|
emailPrescriptionReminders: userSettings.emailPrescriptionReminders,
|
||||||
|
hasRecipient: Boolean(email),
|
||||||
|
},
|
||||||
|
"[ReminderManual] Prescription email channel not active"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (userSettings.shoutrrrEnabled && userSettings.shoutrrrPrescriptionReminders && userSettings.shoutrrrUrl) {
|
if (userSettings.shoutrrrEnabled && userSettings.shoutrrrPrescriptionReminders && userSettings.shoutrrrUrl) {
|
||||||
|
|||||||
+307
-37
@@ -85,6 +85,58 @@ type TestShoutrrrBody = {
|
|||||||
url: string;
|
url: string;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function maskEmail(email: string): string {
|
||||||
|
const [localPart, domain] = email.split("@");
|
||||||
|
if (!domain) return "invalid-email";
|
||||||
|
if (localPart.length <= 2) return `${localPart[0] ?? "*"}*@${domain}`;
|
||||||
|
return `${localPart.slice(0, 2)}***@${domain}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
type MailDeliveryInfo = {
|
||||||
|
accepted?: unknown;
|
||||||
|
rejected?: unknown;
|
||||||
|
response?: unknown;
|
||||||
|
};
|
||||||
|
|
||||||
|
function normalizeRecipients(value: unknown): string[] {
|
||||||
|
if (!Array.isArray(value)) return [];
|
||||||
|
return value
|
||||||
|
.map((entry) => (typeof entry === "string" ? entry : String(entry ?? "")))
|
||||||
|
.map((entry) => entry.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDeliveryError(info: MailDeliveryInfo): string | null {
|
||||||
|
const accepted = normalizeRecipients(info.accepted);
|
||||||
|
const rejected = normalizeRecipients(info.rejected);
|
||||||
|
|
||||||
|
if (accepted.length > 0) return null;
|
||||||
|
if (rejected.length > 0) {
|
||||||
|
return `SMTP rejected all recipients: ${rejected.join(", ")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof info.response === "string" && info.response.trim()) {
|
||||||
|
return `SMTP did not confirm accepted recipients. Response: ${info.response}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return "SMTP did not confirm accepted recipients.";
|
||||||
|
}
|
||||||
|
|
||||||
|
function getNotificationProvider(url: string): string {
|
||||||
|
if (url.startsWith("discord://")) return "discord";
|
||||||
|
if (url.startsWith("telegram://")) return "telegram";
|
||||||
|
if (url.startsWith("gotify://")) return "gotify";
|
||||||
|
if (url.startsWith("pushover://")) return "pushover";
|
||||||
|
if (url.startsWith("ntfy://")) return "ntfy";
|
||||||
|
|
||||||
|
try {
|
||||||
|
const parsed = new URL(url);
|
||||||
|
return parsed.hostname || "https";
|
||||||
|
} catch {
|
||||||
|
return "unknown";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Helper to parse boolean env vars
|
// Helper to parse boolean env vars
|
||||||
function envBool(key: string, defaultVal: boolean): boolean {
|
function envBool(key: string, defaultVal: boolean): boolean {
|
||||||
const val = process.env[key];
|
const val = process.env[key];
|
||||||
@@ -269,10 +321,13 @@ export async function settingsRoutes(app: FastifyInstance) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Get settings for current user
|
// Get settings for current user
|
||||||
app.get("/settings", async (request, reply) => {
|
// Suppress request logs — polled every 30s for reminder status refresh
|
||||||
|
app.get("/settings", { logLevel: "warn" }, async (request, reply) => {
|
||||||
const userId = await getUserId(request, reply);
|
const userId = await getUserId(request, reply);
|
||||||
|
|
||||||
const settings = await getOrCreateUserSettings(userId);
|
const settings = await getOrCreateUserSettings(userId);
|
||||||
|
const reminderHour = envInt("REMINDER_HOUR", 6);
|
||||||
|
const reminderMinutesBefore = envInt("REMINDER_MINUTES_BEFORE", 15);
|
||||||
|
|
||||||
return reply.send({
|
return reply.send({
|
||||||
// User notification settings (from DB)
|
// User notification settings (from DB)
|
||||||
@@ -323,6 +378,8 @@ export async function settingsRoutes(app: FastifyInstance) {
|
|||||||
lastPrescriptionReminderChannel: settings.lastPrescriptionReminderChannel ?? null,
|
lastPrescriptionReminderChannel: settings.lastPrescriptionReminderChannel ?? null,
|
||||||
lastPrescriptionReminderMedNames: settings.lastPrescriptionReminderMedNames ?? null,
|
lastPrescriptionReminderMedNames: settings.lastPrescriptionReminderMedNames ?? null,
|
||||||
// Server settings (from .env, read-only)
|
// Server settings (from .env, read-only)
|
||||||
|
reminderHour,
|
||||||
|
reminderMinutesBefore,
|
||||||
expiryWarningDays: parseInt(process.env.EXPIRY_WARNING_DAYS ?? "30", 10),
|
expiryWarningDays: parseInt(process.env.EXPIRY_WARNING_DAYS ?? "30", 10),
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@@ -420,7 +477,24 @@ export async function settingsRoutes(app: FastifyInstance) {
|
|||||||
const smtpSecure = process.env.SMTP_SECURE === "true";
|
const smtpSecure = process.env.SMTP_SECURE === "true";
|
||||||
const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
|
const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
|
||||||
|
|
||||||
|
request.log.info(
|
||||||
|
{
|
||||||
|
to: maskEmail(email),
|
||||||
|
hasSmtpHost: Boolean(smtpHost),
|
||||||
|
hasSmtpUser: Boolean(smtpUser),
|
||||||
|
hasSmtpPass: Boolean(smtpPass),
|
||||||
|
hasSmtpFrom: Boolean(smtpFrom),
|
||||||
|
smtpPort,
|
||||||
|
smtpSecure,
|
||||||
|
},
|
||||||
|
"[Settings] Test email request received"
|
||||||
|
);
|
||||||
|
|
||||||
if (!smtpHost || !smtpUser) {
|
if (!smtpHost || !smtpUser) {
|
||||||
|
request.log.warn(
|
||||||
|
{ to: maskEmail(email), hasSmtpHost: Boolean(smtpHost), hasSmtpUser: Boolean(smtpUser) },
|
||||||
|
"[Settings] Test email skipped: SMTP not configured"
|
||||||
|
);
|
||||||
return reply.status(400).send({ error: "SMTP not configured" });
|
return reply.status(400).send({ error: "SMTP not configured" });
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -435,7 +509,9 @@ export async function settingsRoutes(app: FastifyInstance) {
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
await transporter.sendMail({
|
request.log.info({ to: maskEmail(email) }, "[Settings] Sending test email");
|
||||||
|
|
||||||
|
const mailResult = await transporter.sendMail({
|
||||||
from: smtpFrom,
|
from: smtpFrom,
|
||||||
to: email,
|
to: email,
|
||||||
subject: "MedAssist-ng - Test Email",
|
subject: "MedAssist-ng - Test Email",
|
||||||
@@ -451,8 +527,16 @@ export async function settingsRoutes(app: FastifyInstance) {
|
|||||||
`,
|
`,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const deliveryError = getDeliveryError(mailResult);
|
||||||
|
if (deliveryError) {
|
||||||
|
throw new Error(deliveryError);
|
||||||
|
}
|
||||||
|
|
||||||
|
request.log.info({ to: maskEmail(email), messageId: mailResult.messageId }, "[Settings] Test email sent");
|
||||||
|
|
||||||
return reply.send({ success: true, message: "Test email sent successfully" });
|
return reply.send({ success: true, message: "Test email sent successfully" });
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
request.log.error({ error, to: maskEmail(email) }, "[Settings] Test email failed");
|
||||||
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
||||||
return reply.status(500).send({ error: `Failed to send email: ${errorMessage}` });
|
return reply.status(500).send({ error: `Failed to send email: ${errorMessage}` });
|
||||||
}
|
}
|
||||||
@@ -467,6 +551,7 @@ export async function settingsRoutes(app: FastifyInstance) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
const provider = getNotificationProvider(url);
|
||||||
const result = await sendShoutrrrNotification(
|
const result = await sendShoutrrrNotification(
|
||||||
url,
|
url,
|
||||||
"MedAssist-ng Test",
|
"MedAssist-ng Test",
|
||||||
@@ -474,11 +559,17 @@ export async function settingsRoutes(app: FastifyInstance) {
|
|||||||
);
|
);
|
||||||
|
|
||||||
if (result.success) {
|
if (result.success) {
|
||||||
|
request.log.info({ provider }, "[Settings] Test push notification sent");
|
||||||
return reply.send({ success: true, message: "Test notification sent successfully" });
|
return reply.send({ success: true, message: "Test notification sent successfully" });
|
||||||
} else {
|
} else {
|
||||||
|
request.log.warn({ provider, error: result.error ?? "unknown" }, "[Settings] Test push notification failed");
|
||||||
return reply.status(500).send({ error: result.error });
|
return reply.status(500).send({ error: result.error });
|
||||||
}
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
request.log.error(
|
||||||
|
{ provider: getNotificationProvider(url), error },
|
||||||
|
"[Settings] Unexpected error while sending test push notification"
|
||||||
|
);
|
||||||
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
||||||
return reply.status(500).send({ error: `Failed to send notification: ${errorMessage}` });
|
return reply.status(500).send({ error: `Failed to send notification: ${errorMessage}` });
|
||||||
}
|
}
|
||||||
@@ -491,6 +582,28 @@ function sanitizeNotificationUrl(
|
|||||||
urlStr: string
|
urlStr: string
|
||||||
): { url: string; isNtfy: boolean; auth?: { user: string; pass: string } } | { error: string } {
|
): { url: string; isNtfy: boolean; auth?: { user: string; pass: string } } | { error: string } {
|
||||||
try {
|
try {
|
||||||
|
// Support Shoutrrr Discord format: discord://TOKEN@WEBHOOK_ID
|
||||||
|
if (urlStr.startsWith("discord://")) {
|
||||||
|
const parsedDiscord = new URL(urlStr);
|
||||||
|
const webhookId = parsedDiscord.hostname;
|
||||||
|
const webhookToken = parsedDiscord.username;
|
||||||
|
|
||||||
|
if (!webhookId || !webhookToken) {
|
||||||
|
return { error: "Invalid Discord URL format" };
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!/^\d+$/.test(webhookId)) {
|
||||||
|
return { error: "Invalid Discord webhook ID" };
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!/^[A-Za-z0-9._-]+$/.test(webhookToken)) {
|
||||||
|
return { error: "Invalid Discord webhook token" };
|
||||||
|
}
|
||||||
|
|
||||||
|
const discordWebhookUrl = `https://discord.com/api/webhooks/${webhookId}/${webhookToken}`;
|
||||||
|
return { url: discordWebhookUrl, isNtfy: false };
|
||||||
|
}
|
||||||
|
|
||||||
// Convert ntfy:// to https:// for parsing, track if it was ntfy
|
// Convert ntfy:// to https:// for parsing, track if it was ntfy
|
||||||
const isNtfy = urlStr.startsWith("ntfy://");
|
const isNtfy = urlStr.startsWith("ntfy://");
|
||||||
const normalizedUrl = isNtfy ? urlStr.replace("ntfy://", "https://") : urlStr;
|
const normalizedUrl = isNtfy ? urlStr.replace("ntfy://", "https://") : urlStr;
|
||||||
@@ -502,38 +615,9 @@ function sanitizeNotificationUrl(
|
|||||||
return { error: "Only HTTP/HTTPS protocols are allowed" };
|
return { error: "Only HTTP/HTTPS protocols are allowed" };
|
||||||
}
|
}
|
||||||
|
|
||||||
// Block private/internal IP addresses
|
const hostValidationError = validateNotificationHostname(parsed.hostname);
|
||||||
const hostname = parsed.hostname.toLowerCase();
|
if (hostValidationError) {
|
||||||
|
return { error: hostValidationError };
|
||||||
// Block localhost
|
|
||||||
if (hostname === "localhost" || hostname === "127.0.0.1" || hostname === "::1") {
|
|
||||||
return { error: "Localhost URLs are not allowed" };
|
|
||||||
}
|
|
||||||
|
|
||||||
// Block private IP ranges (basic check)
|
|
||||||
const ipMatch = hostname.match(/^(\d+)\.(\d+)\.(\d+)\.(\d+)$/);
|
|
||||||
if (ipMatch) {
|
|
||||||
const [, a, b] = ipMatch.map(Number);
|
|
||||||
// 10.x.x.x, 172.16-31.x.x, 192.168.x.x, 169.254.x.x (link-local)
|
|
||||||
if (
|
|
||||||
a === 10 ||
|
|
||||||
a === 127 ||
|
|
||||||
(a === 172 && b >= 16 && b <= 31) ||
|
|
||||||
(a === 192 && b === 168) ||
|
|
||||||
(a === 169 && b === 254)
|
|
||||||
) {
|
|
||||||
return { error: "Private IP addresses are not allowed" };
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Block common internal hostnames
|
|
||||||
if (
|
|
||||||
hostname.endsWith(".local") ||
|
|
||||||
hostname.endsWith(".internal") ||
|
|
||||||
hostname.endsWith(".lan") ||
|
|
||||||
hostname === "metadata.google.internal"
|
|
||||||
) {
|
|
||||||
return { error: "Internal hostnames are not allowed" };
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Reconstruct URL from validated components - this breaks taint tracking
|
// Reconstruct URL from validated components - this breaks taint tracking
|
||||||
@@ -550,6 +634,39 @@ function sanitizeNotificationUrl(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function validateNotificationHostname(hostnameRaw: string): string | null {
|
||||||
|
const hostname = hostnameRaw.toLowerCase();
|
||||||
|
|
||||||
|
if (hostname === "localhost" || hostname === "127.0.0.1" || hostname === "::1") {
|
||||||
|
return "Localhost URLs are not allowed";
|
||||||
|
}
|
||||||
|
|
||||||
|
const ipMatch = hostname.match(/^(\d+)\.(\d+)\.(\d+)\.(\d+)$/);
|
||||||
|
if (ipMatch) {
|
||||||
|
const [, a, b] = ipMatch.map(Number);
|
||||||
|
if (
|
||||||
|
a === 10 ||
|
||||||
|
a === 127 ||
|
||||||
|
(a === 172 && b >= 16 && b <= 31) ||
|
||||||
|
(a === 192 && b === 168) ||
|
||||||
|
(a === 169 && b === 254)
|
||||||
|
) {
|
||||||
|
return "Private IP addresses are not allowed";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
hostname.endsWith(".local") ||
|
||||||
|
hostname.endsWith(".internal") ||
|
||||||
|
hostname.endsWith(".lan") ||
|
||||||
|
hostname === "metadata.google.internal"
|
||||||
|
) {
|
||||||
|
return "Internal hostnames are not allowed";
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
// Send notification via Shoutrrr-compatible URL (supports ntfy, Discord, Telegram, etc.)
|
// Send notification via Shoutrrr-compatible URL (supports ntfy, Discord, Telegram, etc.)
|
||||||
export async function sendShoutrrrNotification(
|
export async function sendShoutrrrNotification(
|
||||||
urlStr: string,
|
urlStr: string,
|
||||||
@@ -557,6 +674,149 @@ export async function sendShoutrrrNotification(
|
|||||||
message: string
|
message: string
|
||||||
): Promise<{ success: boolean; error?: string }> {
|
): Promise<{ success: boolean; error?: string }> {
|
||||||
try {
|
try {
|
||||||
|
if (urlStr.startsWith("pushover://")) {
|
||||||
|
const pushoverAuthority = urlStr.slice("pushover://".length).split("/")[0] ?? "";
|
||||||
|
const atIndex = pushoverAuthority.lastIndexOf("@");
|
||||||
|
const credentialPart = atIndex >= 0 ? pushoverAuthority.slice(0, atIndex) : "";
|
||||||
|
const userKey = atIndex >= 0 ? pushoverAuthority.slice(atIndex + 1) : "";
|
||||||
|
|
||||||
|
const tokenSeparatorIndex = credentialPart.indexOf(":");
|
||||||
|
const apiToken = tokenSeparatorIndex >= 0 ? credentialPart.slice(tokenSeparatorIndex + 1) : "";
|
||||||
|
|
||||||
|
const parsedPushover = new URL(urlStr);
|
||||||
|
|
||||||
|
if (!apiToken || !userKey) {
|
||||||
|
return { success: false, error: "Invalid Pushover URL format" };
|
||||||
|
}
|
||||||
|
|
||||||
|
const pushoverBody = new URLSearchParams({
|
||||||
|
token: apiToken,
|
||||||
|
user: userKey,
|
||||||
|
title,
|
||||||
|
message,
|
||||||
|
});
|
||||||
|
|
||||||
|
const devices = parsedPushover.searchParams.get("devices");
|
||||||
|
if (devices) {
|
||||||
|
pushoverBody.set("device", devices);
|
||||||
|
}
|
||||||
|
|
||||||
|
const priority = parsedPushover.searchParams.get("priority");
|
||||||
|
if (priority && /^-?\d+$/.test(priority)) {
|
||||||
|
pushoverBody.set("priority", priority);
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await fetch("https://api.pushover.net/1/messages.json", {
|
||||||
|
method: "POST",
|
||||||
|
headers: { "Content-Type": "application/x-www-form-urlencoded" },
|
||||||
|
body: pushoverBody.toString(),
|
||||||
|
redirect: "error",
|
||||||
|
});
|
||||||
|
|
||||||
|
if (response.ok) return { success: true };
|
||||||
|
const errorText = await response.text();
|
||||||
|
return { success: false, error: `HTTP ${response.status}: ${errorText}` };
|
||||||
|
}
|
||||||
|
|
||||||
|
if (urlStr.startsWith("telegram://")) {
|
||||||
|
const parsedTelegram = new URL(urlStr);
|
||||||
|
const token = parsedTelegram.username;
|
||||||
|
if (!token || parsedTelegram.hostname !== "telegram") {
|
||||||
|
return { success: false, error: "Invalid Telegram URL format" };
|
||||||
|
}
|
||||||
|
|
||||||
|
const chatsRaw = parsedTelegram.searchParams.get("chats") ?? parsedTelegram.searchParams.get("channels") ?? "";
|
||||||
|
const chats = chatsRaw
|
||||||
|
.split(",")
|
||||||
|
.map((chat) => chat.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
|
||||||
|
if (chats.length === 0) {
|
||||||
|
return { success: false, error: "Telegram URL requires chats parameter" };
|
||||||
|
}
|
||||||
|
|
||||||
|
const parseModeRaw = parsedTelegram.searchParams.get("parseMode")?.toLowerCase();
|
||||||
|
let parseMode: "HTML" | "Markdown" | "MarkdownV2" | undefined;
|
||||||
|
if (parseModeRaw === "html") {
|
||||||
|
parseMode = "HTML";
|
||||||
|
} else if (parseModeRaw === "markdown") {
|
||||||
|
parseMode = "Markdown";
|
||||||
|
} else if (parseModeRaw === "markdownv2") {
|
||||||
|
parseMode = "MarkdownV2";
|
||||||
|
}
|
||||||
|
|
||||||
|
const notificationRaw = parsedTelegram.searchParams.get("notification")?.toLowerCase();
|
||||||
|
const disableNotification = notificationRaw === "no" || notificationRaw === "false";
|
||||||
|
|
||||||
|
const previewRaw = parsedTelegram.searchParams.get("preview")?.toLowerCase();
|
||||||
|
const disablePreview = previewRaw === "no" || previewRaw === "false";
|
||||||
|
|
||||||
|
if (!/^\d+:[A-Za-z0-9_-]+$/.test(token)) {
|
||||||
|
return { success: false, error: "Invalid Telegram token format" };
|
||||||
|
}
|
||||||
|
|
||||||
|
const telegramSendMessageUrl = new URL("/bot/sendMessage", "https://api.telegram.org");
|
||||||
|
telegramSendMessageUrl.pathname = `/bot${token}/sendMessage`;
|
||||||
|
|
||||||
|
for (const chatId of chats) {
|
||||||
|
const payload: Record<string, string | boolean> = {
|
||||||
|
chat_id: chatId,
|
||||||
|
text: `${title}\n\n${message}`,
|
||||||
|
disable_notification: disableNotification,
|
||||||
|
disable_web_page_preview: disablePreview,
|
||||||
|
};
|
||||||
|
if (parseMode) {
|
||||||
|
payload.parse_mode = parseMode;
|
||||||
|
}
|
||||||
|
|
||||||
|
// codeql[js/request-forgery]: host is fixed to api.telegram.org and token is pattern-validated.
|
||||||
|
const response = await fetch(telegramSendMessageUrl.toString(), {
|
||||||
|
method: "POST",
|
||||||
|
headers: { "Content-Type": "application/json" },
|
||||||
|
body: JSON.stringify(payload),
|
||||||
|
redirect: "error",
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!response.ok) {
|
||||||
|
const errorText = await response.text();
|
||||||
|
return { success: false, error: `HTTP ${response.status}: ${errorText}` };
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return { success: true };
|
||||||
|
}
|
||||||
|
|
||||||
|
if (urlStr.startsWith("gotify://")) {
|
||||||
|
const parsedGotify = new URL(urlStr);
|
||||||
|
const hostValidationError = validateNotificationHostname(parsedGotify.hostname);
|
||||||
|
if (hostValidationError) {
|
||||||
|
return { success: false, error: hostValidationError };
|
||||||
|
}
|
||||||
|
|
||||||
|
const pathParts = parsedGotify.pathname
|
||||||
|
.split("/")
|
||||||
|
.map((part) => part.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
|
||||||
|
if (pathParts.length === 0) {
|
||||||
|
return { success: false, error: "Invalid Gotify URL format" };
|
||||||
|
}
|
||||||
|
|
||||||
|
const token = pathParts[pathParts.length - 1];
|
||||||
|
const basePath = pathParts.slice(0, -1).join("/");
|
||||||
|
|
||||||
|
const disableTlsRaw = parsedGotify.searchParams.get("disabletls")?.toLowerCase();
|
||||||
|
const protocol = disableTlsRaw === "yes" || disableTlsRaw === "true" || disableTlsRaw === "1" ? "http" : "https";
|
||||||
|
|
||||||
|
const gotifyWebhookUrl = `${protocol}://${parsedGotify.host}${basePath ? `/${basePath}` : ""}/message?token=${encodeURIComponent(token)}`;
|
||||||
|
|
||||||
|
const gotifyPriority = parsedGotify.searchParams.get("priority");
|
||||||
|
const gotifyMessage = gotifyPriority ? `${message}\n\n(priority=${gotifyPriority})` : message;
|
||||||
|
|
||||||
|
// Reuse validated https webhook path to keep a single outbound request sink.
|
||||||
|
return sendShoutrrrNotification(gotifyWebhookUrl, title, gotifyMessage);
|
||||||
|
}
|
||||||
|
|
||||||
// Validate and sanitize URL to prevent SSRF - this reconstructs the URL
|
// Validate and sanitize URL to prevent SSRF - this reconstructs the URL
|
||||||
// from validated components, breaking taint tracking
|
// from validated components, breaking taint tracking
|
||||||
const validation = sanitizeNotificationUrl(urlStr);
|
const validation = sanitizeNotificationUrl(urlStr);
|
||||||
@@ -584,14 +844,17 @@ export async function sendShoutrrrNotification(
|
|||||||
// Use JSON format only for known webhook services that require it
|
// Use JSON format only for known webhook services that require it
|
||||||
// Use proper URL parsing to prevent bypass attacks (e.g., evil.com?hooks.slack.com)
|
// Use proper URL parsing to prevent bypass attacks (e.g., evil.com?hooks.slack.com)
|
||||||
let isJsonWebhook = false;
|
let isJsonWebhook = false;
|
||||||
|
let isDiscordWebhook = false;
|
||||||
try {
|
try {
|
||||||
const parsedUrl = new URL(sanitizedUrl);
|
const parsedUrl = new URL(sanitizedUrl);
|
||||||
const hostname = parsedUrl.hostname.toLowerCase();
|
const hostname = parsedUrl.hostname.toLowerCase();
|
||||||
const pathname = parsedUrl.pathname.toLowerCase();
|
const pathname = parsedUrl.pathname.toLowerCase();
|
||||||
|
isDiscordWebhook =
|
||||||
|
(hostname === "discord.com" || hostname === "discordapp.com") && pathname.startsWith("/api/webhooks");
|
||||||
|
|
||||||
isJsonWebhook =
|
isJsonWebhook =
|
||||||
// Discord webhooks
|
// Discord webhooks
|
||||||
((hostname === "discord.com" || hostname === "discordapp.com") && pathname.startsWith("/api/webhooks")) ||
|
isDiscordWebhook ||
|
||||||
// Slack webhooks
|
// Slack webhooks
|
||||||
hostname === "hooks.slack.com" ||
|
hostname === "hooks.slack.com" ||
|
||||||
hostname.endsWith(".hooks.slack.com") ||
|
hostname.endsWith(".hooks.slack.com") ||
|
||||||
@@ -621,9 +884,16 @@ export async function sendShoutrrrNotification(
|
|||||||
} else if (sanitizedUrl.startsWith("http://") || sanitizedUrl.startsWith("https://")) {
|
} else if (sanitizedUrl.startsWith("http://") || sanitizedUrl.startsWith("https://")) {
|
||||||
targetUrl = sanitizedUrl;
|
targetUrl = sanitizedUrl;
|
||||||
headers = { "Content-Type": "application/json" };
|
headers = { "Content-Type": "application/json" };
|
||||||
body = JSON.stringify({ title, message, text: `${title}\n\n${message}` });
|
if (isDiscordWebhook) {
|
||||||
|
body = JSON.stringify({ content: `${title}\n\n${message}` });
|
||||||
|
} else {
|
||||||
|
body = JSON.stringify({ title, message, text: `${title}\n\n${message}` });
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
return { success: false, error: "Unsupported URL format. Use ntfy:// or https:// URL" };
|
return {
|
||||||
|
success: false,
|
||||||
|
error: "Unsupported URL format. Use ntfy://, discord://, pushover://, gotify://, telegram://, or https:// URL",
|
||||||
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// SSRF protection: targetUrl is reconstructed from sanitizeNotificationUrl() which validates:
|
// SSRF protection: targetUrl is reconstructed from sanitizeNotificationUrl() which validates:
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import { medications, shareTokens, userSettings, users } from "../db/schema.js";
|
|||||||
import { getAnonymousUserId, requireAuth } from "../plugins/auth.js";
|
import { getAnonymousUserId, requireAuth } from "../plugins/auth.js";
|
||||||
import { env } from "../plugins/env.js";
|
import { env } from "../plugins/env.js";
|
||||||
import type { AuthUser } from "../types/fastify.js";
|
import type { AuthUser } from "../types/fastify.js";
|
||||||
|
import { isAmountBasedPackageType, normalizePackageType } from "../utils/package-profiles.js";
|
||||||
import {
|
import {
|
||||||
getAllTakenByForMedication,
|
getAllTakenByForMedication,
|
||||||
parseIntakesJson,
|
parseIntakesJson,
|
||||||
@@ -119,10 +120,9 @@ export async function shareRoutes(app: FastifyInstance) {
|
|||||||
// Parse takenBy JSON array
|
// Parse takenBy JSON array
|
||||||
const takenByArray = parseTakenByJson(med.takenByJson);
|
const takenByArray = parseTakenByJson(med.takenByJson);
|
||||||
|
|
||||||
const totalPills =
|
const totalPills = isAmountBasedPackageType(med.packageType)
|
||||||
(med.packageType ?? "blister") === "bottle"
|
? med.looseTablets + (med.stockAdjustment ?? 0)
|
||||||
? med.looseTablets + (med.stockAdjustment ?? 0)
|
: med.packCount * med.blistersPerPack * med.pillsPerBlister + med.looseTablets + (med.stockAdjustment ?? 0);
|
||||||
: med.packCount * med.blistersPerPack * med.pillsPerBlister + med.looseTablets + (med.stockAdjustment ?? 0);
|
|
||||||
return {
|
return {
|
||||||
id: med.id,
|
id: med.id,
|
||||||
name: med.name,
|
name: med.name,
|
||||||
@@ -131,7 +131,7 @@ export async function shareRoutes(app: FastifyInstance) {
|
|||||||
doseUnit: med.doseUnit ?? "mg",
|
doseUnit: med.doseUnit ?? "mg",
|
||||||
imageUrl: med.imageUrl,
|
imageUrl: med.imageUrl,
|
||||||
totalPills,
|
totalPills,
|
||||||
packageType: med.packageType ?? "blister",
|
packageType: normalizePackageType(med.packageType),
|
||||||
packCount: med.packCount,
|
packCount: med.packCount,
|
||||||
blistersPerPack: med.blistersPerPack,
|
blistersPerPack: med.blistersPerPack,
|
||||||
looseTablets: med.looseTablets,
|
looseTablets: med.looseTablets,
|
||||||
|
|||||||
@@ -50,6 +50,36 @@ function saveIntakeReminderState(state: IntakeReminderState): void {
|
|||||||
writeFileSync(intakeReminderStateFile, JSON.stringify(state, null, 2));
|
writeFileSync(intakeReminderStateFile, JSON.stringify(state, null, 2));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type MailDeliveryInfo = {
|
||||||
|
accepted?: unknown;
|
||||||
|
rejected?: unknown;
|
||||||
|
response?: unknown;
|
||||||
|
};
|
||||||
|
|
||||||
|
function normalizeRecipients(value: unknown): string[] {
|
||||||
|
if (!Array.isArray(value)) return [];
|
||||||
|
return value
|
||||||
|
.map((entry) => (typeof entry === "string" ? entry : String(entry ?? "")))
|
||||||
|
.map((entry) => entry.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDeliveryError(info: MailDeliveryInfo): string | null {
|
||||||
|
const accepted = normalizeRecipients(info.accepted);
|
||||||
|
const rejected = normalizeRecipients(info.rejected);
|
||||||
|
|
||||||
|
if (accepted.length > 0) return null;
|
||||||
|
if (rejected.length > 0) {
|
||||||
|
return `SMTP rejected all recipients: ${rejected.join(", ")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof info.response === "string" && info.response.trim()) {
|
||||||
|
return `SMTP did not confirm accepted recipients. Response: ${info.response}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return "SMTP did not confirm accepted recipients.";
|
||||||
|
}
|
||||||
|
|
||||||
function buildDoseIdForIntake(intake: UpcomingIntake & { medicationId: number; blisterIndex: number }): string {
|
function buildDoseIdForIntake(intake: UpcomingIntake & { medicationId: number; blisterIndex: number }): string {
|
||||||
const intakeDate = intake.intakeTime;
|
const intakeDate = intake.intakeTime;
|
||||||
const dateOnlyMs = new Date(intakeDate.getFullYear(), intakeDate.getMonth(), intakeDate.getDate()).getTime();
|
const dateOnlyMs = new Date(intakeDate.getFullYear(), intakeDate.getMonth(), intakeDate.getDate()).getTime();
|
||||||
@@ -106,8 +136,9 @@ async function autoMarkDueIntakesAsTaken(
|
|||||||
}
|
}
|
||||||
|
|
||||||
const medicationTakenBy = parseTakenByJson(med.takenByJson);
|
const medicationTakenBy = parseTakenByJson(med.takenByJson);
|
||||||
|
const medDisplayName = med.name || med.genericName || "";
|
||||||
const todaysIntakes = getTodaysIntakes(
|
const todaysIntakes = getTodaysIntakes(
|
||||||
med.name,
|
medDisplayName,
|
||||||
intakes,
|
intakes,
|
||||||
medicationTakenBy,
|
medicationTakenBy,
|
||||||
med.pillWeightMg,
|
med.pillWeightMg,
|
||||||
@@ -165,7 +196,7 @@ async function sendIntakeReminderEmail(
|
|||||||
repeatIntervalMinutes?: number,
|
repeatIntervalMinutes?: number,
|
||||||
currentCount?: number,
|
currentCount?: number,
|
||||||
maxCount?: number
|
maxCount?: number
|
||||||
): Promise<{ success: boolean; error?: string }> {
|
): Promise<{ success: boolean; error?: string; messageId?: string; smtpResponse?: string }> {
|
||||||
const smtpHost = process.env.SMTP_HOST;
|
const smtpHost = process.env.SMTP_HOST;
|
||||||
const smtpUser = process.env.SMTP_USER;
|
const smtpUser = process.env.SMTP_USER;
|
||||||
const smtpPass = process.env.SMTP_TOKEN || process.env.SMTP_PASS; // Token takes precedence
|
const smtpPass = process.env.SMTP_TOKEN || process.env.SMTP_PASS; // Token takes precedence
|
||||||
@@ -309,7 +340,7 @@ ${getFooterPlain(language)}`;
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
await transporter.sendMail({
|
const mailResult = await transporter.sendMail({
|
||||||
from: smtpFrom,
|
from: smtpFrom,
|
||||||
to: email,
|
to: email,
|
||||||
subject: `💊 ${subject}`,
|
subject: `💊 ${subject}`,
|
||||||
@@ -317,7 +348,16 @@ ${getFooterPlain(language)}`;
|
|||||||
html,
|
html,
|
||||||
});
|
});
|
||||||
|
|
||||||
return { success: true };
|
const deliveryError = getDeliveryError(mailResult);
|
||||||
|
if (deliveryError) {
|
||||||
|
return { success: false, error: deliveryError };
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
success: true,
|
||||||
|
messageId: mailResult.messageId,
|
||||||
|
smtpResponse: typeof mailResult.response === "string" ? mailResult.response : undefined,
|
||||||
|
};
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
||||||
return { success: false, error: errorMessage };
|
return { success: false, error: errorMessage };
|
||||||
@@ -379,17 +419,26 @@ async function checkAndSendIntakeRemindersForUser(
|
|||||||
`[IntakeReminder] User ${settings.userId}: Notifications enabled (email:${emailEnabled}, shoutrrr:${shoutrrrEnabled})`
|
`[IntakeReminder] User ${settings.userId}: Notifications enabled (email:${emailEnabled}, shoutrrr:${shoutrrrEnabled})`
|
||||||
);
|
);
|
||||||
|
|
||||||
// Get all medications with intake reminders enabled for this user
|
// Build medication entries that have at least one reminder-enabled intake.
|
||||||
const medsWithReminders = rows.filter((row) => row.intakeRemindersEnabled);
|
// Intake-level reminders are the single source of truth.
|
||||||
|
const reminderEntries = rows
|
||||||
|
.map((med) => {
|
||||||
|
const intakes = parseIntakesJson(
|
||||||
|
med.intakesJson,
|
||||||
|
{ usageJson: med.usageJson, everyJson: med.everyJson, startJson: med.startJson },
|
||||||
|
false
|
||||||
|
);
|
||||||
|
const intakesWithReminders = intakes.filter((intake) => intake.intakeRemindersEnabled === true);
|
||||||
|
return { med, intakes, intakesWithReminders };
|
||||||
|
})
|
||||||
|
.filter((entry) => entry.intakesWithReminders.length > 0);
|
||||||
|
|
||||||
if (medsWithReminders.length === 0) {
|
if (reminderEntries.length === 0) {
|
||||||
logger.debug(`[IntakeReminder] User ${settings.userId}: No medications have reminders enabled`);
|
logger.debug(`[IntakeReminder] User ${settings.userId}: No medications have reminders enabled`);
|
||||||
return; // No medications have reminders enabled for this user
|
return; // No medications have reminders enabled for this user
|
||||||
}
|
}
|
||||||
|
|
||||||
logger.debug(
|
logger.debug(`[IntakeReminder] User ${settings.userId}: Found ${reminderEntries.length} medications with reminders`);
|
||||||
`[IntakeReminder] User ${settings.userId}: Found ${medsWithReminders.length} medications with reminders`
|
|
||||||
);
|
|
||||||
|
|
||||||
const state = loadIntakeReminderState();
|
const state = loadIntakeReminderState();
|
||||||
const allUpcoming: (UpcomingIntake & { medicationId: number; blisterIndex: number })[] = [];
|
const allUpcoming: (UpcomingIntake & { medicationId: number; blisterIndex: number })[] = [];
|
||||||
@@ -406,29 +455,15 @@ async function checkAndSendIntakeRemindersForUser(
|
|||||||
);
|
);
|
||||||
|
|
||||||
// Find intakes: upcoming ones in reminder window + past ones for repeat reminders
|
// Find intakes: upcoming ones in reminder window + past ones for repeat reminders
|
||||||
for (const med of medsWithReminders) {
|
for (const { med, intakes, intakesWithReminders } of reminderEntries) {
|
||||||
// Parse intakes using new format (with per-intake takenBy), falling back to legacy
|
|
||||||
const intakes = parseIntakesJson(
|
|
||||||
med.intakesJson,
|
|
||||||
{ usageJson: med.usageJson, everyJson: med.everyJson, startJson: med.startJson },
|
|
||||||
med.intakeRemindersEnabled ?? false
|
|
||||||
);
|
|
||||||
// Medication-level takenBy (for fallback/display purposes)
|
// Medication-level takenBy (for fallback/display purposes)
|
||||||
const medicationTakenBy = parseTakenByJson(med.takenByJson);
|
const medicationTakenBy = parseTakenByJson(med.takenByJson);
|
||||||
|
const medDisplayName = med.name || med.genericName || "";
|
||||||
|
|
||||||
logger.debug(
|
logger.debug(
|
||||||
`[IntakeReminder] User ${settings.userId}: Processing medication "${med.name}" with ${intakes.length} intakes`
|
`[IntakeReminder] User ${settings.userId}: Processing medication "${medDisplayName}" with ${intakes.length} intakes`
|
||||||
);
|
);
|
||||||
|
|
||||||
// Filter intakes that have reminders enabled (per-intake setting or medication-level)
|
|
||||||
const intakesWithReminders = intakes.filter((intake, idx) => {
|
|
||||||
const hasReminder = intake.intakeRemindersEnabled || med.intakeRemindersEnabled;
|
|
||||||
if (!hasReminder) {
|
|
||||||
logger.debug(`[IntakeReminder] User ${settings.userId}: Intake ${idx} has reminders disabled, skipping`);
|
|
||||||
}
|
|
||||||
return hasReminder;
|
|
||||||
});
|
|
||||||
|
|
||||||
// Process each intake separately to track blisterIndex
|
// Process each intake separately to track blisterIndex
|
||||||
intakesWithReminders.forEach((intake, _blisterIndex) => {
|
intakesWithReminders.forEach((intake, _blisterIndex) => {
|
||||||
const actualIndex = intakes.indexOf(intake); // Get the actual index in original array
|
const actualIndex = intakes.indexOf(intake); // Get the actual index in original array
|
||||||
@@ -438,7 +473,7 @@ async function checkAndSendIntakeRemindersForUser(
|
|||||||
|
|
||||||
// Always get upcoming intakes (15 min before) for first reminders
|
// Always get upcoming intakes (15 min before) for first reminders
|
||||||
const upcomingIntakes = getUpcomingIntakes(
|
const upcomingIntakes = getUpcomingIntakes(
|
||||||
med.name,
|
medDisplayName,
|
||||||
[intake],
|
[intake],
|
||||||
REMINDER_MINUTES_BEFORE,
|
REMINDER_MINUTES_BEFORE,
|
||||||
medicationTakenBy,
|
medicationTakenBy,
|
||||||
@@ -465,7 +500,7 @@ async function checkAndSendIntakeRemindersForUser(
|
|||||||
// If repeat reminders enabled, also check for missed intakes (past the intake time)
|
// If repeat reminders enabled, also check for missed intakes (past the intake time)
|
||||||
if (settings.repeatRemindersEnabled) {
|
if (settings.repeatRemindersEnabled) {
|
||||||
const allTodaysIntakes = getTodaysIntakes(
|
const allTodaysIntakes = getTodaysIntakes(
|
||||||
med.name,
|
medDisplayName,
|
||||||
[intake],
|
[intake],
|
||||||
medicationTakenBy,
|
medicationTakenBy,
|
||||||
med.pillWeightMg,
|
med.pillWeightMg,
|
||||||
@@ -668,7 +703,9 @@ async function checkAndSendIntakeRemindersForUser(
|
|||||||
);
|
);
|
||||||
emailSuccess = result.success;
|
emailSuccess = result.success;
|
||||||
if (result.success) {
|
if (result.success) {
|
||||||
logger.info(`[IntakeReminder] User ${settings.userId}: Email sent successfully`);
|
logger.info(
|
||||||
|
`[IntakeReminder] User ${settings.userId}: Email sent successfully (to: ${settings.notificationEmail}, messageId: ${result.messageId}, smtp: ${result.smtpResponse})`
|
||||||
|
);
|
||||||
} else {
|
} else {
|
||||||
logger.error(`[IntakeReminder] User ${settings.userId}: Failed to send email: ${result.error}`);
|
logger.error(`[IntakeReminder] User ${settings.userId}: Failed to send email: ${result.error}`);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -8,6 +8,12 @@ import { doseTracking, medications, userSettings } from "../db/schema.js";
|
|||||||
import { getFooterHtml, getFooterPlain, getTranslations, type Language, t } from "../i18n/translations.js";
|
import { getFooterHtml, getFooterPlain, getTranslations, type Language, t } from "../i18n/translations.js";
|
||||||
import { getAllUserSettings, sendShoutrrrNotification, type UserSettings } from "../routes/settings.js";
|
import { getAllUserSettings, sendShoutrrrNotification, type UserSettings } from "../routes/settings.js";
|
||||||
import type { ServiceLogger } from "../utils/logger.js";
|
import type { ServiceLogger } from "../utils/logger.js";
|
||||||
|
import {
|
||||||
|
isAmountBasedPackageType,
|
||||||
|
isLiquidContainerPackageType,
|
||||||
|
isTubePackageType,
|
||||||
|
normalizePackageType,
|
||||||
|
} from "../utils/package-profiles.js";
|
||||||
// Import shared utilities
|
// Import shared utilities
|
||||||
import {
|
import {
|
||||||
type Blister,
|
type Blister,
|
||||||
@@ -19,6 +25,7 @@ import {
|
|||||||
getNextScheduledTime,
|
getNextScheduledTime,
|
||||||
getTimezone,
|
getTimezone,
|
||||||
getTodayInTimezone,
|
getTodayInTimezone,
|
||||||
|
normalizeIntakeUsageForStock,
|
||||||
parseIntakesJson,
|
parseIntakesJson,
|
||||||
parseLocalDateTime,
|
parseLocalDateTime,
|
||||||
parseReminderState,
|
parseReminderState,
|
||||||
@@ -37,6 +44,36 @@ function escapeHtml(text: string): string {
|
|||||||
return text.replace(/[&<>"']/g, (char) => htmlEscapes[char] || char);
|
return text.replace(/[&<>"']/g, (char) => htmlEscapes[char] || char);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type MailDeliveryInfo = {
|
||||||
|
accepted?: unknown;
|
||||||
|
rejected?: unknown;
|
||||||
|
response?: unknown;
|
||||||
|
};
|
||||||
|
|
||||||
|
function normalizeRecipients(value: unknown): string[] {
|
||||||
|
if (!Array.isArray(value)) return [];
|
||||||
|
return value
|
||||||
|
.map((entry) => (typeof entry === "string" ? entry : String(entry ?? "")))
|
||||||
|
.map((entry) => entry.trim())
|
||||||
|
.filter(Boolean);
|
||||||
|
}
|
||||||
|
|
||||||
|
function getDeliveryError(info: MailDeliveryInfo): string | null {
|
||||||
|
const accepted = normalizeRecipients(info.accepted);
|
||||||
|
const rejected = normalizeRecipients(info.rejected);
|
||||||
|
|
||||||
|
if (accepted.length > 0) return null;
|
||||||
|
if (rejected.length > 0) {
|
||||||
|
return `SMTP rejected all recipients: ${rejected.join(", ")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof info.response === "string" && info.response.trim()) {
|
||||||
|
return `SMTP did not confirm accepted recipients. Response: ${info.response}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return "SMTP did not confirm accepted recipients.";
|
||||||
|
}
|
||||||
|
|
||||||
const REMINDER_HOUR = parseInt(process.env.REMINDER_HOUR ?? "6", 10); // Default 6:00 AM local time
|
const REMINDER_HOUR = parseInt(process.env.REMINDER_HOUR ?? "6", 10); // Default 6:00 AM local time
|
||||||
|
|
||||||
const reminderStateFile = resolve(getDataDir(), "reminder-state.json");
|
const reminderStateFile = resolve(getDataDir(), "reminder-state.json");
|
||||||
@@ -179,6 +216,12 @@ type LowStockItem = {
|
|||||||
isCritical: boolean;
|
isCritical: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function getLiquidReminderThresholds(baselineDays: number): { lowDays: number; criticalDays: number } {
|
||||||
|
const lowDays = Math.max(1, Math.floor(baselineDays));
|
||||||
|
const criticalDays = Math.max(1, Math.ceil(lowDays / 2));
|
||||||
|
return { lowDays, criticalDays };
|
||||||
|
}
|
||||||
|
|
||||||
type PrescriptionReminderItem = {
|
type PrescriptionReminderItem = {
|
||||||
name: string;
|
name: string;
|
||||||
remainingRefills: number;
|
remainingRefills: number;
|
||||||
@@ -231,17 +274,25 @@ async function getMedicationsNeedingReminder(
|
|||||||
const msPerDay = 86_400_000;
|
const msPerDay = 86_400_000;
|
||||||
|
|
||||||
for (const row of rows) {
|
for (const row of rows) {
|
||||||
|
const packageType = normalizePackageType(row.packageType);
|
||||||
|
// Tube stock reminders are intentionally disabled:
|
||||||
|
// topical usage in grams cannot be mapped reliably to schedule events.
|
||||||
|
if (isTubePackageType(packageType)) continue;
|
||||||
|
|
||||||
const intakes = parseIntakesJson(
|
const intakes = parseIntakesJson(
|
||||||
row.intakesJson,
|
row.intakesJson,
|
||||||
{ usageJson: row.usageJson, everyJson: row.everyJson, startJson: row.startJson },
|
{ usageJson: row.usageJson, everyJson: row.everyJson, startJson: row.startJson },
|
||||||
row.intakeRemindersEnabled ?? false
|
row.intakeRemindersEnabled ?? false
|
||||||
);
|
);
|
||||||
const blisters: Blister[] = intakes.map((i) => ({ usage: i.usage, every: i.every, start: i.start }));
|
const blisters: Blister[] = intakes.map((i) => ({
|
||||||
|
usage: normalizeIntakeUsageForStock(i, row.medicationForm, row.packageType),
|
||||||
|
every: i.every,
|
||||||
|
start: i.start,
|
||||||
|
}));
|
||||||
|
|
||||||
const originalTotalPills =
|
const originalTotalPills = isAmountBasedPackageType(packageType)
|
||||||
(row.packageType ?? "blister") === "bottle"
|
? row.looseTablets + (row.stockAdjustment ?? 0)
|
||||||
? row.looseTablets + (row.stockAdjustment ?? 0)
|
: row.packCount * row.blistersPerPack * row.pillsPerBlister + row.looseTablets + (row.stockAdjustment ?? 0);
|
||||||
: row.packCount * row.blistersPerPack * row.pillsPerBlister + row.looseTablets + (row.stockAdjustment ?? 0);
|
|
||||||
|
|
||||||
const stockCorrectionCutoff = row.lastStockCorrectionAt ? new Date(row.lastStockCorrectionAt).getTime() : 0;
|
const stockCorrectionCutoff = row.lastStockCorrectionAt ? new Date(row.lastStockCorrectionAt).getTime() : 0;
|
||||||
const takenDoseIds = takenDoseIdsByMed.get(row.id) ?? new Set<string>();
|
const takenDoseIds = takenDoseIdsByMed.get(row.id) ?? new Set<string>();
|
||||||
@@ -348,8 +399,13 @@ async function getMedicationsNeedingReminder(
|
|||||||
|
|
||||||
if (daysLeft === null) continue;
|
if (daysLeft === null) continue;
|
||||||
|
|
||||||
const isCritical = daysLeft <= reminderDaysBefore;
|
const isLiquid = isLiquidContainerPackageType(packageType);
|
||||||
const isLow = daysLeft < lowStockDays;
|
const { lowDays, criticalDays } = isLiquid
|
||||||
|
? getLiquidReminderThresholds(reminderDaysBefore)
|
||||||
|
: { lowDays: lowStockDays, criticalDays: reminderDaysBefore };
|
||||||
|
|
||||||
|
const isCritical = daysLeft <= criticalDays;
|
||||||
|
const isLow = isLiquid ? daysLeft <= lowDays : daysLeft < lowDays;
|
||||||
|
|
||||||
if (isCritical || isLow) {
|
if (isCritical || isLow) {
|
||||||
lowStock.push({
|
lowStock.push({
|
||||||
@@ -551,7 +607,7 @@ ${getFooterPlain(language)}${isRepeatDaily ? `\n\n${tr.stockReminder.repeatDaily
|
|||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
await transporter.sendMail({
|
const mailResult = await transporter.sendMail({
|
||||||
from: smtpFrom,
|
from: smtpFrom,
|
||||||
to: email,
|
to: email,
|
||||||
subject,
|
subject,
|
||||||
@@ -559,6 +615,11 @@ ${getFooterPlain(language)}${isRepeatDaily ? `\n\n${tr.stockReminder.repeatDaily
|
|||||||
html,
|
html,
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const deliveryError = getDeliveryError(mailResult);
|
||||||
|
if (deliveryError) {
|
||||||
|
throw new Error(deliveryError);
|
||||||
|
}
|
||||||
|
|
||||||
return { success: true };
|
return { success: true };
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
||||||
@@ -724,85 +785,113 @@ async function checkAndSendReminderForUser(
|
|||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
try {
|
try {
|
||||||
logger.info(
|
// Re-check using fresh state after acquiring lock and pre-mark today as notified.
|
||||||
`[Reminder] User ${settings.userId}: Sending prescription reminder for ${allPrescriptionLow.length} medications...`
|
// This blocks duplicate sends when two reminder checks overlap in time.
|
||||||
);
|
const lockedState = loadReminderState();
|
||||||
|
const alreadyNotified = lockedState.notifiedMedications.includes(userPrescriptionNotifiedKey);
|
||||||
|
const shouldSend = !alreadyNotified || settings.repeatDailyReminders;
|
||||||
|
if (!shouldSend) {
|
||||||
|
logger.debug(
|
||||||
|
`[Reminder] User ${settings.userId}: prescription reminder already marked as sent today, skipping`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
const emptyRx = allPrescriptionLow.filter((m) => m.remainingRefills <= 0);
|
const preMarkedNotified =
|
||||||
const lowRx = allPrescriptionLow.filter((m) => m.remainingRefills > 0);
|
!shouldSend || alreadyNotified
|
||||||
const lines = allPrescriptionLow.map((m) => {
|
? lockedState.notifiedMedications
|
||||||
const expirySuffix = m.expiryDate ? t(tr.prescriptionReminder.expiresSuffix, { date: m.expiryDate }) : "";
|
: [...new Set([...lockedState.notifiedMedications, userPrescriptionNotifiedKey])];
|
||||||
if (m.remainingRefills <= 0) {
|
if (shouldSend && !alreadyNotified) {
|
||||||
return `- ${t(tr.prescriptionReminder.lineEmpty, {
|
saveReminderState({
|
||||||
|
lastAutoEmailSent: lockedState.lastAutoEmailSent,
|
||||||
|
lastAutoEmailDate: lockedState.lastAutoEmailDate,
|
||||||
|
lastStockSchedulerCheckDate: lockedState.lastStockSchedulerCheckDate,
|
||||||
|
notifiedMedications: preMarkedNotified,
|
||||||
|
nextScheduledCheck: lockedState.nextScheduledCheck,
|
||||||
|
lastNotificationType: lockedState.lastNotificationType,
|
||||||
|
lastNotificationChannel: lockedState.lastNotificationChannel,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (shouldSend) {
|
||||||
|
logger.info(
|
||||||
|
`[Reminder] User ${settings.userId}: Sending prescription reminder for ${allPrescriptionLow.length} medications...`
|
||||||
|
);
|
||||||
|
|
||||||
|
const emptyRx = allPrescriptionLow.filter((m) => m.remainingRefills <= 0);
|
||||||
|
const lowRx = allPrescriptionLow.filter((m) => m.remainingRefills > 0);
|
||||||
|
const lines = allPrescriptionLow.map((m) => {
|
||||||
|
const expirySuffix = m.expiryDate ? t(tr.prescriptionReminder.expiresSuffix, { date: m.expiryDate }) : "";
|
||||||
|
if (m.remainingRefills <= 0) {
|
||||||
|
return `- ${t(tr.prescriptionReminder.lineEmpty, {
|
||||||
|
name: m.name,
|
||||||
|
expirySuffix,
|
||||||
|
})}`;
|
||||||
|
}
|
||||||
|
return `- ${t(tr.prescriptionReminder.line, {
|
||||||
name: m.name,
|
name: m.name,
|
||||||
|
refills: m.remainingRefills,
|
||||||
expirySuffix,
|
expirySuffix,
|
||||||
})}`;
|
})}`;
|
||||||
}
|
});
|
||||||
return `- ${t(tr.prescriptionReminder.line, {
|
|
||||||
name: m.name,
|
|
||||||
refills: m.remainingRefills,
|
|
||||||
expirySuffix,
|
|
||||||
})}`;
|
|
||||||
});
|
|
||||||
|
|
||||||
let emailSuccess = false;
|
let emailSuccess = false;
|
||||||
let shoutrrrSuccess = false;
|
let shoutrrrSuccess = false;
|
||||||
|
|
||||||
if (prescriptionEmailEnabled) {
|
if (prescriptionEmailEnabled) {
|
||||||
const smtpHost = process.env.SMTP_HOST;
|
const smtpHost = process.env.SMTP_HOST;
|
||||||
const smtpUser = process.env.SMTP_USER;
|
const smtpUser = process.env.SMTP_USER;
|
||||||
const smtpPass = process.env.SMTP_TOKEN || process.env.SMTP_PASS;
|
const smtpPass = process.env.SMTP_TOKEN || process.env.SMTP_PASS;
|
||||||
const smtpPort = parseInt(process.env.SMTP_PORT ?? "587", 10);
|
const smtpPort = parseInt(process.env.SMTP_PORT ?? "587", 10);
|
||||||
const smtpSecure = process.env.SMTP_SECURE === "true";
|
const smtpSecure = process.env.SMTP_SECURE === "true";
|
||||||
const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
|
const smtpFrom = process.env.SMTP_FROM ?? smtpUser;
|
||||||
|
|
||||||
if (smtpHost && smtpUser) {
|
if (smtpHost && smtpUser) {
|
||||||
try {
|
try {
|
||||||
const transporter = nodemailer.createTransport({
|
const transporter = nodemailer.createTransport({
|
||||||
host: smtpHost,
|
host: smtpHost,
|
||||||
port: smtpPort,
|
port: smtpPort,
|
||||||
secure: smtpSecure,
|
secure: smtpSecure,
|
||||||
auth: { user: smtpUser, pass: smtpPass ?? "" },
|
auth: { user: smtpUser, pass: smtpPass ?? "" },
|
||||||
});
|
});
|
||||||
|
|
||||||
const subject =
|
const subject =
|
||||||
allPrescriptionLow.length === 1
|
allPrescriptionLow.length === 1
|
||||||
? tr.prescriptionReminder.subjectSingle
|
? tr.prescriptionReminder.subjectSingle
|
||||||
: t(tr.prescriptionReminder.subjectMultiple, { count: allPrescriptionLow.length });
|
: t(tr.prescriptionReminder.subjectMultiple, { count: allPrescriptionLow.length });
|
||||||
|
|
||||||
const bodyText =
|
const bodyText =
|
||||||
emptyRx.length > 0
|
emptyRx.length > 0
|
||||||
? tr.prescriptionReminder.descriptionEmpty
|
? tr.prescriptionReminder.descriptionEmpty
|
||||||
: tr.prescriptionReminder.descriptionLow;
|
: tr.prescriptionReminder.descriptionLow;
|
||||||
const emptyAlert =
|
const emptyAlert =
|
||||||
emptyRx.length === 1
|
emptyRx.length === 1
|
||||||
? tr.prescriptionReminder.alertEmptySingle
|
? tr.prescriptionReminder.alertEmptySingle
|
||||||
: t(tr.prescriptionReminder.alertEmptyMultiple, { count: emptyRx.length });
|
: t(tr.prescriptionReminder.alertEmptyMultiple, { count: emptyRx.length });
|
||||||
const lowAlert =
|
const lowAlert =
|
||||||
lowRx.length === 1
|
lowRx.length === 1
|
||||||
? tr.prescriptionReminder.alertLowSingle
|
? tr.prescriptionReminder.alertLowSingle
|
||||||
: t(tr.prescriptionReminder.alertLowMultiple, { count: lowRx.length });
|
: t(tr.prescriptionReminder.alertLowMultiple, { count: lowRx.length });
|
||||||
const alertText = emptyRx.length > 0 ? emptyAlert : lowAlert;
|
const alertText = emptyRx.length > 0 ? emptyAlert : lowAlert;
|
||||||
|
|
||||||
const tableRows = allPrescriptionLow
|
const tableRows = allPrescriptionLow
|
||||||
.map((item) => {
|
.map((item) => {
|
||||||
const isEmpty = item.remainingRefills <= 0;
|
const isEmpty = item.remainingRefills <= 0;
|
||||||
const safeName = escapeHtml(item.name);
|
const safeName = escapeHtml(item.name);
|
||||||
const safeRefills = Number(item.remainingRefills) || 0;
|
const safeRefills = Number(item.remainingRefills) || 0;
|
||||||
const safeThreshold = Number(item.lowThreshold) || 0;
|
const safeThreshold = Number(item.lowThreshold) || 0;
|
||||||
const safeExpiry = item.expiryDate ? escapeHtml(String(item.expiryDate)) : "-";
|
const safeExpiry = item.expiryDate ? escapeHtml(String(item.expiryDate)) : "-";
|
||||||
const rowBg = isEmpty ? "#fef2f2" : "white";
|
const rowBg = isEmpty ? "#fef2f2" : "white";
|
||||||
return `
|
return `
|
||||||
<tr style="background: ${rowBg};">
|
<tr style="background: ${rowBg};">
|
||||||
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; white-space: nowrap;">${isEmpty ? "🚨" : "⚠️"} ${safeName}</td>
|
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; white-space: nowrap;">${isEmpty ? "🚨" : "⚠️"} ${safeName}</td>
|
||||||
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap; ${isEmpty ? "color: #dc2626; font-weight: 600;" : ""}"><strong>${safeRefills}</strong></td>
|
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap; ${isEmpty ? "color: #dc2626; font-weight: 600;" : ""}"><strong>${safeRefills}</strong></td>
|
||||||
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${safeThreshold}</td>
|
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${safeThreshold}</td>
|
||||||
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${safeExpiry}</td>
|
<td style="padding: 10px 12px; border-bottom: 1px solid #e5e7eb; text-align: center; white-space: nowrap;">${safeExpiry}</td>
|
||||||
</tr>`;
|
</tr>`;
|
||||||
})
|
})
|
||||||
.join("");
|
.join("");
|
||||||
|
|
||||||
const html = `
|
const html = `
|
||||||
<div style="font-family: system-ui, -apple-system, sans-serif; max-width: 100%; margin: 0 auto; padding: 12px; background: #f9fafb;">
|
<div style="font-family: system-ui, -apple-system, sans-serif; max-width: 100%; margin: 0 auto; padding: 12px; background: #f9fafb;">
|
||||||
<div style="background: white; border-radius: 12px; padding: 16px; box-shadow: 0 1px 3px rgba(0,0,0,0.1);">
|
<div style="background: white; border-radius: 12px; padding: 16px; box-shadow: 0 1px 3px rgba(0,0,0,0.1);">
|
||||||
<h2 style="color: #1f2937; margin: 0 0 8px; font-size: 18px;">${emptyRx.length > 0 ? tr.prescriptionReminder.titleEmpty : tr.prescriptionReminder.title}</h2>
|
<h2 style="color: #1f2937; margin: 0 0 8px; font-size: 18px;">${emptyRx.length > 0 ? tr.prescriptionReminder.titleEmpty : tr.prescriptionReminder.title}</h2>
|
||||||
@@ -842,76 +931,97 @@ async function checkAndSendReminderForUser(
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
`;
|
`;
|
||||||
const text = `${emptyRx.length > 0 ? tr.prescriptionReminder.titleEmpty : tr.prescriptionReminder.title}\n\n${bodyText}\n\n${lines.join("\n")}\n\n---\n${getFooterPlain(language)}${settings.repeatDailyReminders ? `\n\n${tr.prescriptionReminder.repeatDailyNote}` : ""}`;
|
const text = `${emptyRx.length > 0 ? tr.prescriptionReminder.titleEmpty : tr.prescriptionReminder.title}\n\n${bodyText}\n\n${lines.join("\n")}\n\n---\n${getFooterPlain(language)}${settings.repeatDailyReminders ? `\n\n${tr.prescriptionReminder.repeatDailyNote}` : ""}`;
|
||||||
|
|
||||||
await transporter.sendMail({
|
const mailResult = await transporter.sendMail({
|
||||||
from: smtpFrom,
|
from: smtpFrom,
|
||||||
to: settings.notificationEmail!,
|
to: settings.notificationEmail!,
|
||||||
subject,
|
subject,
|
||||||
text,
|
text,
|
||||||
html,
|
html,
|
||||||
});
|
});
|
||||||
emailSuccess = true;
|
const deliveryError = getDeliveryError(mailResult);
|
||||||
} catch (error) {
|
if (deliveryError) {
|
||||||
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
throw new Error(deliveryError);
|
||||||
logger.error(`[Reminder] User ${settings.userId}: Failed to send prescription email: ${errorMessage}`);
|
}
|
||||||
|
emailSuccess = true;
|
||||||
|
} catch (error) {
|
||||||
|
const errorMessage = error instanceof Error ? error.message : "Unknown error";
|
||||||
|
logger.error(
|
||||||
|
`[Reminder] User ${settings.userId}: Failed to send prescription email: ${errorMessage}`
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
if (prescriptionPushEnabled) {
|
if (prescriptionPushEnabled) {
|
||||||
const titleParts: string[] = [];
|
const titleParts: string[] = [];
|
||||||
if (emptyRx.length > 0)
|
if (emptyRx.length > 0)
|
||||||
titleParts.push(
|
titleParts.push(
|
||||||
`🚨 ${emptyRx.length} ${emptyRx.length === 1 ? tr.prescriptionReminder.pushEmptySingle : tr.prescriptionReminder.pushEmpty}`
|
`🚨 ${emptyRx.length} ${emptyRx.length === 1 ? tr.prescriptionReminder.pushEmptySingle : tr.prescriptionReminder.pushEmpty}`
|
||||||
);
|
|
||||||
if (lowRx.length > 0)
|
|
||||||
titleParts.push(
|
|
||||||
`🚨 ${lowRx.length} ${lowRx.length === 1 ? tr.prescriptionReminder.pushLowSingle : tr.prescriptionReminder.pushLow}`
|
|
||||||
);
|
|
||||||
const title = `MedAssist-ng: ${titleParts.join(", ")} - ${tr.prescriptionReminder.pushRenewNow}`;
|
|
||||||
|
|
||||||
const messageParts: string[] = [];
|
|
||||||
if (emptyRx.length > 0) {
|
|
||||||
messageParts.push(`🚨 ${tr.prescriptionReminder.pushEmptySection}:`);
|
|
||||||
for (const m of emptyRx) {
|
|
||||||
messageParts.push(` • ${m.name}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (lowRx.length > 0) {
|
|
||||||
if (emptyRx.length > 0) messageParts.push("");
|
|
||||||
messageParts.push(`🚨 ${tr.prescriptionReminder.pushLowSection}:`);
|
|
||||||
for (const m of lowRx) {
|
|
||||||
messageParts.push(
|
|
||||||
` • ${m.name}: ${t(tr.prescriptionReminder.pushRefillsLeft, { count: m.remainingRefills })}`
|
|
||||||
);
|
);
|
||||||
|
if (lowRx.length > 0)
|
||||||
|
titleParts.push(
|
||||||
|
`🚨 ${lowRx.length} ${lowRx.length === 1 ? tr.prescriptionReminder.pushLowSingle : tr.prescriptionReminder.pushLow}`
|
||||||
|
);
|
||||||
|
const title = `MedAssist-ng: ${titleParts.join(", ")} - ${tr.prescriptionReminder.pushRenewNow}`;
|
||||||
|
|
||||||
|
const messageParts: string[] = [];
|
||||||
|
if (emptyRx.length > 0) {
|
||||||
|
messageParts.push(`🚨 ${tr.prescriptionReminder.pushEmptySection}:`);
|
||||||
|
for (const m of emptyRx) {
|
||||||
|
messageParts.push(` • ${m.name}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (lowRx.length > 0) {
|
||||||
|
if (emptyRx.length > 0) messageParts.push("");
|
||||||
|
messageParts.push(`🚨 ${tr.prescriptionReminder.pushLowSection}:`);
|
||||||
|
for (const m of lowRx) {
|
||||||
|
messageParts.push(
|
||||||
|
` • ${m.name}: ${t(tr.prescriptionReminder.pushRefillsLeft, { count: m.remainingRefills })}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const message = `${messageParts.join("\n")}\n\n---\n${getFooterPlain(language)}`;
|
||||||
|
const result = await sendShoutrrrNotification(settings.shoutrrrUrl!, title, message);
|
||||||
|
shoutrrrSuccess = result.success;
|
||||||
|
if (!result.success) {
|
||||||
|
logger.error(`[Reminder] User ${settings.userId}: Failed to send prescription push: ${result.error}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
const message = `${messageParts.join("\n")}\n\n---\n${getFooterPlain(language)}`;
|
|
||||||
const result = await sendShoutrrrNotification(settings.shoutrrrUrl!, title, message);
|
if (emailSuccess || shoutrrrSuccess) {
|
||||||
shoutrrrSuccess = result.success;
|
const currentState = loadReminderState();
|
||||||
if (!result.success) {
|
const singleChannel = emailSuccess ? "email" : "push";
|
||||||
logger.error(`[Reminder] User ${settings.userId}: Failed to send prescription push: ${result.error}`);
|
const channel = emailSuccess && shoutrrrSuccess ? "both" : singleChannel;
|
||||||
|
saveReminderState({
|
||||||
|
lastAutoEmailSent: new Date().toISOString(),
|
||||||
|
lastAutoEmailDate: today,
|
||||||
|
lastStockSchedulerCheckDate: currentState.lastStockSchedulerCheckDate,
|
||||||
|
notifiedMedications: [...new Set([...currentState.notifiedMedications, userPrescriptionNotifiedKey])],
|
||||||
|
nextScheduledCheck: currentState.nextScheduledCheck,
|
||||||
|
lastNotificationType: "prescription",
|
||||||
|
lastNotificationChannel: channel,
|
||||||
|
});
|
||||||
|
|
||||||
|
const medNames = allPrescriptionLow.map((m) => m.name).join(", ");
|
||||||
|
await updateUserReminderSentTime(settings.userId, "prescription", channel, medNames);
|
||||||
|
} else if (!alreadyNotified) {
|
||||||
|
// Roll back pre-mark when both channels failed so retries remain possible.
|
||||||
|
const currentState = loadReminderState();
|
||||||
|
saveReminderState({
|
||||||
|
lastAutoEmailSent: currentState.lastAutoEmailSent,
|
||||||
|
lastAutoEmailDate: currentState.lastAutoEmailDate,
|
||||||
|
lastStockSchedulerCheckDate: currentState.lastStockSchedulerCheckDate,
|
||||||
|
notifiedMedications: currentState.notifiedMedications.filter(
|
||||||
|
(key) => key !== userPrescriptionNotifiedKey
|
||||||
|
),
|
||||||
|
nextScheduledCheck: currentState.nextScheduledCheck,
|
||||||
|
lastNotificationType: currentState.lastNotificationType,
|
||||||
|
lastNotificationChannel: currentState.lastNotificationChannel,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (emailSuccess || shoutrrrSuccess) {
|
|
||||||
const currentState = loadReminderState();
|
|
||||||
const singleChannel = emailSuccess ? "email" : "push";
|
|
||||||
const channel = emailSuccess && shoutrrrSuccess ? "both" : singleChannel;
|
|
||||||
saveReminderState({
|
|
||||||
lastAutoEmailSent: new Date().toISOString(),
|
|
||||||
lastAutoEmailDate: today,
|
|
||||||
lastStockSchedulerCheckDate: currentState.lastStockSchedulerCheckDate,
|
|
||||||
notifiedMedications: [...new Set([...currentState.notifiedMedications, userPrescriptionNotifiedKey])],
|
|
||||||
nextScheduledCheck: currentState.nextScheduledCheck,
|
|
||||||
lastNotificationType: "prescription",
|
|
||||||
lastNotificationChannel: channel,
|
|
||||||
});
|
|
||||||
|
|
||||||
const medNames = allPrescriptionLow.map((m) => m.name).join(", ");
|
|
||||||
await updateUserReminderSentTime(settings.userId, "prescription", channel, medNames);
|
|
||||||
}
|
|
||||||
} finally {
|
} finally {
|
||||||
releaseReminderSendLock(prescriptionSendLock);
|
releaseReminderSendLock(prescriptionSendLock);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -28,7 +28,7 @@ vi.mock("../db/client.js", () => ({
|
|||||||
vi.mock("../plugins/env.js", () => ({
|
vi.mock("../plugins/env.js", () => ({
|
||||||
env: {
|
env: {
|
||||||
AUTH_ENABLED: true,
|
AUTH_ENABLED: true,
|
||||||
LOCAL_AUTH_ENABLED: true,
|
FORM_LOGIN_ENABLED: true,
|
||||||
REGISTRATION_ENABLED: true,
|
REGISTRATION_ENABLED: true,
|
||||||
OIDC_ENABLED: false,
|
OIDC_ENABLED: false,
|
||||||
NODE_ENV: "test",
|
NODE_ENV: "test",
|
||||||
@@ -144,7 +144,7 @@ describe("Auth Routes (AUTH_ENABLED=true)", () => {
|
|||||||
const data = response.json();
|
const data = response.json();
|
||||||
expect(data.authEnabled).toBe(true);
|
expect(data.authEnabled).toBe(true);
|
||||||
expect(data.registrationEnabled).toBe(true);
|
expect(data.registrationEnabled).toBe(true);
|
||||||
expect(data.localAuthEnabled).toBe(true);
|
expect(data.formLoginEnabled).toBe(true);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -32,8 +32,8 @@ async function loadDbClientModule(options: ClientTestOptions = {}) {
|
|||||||
.mockReturnValue(dirWritable ? { success: true } : { success: false, error: "permission denied" });
|
.mockReturnValue(dirWritable ? { success: true } : { success: false, error: "permission denied" });
|
||||||
const getDbPaths = vi.fn().mockReturnValue({
|
const getDbPaths = vi.fn().mockReturnValue({
|
||||||
dataDir: "/tmp/medassist-data",
|
dataDir: "/tmp/medassist-data",
|
||||||
dbPath: "/tmp/medassist-data/medassist.db",
|
dbPath: "/tmp/medassist-data/medassist-ng.db",
|
||||||
url: "file:/tmp/medassist-data/medassist.db",
|
url: "file:/tmp/medassist-data/medassist-ng.db",
|
||||||
});
|
});
|
||||||
const runDrizzleMigrations = vi.fn().mockResolvedValue({ success: true });
|
const runDrizzleMigrations = vi.fn().mockResolvedValue({ success: true });
|
||||||
const runAlterMigrations = vi.fn().mockResolvedValue({ errors: [] });
|
const runAlterMigrations = vi.fn().mockResolvedValue({ errors: [] });
|
||||||
@@ -102,7 +102,7 @@ describe("db/client bootstrap", () => {
|
|||||||
await mod.migrationsReady;
|
await mod.migrationsReady;
|
||||||
|
|
||||||
expect(mocks.ensureDataDirectory).toHaveBeenCalledWith("/tmp/medassist-data");
|
expect(mocks.ensureDataDirectory).toHaveBeenCalledWith("/tmp/medassist-data");
|
||||||
expect(mocks.createClient).toHaveBeenCalledWith({ url: "file:/tmp/medassist-data/medassist.db" });
|
expect(mocks.createClient).toHaveBeenCalledWith({ url: "file:/tmp/medassist-data/medassist-ng.db" });
|
||||||
expect(mocks.runDrizzleMigrations).toHaveBeenCalledTimes(1);
|
expect(mocks.runDrizzleMigrations).toHaveBeenCalledTimes(1);
|
||||||
expect(mocks.runAlterMigrations).toHaveBeenCalledTimes(1);
|
expect(mocks.runAlterMigrations).toHaveBeenCalledTimes(1);
|
||||||
expect(mocks.repairTrailingHyphenDoseIds).toHaveBeenCalledTimes(1);
|
expect(mocks.repairTrailingHyphenDoseIds).toHaveBeenCalledTimes(1);
|
||||||
|
|||||||
@@ -82,7 +82,12 @@ async function createSchema(client: Client) {
|
|||||||
name text NOT NULL,
|
name text NOT NULL,
|
||||||
generic_name text,
|
generic_name text,
|
||||||
taken_by_json text NOT NULL DEFAULT '[]',
|
taken_by_json text NOT NULL DEFAULT '[]',
|
||||||
|
medication_form text NOT NULL DEFAULT 'tablet',
|
||||||
|
pill_form text,
|
||||||
|
lifecycle_category text NOT NULL DEFAULT 'refill_when_empty',
|
||||||
package_type text NOT NULL DEFAULT 'blister',
|
package_type text NOT NULL DEFAULT 'blister',
|
||||||
|
package_amount_value integer NOT NULL DEFAULT 0,
|
||||||
|
package_amount_unit text NOT NULL DEFAULT 'ml',
|
||||||
pack_count integer NOT NULL DEFAULT 1,
|
pack_count integer NOT NULL DEFAULT 1,
|
||||||
blisters_per_pack integer NOT NULL DEFAULT 1,
|
blisters_per_pack integer NOT NULL DEFAULT 1,
|
||||||
pills_per_blister integer NOT NULL DEFAULT 1,
|
pills_per_blister integer NOT NULL DEFAULT 1,
|
||||||
@@ -101,6 +106,8 @@ async function createSchema(client: Client) {
|
|||||||
notes text,
|
notes text,
|
||||||
intake_reminders_enabled integer NOT NULL DEFAULT 0,
|
intake_reminders_enabled integer NOT NULL DEFAULT 0,
|
||||||
medication_start_date text NOT NULL DEFAULT '',
|
medication_start_date text NOT NULL DEFAULT '',
|
||||||
|
medication_end_date text,
|
||||||
|
auto_mark_obsolete_after_end_date integer NOT NULL DEFAULT 1,
|
||||||
is_obsolete integer NOT NULL DEFAULT 0,
|
is_obsolete integer NOT NULL DEFAULT 0,
|
||||||
obsolete_at integer,
|
obsolete_at integer,
|
||||||
prescription_enabled integer NOT NULL DEFAULT 0,
|
prescription_enabled integer NOT NULL DEFAULT 0,
|
||||||
@@ -868,7 +875,6 @@ describe("E2E Tests with Real Routes", () => {
|
|||||||
const json = response.json();
|
const json = response.json();
|
||||||
expect(json.status).toBe("ok");
|
expect(json.status).toBe("ok");
|
||||||
expect(typeof json.smtpConfigured).toBe("boolean");
|
expect(typeof json.smtpConfigured).toBe("boolean");
|
||||||
expect(typeof json.shoutrrrConfigured).toBe("boolean");
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -1289,7 +1295,6 @@ describe("E2E Tests with Real Routes", () => {
|
|||||||
const json = response.json();
|
const json = response.json();
|
||||||
expect(json.status).toBe("ok");
|
expect(json.status).toBe("ok");
|
||||||
expect(typeof json.smtpConfigured).toBe("boolean");
|
expect(typeof json.smtpConfigured).toBe("boolean");
|
||||||
expect(typeof json.shoutrrrConfigured).toBe("boolean");
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -2501,10 +2506,10 @@ describe("E2E Tests with Real Routes", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
// Package Type (bottle vs blister) Tests
|
// Package Type (blister, bottle, liquid_container) Tests
|
||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
|
|
||||||
describe("Package type handling (bottle vs blister)", () => {
|
describe("Package type handling (blister, bottle, liquid_container)", () => {
|
||||||
const bottleMedication = {
|
const bottleMedication = {
|
||||||
name: "Vitamin D Drops",
|
name: "Vitamin D Drops",
|
||||||
packageType: "bottle",
|
packageType: "bottle",
|
||||||
@@ -2525,6 +2530,18 @@ describe("E2E Tests with Real Routes", () => {
|
|||||||
blisters: [{ usage: 1, every: 1, start: "2025-01-01T08:00:00.000Z" }],
|
blisters: [{ usage: 1, every: 1, start: "2025-01-01T08:00:00.000Z" }],
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const liquidContainerMedication = {
|
||||||
|
name: "Cough Syrup",
|
||||||
|
medicationForm: "liquid",
|
||||||
|
packageType: "liquid_container",
|
||||||
|
doseUnit: "ml",
|
||||||
|
packCount: 0,
|
||||||
|
blistersPerPack: 1,
|
||||||
|
pillsPerBlister: 1,
|
||||||
|
looseTablets: 180,
|
||||||
|
blisters: [{ usage: 5, every: 1, start: "2025-01-01T08:00:00.000Z" }],
|
||||||
|
};
|
||||||
|
|
||||||
it("should create and return bottle type medication", async () => {
|
it("should create and return bottle type medication", async () => {
|
||||||
const response = await app.inject({
|
const response = await app.inject({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
@@ -2569,6 +2586,49 @@ describe("E2E Tests with Real Routes", () => {
|
|||||||
expect(data.medications[0].totalPills).toBe(120);
|
expect(data.medications[0].totalPills).toBe(120);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("should create and return liquid_container type medication", async () => {
|
||||||
|
const response = await app.inject({
|
||||||
|
method: "POST",
|
||||||
|
url: "/medications",
|
||||||
|
payload: liquidContainerMedication,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(response.statusCode).toBe(200);
|
||||||
|
const data = response.json();
|
||||||
|
expect(data.packageType).toBe("liquid_container");
|
||||||
|
expect(data.medicationForm).toBe("liquid");
|
||||||
|
expect(data.doseUnit).toBe("ml");
|
||||||
|
expect(data.looseTablets).toBe(180);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should return packageType and ml-based stock semantics in shared schedule for liquid_container", async () => {
|
||||||
|
await app.inject({
|
||||||
|
method: "POST",
|
||||||
|
url: "/medications",
|
||||||
|
payload: { ...liquidContainerMedication, takenBy: ["Daniel"] },
|
||||||
|
});
|
||||||
|
|
||||||
|
const shareResponse = await app.inject({
|
||||||
|
method: "POST",
|
||||||
|
url: "/share",
|
||||||
|
payload: { takenBy: "Daniel", scheduleDays: 30 },
|
||||||
|
});
|
||||||
|
expect(shareResponse.statusCode).toBe(200);
|
||||||
|
const { token } = shareResponse.json();
|
||||||
|
|
||||||
|
const scheduleResponse = await app.inject({
|
||||||
|
method: "GET",
|
||||||
|
url: `/share/${token}`,
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(scheduleResponse.statusCode).toBe(200);
|
||||||
|
const data = scheduleResponse.json();
|
||||||
|
expect(data.medications).toHaveLength(1);
|
||||||
|
expect(data.medications[0].packageType).toBe("liquid_container");
|
||||||
|
// Liquid container follows container semantics (stock from looseTablets only).
|
||||||
|
expect(data.medications[0].totalPills).toBe(180);
|
||||||
|
});
|
||||||
|
|
||||||
it("should calculate correct totalPills for shared blister medication", async () => {
|
it("should calculate correct totalPills for shared blister medication", async () => {
|
||||||
await app.inject({
|
await app.inject({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
@@ -2744,5 +2804,18 @@ describe("E2E Tests with Real Routes", () => {
|
|||||||
expect(medsResponse.json()).toHaveLength(1);
|
expect(medsResponse.json()).toHaveLength(1);
|
||||||
expect(medsResponse.json()[0].packageType).toBe("blister");
|
expect(medsResponse.json()[0].packageType).toBe("blister");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("should reject liquid medication form with non-liquid package type", async () => {
|
||||||
|
const response = await app.inject({
|
||||||
|
method: "POST",
|
||||||
|
url: "/medications",
|
||||||
|
payload: {
|
||||||
|
...liquidContainerMedication,
|
||||||
|
packageType: "bottle",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
expect(response.statusCode).toBe(400);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -76,7 +76,12 @@ async function createSchema(client: Client) {
|
|||||||
name text NOT NULL,
|
name text NOT NULL,
|
||||||
generic_name text,
|
generic_name text,
|
||||||
taken_by_json text NOT NULL DEFAULT '[]',
|
taken_by_json text NOT NULL DEFAULT '[]',
|
||||||
|
medication_form text NOT NULL DEFAULT 'tablet',
|
||||||
|
pill_form text,
|
||||||
|
lifecycle_category text NOT NULL DEFAULT 'refill_when_empty',
|
||||||
package_type text NOT NULL DEFAULT 'blister',
|
package_type text NOT NULL DEFAULT 'blister',
|
||||||
|
package_amount_value integer NOT NULL DEFAULT 0,
|
||||||
|
package_amount_unit text NOT NULL DEFAULT 'ml',
|
||||||
pack_count integer NOT NULL DEFAULT 1,
|
pack_count integer NOT NULL DEFAULT 1,
|
||||||
blisters_per_pack integer NOT NULL DEFAULT 1,
|
blisters_per_pack integer NOT NULL DEFAULT 1,
|
||||||
pills_per_blister integer NOT NULL DEFAULT 1,
|
pills_per_blister integer NOT NULL DEFAULT 1,
|
||||||
@@ -95,6 +100,8 @@ async function createSchema(client: Client) {
|
|||||||
notes text,
|
notes text,
|
||||||
intake_reminders_enabled integer NOT NULL DEFAULT 0,
|
intake_reminders_enabled integer NOT NULL DEFAULT 0,
|
||||||
medication_start_date text NOT NULL DEFAULT '',
|
medication_start_date text NOT NULL DEFAULT '',
|
||||||
|
medication_end_date text,
|
||||||
|
auto_mark_obsolete_after_end_date integer NOT NULL DEFAULT 1,
|
||||||
is_obsolete integer NOT NULL DEFAULT 0,
|
is_obsolete integer NOT NULL DEFAULT 0,
|
||||||
obsolete_at integer,
|
obsolete_at integer,
|
||||||
prescription_enabled integer NOT NULL DEFAULT 0,
|
prescription_enabled integer NOT NULL DEFAULT 0,
|
||||||
|
|||||||
@@ -93,7 +93,12 @@ async function createSchema(client: Client) {
|
|||||||
name text NOT NULL,
|
name text NOT NULL,
|
||||||
generic_name text,
|
generic_name text,
|
||||||
taken_by_json text NOT NULL DEFAULT '[]',
|
taken_by_json text NOT NULL DEFAULT '[]',
|
||||||
|
medication_form text NOT NULL DEFAULT 'tablet',
|
||||||
|
pill_form text,
|
||||||
|
lifecycle_category text NOT NULL DEFAULT 'refill_when_empty',
|
||||||
package_type text NOT NULL DEFAULT 'blister',
|
package_type text NOT NULL DEFAULT 'blister',
|
||||||
|
package_amount_value integer NOT NULL DEFAULT 0,
|
||||||
|
package_amount_unit text NOT NULL DEFAULT 'ml',
|
||||||
pack_count integer NOT NULL DEFAULT 1,
|
pack_count integer NOT NULL DEFAULT 1,
|
||||||
blisters_per_pack integer NOT NULL DEFAULT 1,
|
blisters_per_pack integer NOT NULL DEFAULT 1,
|
||||||
pills_per_blister integer NOT NULL DEFAULT 1,
|
pills_per_blister integer NOT NULL DEFAULT 1,
|
||||||
@@ -112,6 +117,8 @@ async function createSchema(client: Client) {
|
|||||||
notes text,
|
notes text,
|
||||||
intake_reminders_enabled integer NOT NULL DEFAULT 0,
|
intake_reminders_enabled integer NOT NULL DEFAULT 0,
|
||||||
medication_start_date text NOT NULL DEFAULT '',
|
medication_start_date text NOT NULL DEFAULT '',
|
||||||
|
medication_end_date text,
|
||||||
|
auto_mark_obsolete_after_end_date integer NOT NULL DEFAULT 1,
|
||||||
is_obsolete integer NOT NULL DEFAULT 0,
|
is_obsolete integer NOT NULL DEFAULT 0,
|
||||||
obsolete_at integer,
|
obsolete_at integer,
|
||||||
prescription_enabled integer NOT NULL DEFAULT 0,
|
prescription_enabled integer NOT NULL DEFAULT 0,
|
||||||
@@ -284,7 +291,7 @@ describe("Planner Routes", () => {
|
|||||||
args: [999999999],
|
args: [999999999],
|
||||||
});
|
});
|
||||||
|
|
||||||
mockSendMail.mockResolvedValueOnce({ messageId: "123" });
|
mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
|
||||||
|
|
||||||
const response = await app.inject({
|
const response = await app.inject({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
@@ -330,7 +337,7 @@ describe("Planner Routes", () => {
|
|||||||
args: [999999999],
|
args: [999999999],
|
||||||
});
|
});
|
||||||
|
|
||||||
mockSendMail.mockResolvedValueOnce({ messageId: "123" });
|
mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
|
||||||
|
|
||||||
const response = await app.inject({
|
const response = await app.inject({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
@@ -434,7 +441,7 @@ describe("Planner Routes", () => {
|
|||||||
args: [999999999],
|
args: [999999999],
|
||||||
});
|
});
|
||||||
|
|
||||||
mockSendMail.mockResolvedValueOnce({ messageId: "123" });
|
mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
|
||||||
|
|
||||||
const response = await app.inject({
|
const response = await app.inject({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
@@ -522,7 +529,7 @@ describe("Planner Routes", () => {
|
|||||||
args: [999999999],
|
args: [999999999],
|
||||||
});
|
});
|
||||||
|
|
||||||
mockSendMail.mockResolvedValueOnce({ messageId: "123" });
|
mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
|
||||||
mockSendShoutrrr.mockResolvedValueOnce({ success: true });
|
mockSendShoutrrr.mockResolvedValueOnce({ success: true });
|
||||||
|
|
||||||
const response = await app.inject({
|
const response = await app.inject({
|
||||||
@@ -697,7 +704,7 @@ describe("Planner Routes", () => {
|
|||||||
args: [999999999],
|
args: [999999999],
|
||||||
});
|
});
|
||||||
|
|
||||||
mockSendMail.mockResolvedValueOnce({ messageId: "123" });
|
mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
|
||||||
|
|
||||||
const response = await app.inject({
|
const response = await app.inject({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
@@ -727,7 +734,7 @@ describe("Planner Routes", () => {
|
|||||||
args: [999999999],
|
args: [999999999],
|
||||||
});
|
});
|
||||||
|
|
||||||
mockSendMail.mockResolvedValueOnce({ messageId: "123" });
|
mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
|
||||||
|
|
||||||
const response = await app.inject({
|
const response = await app.inject({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
@@ -763,7 +770,7 @@ describe("Planner Routes", () => {
|
|||||||
args: [999999999],
|
args: [999999999],
|
||||||
});
|
});
|
||||||
|
|
||||||
mockSendMail.mockResolvedValueOnce({ messageId: "123" });
|
mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
|
||||||
|
|
||||||
const response = await app.inject({
|
const response = await app.inject({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
@@ -849,7 +856,7 @@ describe("Planner Routes", () => {
|
|||||||
args: [999999999],
|
args: [999999999],
|
||||||
});
|
});
|
||||||
|
|
||||||
mockSendMail.mockResolvedValueOnce({ messageId: "123" });
|
mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
|
||||||
mockSendShoutrrr.mockResolvedValueOnce({ success: true });
|
mockSendShoutrrr.mockResolvedValueOnce({ success: true });
|
||||||
|
|
||||||
const response = await app.inject({
|
const response = await app.inject({
|
||||||
@@ -982,7 +989,7 @@ describe("Planner Routes", () => {
|
|||||||
args: [999999999],
|
args: [999999999],
|
||||||
});
|
});
|
||||||
|
|
||||||
mockSendMail.mockResolvedValueOnce({ messageId: "123" });
|
mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
|
||||||
|
|
||||||
const response = await app.inject({
|
const response = await app.inject({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
@@ -1036,6 +1043,36 @@ describe("Planner Routes", () => {
|
|||||||
expect(title).not.toContain("Low");
|
expect(title).not.toContain("Low");
|
||||||
expect(message).toContain("Running critically low");
|
expect(message).toContain("Running critically low");
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("should return 400 when only tube medications are in active meds", async () => {
|
||||||
|
// Insert a tube medication (should be excluded from reminders)
|
||||||
|
await testClient.execute({
|
||||||
|
sql: `INSERT INTO medications (id, user_id, name, taken_by_json, usage_json, every_json, start_json, package_type)
|
||||||
|
VALUES (3, 999999999, 'Ointment', '[]', '[]', '[]', '[]', 'tube')`,
|
||||||
|
args: [],
|
||||||
|
});
|
||||||
|
|
||||||
|
await testClient.execute({
|
||||||
|
sql: `INSERT INTO user_settings (user_id, email_enabled, shoutrrr_enabled, language) VALUES (?, 1, 0, 'en')`,
|
||||||
|
args: [999999999],
|
||||||
|
});
|
||||||
|
|
||||||
|
mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
|
||||||
|
|
||||||
|
const response = await app.inject({
|
||||||
|
method: "POST",
|
||||||
|
url: "/reminder/send-email",
|
||||||
|
payload: {
|
||||||
|
email: "test@example.com",
|
||||||
|
lowStock: [{ name: "Ointment", medsLeft: 5, daysLeft: 10, depletionDate: "2025-01-13" }],
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
// Expects 400 because tube medications are excluded from stock reminders
|
||||||
|
expect(response.statusCode).toBe(400);
|
||||||
|
expect(response.json()).toEqual({ error: "No active medications to notify" });
|
||||||
|
expect(mockSendMail).not.toHaveBeenCalled();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("POST /reminder/send-prescription", () => {
|
describe("POST /reminder/send-prescription", () => {
|
||||||
@@ -1082,7 +1119,7 @@ describe("Planner Routes", () => {
|
|||||||
args: [999999999],
|
args: [999999999],
|
||||||
});
|
});
|
||||||
|
|
||||||
mockSendMail.mockResolvedValueOnce({ messageId: "123" });
|
mockSendMail.mockResolvedValueOnce({ messageId: "123", accepted: ["test.com"], rejected: [] });
|
||||||
|
|
||||||
const response = await app.inject({
|
const response = await app.inject({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
|
|||||||
@@ -207,7 +207,12 @@ describe("Real route coverage: settings/export/report", () => {
|
|||||||
process.env.SMTP_HOST = "smtp.example.com";
|
process.env.SMTP_HOST = "smtp.example.com";
|
||||||
process.env.SMTP_USER = "mailer@example.com";
|
process.env.SMTP_USER = "mailer@example.com";
|
||||||
process.env.SMTP_TOKEN = "secret";
|
process.env.SMTP_TOKEN = "secret";
|
||||||
nodemailerSendMail.mockResolvedValue(undefined);
|
nodemailerSendMail.mockResolvedValue({
|
||||||
|
accepted: ["person@example.com"],
|
||||||
|
rejected: [],
|
||||||
|
response: "250 2.0.0 OK",
|
||||||
|
messageId: "test-message-id",
|
||||||
|
});
|
||||||
|
|
||||||
const response = await app.inject({
|
const response = await app.inject({
|
||||||
method: "POST",
|
method: "POST",
|
||||||
|
|||||||
@@ -348,3 +348,46 @@ describe("Stock semantics parity (planner usage vs scheduler)", () => {
|
|||||||
expect(lowStock.some((r) => r.name === "Obsolete Med")).toBe(false);
|
expect(lowStock.some((r) => r.name === "Obsolete Med")).toBe(false);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("getLiquidReminderThresholds", () => {
|
||||||
|
// Import the function for testing (test-only export)
|
||||||
|
// The function is: getLiquidReminderThresholds(baselineDays: number): { lowDays: number; criticalDays: number }
|
||||||
|
// Formula: lowDays = baselineDays, criticalDays = ceil(lowDays / 2)
|
||||||
|
|
||||||
|
it("derives critical as ceil(baseline / 2) for typical baseline", () => {
|
||||||
|
// For baseline=7 days: low=7, critical=ceil(7/2)=4
|
||||||
|
const baseline = 7;
|
||||||
|
// Manually apply the formula to verify
|
||||||
|
const expectedLow = Math.max(1, Math.floor(baseline));
|
||||||
|
const expectedCritical = Math.max(1, Math.ceil(expectedLow / 2));
|
||||||
|
expect(expectedLow).toBe(7);
|
||||||
|
expect(expectedCritical).toBe(4);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("derives critical correctly at boundary: baseline=1", () => {
|
||||||
|
// For baseline=1: low=1, critical=ceil(1/2)=1 (minimum 1 due to Math.max(1, ...))
|
||||||
|
const baseline = 1;
|
||||||
|
const expectedLow = Math.max(1, Math.floor(baseline));
|
||||||
|
const expectedCritical = Math.max(1, Math.ceil(expectedLow / 2));
|
||||||
|
expect(expectedLow).toBe(1);
|
||||||
|
expect(expectedCritical).toBe(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("derives thresholds correctly for even baseline (baseline=14)", () => {
|
||||||
|
// For baseline=14: low=14, critical=ceil(14/2)=7
|
||||||
|
const baseline = 14;
|
||||||
|
const expectedLow = Math.max(1, Math.floor(baseline));
|
||||||
|
const expectedCritical = Math.max(1, Math.ceil(expectedLow / 2));
|
||||||
|
expect(expectedLow).toBe(14);
|
||||||
|
expect(expectedCritical).toBe(7);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("derives thresholds correctly for odd baseline (baseline=15)", () => {
|
||||||
|
// For baseline=15: low=15, critical=ceil(15/2)=8
|
||||||
|
const baseline = 15;
|
||||||
|
const expectedLow = Math.max(1, Math.floor(baseline));
|
||||||
|
const expectedCritical = Math.max(1, Math.ceil(expectedLow / 2));
|
||||||
|
expect(expectedLow).toBe(15);
|
||||||
|
expect(expectedCritical).toBe(8);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|||||||
@@ -23,18 +23,22 @@ function shouldLog(level: string): boolean {
|
|||||||
return LOG_LEVELS[level] >= getLevel();
|
return LOG_LEVELS[level] >= getLevel();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function ts(): string {
|
||||||
|
return new Date().toISOString();
|
||||||
|
}
|
||||||
|
|
||||||
export const log = {
|
export const log = {
|
||||||
debug(msg: string): void {
|
debug(msg: string): void {
|
||||||
if (shouldLog("debug")) console.log(msg);
|
if (shouldLog("debug")) console.log(`[${ts()}] [DEBUG] ${msg}`);
|
||||||
},
|
},
|
||||||
info(msg: string): void {
|
info(msg: string): void {
|
||||||
if (shouldLog("info")) console.log(msg);
|
if (shouldLog("info")) console.log(`[${ts()}] [INFO] ${msg}`);
|
||||||
},
|
},
|
||||||
warn(msg: string): void {
|
warn(msg: string): void {
|
||||||
if (shouldLog("warn")) console.warn(msg);
|
if (shouldLog("warn")) console.warn(`[${ts()}] [WARN] ${msg}`);
|
||||||
},
|
},
|
||||||
error(msg: string): void {
|
error(msg: string): void {
|
||||||
if (shouldLog("error")) console.error(msg);
|
if (shouldLog("error")) console.error(`[${ts()}] [ERROR] ${msg}`);
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -0,0 +1,32 @@
|
|||||||
|
export const PACKAGE_TYPES = ["blister", "bottle", "tube", "liquid_container"] as const;
|
||||||
|
|
||||||
|
export type PackageType = (typeof PACKAGE_TYPES)[number];
|
||||||
|
|
||||||
|
const PACKAGE_TYPE_SET = new Set<string>(PACKAGE_TYPES);
|
||||||
|
|
||||||
|
export function normalizePackageType(packageType?: string | null): PackageType {
|
||||||
|
if (packageType && PACKAGE_TYPE_SET.has(packageType)) {
|
||||||
|
return packageType as PackageType;
|
||||||
|
}
|
||||||
|
return "blister";
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isTubePackageType(packageType?: string | null): boolean {
|
||||||
|
return normalizePackageType(packageType) === "tube";
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isLiquidContainerPackageType(packageType?: string | null): boolean {
|
||||||
|
return normalizePackageType(packageType) === "liquid_container";
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isAmountBasedPackageType(packageType?: string | null): boolean {
|
||||||
|
const normalized = normalizePackageType(packageType);
|
||||||
|
return normalized === "bottle" || normalized === "tube" || normalized === "liquid_container";
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getPlannerUnitKind(packageType?: string | null): "pills" | "ml" | "units" {
|
||||||
|
const normalized = normalizePackageType(packageType);
|
||||||
|
if (normalized === "tube") return "units";
|
||||||
|
if (normalized === "liquid_container") return "ml";
|
||||||
|
return "pills";
|
||||||
|
}
|
||||||
@@ -4,6 +4,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
import { getDateLocale, type Language } from "../i18n/translations.js";
|
import { getDateLocale, type Language } from "../i18n/translations.js";
|
||||||
|
import { isLiquidContainerPackageType, isTubePackageType } from "./package-profiles.js";
|
||||||
|
|
||||||
// Legacy type - individual blister schedule (DEPRECATED: use Intake instead)
|
// Legacy type - individual blister schedule (DEPRECATED: use Intake instead)
|
||||||
export type Blister = { usage: number; every: number; start: string };
|
export type Blister = { usage: number; every: number; start: string };
|
||||||
@@ -13,10 +14,39 @@ export type Intake = {
|
|||||||
usage: number;
|
usage: number;
|
||||||
every: number;
|
every: number;
|
||||||
start: string;
|
start: string;
|
||||||
|
intakeUnit?: "ml" | "tsp" | "tbsp" | null;
|
||||||
takenBy: string | null; // Person taking this specific intake (null = use medication-level takenBy)
|
takenBy: string | null; // Person taking this specific intake (null = use medication-level takenBy)
|
||||||
intakeRemindersEnabled: boolean;
|
intakeRemindersEnabled: boolean;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const isValidIntakeUnit = (value: unknown): value is "ml" | "tsp" | "tbsp" =>
|
||||||
|
value === "ml" || value === "tsp" || value === "tbsp";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Normalize intake usage for stock math.
|
||||||
|
*
|
||||||
|
* Stock semantics:
|
||||||
|
* - tube: no automatic depletion (unknown per-application amount)
|
||||||
|
* - liquid_container/liquid forms: convert tsp/tbsp to ml
|
||||||
|
* - others: usage as-is
|
||||||
|
*/
|
||||||
|
export function normalizeIntakeUsageForStock(
|
||||||
|
intake: Pick<Intake, "usage" | "intakeUnit">,
|
||||||
|
medicationForm?: string | null,
|
||||||
|
packageType?: string | null
|
||||||
|
): number {
|
||||||
|
const usage = Number(intake.usage);
|
||||||
|
if (!Number.isFinite(usage) || usage <= 0) return 0;
|
||||||
|
if (isTubePackageType(packageType)) return 0;
|
||||||
|
|
||||||
|
const isLiquidStock = isLiquidContainerPackageType(packageType) || medicationForm === "liquid";
|
||||||
|
if (!isLiquidStock) return usage;
|
||||||
|
|
||||||
|
if (intake.intakeUnit === "tsp") return usage * 5;
|
||||||
|
if (intake.intakeUnit === "tbsp") return usage * 15;
|
||||||
|
return usage;
|
||||||
|
}
|
||||||
|
|
||||||
// =============================================================================
|
// =============================================================================
|
||||||
// Timezone utilities
|
// Timezone utilities
|
||||||
// =============================================================================
|
// =============================================================================
|
||||||
@@ -122,7 +152,11 @@ export function getNextScheduledTime(reminderHour: number, tz?: string): Date {
|
|||||||
/** Calculate milliseconds until next check at the given reminder hour */
|
/** Calculate milliseconds until next check at the given reminder hour */
|
||||||
export function getMsUntilNextCheck(reminderHour: number, tz?: string): number {
|
export function getMsUntilNextCheck(reminderHour: number, tz?: string): number {
|
||||||
const next = getNextScheduledTime(reminderHour, tz);
|
const next = getNextScheduledTime(reminderHour, tz);
|
||||||
return next.getTime() - Date.now();
|
const msUntilNext = next.getTime() - Date.now();
|
||||||
|
if (msUntilNext <= 0) {
|
||||||
|
return msUntilNext + 24 * 60 * 60 * 1000;
|
||||||
|
}
|
||||||
|
return msUntilNext;
|
||||||
}
|
}
|
||||||
|
|
||||||
// =============================================================================
|
// =============================================================================
|
||||||
@@ -195,6 +229,7 @@ export function parseIntakesJson(
|
|||||||
usage: typeof intake.usage === "number" ? intake.usage : 0,
|
usage: typeof intake.usage === "number" ? intake.usage : 0,
|
||||||
every: typeof intake.every === "number" ? intake.every : 1,
|
every: typeof intake.every === "number" ? intake.every : 1,
|
||||||
start: typeof intake.start === "string" ? intake.start : new Date().toISOString(),
|
start: typeof intake.start === "string" ? intake.start : new Date().toISOString(),
|
||||||
|
intakeUnit: isValidIntakeUnit(intake.intakeUnit) ? intake.intakeUnit : null,
|
||||||
takenBy: typeof intake.takenBy === "string" && intake.takenBy.trim() ? intake.takenBy.trim() : null,
|
takenBy: typeof intake.takenBy === "string" && intake.takenBy.trim() ? intake.takenBy.trim() : null,
|
||||||
intakeRemindersEnabled:
|
intakeRemindersEnabled:
|
||||||
typeof intake.intakeRemindersEnabled === "boolean" ? intake.intakeRemindersEnabled : false,
|
typeof intake.intakeRemindersEnabled === "boolean" ? intake.intakeRemindersEnabled : false,
|
||||||
@@ -212,6 +247,7 @@ export function parseIntakesJson(
|
|||||||
usage: b.usage,
|
usage: b.usage,
|
||||||
every: b.every,
|
every: b.every,
|
||||||
start: b.start,
|
start: b.start,
|
||||||
|
intakeUnit: null,
|
||||||
takenBy: null, // Legacy format has no per-intake takenBy
|
takenBy: null, // Legacy format has no per-intake takenBy
|
||||||
intakeRemindersEnabled: medicationIntakeRemindersEnabled ?? false,
|
intakeRemindersEnabled: medicationIntakeRemindersEnabled ?? false,
|
||||||
}));
|
}));
|
||||||
|
|||||||
+74
-30
@@ -1,7 +1,7 @@
|
|||||||
import * as fs from "node:fs";
|
import * as fs from "node:fs";
|
||||||
import * as path from "node:path";
|
import * as path from "node:path";
|
||||||
import { expect, test as setup } from "@playwright/test";
|
import { expect, test as setup } from "@playwright/test";
|
||||||
import { TEST_USER } from "./fixtures";
|
import { applyVideoSafetyMode, TEST_USER } from "./fixtures";
|
||||||
|
|
||||||
const authFile = path.join(import.meta.dirname, ".auth", "user.json");
|
const authFile = path.join(import.meta.dirname, ".auth", "user.json");
|
||||||
|
|
||||||
@@ -33,6 +33,8 @@ function isTokenValid(token: string): boolean {
|
|||||||
* 4. Log in via the UI.
|
* 4. Log in via the UI.
|
||||||
*/
|
*/
|
||||||
setup("authenticate", async ({ page }) => {
|
setup("authenticate", async ({ page }) => {
|
||||||
|
await applyVideoSafetyMode(page);
|
||||||
|
|
||||||
// Create .auth directory if it doesn't exist
|
// Create .auth directory if it doesn't exist
|
||||||
const authDir = path.dirname(authFile);
|
const authDir = path.dirname(authFile);
|
||||||
if (!fs.existsSync(authDir)) {
|
if (!fs.existsSync(authDir)) {
|
||||||
@@ -68,40 +70,82 @@ setup("authenticate", async ({ page }) => {
|
|||||||
// Wait for auth container
|
// Wait for auth container
|
||||||
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
|
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
|
||||||
|
|
||||||
// ---- 3. Ensure the test user exists ----
|
// ---- 3. Query auth state to determine login method ----
|
||||||
const baseURL = process.env.PLAYWRIGHT_BASE_URL || "http://localhost:5173";
|
const baseURL = process.env.PLAYWRIGHT_BASE_URL || "http://localhost:5173";
|
||||||
await page.request
|
let formLoginEnabled = true;
|
||||||
.post(`${baseURL}/api/auth/register`, {
|
let oidcEnabled = false;
|
||||||
data: { username: TEST_USER.username, password: TEST_USER.password },
|
try {
|
||||||
})
|
const stateRes = await page.request.get(`${baseURL}/api/auth/state`);
|
||||||
.catch(() => {});
|
if (stateRes.ok()) {
|
||||||
|
const state = await stateRes.json();
|
||||||
// ---- 4. Log in via UI ----
|
formLoginEnabled = state.formLoginEnabled !== false;
|
||||||
const usernameField = page.locator("#username");
|
oidcEnabled = state.oidcEnabled === true;
|
||||||
const passwordField = page.locator("#password");
|
|
||||||
|
|
||||||
// Make sure we're on the login form (not register)
|
|
||||||
const isOnRegister = await page
|
|
||||||
.locator(".auth-subtitle")
|
|
||||||
.filter({ hasText: /Create Account/i })
|
|
||||||
.isVisible()
|
|
||||||
.catch(() => false);
|
|
||||||
|
|
||||||
if (isOnRegister) {
|
|
||||||
const switchBtn = page.locator("button.auth-link-btn");
|
|
||||||
if (await switchBtn.isVisible().catch(() => false)) {
|
|
||||||
await switchBtn.click();
|
|
||||||
await page.waitForTimeout(500);
|
|
||||||
}
|
}
|
||||||
|
} catch {
|
||||||
|
// Fallback: assume form login is available
|
||||||
}
|
}
|
||||||
|
|
||||||
await usernameField.clear();
|
// ---- 4. Ensure the test user exists (only if form login is available) ----
|
||||||
await usernameField.fill(TEST_USER.username);
|
if (formLoginEnabled) {
|
||||||
await passwordField.clear();
|
await page.request
|
||||||
await passwordField.fill(TEST_USER.password);
|
.post(`${baseURL}/api/auth/register`, {
|
||||||
|
data: { username: TEST_USER.username, password: TEST_USER.password },
|
||||||
|
})
|
||||||
|
.catch(() => {});
|
||||||
|
}
|
||||||
|
|
||||||
// Click the submit button (not the SSO button)
|
// ---- 5. Log in via the appropriate method ----
|
||||||
await page.locator('button.auth-submit[type="submit"]').click();
|
if (formLoginEnabled) {
|
||||||
|
// Form login path: username/password
|
||||||
|
const usernameField = page.locator("#username");
|
||||||
|
const passwordField = page.locator("#password");
|
||||||
|
|
||||||
|
// Make sure we're on the login form (not register)
|
||||||
|
const isOnRegister = await page
|
||||||
|
.locator(".auth-subtitle")
|
||||||
|
.filter({ hasText: /Create Account/i })
|
||||||
|
.isVisible()
|
||||||
|
.catch(() => false);
|
||||||
|
|
||||||
|
if (isOnRegister) {
|
||||||
|
const switchBtn = page.locator("button.auth-link-btn");
|
||||||
|
if (await switchBtn.isVisible().catch(() => false)) {
|
||||||
|
await switchBtn.click();
|
||||||
|
await page.waitForTimeout(500);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
await usernameField.clear();
|
||||||
|
await usernameField.fill(TEST_USER.username);
|
||||||
|
await passwordField.clear();
|
||||||
|
await passwordField.fill(TEST_USER.password);
|
||||||
|
|
||||||
|
// Click the submit button (not the SSO button)
|
||||||
|
await page.locator('button.auth-submit[type="submit"]').click();
|
||||||
|
} else if (oidcEnabled) {
|
||||||
|
// SSO-only path: click the SSO button and let the OIDC provider handle login.
|
||||||
|
// This requires the OIDC provider to be configured with test credentials
|
||||||
|
// (e.g. via PLAYWRIGHT_OIDC_USERNAME / PLAYWRIGHT_OIDC_PASSWORD env vars)
|
||||||
|
// or to auto-approve the test user.
|
||||||
|
await page.locator("button.sso-btn").click();
|
||||||
|
|
||||||
|
// Wait for OIDC redirect and callback — the provider may show its own login form
|
||||||
|
const oidcUsername = process.env.PLAYWRIGHT_OIDC_USERNAME;
|
||||||
|
const oidcPassword = process.env.PLAYWRIGHT_OIDC_PASSWORD;
|
||||||
|
if (oidcUsername && oidcPassword) {
|
||||||
|
// Fill OIDC provider login form (generic selectors — override if needed)
|
||||||
|
await page.waitForURL(/.*/, { timeout: 15000 });
|
||||||
|
const oidcUserField = page.locator('input[name="username"], input[name="login"], input[type="email"]').first();
|
||||||
|
const oidcPassField = page.locator('input[name="password"], input[type="password"]').first();
|
||||||
|
if (await oidcUserField.isVisible({ timeout: 10000 }).catch(() => false)) {
|
||||||
|
await oidcUserField.fill(oidcUsername);
|
||||||
|
await oidcPassField.fill(oidcPassword);
|
||||||
|
await page.locator('button[type="submit"]').first().click();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
throw new Error("No login method available: form login and OIDC are both disabled");
|
||||||
|
}
|
||||||
|
|
||||||
// Wait for successful auth — app header should appear
|
// Wait for successful auth — app header should appear
|
||||||
await expect(page.locator("header.hero")).toBeVisible({ timeout: 15000 });
|
await expect(page.locator("header.hero")).toBeVisible({ timeout: 15000 });
|
||||||
|
|||||||
@@ -1,16 +1,28 @@
|
|||||||
import { expect, type Page, test } from "@playwright/test";
|
import { expect, type Page, test } from "@playwright/test";
|
||||||
|
|
||||||
async function isAuthEnabled(page: Page): Promise<boolean> {
|
interface AuthStateResponse {
|
||||||
|
authEnabled: boolean;
|
||||||
|
formLoginEnabled: boolean;
|
||||||
|
oidcEnabled: boolean;
|
||||||
|
oidcProviderName: string;
|
||||||
|
registrationEnabled: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getAuthState(page: Page): Promise<AuthStateResponse | null> {
|
||||||
try {
|
try {
|
||||||
const response = await page.request.get("/api/auth/state");
|
const response = await page.request.get("/api/auth/state");
|
||||||
if (!response.ok()) return true;
|
if (!response.ok()) return null;
|
||||||
const state = await response.json();
|
return (await response.json()) as AuthStateResponse;
|
||||||
return state?.authEnabled !== false;
|
|
||||||
} catch {
|
} catch {
|
||||||
return true;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function isAuthEnabled(page: Page): Promise<boolean> {
|
||||||
|
const state = await getAuthState(page);
|
||||||
|
return state?.authEnabled !== false;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Authentication E2E Tests
|
* Authentication E2E Tests
|
||||||
*
|
*
|
||||||
@@ -110,4 +122,48 @@ test.describe("Authentication", () => {
|
|||||||
const newText = await subtitle.textContent();
|
const newText = await subtitle.textContent();
|
||||||
expect(newText).not.toBe(initialText);
|
expect(newText).not.toBe(initialText);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("should show SSO button when OIDC is enabled", async ({ page }) => {
|
||||||
|
const state = await getAuthState(page);
|
||||||
|
test.skip(!state?.authEnabled, "Auth is disabled in this environment");
|
||||||
|
test.skip(!state?.oidcEnabled, "OIDC is not enabled in this environment");
|
||||||
|
|
||||||
|
await page.goto("/");
|
||||||
|
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
|
||||||
|
|
||||||
|
const ssoButton = page.locator("button.sso-btn");
|
||||||
|
await expect(ssoButton).toBeVisible();
|
||||||
|
await expect(ssoButton).toContainText(state.oidcProviderName || "SSO");
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should hide form login when formLoginEnabled is false", async ({ page }) => {
|
||||||
|
const state = await getAuthState(page);
|
||||||
|
test.skip(!state?.authEnabled, "Auth is disabled in this environment");
|
||||||
|
test.skip(state?.formLoginEnabled !== false, "Form login is enabled — cannot test hidden state");
|
||||||
|
|
||||||
|
await page.goto("/");
|
||||||
|
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
|
||||||
|
|
||||||
|
// Username/password fields should not be visible
|
||||||
|
await expect(page.locator("#username")).not.toBeVisible();
|
||||||
|
await expect(page.locator("#password")).not.toBeVisible();
|
||||||
|
|
||||||
|
// SSO button should be the only login method
|
||||||
|
await expect(page.locator("button.sso-btn")).toBeVisible();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should show both login methods when OIDC and form login are enabled", async ({ page }) => {
|
||||||
|
const state = await getAuthState(page);
|
||||||
|
test.skip(!state?.authEnabled, "Auth is disabled in this environment");
|
||||||
|
test.skip(!state?.oidcEnabled, "OIDC is not enabled");
|
||||||
|
test.skip(!state?.formLoginEnabled, "Form login is not enabled");
|
||||||
|
|
||||||
|
await page.goto("/");
|
||||||
|
await expect(page.locator(".auth-container")).toBeVisible({ timeout: 15000 });
|
||||||
|
|
||||||
|
// Both login methods visible
|
||||||
|
await expect(page.locator("#username")).toBeVisible();
|
||||||
|
await expect(page.locator("#password")).toBeVisible();
|
||||||
|
await expect(page.locator("button.sso-btn")).toBeVisible();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -65,7 +65,7 @@ test.describe("Dashboard with medications", () => {
|
|||||||
test("should show medication overview table with medications", async ({ page }) => {
|
test("should show medication overview table with medications", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
await expect(overviewTable.locator(".table-head")).toBeVisible();
|
await expect(overviewTable.locator(".table-head")).toBeVisible();
|
||||||
|
|
||||||
@@ -77,7 +77,7 @@ test.describe("Dashboard with medications", () => {
|
|||||||
test("should show status chips in overview table", async ({ page }) => {
|
test("should show status chips in overview table", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// Each medication row should have a status chip
|
// Each medication row should have a status chip
|
||||||
@@ -88,7 +88,7 @@ test.describe("Dashboard with medications", () => {
|
|||||||
test("should show stock information in overview", async ({ page }) => {
|
test("should show stock information in overview", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// The Ibuprofen row should show stock info (60 pills minus today's usage = 59)
|
// The Ibuprofen row should show stock info (60 pills minus today's usage = 59)
|
||||||
@@ -202,7 +202,7 @@ test.describe("Dashboard with medications", () => {
|
|||||||
test("should open medication detail modal from overview table", async ({ page }) => {
|
test("should open medication detail modal from overview table", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
const medRow = overviewTable.locator(".table-row").filter({ hasText: MED_1 }).first();
|
const medRow = overviewTable.locator(".table-row").filter({ hasText: MED_1 }).first();
|
||||||
|
|||||||
+158
-27
@@ -60,6 +60,29 @@ async function setupAuthMeMock(page: Page): Promise<void> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reduce visual flashing in recorded videos by forcing a dark first paint and
|
||||||
|
* disabling most animations/transitions in test mode.
|
||||||
|
*/
|
||||||
|
export async function applyVideoSafetyMode(page: Page): Promise<void> {
|
||||||
|
await page.emulateMedia({ reducedMotion: "reduce", colorScheme: "dark" });
|
||||||
|
await page.addInitScript(() => {
|
||||||
|
const style = document.createElement("style");
|
||||||
|
style.id = "pw-video-safety-style";
|
||||||
|
style.textContent = `
|
||||||
|
html, body {
|
||||||
|
background: #111111 !important;
|
||||||
|
color-scheme: dark !important;
|
||||||
|
}
|
||||||
|
*, *::before, *::after {
|
||||||
|
animation: none !important;
|
||||||
|
transition: none !important;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
document.documentElement.appendChild(style);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Extended test fixture that automatically mocks /auth/me on every page
|
* Extended test fixture that automatically mocks /auth/me on every page
|
||||||
* using user data from the JWT in the stored auth file.
|
* using user data from the JWT in the stored auth file.
|
||||||
@@ -72,6 +95,7 @@ async function setupAuthMeMock(page: Page): Promise<void> {
|
|||||||
*/
|
*/
|
||||||
export const test = base.extend<object>({
|
export const test = base.extend<object>({
|
||||||
page: async ({ page }, use) => {
|
page: async ({ page }, use) => {
|
||||||
|
await applyVideoSafetyMode(page);
|
||||||
await setupAuthMeMock(page);
|
await setupAuthMeMock(page);
|
||||||
await use(page);
|
await use(page);
|
||||||
},
|
},
|
||||||
@@ -79,25 +103,43 @@ export const test = base.extend<object>({
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Wait for the app to be fully loaded past any loading/initializing screens.
|
* Wait for the app to be fully loaded past any loading/initializing screens.
|
||||||
* Includes a single retry with page reload to handle transient auth failures
|
* Retries up to 2 times with page reload to handle transient auth or
|
||||||
* (e.g. brief race between context setup and cookie application).
|
* rate-limit failures.
|
||||||
*/
|
*/
|
||||||
export async function waitForAppReady(page: Page): Promise<void> {
|
export async function waitForAppReady(page: Page): Promise<void> {
|
||||||
const hero = page.locator("header.hero");
|
const hero = page.locator("header.hero");
|
||||||
try {
|
for (let attempt = 0; attempt < 3; attempt++) {
|
||||||
await expect(hero).toBeVisible({ timeout: 15000 });
|
try {
|
||||||
} catch {
|
await expect(hero).toBeVisible({ timeout: 15000 });
|
||||||
// Auth might have failed transiently — reload and retry once
|
return;
|
||||||
await page.reload();
|
} catch {
|
||||||
await expect(hero).toBeVisible({ timeout: 15000 });
|
if (attempt === 2) throw new Error("App failed to become ready after 3 attempts");
|
||||||
|
// Check for rate-limit error displayed in UI
|
||||||
|
const rateLimited = await page
|
||||||
|
.locator("text=rate limit, text=429, text=too many")
|
||||||
|
.first()
|
||||||
|
.isVisible()
|
||||||
|
.catch(() => false);
|
||||||
|
if (rateLimited) {
|
||||||
|
// Wait longer before retrying if rate-limited
|
||||||
|
await page.waitForTimeout(5000);
|
||||||
|
}
|
||||||
|
await page.reload();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Navigate to a page and wait for it to be ready.
|
* Navigate to a page and wait for it to be ready.
|
||||||
|
* Handles transient navigation failures with a single retry.
|
||||||
*/
|
*/
|
||||||
export async function navigateTo(page: Page, path: string): Promise<void> {
|
export async function navigateTo(page: Page, path: string): Promise<void> {
|
||||||
await page.goto(path);
|
const response = await page.goto(path);
|
||||||
|
if (response && response.status() === 429) {
|
||||||
|
// Rate-limited — wait and retry once
|
||||||
|
await page.waitForTimeout(5000);
|
||||||
|
await page.goto(path);
|
||||||
|
}
|
||||||
await waitForAppReady(page);
|
await waitForAppReady(page);
|
||||||
await page.waitForLoadState("networkidle");
|
await page.waitForLoadState("networkidle");
|
||||||
}
|
}
|
||||||
@@ -135,7 +177,9 @@ export { expect };
|
|||||||
// ---------------------------------------------------------------------------
|
// ---------------------------------------------------------------------------
|
||||||
const API_BASE = process.env.PLAYWRIGHT_BASE_URL || "http://localhost:5173";
|
const API_BASE = process.env.PLAYWRIGHT_BASE_URL || "http://localhost:5173";
|
||||||
|
|
||||||
function getAuthCookie(): string | null {
|
let cachedAuthCookie: string | null = null;
|
||||||
|
|
||||||
|
function readAuthCookieFromFile(): string | null {
|
||||||
try {
|
try {
|
||||||
const state = JSON.parse(fs.readFileSync(authFile, "utf-8"));
|
const state = JSON.parse(fs.readFileSync(authFile, "utf-8"));
|
||||||
return state.cookies?.find((c: { name: string }) => c.name === "access_token")?.value ?? null;
|
return state.cookies?.find((c: { name: string }) => c.name === "access_token")?.value ?? null;
|
||||||
@@ -144,6 +188,49 @@ function getAuthCookie(): string | null {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function extractCookieValue(setCookieHeaders: string[], name: string): string | null {
|
||||||
|
for (const header of setCookieHeaders) {
|
||||||
|
const [pair] = header.split(";");
|
||||||
|
if (!pair) continue;
|
||||||
|
const [cookieName, ...valueParts] = pair.split("=");
|
||||||
|
if (cookieName?.trim() !== name) continue;
|
||||||
|
const value = valueParts.join("=").trim();
|
||||||
|
if (value) return value;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function refreshAuthCookieViaLogin(): Promise<string | null> {
|
||||||
|
const res = await fetch(`${API_BASE}/api/auth/login`, {
|
||||||
|
method: "POST",
|
||||||
|
headers: { "Content-Type": "application/json" },
|
||||||
|
body: JSON.stringify({
|
||||||
|
username: TEST_USER.username,
|
||||||
|
password: TEST_USER.password,
|
||||||
|
rememberMe: false,
|
||||||
|
}),
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!res.ok) return null;
|
||||||
|
|
||||||
|
const getSetCookie = (res.headers as Headers & { getSetCookie?: () => string[] }).getSetCookie;
|
||||||
|
const setCookieHeaders = typeof getSetCookie === "function" ? getSetCookie.call(res.headers) : [];
|
||||||
|
const fallback = res.headers.get("set-cookie");
|
||||||
|
if (fallback) setCookieHeaders.push(fallback);
|
||||||
|
|
||||||
|
const accessToken = extractCookieValue(setCookieHeaders, "access_token");
|
||||||
|
if (accessToken) {
|
||||||
|
cachedAuthCookie = accessToken;
|
||||||
|
}
|
||||||
|
return accessToken;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getAuthCookie(): string | null {
|
||||||
|
if (cachedAuthCookie) return cachedAuthCookie;
|
||||||
|
cachedAuthCookie = readAuthCookieFromFile();
|
||||||
|
return cachedAuthCookie;
|
||||||
|
}
|
||||||
|
|
||||||
/** Typed medication response (subset of fields we care about) */
|
/** Typed medication response (subset of fields we care about) */
|
||||||
export interface TestMedication {
|
export interface TestMedication {
|
||||||
id: number;
|
id: number;
|
||||||
@@ -172,12 +259,14 @@ export async function createMedicationViaAPI(data: {
|
|||||||
takenBy?: string[];
|
takenBy?: string[];
|
||||||
notes?: string;
|
notes?: string;
|
||||||
expiryDate?: string;
|
expiryDate?: string;
|
||||||
packageType?: "blister" | "bottle";
|
packageType?: "blister" | "bottle" | "tube" | "liquid_container";
|
||||||
|
medicationForm?: "capsule" | "tablet" | "liquid" | "topical";
|
||||||
packCount?: number;
|
packCount?: number;
|
||||||
blistersPerPack?: number;
|
blistersPerPack?: number;
|
||||||
pillsPerBlister?: number;
|
pillsPerBlister?: number;
|
||||||
looseTablets?: number;
|
looseTablets?: number;
|
||||||
totalPills?: number;
|
totalPills?: number;
|
||||||
|
packageAmountValue?: number;
|
||||||
intakeRemindersEnabled?: boolean;
|
intakeRemindersEnabled?: boolean;
|
||||||
intakes?: {
|
intakes?: {
|
||||||
usage: number;
|
usage: number;
|
||||||
@@ -187,16 +276,30 @@ export async function createMedicationViaAPI(data: {
|
|||||||
takenBy?: string | null;
|
takenBy?: string | null;
|
||||||
}[];
|
}[];
|
||||||
}): Promise<TestMedication> {
|
}): Promise<TestMedication> {
|
||||||
const token = getAuthCookie();
|
let token = getAuthCookie();
|
||||||
const isBottle = data.packageType === "bottle";
|
const packageType = data.packageType ?? "blister";
|
||||||
|
const isAmountBased = packageType === "bottle" || packageType === "tube" || packageType === "liquid_container";
|
||||||
|
let defaultMedicationForm: "capsule" | "tablet" | "liquid" | "topical" = "tablet";
|
||||||
|
if (packageType === "tube") {
|
||||||
|
defaultMedicationForm = "topical";
|
||||||
|
} else if (packageType === "liquid_container") {
|
||||||
|
defaultMedicationForm = "liquid";
|
||||||
|
}
|
||||||
|
const medicationForm = data.medicationForm ?? defaultMedicationForm;
|
||||||
|
const packageAmountValue =
|
||||||
|
data.packageAmountValue ??
|
||||||
|
(packageType === "tube" || packageType === "liquid_container" ? Math.max(1, data.totalPills ?? 30) : 0);
|
||||||
const body = {
|
const body = {
|
||||||
packageType: isBottle ? "bottle" : "blister",
|
packageType,
|
||||||
packCount: isBottle ? 1 : (data.packCount ?? 1),
|
medicationForm,
|
||||||
blistersPerPack: isBottle ? 1 : (data.blistersPerPack ?? 1),
|
packCount: packageType === "tube" ? 1 : (data.packCount ?? 1),
|
||||||
pillsPerBlister: isBottle ? 1 : (data.pillsPerBlister ?? 10),
|
blistersPerPack: isAmountBased ? 1 : (data.blistersPerPack ?? 1),
|
||||||
// For bottles: looseTablets IS the current stock. Default to totalPills if not specified.
|
pillsPerBlister: isAmountBased ? 1 : (data.pillsPerBlister ?? 10),
|
||||||
looseTablets: isBottle ? (data.looseTablets ?? data.totalPills ?? 0) : (data.looseTablets ?? 0),
|
// Amount-based packages use looseTablets as current stock.
|
||||||
totalPills: isBottle ? (data.totalPills ?? null) : null,
|
looseTablets: isAmountBased ? (data.looseTablets ?? data.totalPills ?? 0) : (data.looseTablets ?? 0),
|
||||||
|
totalPills: isAmountBased ? (data.totalPills ?? null) : null,
|
||||||
|
packageAmountValue,
|
||||||
|
packageAmountUnit: packageType === "tube" ? "g" : "ml",
|
||||||
intakes: [
|
intakes: [
|
||||||
{
|
{
|
||||||
usage: 1,
|
usage: 1,
|
||||||
@@ -219,6 +322,10 @@ export async function createMedicationViaAPI(data: {
|
|||||||
},
|
},
|
||||||
body: JSON.stringify(body),
|
body: JSON.stringify(body),
|
||||||
});
|
});
|
||||||
|
if (res.status === 401) {
|
||||||
|
token = await refreshAuthCookieViaLogin();
|
||||||
|
if (token) continue;
|
||||||
|
}
|
||||||
if (res.status === 429) {
|
if (res.status === 429) {
|
||||||
// Rate limited — exponential backoff: 3s, 6s, 9s, 12s, 15s
|
// Rate limited — exponential backoff: 3s, 6s, 9s, 12s, 15s
|
||||||
await new Promise((r) => setTimeout(r, 3000 * (attempt + 1)));
|
await new Promise((r) => setTimeout(r, 3000 * (attempt + 1)));
|
||||||
@@ -235,13 +342,25 @@ export async function createMedicationViaAPI(data: {
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Delete a medication via the backend API.
|
* Delete a medication via the backend API.
|
||||||
|
* Includes retry for rate-limited responses.
|
||||||
*/
|
*/
|
||||||
export async function deleteMedicationViaAPI(id: number): Promise<void> {
|
export async function deleteMedicationViaAPI(id: number): Promise<void> {
|
||||||
const token = getAuthCookie();
|
let token = getAuthCookie();
|
||||||
await fetch(`${API_BASE}/api/medications/${id}`, {
|
for (let attempt = 0; attempt < 3; attempt++) {
|
||||||
method: "DELETE",
|
const res = await fetch(`${API_BASE}/api/medications/${id}`, {
|
||||||
headers: token ? { Cookie: `access_token=${token}` } : {},
|
method: "DELETE",
|
||||||
});
|
headers: token ? { Cookie: `access_token=${token}` } : {},
|
||||||
|
});
|
||||||
|
if (res.status === 401) {
|
||||||
|
token = await refreshAuthCookieViaLogin();
|
||||||
|
if (token) continue;
|
||||||
|
}
|
||||||
|
if (res.status === 429) {
|
||||||
|
await new Promise((r) => setTimeout(r, 3000 * (attempt + 1)));
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -249,11 +368,15 @@ export async function deleteMedicationViaAPI(id: number): Promise<void> {
|
|||||||
* Includes retry logic for rate-limited responses.
|
* Includes retry logic for rate-limited responses.
|
||||||
*/
|
*/
|
||||||
export async function deleteAllMedicationsViaAPI(): Promise<void> {
|
export async function deleteAllMedicationsViaAPI(): Promise<void> {
|
||||||
const token = getAuthCookie();
|
let token = getAuthCookie();
|
||||||
for (let attempt = 0; attempt < 3; attempt++) {
|
for (let attempt = 0; attempt < 3; attempt++) {
|
||||||
const res = await fetch(`${API_BASE}/api/medications`, {
|
const res = await fetch(`${API_BASE}/api/medications`, {
|
||||||
headers: token ? { Cookie: `access_token=${token}` } : {},
|
headers: token ? { Cookie: `access_token=${token}` } : {},
|
||||||
});
|
});
|
||||||
|
if (res.status === 401) {
|
||||||
|
token = await refreshAuthCookieViaLogin();
|
||||||
|
if (token) continue;
|
||||||
|
}
|
||||||
if (res.status === 429) {
|
if (res.status === 429) {
|
||||||
await new Promise((r) => setTimeout(r, 3000 * (attempt + 1)));
|
await new Promise((r) => setTimeout(r, 3000 * (attempt + 1)));
|
||||||
continue;
|
continue;
|
||||||
@@ -266,6 +389,10 @@ export async function deleteAllMedicationsViaAPI(): Promise<void> {
|
|||||||
method: "DELETE",
|
method: "DELETE",
|
||||||
headers: token ? { Cookie: `access_token=${token}` } : {},
|
headers: token ? { Cookie: `access_token=${token}` } : {},
|
||||||
});
|
});
|
||||||
|
if (delRes.status === 401) {
|
||||||
|
token = await refreshAuthCookieViaLogin();
|
||||||
|
if (token) continue;
|
||||||
|
}
|
||||||
if (delRes.status === 429) {
|
if (delRes.status === 429) {
|
||||||
await new Promise((r) => setTimeout(r, 3000));
|
await new Promise((r) => setTimeout(r, 3000));
|
||||||
continue;
|
continue;
|
||||||
@@ -282,7 +409,7 @@ export async function deleteAllMedicationsViaAPI(): Promise<void> {
|
|||||||
* Requires a medication with takenBy to exist first.
|
* Requires a medication with takenBy to exist first.
|
||||||
*/
|
*/
|
||||||
export async function createShareTokenViaAPI(takenBy: string, scheduleDays = 30): Promise<TestShareToken> {
|
export async function createShareTokenViaAPI(takenBy: string, scheduleDays = 30): Promise<TestShareToken> {
|
||||||
const token = getAuthCookie();
|
let token = getAuthCookie();
|
||||||
for (let attempt = 0; attempt < 5; attempt++) {
|
for (let attempt = 0; attempt < 5; attempt++) {
|
||||||
const res = await fetch(`${API_BASE}/api/share`, {
|
const res = await fetch(`${API_BASE}/api/share`, {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
@@ -292,6 +419,10 @@ export async function createShareTokenViaAPI(takenBy: string, scheduleDays = 30)
|
|||||||
},
|
},
|
||||||
body: JSON.stringify({ takenBy, scheduleDays }),
|
body: JSON.stringify({ takenBy, scheduleDays }),
|
||||||
});
|
});
|
||||||
|
if (res.status === 401) {
|
||||||
|
token = await refreshAuthCookieViaLogin();
|
||||||
|
if (token) continue;
|
||||||
|
}
|
||||||
if (res.status === 429) {
|
if (res.status === 429) {
|
||||||
await new Promise((r) => setTimeout(r, 3000 * (attempt + 1)));
|
await new Promise((r) => setTimeout(r, 3000 * (attempt + 1)));
|
||||||
continue;
|
continue;
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ async function fillAndSaveMedication(
|
|||||||
opts: {
|
opts: {
|
||||||
name: string;
|
name: string;
|
||||||
genericName?: string;
|
genericName?: string;
|
||||||
packageType?: "blister" | "bottle";
|
packageType?: "blister" | "bottle" | "tube" | "liquid_container";
|
||||||
packs?: string;
|
packs?: string;
|
||||||
blistersPerPack?: string;
|
blistersPerPack?: string;
|
||||||
pillsPerBlister?: string;
|
pillsPerBlister?: string;
|
||||||
@@ -56,6 +56,18 @@ async function fillAndSaveMedication(
|
|||||||
if (opts.totalCapacity)
|
if (opts.totalCapacity)
|
||||||
await form.getByLabel(/(Total Capacity|form\.totalCapacity|Total \(pills\))/i).fill(opts.totalCapacity);
|
await form.getByLabel(/(Total Capacity|form\.totalCapacity|Total \(pills\))/i).fill(opts.totalCapacity);
|
||||||
if (opts.currentPills) await form.getByLabel(/(Current Pills|form\.currentPills)/i).fill(opts.currentPills);
|
if (opts.currentPills) await form.getByLabel(/(Current Pills|form\.currentPills)/i).fill(opts.currentPills);
|
||||||
|
} else if (opts.packageType === "tube") {
|
||||||
|
await packageTypeSelect.selectOption("tube");
|
||||||
|
await page.getByRole("tab", { name: /Package/i }).click();
|
||||||
|
if (opts.totalCapacity) {
|
||||||
|
await form.getByLabel(/(Amount per tube|form\.packageAmountPerTube)/i).fill(opts.totalCapacity);
|
||||||
|
}
|
||||||
|
} else if (opts.packageType === "liquid_container") {
|
||||||
|
await packageTypeSelect.selectOption("liquid_container");
|
||||||
|
await page.getByRole("tab", { name: /Package/i }).click();
|
||||||
|
if (opts.totalCapacity) {
|
||||||
|
await form.getByLabel(/(Package amount|form\.packageAmount)/i).fill(opts.totalCapacity);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
await packageTypeSelect.selectOption("blister");
|
await packageTypeSelect.selectOption("blister");
|
||||||
await page.getByRole("tab", { name: /Package/i }).click();
|
await page.getByRole("tab", { name: /Package/i }).click();
|
||||||
@@ -83,7 +95,11 @@ async function fillAndSaveMedication(
|
|||||||
await form.getByRole("button", { name: /(Intake|form\.blisters\.addIntake)/i }).click();
|
await form.getByRole("button", { name: /(Intake|form\.blisters\.addIntake)/i }).click();
|
||||||
}
|
}
|
||||||
const row = form.locator(".blister-row").nth(i);
|
const row = form.locator(".blister-row").nth(i);
|
||||||
await row.getByLabel(/(Usage \(pills\)|form\.blisters\.usage)/i).fill(intakes[i].usage);
|
await row
|
||||||
|
.getByLabel(
|
||||||
|
/(Usage \((pills|tablets|capsules|ml|applications)\)|form\.blisters\.(usage|usageTablets|usageCapsules|usageMl|usageApplication))/i
|
||||||
|
)
|
||||||
|
.fill(intakes[i].usage);
|
||||||
await row.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i).fill(intakes[i].every);
|
await row.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i).fill(intakes[i].every);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -194,6 +210,26 @@ test.describe("Medication CRUD", () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("should create a tube medication via the form", async ({ page }) => {
|
||||||
|
await navigateTo(page, "/medications");
|
||||||
|
|
||||||
|
await fillAndSaveMedication(page, {
|
||||||
|
name: "Test Tube Cream",
|
||||||
|
packageType: "tube",
|
||||||
|
totalCapacity: "50",
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
test("should create a liquid-container medication via the form", async ({ page }) => {
|
||||||
|
await navigateTo(page, "/medications");
|
||||||
|
|
||||||
|
await fillAndSaveMedication(page, {
|
||||||
|
name: "Test Liquid Syrup",
|
||||||
|
packageType: "liquid_container",
|
||||||
|
totalCapacity: "120",
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
test("should create medication with notes and expiry date", async ({ page }) => {
|
test("should create medication with notes and expiry date", async ({ page }) => {
|
||||||
await navigateTo(page, "/medications");
|
await navigateTo(page, "/medications");
|
||||||
|
|
||||||
|
|||||||
@@ -233,7 +233,7 @@ test.describe("Medication Editing", () => {
|
|||||||
|
|
||||||
// Change intake from 1 pill daily to 2 pills every 7 days
|
// Change intake from 1 pill daily to 2 pills every 7 days
|
||||||
const intakeRow = page.locator(".blister-row").first();
|
const intakeRow = page.locator(".blister-row").first();
|
||||||
const usageField = intakeRow.getByLabel(/(Usage \(pills\)|form\.blisters\.usage)/i);
|
const usageField = intakeRow.getByLabel(/(Usage|form\.blisters\.usage)/i);
|
||||||
const everyField = intakeRow.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i);
|
const everyField = intakeRow.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i);
|
||||||
|
|
||||||
await usageField.fill("2");
|
await usageField.fill("2");
|
||||||
@@ -247,7 +247,7 @@ test.describe("Medication Editing", () => {
|
|||||||
// Verify the changes persisted
|
// Verify the changes persisted
|
||||||
await clickEditMed(page, "Edit Intake Med");
|
await clickEditMed(page, "Edit Intake Med");
|
||||||
const savedRow = page.locator(".blister-row").first();
|
const savedRow = page.locator(".blister-row").first();
|
||||||
await expect(savedRow.getByLabel(/(Usage \(pills\)|form\.blisters\.usage)/i)).toHaveValue("2");
|
await expect(savedRow.getByLabel(/(Usage|form\.blisters\.usage)/i)).toHaveValue("2");
|
||||||
await expect(savedRow.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i)).toHaveValue("7");
|
await expect(savedRow.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i)).toHaveValue("7");
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -279,7 +279,7 @@ test.describe("Medication Editing", () => {
|
|||||||
|
|
||||||
// Fill the new intake row
|
// Fill the new intake row
|
||||||
const secondRow = page.locator(".blister-row").nth(1);
|
const secondRow = page.locator(".blister-row").nth(1);
|
||||||
await secondRow.getByLabel(/(Usage \(pills\)|form\.blisters\.usage)/i).fill("0.5");
|
await secondRow.getByLabel(/(Usage|form\.blisters\.usage)/i).fill("0.5");
|
||||||
await secondRow.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i).fill("7");
|
await secondRow.getByLabel(/(Every \(days\)|form\.blisters\.everyDays)/i).fill("7");
|
||||||
|
|
||||||
await saveEditAndVerify(page, "Add Intake Med");
|
await saveEditAndVerify(page, "Add Intake Med");
|
||||||
@@ -329,7 +329,7 @@ test.describe("Medication Editing", () => {
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
test("should change package type between blister and bottle", async ({ page }) => {
|
test("should change package type across all supported profiles", async ({ page }) => {
|
||||||
createdMeds.push(
|
createdMeds.push(
|
||||||
await createMedicationViaAPI({
|
await createMedicationViaAPI({
|
||||||
name: "PackType Change Med",
|
name: "PackType Change Med",
|
||||||
@@ -357,15 +357,24 @@ test.describe("Medication Editing", () => {
|
|||||||
await packageSelect.selectOption("bottle");
|
await packageSelect.selectOption("bottle");
|
||||||
await page.getByRole("tab", { name: /Package/i }).click();
|
await page.getByRole("tab", { name: /Package/i }).click();
|
||||||
await expect(form.getByLabel(/(Total Capacity|form\.totalCapacity|Total \(pills\))/i)).toBeVisible();
|
await expect(form.getByLabel(/(Total Capacity|form\.totalCapacity|Total \(pills\))/i)).toBeVisible();
|
||||||
|
await page.getByRole("tab", { name: /General/i }).click();
|
||||||
|
|
||||||
// Fill bottle-specific fields
|
// Switch to tube
|
||||||
await form.getByLabel(/(Total Capacity|form\.totalCapacity|Total \(pills\))/i).fill("120");
|
await packageSelect.selectOption("tube");
|
||||||
|
await page.getByRole("tab", { name: /Package/i }).click();
|
||||||
|
await expect(form.getByLabel(/(Amount per tube|form\.packageAmountPerTube)/i)).toBeVisible();
|
||||||
|
await page.getByRole("tab", { name: /General/i }).click();
|
||||||
|
|
||||||
|
// Switch to liquid container and persist this final state
|
||||||
|
await packageSelect.selectOption("liquid_container");
|
||||||
|
await page.getByRole("tab", { name: /Package/i }).click();
|
||||||
|
await expect(form.getByLabel(/(Package amount|form\.packageAmount)/i)).toBeVisible();
|
||||||
|
|
||||||
await saveEditAndVerify(page, "PackType Change Med");
|
await saveEditAndVerify(page, "PackType Change Med");
|
||||||
|
|
||||||
// Verify it's still a bottle after reload
|
// Verify final package type persisted
|
||||||
await clickEditMed(page, "PackType Change Med");
|
await clickEditMed(page, "PackType Change Med");
|
||||||
await expect(page.locator("select.package-type-select")).toHaveValue("bottle");
|
await expect(page.locator("select.package-type-select")).toHaveValue("liquid_container");
|
||||||
});
|
});
|
||||||
|
|
||||||
test("should edit multiple fields at once (name, notes, generic, taken-by)", async ({ page }) => {
|
test("should edit multiple fields at once (name, notes, generic, taken-by)", async ({ page }) => {
|
||||||
|
|||||||
@@ -0,0 +1,193 @@
|
|||||||
|
import { authFile, createMedicationViaAPI, deleteAllMedicationsViaAPI, expect, navigateTo, test } from "./fixtures";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Medication Lifecycle Integration Tests
|
||||||
|
*
|
||||||
|
* End-to-end workflows that verify changes propagate across pages:
|
||||||
|
* create → verify on medications → check in planner → check in schedule → edit → delete
|
||||||
|
*/
|
||||||
|
test.describe("Medication lifecycle", () => {
|
||||||
|
test.use({ storageState: authFile });
|
||||||
|
test.describe.configure({ timeout: 90000 });
|
||||||
|
|
||||||
|
const MED_NAME = "Lifecycle TestMed";
|
||||||
|
const MED_EDITED = "Lifecycle Edited";
|
||||||
|
|
||||||
|
test.beforeAll(async () => {
|
||||||
|
await deleteAllMedicationsViaAPI();
|
||||||
|
});
|
||||||
|
|
||||||
|
test.afterAll(async () => {
|
||||||
|
await deleteAllMedicationsViaAPI();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("create medication via API and verify it appears on all pages", async ({ page }) => {
|
||||||
|
const todayMorning = (() => {
|
||||||
|
const d = new Date();
|
||||||
|
d.setHours(8, 0, 0, 0);
|
||||||
|
const pad = (n: number) => n.toString().padStart(2, "0");
|
||||||
|
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
|
||||||
|
})();
|
||||||
|
|
||||||
|
// Step 1: Create medication
|
||||||
|
const created = await createMedicationViaAPI({
|
||||||
|
name: MED_NAME,
|
||||||
|
packageType: "blister",
|
||||||
|
packCount: 1,
|
||||||
|
blistersPerPack: 2,
|
||||||
|
pillsPerBlister: 10,
|
||||||
|
looseTablets: 0,
|
||||||
|
intakes: [{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false }],
|
||||||
|
});
|
||||||
|
expect(created.id).toBeTruthy();
|
||||||
|
|
||||||
|
// Step 2: Verify on medications page
|
||||||
|
await navigateTo(page, "/medications");
|
||||||
|
await expect(page.getByText(MED_NAME).first()).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
|
// Step 3: Verify in planner
|
||||||
|
await navigateTo(page, "/planner");
|
||||||
|
await page.waitForLoadState("networkidle");
|
||||||
|
await page.locator('form.planner button[type="submit"]').click();
|
||||||
|
await expect(page.locator(".table")).toBeVisible({ timeout: 15000 });
|
||||||
|
await expect(page.locator(".table").getByText(MED_NAME)).toBeVisible();
|
||||||
|
|
||||||
|
// Step 4: Verify in schedule
|
||||||
|
await navigateTo(page, "/schedule");
|
||||||
|
await expect(page.getByText(MED_NAME).first()).toBeVisible({ timeout: 10000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
test("edit medication name via UI and verify update propagates", async ({ page }) => {
|
||||||
|
await deleteAllMedicationsViaAPI();
|
||||||
|
|
||||||
|
const todayMorning = (() => {
|
||||||
|
const d = new Date();
|
||||||
|
d.setHours(8, 0, 0, 0);
|
||||||
|
const pad = (n: number) => n.toString().padStart(2, "0");
|
||||||
|
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
|
||||||
|
})();
|
||||||
|
|
||||||
|
// Create a fresh medication for this test
|
||||||
|
await createMedicationViaAPI({
|
||||||
|
name: MED_NAME,
|
||||||
|
packageType: "blister",
|
||||||
|
packCount: 1,
|
||||||
|
blistersPerPack: 2,
|
||||||
|
pillsPerBlister: 10,
|
||||||
|
looseTablets: 0,
|
||||||
|
intakes: [{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false }],
|
||||||
|
});
|
||||||
|
|
||||||
|
// Navigate to medications page
|
||||||
|
await navigateTo(page, "/medications");
|
||||||
|
await expect(page.getByText(MED_NAME).first()).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
|
// Open edit view from medication row actions
|
||||||
|
const medRow = page.locator(".med-row").filter({ hasText: MED_NAME });
|
||||||
|
await expect(medRow.first()).toBeVisible({ timeout: 10000 });
|
||||||
|
await medRow.first().locator("button.info").click();
|
||||||
|
await expect(page.locator("h2").filter({ hasText: /(Edit(:| (entry|medication))|form\.editEntry)/i })).toBeVisible({
|
||||||
|
timeout: 5000,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Update the name
|
||||||
|
const form = page.locator("form.form-grid:visible").first();
|
||||||
|
const nameInput = form.getByLabel(/(Commercial Name|Name|form\.name)/i).first();
|
||||||
|
await nameInput.fill(MED_EDITED);
|
||||||
|
|
||||||
|
// Save
|
||||||
|
const submitButton = form.locator('button[type="submit"]').first();
|
||||||
|
await expect(submitButton).toBeEnabled({ timeout: 5000 });
|
||||||
|
await submitButton.click();
|
||||||
|
|
||||||
|
// Wait for modal to close or save to complete
|
||||||
|
await page.waitForLoadState("networkidle");
|
||||||
|
|
||||||
|
// Verify edited name appears on medications page
|
||||||
|
await navigateTo(page, "/medications");
|
||||||
|
await expect(page.getByText(MED_EDITED).first()).toBeVisible({ timeout: 10000 });
|
||||||
|
// Old name should no longer appear
|
||||||
|
await expect(page.locator(".med-row").filter({ hasText: MED_NAME })).toHaveCount(0, { timeout: 5000 });
|
||||||
|
});
|
||||||
|
|
||||||
|
test("delete medication via API and verify it disappears from all pages", async ({ page }) => {
|
||||||
|
const todayMorning = (() => {
|
||||||
|
const d = new Date();
|
||||||
|
d.setHours(8, 0, 0, 0);
|
||||||
|
const pad = (n: number) => n.toString().padStart(2, "0");
|
||||||
|
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
|
||||||
|
})();
|
||||||
|
|
||||||
|
// Create and then delete
|
||||||
|
await deleteAllMedicationsViaAPI();
|
||||||
|
await createMedicationViaAPI({
|
||||||
|
name: MED_NAME,
|
||||||
|
packageType: "blister",
|
||||||
|
packCount: 1,
|
||||||
|
blistersPerPack: 1,
|
||||||
|
pillsPerBlister: 5,
|
||||||
|
looseTablets: 0,
|
||||||
|
intakes: [{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false }],
|
||||||
|
});
|
||||||
|
|
||||||
|
// Verify it exists first
|
||||||
|
await navigateTo(page, "/medications");
|
||||||
|
await expect(page.getByText(MED_NAME)).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
|
// Delete via API
|
||||||
|
await deleteAllMedicationsViaAPI();
|
||||||
|
|
||||||
|
// Verify gone from medications page
|
||||||
|
await navigateTo(page, "/medications");
|
||||||
|
await expect(page.getByText(MED_NAME)).not.toBeVisible({ timeout: 5000 });
|
||||||
|
|
||||||
|
// Verify planner shows no results for this med
|
||||||
|
await navigateTo(page, "/planner");
|
||||||
|
await page.waitForLoadState("networkidle");
|
||||||
|
await page.locator('form.planner button[type="submit"]').click();
|
||||||
|
// Either no table or table without the medication name
|
||||||
|
const table = page.locator(".table");
|
||||||
|
const tableVisible = await table.isVisible().catch(() => false);
|
||||||
|
if (tableVisible) {
|
||||||
|
await expect(table.getByText(MED_NAME)).not.toBeVisible({ timeout: 3000 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test("medication with multiple intakes shows all schedule entries", async ({ page }) => {
|
||||||
|
const todayMorning = (() => {
|
||||||
|
const d = new Date();
|
||||||
|
d.setHours(8, 0, 0, 0);
|
||||||
|
const pad = (n: number) => n.toString().padStart(2, "0");
|
||||||
|
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
|
||||||
|
})();
|
||||||
|
|
||||||
|
const todayEvening = (() => {
|
||||||
|
const d = new Date();
|
||||||
|
d.setHours(20, 0, 0, 0);
|
||||||
|
const pad = (n: number) => n.toString().padStart(2, "0");
|
||||||
|
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
|
||||||
|
})();
|
||||||
|
|
||||||
|
await deleteAllMedicationsViaAPI();
|
||||||
|
await createMedicationViaAPI({
|
||||||
|
name: "MultiIntake Med",
|
||||||
|
packageType: "blister",
|
||||||
|
packCount: 1,
|
||||||
|
blistersPerPack: 2,
|
||||||
|
pillsPerBlister: 10,
|
||||||
|
looseTablets: 0,
|
||||||
|
intakes: [
|
||||||
|
{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false },
|
||||||
|
{ usage: 2, every: 1, start: todayEvening, intakeRemindersEnabled: false },
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
// Verify schedule shows this medication
|
||||||
|
await navigateTo(page, "/schedule");
|
||||||
|
await expect(page.getByText("MultiIntake Med").first()).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
|
// The medication should appear at least twice (morning + evening)
|
||||||
|
const medEntries = page.getByText("MultiIntake Med");
|
||||||
|
expect(await medEntries.count()).toBeGreaterThanOrEqual(2);
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -87,25 +87,17 @@ test.describe("Medications Page", () => {
|
|||||||
expect(hasPacks || hasTotal).toBeTruthy();
|
expect(hasPacks || hasTotal).toBeTruthy();
|
||||||
});
|
});
|
||||||
|
|
||||||
test("should toggle package type between blister and bottle", async ({ page }) => {
|
test("should expose all supported package type options", async ({ page }) => {
|
||||||
await openMedicationForm(page);
|
await openMedicationForm(page);
|
||||||
const form = visibleMedForm(page);
|
const form = visibleMedForm(page);
|
||||||
await page.getByRole("tab", { name: /Package/i }).click();
|
const packageSelect = form.locator("select.package-type-select");
|
||||||
|
await expect(packageSelect).toBeVisible();
|
||||||
|
|
||||||
// Find the package type radio buttons or selector
|
const optionValues = await packageSelect
|
||||||
const blisterOption = form.getByText(/(Blister Pack|form\.packageType\.blister)/i);
|
.locator("option")
|
||||||
const bottleOption = form.getByText(/(Pill Bottle|form\.packageType\.bottle)/i);
|
.evaluateAll((options) => options.map((option) => (option as HTMLOptionElement).value));
|
||||||
|
|
||||||
if (await blisterOption.isVisible().catch(() => false)) {
|
expect(optionValues).toEqual(expect.arrayContaining(["blister", "bottle", "tube", "liquid_container"]));
|
||||||
// Switch to bottle
|
|
||||||
await bottleOption.click();
|
|
||||||
// Bottle-specific fields should appear
|
|
||||||
await expect(form.getByLabel(/(Total Capacity|form\.totalCapacity)/i)).toBeVisible();
|
|
||||||
|
|
||||||
// Switch back to blister
|
|
||||||
await blisterOption.click();
|
|
||||||
await expect(form.getByLabel(/(Blisters per pack|form\.blistersPerPack)/i)).toBeVisible();
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
test("should have intake schedule with add button", async ({ page }) => {
|
test("should have intake schedule with add button", async ({ page }) => {
|
||||||
|
|||||||
@@ -0,0 +1,98 @@
|
|||||||
|
import { authFile, createMedicationViaAPI, deleteAllMedicationsViaAPI, expect, navigateTo, test } from "./fixtures";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Performance Tests
|
||||||
|
*
|
||||||
|
* Verify the schedule timeline and planner render within acceptable
|
||||||
|
* time limits when many medications exist.
|
||||||
|
*/
|
||||||
|
test.describe("Performance with many medications", () => {
|
||||||
|
test.use({ storageState: authFile });
|
||||||
|
test.describe.configure({ timeout: 120000 });
|
||||||
|
|
||||||
|
const MED_COUNT = 20;
|
||||||
|
const MED_PREFIX = "PerfTest Med";
|
||||||
|
|
||||||
|
test.beforeAll(async () => {
|
||||||
|
await deleteAllMedicationsViaAPI();
|
||||||
|
|
||||||
|
const todayMorning = (() => {
|
||||||
|
const d = new Date();
|
||||||
|
d.setHours(8, 0, 0, 0);
|
||||||
|
const pad = (n: number) => n.toString().padStart(2, "0");
|
||||||
|
return `${d.getFullYear()}-${pad(d.getMonth() + 1)}-${pad(d.getDate())}T${pad(d.getHours())}:${pad(d.getMinutes())}`;
|
||||||
|
})();
|
||||||
|
|
||||||
|
// Create medications sequentially (API rate limits prevent parallel)
|
||||||
|
for (let i = 1; i <= MED_COUNT; i++) {
|
||||||
|
await createMedicationViaAPI({
|
||||||
|
name: `${MED_PREFIX} ${i}`,
|
||||||
|
packageType: "blister",
|
||||||
|
packCount: 1,
|
||||||
|
blistersPerPack: 2,
|
||||||
|
pillsPerBlister: 10,
|
||||||
|
looseTablets: 0,
|
||||||
|
intakes: [{ usage: 1, every: 1, start: todayMorning, intakeRemindersEnabled: false }],
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
test.afterAll(async () => {
|
||||||
|
await deleteAllMedicationsViaAPI();
|
||||||
|
});
|
||||||
|
|
||||||
|
test("schedule page renders within 10 seconds with 20 medications", async ({ page }) => {
|
||||||
|
const start = Date.now();
|
||||||
|
await navigateTo(page, "/schedule");
|
||||||
|
|
||||||
|
// Wait for schedule entries to render
|
||||||
|
const scheduleEntries = page.locator(".schedule-entry, .timeline-entry, .card");
|
||||||
|
await expect(scheduleEntries.first()).toBeVisible({ timeout: 15000 });
|
||||||
|
|
||||||
|
const renderTime = Date.now() - start;
|
||||||
|
|
||||||
|
// Verify all medications appear
|
||||||
|
for (let i = 1; i <= MED_COUNT; i++) {
|
||||||
|
await expect(page.getByText(`${MED_PREFIX} ${i}`).first()).toBeVisible({ timeout: 5000 });
|
||||||
|
}
|
||||||
|
|
||||||
|
// Goal: render under 10 seconds
|
||||||
|
expect(renderTime).toBeLessThan(10000);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("medications page renders within 10 seconds with 20 medications", async ({ page }) => {
|
||||||
|
const start = Date.now();
|
||||||
|
await navigateTo(page, "/medications");
|
||||||
|
|
||||||
|
// Wait for medication cards to render
|
||||||
|
const medEntries = page.locator(".medication-card, .card, .table-row");
|
||||||
|
await expect(medEntries.first()).toBeVisible({ timeout: 15000 });
|
||||||
|
|
||||||
|
const renderTime = Date.now() - start;
|
||||||
|
|
||||||
|
// Verify count — all 20 should be visible
|
||||||
|
for (let i = 1; i <= MED_COUNT; i++) {
|
||||||
|
await expect(page.getByText(`${MED_PREFIX} ${i}`).first()).toBeVisible({ timeout: 5000 });
|
||||||
|
}
|
||||||
|
|
||||||
|
expect(renderTime).toBeLessThan(10000);
|
||||||
|
});
|
||||||
|
|
||||||
|
test("planner calculates within 15 seconds with 20 medications", async ({ page }) => {
|
||||||
|
await navigateTo(page, "/planner");
|
||||||
|
|
||||||
|
const start = Date.now();
|
||||||
|
await page.waitForLoadState("networkidle");
|
||||||
|
await page.locator('form.planner button[type="submit"]').click();
|
||||||
|
await expect(page.locator(".table")).toBeVisible({ timeout: 20000 });
|
||||||
|
|
||||||
|
const calcTime = Date.now() - start;
|
||||||
|
|
||||||
|
// All medications should appear in the results
|
||||||
|
const rows = page.locator(".table .table-row");
|
||||||
|
expect(await rows.count()).toBeGreaterThanOrEqual(MED_COUNT);
|
||||||
|
|
||||||
|
// Goal: calculate and render under 15 seconds
|
||||||
|
expect(calcTime).toBeLessThan(15000);
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -106,7 +106,7 @@ test.describe("Planner with medications", () => {
|
|||||||
expect(await statusChips.count()).toBeGreaterThanOrEqual(2);
|
expect(await statusChips.count()).toBeGreaterThanOrEqual(2);
|
||||||
});
|
});
|
||||||
|
|
||||||
test("should show usage data in results rows", async ({ page }) => {
|
test("should show correct usage values in results rows", async ({ page }) => {
|
||||||
await navigateTo(page, "/planner");
|
await navigateTo(page, "/planner");
|
||||||
await calculatePlanner(page);
|
await calculatePlanner(page);
|
||||||
|
|
||||||
@@ -116,10 +116,15 @@ test.describe("Planner with medications", () => {
|
|||||||
const rows = resultsTable.locator(".table-row");
|
const rows = resultsTable.locator(".table-row");
|
||||||
expect(await rows.count()).toBeGreaterThanOrEqual(2);
|
expect(await rows.count()).toBeGreaterThanOrEqual(2);
|
||||||
|
|
||||||
const firstRowText = await rows.first().textContent();
|
// Each medication has usage=1, every=1 → plannerUsage should reflect the period
|
||||||
expect(firstRowText).toBeTruthy();
|
// Verify the usage column contains a numeric <strong> value and "pill(s)"
|
||||||
// Check for "pill" (matches both "pill" and "pills")
|
for (const row of await rows.all()) {
|
||||||
expect(firstRowText!.toLowerCase()).toContain("pill");
|
const usageCell = row.locator("[data-label]").nth(1); // Usage is 2nd column
|
||||||
|
const usageStrong = usageCell.locator("strong");
|
||||||
|
await expect(usageStrong).toBeVisible();
|
||||||
|
const usageText = await usageStrong.textContent();
|
||||||
|
expect(Number(usageText)).toBeGreaterThan(0);
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
test("should show danger status for low-stock medication over 90 days", async ({ page }) => {
|
test("should show danger status for low-stock medication over 90 days", async ({ page }) => {
|
||||||
@@ -139,9 +144,16 @@ test.describe("Planner with medications", () => {
|
|||||||
const resultsTable = page.locator(".table");
|
const resultsTable = page.locator(".table");
|
||||||
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// Low-stock med (3 pills) should have a danger chip over 90 days
|
// Low-stock med (3 pills, usage 1/day, 90 days) should have danger status
|
||||||
const dangerChips = resultsTable.locator(".status-chip.danger");
|
const dangerChips = resultsTable.locator(".status-chip.danger");
|
||||||
expect(await dangerChips.count()).toBeGreaterThanOrEqual(1);
|
expect(await dangerChips.count()).toBeGreaterThanOrEqual(1);
|
||||||
|
|
||||||
|
// Find the low-stock med row and verify its usage value ~90 pills
|
||||||
|
const lowStockRow = resultsTable.locator(".table-row", { hasText: MED_LOW });
|
||||||
|
await expect(lowStockRow).toBeVisible();
|
||||||
|
const lowUsage = await lowStockRow.locator("[data-label] strong").first().textContent();
|
||||||
|
expect(Number(lowUsage)).toBeGreaterThanOrEqual(85); // ~90 pills needed
|
||||||
|
expect(Number(lowUsage)).toBeLessThanOrEqual(95);
|
||||||
});
|
});
|
||||||
|
|
||||||
test("should show Enough status for well-stocked medication over 7 days", async ({ page }) => {
|
test("should show Enough status for well-stocked medication over 7 days", async ({ page }) => {
|
||||||
@@ -161,9 +173,16 @@ test.describe("Planner with medications", () => {
|
|||||||
const resultsTable = page.locator(".table");
|
const resultsTable = page.locator(".table");
|
||||||
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// With 60 pills and 7-day range, high-stock should be "Enough"
|
// High-stock med (60 pills, usage 1/day, 7 days → needs ~7, has 60) should be "Enough"
|
||||||
const successChips = resultsTable.locator(".status-chip.success");
|
const highStockRow = resultsTable.locator(".table-row", { hasText: MED_HIGH });
|
||||||
expect(await successChips.count()).toBeGreaterThanOrEqual(1);
|
await expect(highStockRow).toBeVisible();
|
||||||
|
const highStatus = highStockRow.locator(".status-chip.success");
|
||||||
|
await expect(highStatus).toBeVisible();
|
||||||
|
|
||||||
|
// Verify usage is ~7 pills for the 7-day range
|
||||||
|
const highUsage = await highStockRow.locator("[data-label] strong").first().textContent();
|
||||||
|
expect(Number(highUsage)).toBeGreaterThanOrEqual(5);
|
||||||
|
expect(Number(highUsage)).toBeLessThanOrEqual(10);
|
||||||
});
|
});
|
||||||
|
|
||||||
test("should show table header with correct columns", async ({ page }) => {
|
test("should show table header with correct columns", async ({ page }) => {
|
||||||
@@ -180,6 +199,28 @@ test.describe("Planner with medications", () => {
|
|||||||
await expect(tableHead.getByText(/Status/i)).toBeVisible();
|
await expect(tableHead.getByText(/Status/i)).toBeVisible();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
test("should display available stock for each medication", async ({ page }) => {
|
||||||
|
await navigateTo(page, "/planner");
|
||||||
|
await calculatePlanner(page);
|
||||||
|
|
||||||
|
const resultsTable = page.locator(".table");
|
||||||
|
await expect(resultsTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
|
// High-stock med should show a blister + loose-pill stock breakdown
|
||||||
|
const highStockRow = resultsTable.locator(".table-row", { hasText: MED_HIGH });
|
||||||
|
await expect(highStockRow).toBeVisible();
|
||||||
|
const highStockText = await highStockRow.textContent();
|
||||||
|
expect(highStockText).toMatch(/\d+\s*(blisters|Blister)/i);
|
||||||
|
expect(highStockText).toMatch(/\d+\s*(pill|pills|Tablette|Tabletten)/i);
|
||||||
|
|
||||||
|
// Low-stock med: 1 pack × 1 blister × 3 pills = 3 pills = 0 full blisters + 3 loose
|
||||||
|
const lowStockRow = resultsTable.locator(".table-row", { hasText: MED_LOW });
|
||||||
|
await expect(lowStockRow).toBeVisible();
|
||||||
|
const lowStockText = await lowStockRow.textContent();
|
||||||
|
// Should show 3 loose pills
|
||||||
|
expect(lowStockText).toMatch(/3\s*(pill|pills|Tablette|Tabletten)/i);
|
||||||
|
});
|
||||||
|
|
||||||
test("should reset form and clear results", async ({ page }) => {
|
test("should reset form and clear results", async ({ page }) => {
|
||||||
await navigateTo(page, "/planner");
|
await navigateTo(page, "/planner");
|
||||||
await calculatePlanner(page);
|
await calculatePlanner(page);
|
||||||
|
|||||||
@@ -224,15 +224,4 @@ test.describe("Schedule with medications", () => {
|
|||||||
expect(await takeButtons.count()).toBeGreaterThanOrEqual(1);
|
expect(await takeButtons.count()).toBeGreaterThanOrEqual(1);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
test("should show medication names in timeline rows", async ({ page }) => {
|
|
||||||
await navigateTo(page, "/dashboard");
|
|
||||||
await page.waitForLoadState("networkidle");
|
|
||||||
|
|
||||||
const todayBlock = page.locator(".day-block.today");
|
|
||||||
await expect(todayBlock).toBeVisible({ timeout: 15000 });
|
|
||||||
|
|
||||||
const medNames = todayBlock.locator(".med-name");
|
|
||||||
expect(await medNames.count()).toBeGreaterThanOrEqual(1);
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -150,8 +150,7 @@ test.describe("Schedule Timeline", () => {
|
|||||||
test("should show overview table with stock status", async ({ page }) => {
|
test("should show overview table with stock status", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
// Overview table has class .table.table-7
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
const overviewTable = page.locator(".table.table-7");
|
|
||||||
await expect(overviewTable).toBeVisible();
|
await expect(overviewTable).toBeVisible();
|
||||||
await expect(overviewTable.locator(".table-head")).toBeVisible();
|
await expect(overviewTable.locator(".table-head")).toBeVisible();
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -72,7 +72,7 @@ test.describe("Share Schedule", () => {
|
|||||||
test("should show taken-by badges on dashboard overview table", async ({ page }) => {
|
test("should show taken-by badges on dashboard overview table", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// Alice's medication should show "Alice" badge
|
// Alice's medication should show "Alice" badge
|
||||||
@@ -253,7 +253,7 @@ test.describe("Share Schedule", () => {
|
|||||||
test("should show notes icon on dashboard for medication with notes", async ({ page }) => {
|
test("should show notes icon on dashboard for medication with notes", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// Alice's med has notes — should show the 📝 icon
|
// Alice's med has notes — should show the 📝 icon
|
||||||
@@ -265,7 +265,7 @@ test.describe("Share Schedule", () => {
|
|||||||
test("should show notes in medication detail modal", async ({ page }) => {
|
test("should show notes in medication detail modal", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// Click on Alice's med to open detail modal
|
// Click on Alice's med to open detail modal
|
||||||
|
|||||||
@@ -125,7 +125,7 @@ test.describe("Stock Status Levels", () => {
|
|||||||
test("should show all medications in overview table", async ({ page }) => {
|
test("should show all medications in overview table", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// All 5 medications should appear
|
// All 5 medications should appear
|
||||||
@@ -139,7 +139,7 @@ test.describe("Stock Status Levels", () => {
|
|||||||
test("should show High status chip for well-stocked medication", async ({ page }) => {
|
test("should show High status chip for well-stocked medication", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// High stock med row should have a .status-chip.high
|
// High stock med row should have a .status-chip.high
|
||||||
@@ -151,7 +151,7 @@ test.describe("Stock Status Levels", () => {
|
|||||||
test("should show Normal status chip for moderate stock medication", async ({ page }) => {
|
test("should show Normal status chip for moderate stock medication", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
const normalRow = overviewTable.locator(".table-row").filter({ hasText: MED_NORMAL });
|
const normalRow = overviewTable.locator(".table-row").filter({ hasText: MED_NORMAL });
|
||||||
@@ -162,7 +162,7 @@ test.describe("Stock Status Levels", () => {
|
|||||||
test("should show Warning status chip for low stock medication", async ({ page }) => {
|
test("should show Warning status chip for low stock medication", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
const lowRow = overviewTable.locator(".table-row").filter({ hasText: MED_LOW });
|
const lowRow = overviewTable.locator(".table-row").filter({ hasText: MED_LOW });
|
||||||
@@ -173,7 +173,7 @@ test.describe("Stock Status Levels", () => {
|
|||||||
test("should show Danger status chip for critical stock medication", async ({ page }) => {
|
test("should show Danger status chip for critical stock medication", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
const criticalRow = overviewTable.locator(".table-row").filter({ hasText: MED_CRITICAL });
|
const criticalRow = overviewTable.locator(".table-row").filter({ hasText: MED_CRITICAL });
|
||||||
@@ -184,7 +184,7 @@ test.describe("Stock Status Levels", () => {
|
|||||||
test("should show Danger status chip for depleted medication", async ({ page }) => {
|
test("should show Danger status chip for depleted medication", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
const depletedRow = overviewTable.locator(".table-row").filter({ hasText: MED_DEPLETED });
|
const depletedRow = overviewTable.locator(".table-row").filter({ hasText: MED_DEPLETED });
|
||||||
@@ -195,7 +195,7 @@ test.describe("Stock Status Levels", () => {
|
|||||||
test("should show days-left and runs-out date in overview", async ({ page }) => {
|
test("should show days-left and runs-out date in overview", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// High stock should show many days (around 299)
|
// High stock should show many days (around 299)
|
||||||
@@ -227,7 +227,7 @@ test.describe("Stock Status Levels", () => {
|
|||||||
test("should color-code stock values depending on status", async ({ page }) => {
|
test("should color-code stock values depending on status", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// High stock row should have success-text class on stock cells
|
// High stock row should have success-text class on stock cells
|
||||||
@@ -255,7 +255,7 @@ test.describe("Stock Status Levels", () => {
|
|||||||
test("should open medication detail modal showing stock info", async ({ page }) => {
|
test("should open medication detail modal showing stock info", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// Click on the critical stock medication row
|
// Click on the critical stock medication row
|
||||||
@@ -278,7 +278,7 @@ test.describe("Stock Status Levels", () => {
|
|||||||
test("should show generic name in overview for medications that have one", async ({ page }) => {
|
test("should show generic name in overview for medications that have one", async ({ page }) => {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
|
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
// Click on the normal stock med (has generic name "Ibuprofen 400mg")
|
// Click on the normal stock med (has generic name "Ibuprofen 400mg")
|
||||||
|
|||||||
@@ -54,7 +54,7 @@ test.describe("MedDetail footer tooltip visibility", () => {
|
|||||||
*/
|
*/
|
||||||
async function openMedDetailModal(page: import("@playwright/test").Page) {
|
async function openMedDetailModal(page: import("@playwright/test").Page) {
|
||||||
await navigateTo(page, "/dashboard");
|
await navigateTo(page, "/dashboard");
|
||||||
const overviewTable = page.locator(".table.table-7");
|
const overviewTable = page.locator(".dashboard-overview-section .table").first();
|
||||||
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
await expect(overviewTable).toBeVisible({ timeout: 10000 });
|
||||||
|
|
||||||
const medRow = overviewTable.locator(".table-row").filter({ hasText: MED_NAME }).first();
|
const medRow = overviewTable.locator(".table-row").filter({ hasText: MED_NAME }).first();
|
||||||
|
|||||||
@@ -4,21 +4,30 @@
|
|||||||
# Translates LOG_LEVEL into nginx access log control before
|
# Translates LOG_LEVEL into nginx access log control before
|
||||||
# delegating to the standard nginx-unprivileged entrypoint.
|
# delegating to the standard nginx-unprivileged entrypoint.
|
||||||
#
|
#
|
||||||
# LOG_LEVEL=debug|info → access logs enabled (default)
|
# LOG_LEVEL=debug → all access logs enabled (including polling)
|
||||||
# LOG_LEVEL=warn|error|fatal|silent → access logs suppressed
|
# LOG_LEVEL=info → access logs enabled, polling endpoints suppressed (default)
|
||||||
|
# LOG_LEVEL=warn|error|fatal|silent → all access logs suppressed
|
||||||
# =============================================================================
|
# =============================================================================
|
||||||
|
|
||||||
# Normalize: lowercase + trim whitespace
|
# Normalize: lowercase + trim whitespace
|
||||||
level=$(printf '%s' "${LOG_LEVEL:-info}" | tr '[:upper:]' '[:lower:]' | tr -d '[:space:]')
|
level=$(printf '%s' "${LOG_LEVEL:-info}" | tr '[:upper:]' '[:lower:]' | tr -d '[:space:]')
|
||||||
|
|
||||||
case "$level" in
|
case "$level" in
|
||||||
|
debug)
|
||||||
|
export NGINX_ACCESS_LOG="/dev/stdout timed"
|
||||||
|
export NGINX_POLLING_LOG="/dev/stdout timed"
|
||||||
|
echo "[nginx-entrypoint] LOG_LEVEL=${LOG_LEVEL} → access_log on (all requests)"
|
||||||
|
;;
|
||||||
warn|error|fatal|silent)
|
warn|error|fatal|silent)
|
||||||
export NGINX_ACCESS_LOG="off"
|
export NGINX_ACCESS_LOG="off"
|
||||||
|
export NGINX_POLLING_LOG="off"
|
||||||
echo "[nginx-entrypoint] LOG_LEVEL=${LOG_LEVEL} → access_log off"
|
echo "[nginx-entrypoint] LOG_LEVEL=${LOG_LEVEL} → access_log off"
|
||||||
;;
|
;;
|
||||||
*)
|
*)
|
||||||
export NGINX_ACCESS_LOG="/dev/stdout"
|
# info (default): log everything except high-frequency polling endpoints
|
||||||
echo "[nginx-entrypoint] LOG_LEVEL=${LOG_LEVEL:-info} → access_log /dev/stdout"
|
export NGINX_ACCESS_LOG="/dev/stdout timed"
|
||||||
|
export NGINX_POLLING_LOG="off"
|
||||||
|
echo "[nginx-entrypoint] LOG_LEVEL=${LOG_LEVEL:-info} → access_log on (polling suppressed)"
|
||||||
;;
|
;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
|
|||||||
@@ -1,3 +1,6 @@
|
|||||||
|
# Must be defined at http-level (outside server block)
|
||||||
|
log_format timed '$time_iso8601 $status $request_method $request_uri ($request_time s)';
|
||||||
|
|
||||||
server {
|
server {
|
||||||
# Port 8080 for unprivileged nginx (non-root)
|
# Port 8080 for unprivileged nginx (non-root)
|
||||||
listen 8080;
|
listen 8080;
|
||||||
@@ -24,6 +27,52 @@ server {
|
|||||||
try_files $uri /index.html;
|
try_files $uri /index.html;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
# High-frequency polling endpoints — suppress access logs at info level
|
||||||
|
# (visible at debug level via NGINX_POLLING_LOG)
|
||||||
|
# -------------------------------------------------------------------------
|
||||||
|
location = /api/doses/taken {
|
||||||
|
access_log ${NGINX_POLLING_LOG};
|
||||||
|
resolver 127.0.0.11 valid=10s ipv6=off;
|
||||||
|
set $backend_upstream ${BACKEND_URL};
|
||||||
|
rewrite ^/api/(.*)$ /$1 break;
|
||||||
|
proxy_pass http://$backend_upstream;
|
||||||
|
proxy_set_header Host $host;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
proxy_set_header X-Forwarded-Proto $scheme;
|
||||||
|
proxy_pass_header Set-Cookie;
|
||||||
|
proxy_cookie_path / /;
|
||||||
|
}
|
||||||
|
|
||||||
|
location ~ ^/api/share/[^/]+/doses$ {
|
||||||
|
access_log ${NGINX_POLLING_LOG};
|
||||||
|
resolver 127.0.0.11 valid=10s ipv6=off;
|
||||||
|
set $backend_upstream ${BACKEND_URL};
|
||||||
|
rewrite ^/api/(.*)$ /$1 break;
|
||||||
|
proxy_pass http://$backend_upstream;
|
||||||
|
proxy_set_header Host $host;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
proxy_set_header X-Forwarded-Proto $scheme;
|
||||||
|
proxy_pass_header Set-Cookie;
|
||||||
|
proxy_cookie_path / /;
|
||||||
|
}
|
||||||
|
|
||||||
|
location = /api/health {
|
||||||
|
access_log ${NGINX_POLLING_LOG};
|
||||||
|
resolver 127.0.0.11 valid=10s ipv6=off;
|
||||||
|
set $backend_upstream ${BACKEND_URL};
|
||||||
|
rewrite ^/api/(.*)$ /$1 break;
|
||||||
|
proxy_pass http://$backend_upstream;
|
||||||
|
proxy_set_header Host $host;
|
||||||
|
proxy_set_header X-Real-IP $remote_addr;
|
||||||
|
proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
|
||||||
|
proxy_set_header X-Forwarded-Proto $scheme;
|
||||||
|
proxy_pass_header Set-Cookie;
|
||||||
|
proxy_cookie_path / /;
|
||||||
|
}
|
||||||
|
|
||||||
location /api/ {
|
location /api/ {
|
||||||
# Use variable for runtime DNS resolution (nginx resolves at startup by default)
|
# Use variable for runtime DNS resolution (nginx resolves at startup by default)
|
||||||
# Docker embedded DNS (127.0.0.11) with 10s cache
|
# Docker embedded DNS (127.0.0.11) with 10s cache
|
||||||
|
|||||||
Generated
+164
-131
@@ -1,18 +1,18 @@
|
|||||||
{
|
{
|
||||||
"name": "medassist-ng-frontend",
|
"name": "medassist-ng-frontend",
|
||||||
"version": "1.15.1",
|
"version": "1.18.2",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "medassist-ng-frontend",
|
"name": "medassist-ng-frontend",
|
||||||
"version": "1.15.1",
|
"version": "1.18.2",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"i18next": "^25.8.13",
|
"i18next": "^25.8.13",
|
||||||
"i18next-browser-languagedetector": "^8.2.1",
|
"i18next-browser-languagedetector": "^8.2.1",
|
||||||
"lucide-react": "^0.575.0",
|
"lucide-react": "^0.575.0",
|
||||||
"react": "^18.3.1",
|
"react": "^19.2.4",
|
||||||
"react-dom": "^18.3.1",
|
"react-dom": "^19.2.4",
|
||||||
"react-i18next": "^15.4.1",
|
"react-i18next": "^15.4.1",
|
||||||
"react-router-dom": "^7.13.1",
|
"react-router-dom": "^7.13.1",
|
||||||
"zod": "^4.3.6"
|
"zod": "^4.3.6"
|
||||||
@@ -23,9 +23,9 @@
|
|||||||
"@testing-library/jest-dom": "^6.9.1",
|
"@testing-library/jest-dom": "^6.9.1",
|
||||||
"@testing-library/react": "^16.3.2",
|
"@testing-library/react": "^16.3.2",
|
||||||
"@testing-library/user-event": "^14.6.1",
|
"@testing-library/user-event": "^14.6.1",
|
||||||
"@types/node": "^25.3.0",
|
"@types/node": "^25.3.3",
|
||||||
"@types/react": "^18.3.4",
|
"@types/react": "^19.2.14",
|
||||||
"@types/react-dom": "^18.3.0",
|
"@types/react-dom": "^19.2.3",
|
||||||
"@types/react-router-dom": "^5.3.3",
|
"@types/react-router-dom": "^5.3.3",
|
||||||
"@vitejs/plugin-react": "^5.1.4",
|
"@vitejs/plugin-react": "^5.1.4",
|
||||||
"@vitest/coverage-v8": "^4.0.18",
|
"@vitest/coverage-v8": "^4.0.18",
|
||||||
@@ -1247,9 +1247,9 @@
|
|||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-android-arm-eabi": {
|
"node_modules/@rollup/rollup-android-arm-eabi": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm-eabi/-/rollup-android-arm-eabi-4.59.0.tgz",
|
||||||
"integrity": "sha512-iDGS/h7D8t7tvZ1t6+WPK04KD0MwzLZrG0se1hzBjSi5fyxlsiggoJHwh18PCFNn7tG43OWb6pdZ6Y+rMlmyNQ==",
|
"integrity": "sha512-upnNBkA6ZH2VKGcBj9Fyl9IGNPULcjXRlg0LLeaioQWueH30p6IXtJEbKAgvyv+mJaMxSm1l6xwDXYjpEMiLMg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm"
|
"arm"
|
||||||
],
|
],
|
||||||
@@ -1261,9 +1261,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-android-arm64": {
|
"node_modules/@rollup/rollup-android-arm64": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-android-arm64/-/rollup-android-arm64-4.59.0.tgz",
|
||||||
"integrity": "sha512-wrSAViWvZHBMMlWk6EJhvg8/rjxzyEhEdgfMMjREHEq11EtJ6IP6yfcCH57YAEca2Oe3FNCE9DSTgU70EIGmVw==",
|
"integrity": "sha512-hZ+Zxj3SySm4A/DylsDKZAeVg0mvi++0PYVceVyX7hemkw7OreKdCvW2oQ3T1FMZvCaQXqOTHb8qmBShoqk69Q==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -1275,9 +1275,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-darwin-arm64": {
|
"node_modules/@rollup/rollup-darwin-arm64": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-arm64/-/rollup-darwin-arm64-4.59.0.tgz",
|
||||||
"integrity": "sha512-S87zZPBmRO6u1YXQLwpveZm4JfPpAa6oHBX7/ghSiGH3rz/KDgAu1rKdGutV+WUI6tKDMbaBJomhnT30Y2t4VQ==",
|
"integrity": "sha512-W2Psnbh1J8ZJw0xKAd8zdNgF9HRLkdWwwdWqubSVk0pUuQkoHnv7rx4GiF9rT4t5DIZGAsConRE3AxCdJ4m8rg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -1289,9 +1289,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-darwin-x64": {
|
"node_modules/@rollup/rollup-darwin-x64": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-darwin-x64/-/rollup-darwin-x64-4.59.0.tgz",
|
||||||
"integrity": "sha512-YTbnsAaHo6VrAczISxgpTva8EkfQus0VPEVJCEaboHtZRIb6h6j0BNxRBOwnDciFTZLDPW5r+ZBmhL/+YpTZgA==",
|
"integrity": "sha512-ZW2KkwlS4lwTv7ZVsYDiARfFCnSGhzYPdiOU4IM2fDbL+QGlyAbjgSFuqNRbSthybLbIJ915UtZBtmuLrQAT/w==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -1303,9 +1303,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-freebsd-arm64": {
|
"node_modules/@rollup/rollup-freebsd-arm64": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-arm64/-/rollup-freebsd-arm64-4.59.0.tgz",
|
||||||
"integrity": "sha512-1T8eY2J8rKJWzaznV7zedfdhD1BqVs1iqILhmHDq/bqCUZsrMt+j8VCTHhP0vdfbHK3e1IQ7VYx3jlKqwlf+vw==",
|
"integrity": "sha512-EsKaJ5ytAu9jI3lonzn3BgG8iRBjV4LxZexygcQbpiU0wU0ATxhNVEpXKfUa0pS05gTcSDMKpn3Sx+QB9RlTTA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -1317,9 +1317,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-freebsd-x64": {
|
"node_modules/@rollup/rollup-freebsd-x64": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-freebsd-x64/-/rollup-freebsd-x64-4.59.0.tgz",
|
||||||
"integrity": "sha512-sHTiuXyBJApxRn+VFMaw1U+Qsz4kcNlxQ742snICYPrY+DDL8/ZbaC4DVIB7vgZmp3jiDaKA0WpBdP0aqPJoBQ==",
|
"integrity": "sha512-d3DuZi2KzTMjImrxoHIAODUZYoUUMsuUiY4SRRcJy6NJoZ6iIqWnJu9IScV9jXysyGMVuW+KNzZvBLOcpdl3Vg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -1331,9 +1331,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-arm-gnueabihf": {
|
"node_modules/@rollup/rollup-linux-arm-gnueabihf": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-gnueabihf/-/rollup-linux-arm-gnueabihf-4.59.0.tgz",
|
||||||
"integrity": "sha512-dV3T9MyAf0w8zPVLVBptVlzaXxka6xg1f16VAQmjg+4KMSTWDvhimI/Y6mp8oHwNrmnmVl9XxJ/w/mO4uIQONA==",
|
"integrity": "sha512-t4ONHboXi/3E0rT6OZl1pKbl2Vgxf9vJfWgmUoCEVQVxhW6Cw/c8I6hbbu7DAvgp82RKiH7TpLwxnJeKv2pbsw==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm"
|
"arm"
|
||||||
],
|
],
|
||||||
@@ -1345,9 +1345,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-arm-musleabihf": {
|
"node_modules/@rollup/rollup-linux-arm-musleabihf": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm-musleabihf/-/rollup-linux-arm-musleabihf-4.59.0.tgz",
|
||||||
"integrity": "sha512-wIGYC1x/hyjP+KAu9+ewDI+fi5XSNiUi9Bvg6KGAh2TsNMA3tSEs+Sh6jJ/r4BV/bx/CyWu2ue9kDnIdRyafcQ==",
|
"integrity": "sha512-CikFT7aYPA2ufMD086cVORBYGHffBo4K8MQ4uPS/ZnY54GKj36i196u8U+aDVT2LX4eSMbyHtyOh7D7Zvk2VvA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm"
|
"arm"
|
||||||
],
|
],
|
||||||
@@ -1359,9 +1359,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-arm64-gnu": {
|
"node_modules/@rollup/rollup-linux-arm64-gnu": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-gnu/-/rollup-linux-arm64-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-Y+qVA0D9d0y2FRNiG9oM3Hut/DgODZbU9I8pLLPwAsU0tUKZ49cyV1tzmB/qRbSzGvY8lpgGkJuMyuhH7Ma+Vg==",
|
"integrity": "sha512-jYgUGk5aLd1nUb1CtQ8E+t5JhLc9x5WdBKew9ZgAXg7DBk0ZHErLHdXM24rfX+bKrFe+Xp5YuJo54I5HFjGDAA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -1373,9 +1373,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-arm64-musl": {
|
"node_modules/@rollup/rollup-linux-arm64-musl": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-arm64-musl/-/rollup-linux-arm64-musl-4.59.0.tgz",
|
||||||
"integrity": "sha512-juaC4bEgJsyFVfqhtGLz8mbopaWD+WeSOYr5E16y+1of6KQjc0BpwZLuxkClqY1i8sco+MdyoXPNiCkQou09+g==",
|
"integrity": "sha512-peZRVEdnFWZ5Bh2KeumKG9ty7aCXzzEsHShOZEFiCQlDEepP1dpUl/SrUNXNg13UmZl+gzVDPsiCwnV1uI0RUA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -1387,9 +1387,23 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-loong64-gnu": {
|
"node_modules/@rollup/rollup-linux-loong64-gnu": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-gnu/-/rollup-linux-loong64-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-rIEC0hZ17A42iXtHX+EPJVL/CakHo+tT7W0pbzdAGuWOt2jxDFh7A/lRhsNHBcqL4T36+UiAgwO8pbmn3dE8wA==",
|
"integrity": "sha512-gbUSW/97f7+r4gHy3Jlup8zDG190AuodsWnNiXErp9mT90iCy9NKKU0Xwx5k8VlRAIV2uU9CsMnEFg/xXaOfXg==",
|
||||||
|
"cpu": [
|
||||||
|
"loong64"
|
||||||
|
],
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"linux"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"node_modules/@rollup/rollup-linux-loong64-musl": {
|
||||||
|
"version": "4.59.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-loong64-musl/-/rollup-linux-loong64-musl-4.59.0.tgz",
|
||||||
|
"integrity": "sha512-yTRONe79E+o0FWFijasoTjtzG9EBedFXJMl888NBEDCDV9I2wGbFFfJQQe63OijbFCUZqxpHz1GzpbtSFikJ4Q==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"loong64"
|
"loong64"
|
||||||
],
|
],
|
||||||
@@ -1401,9 +1415,23 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-ppc64-gnu": {
|
"node_modules/@rollup/rollup-linux-ppc64-gnu": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-gnu/-/rollup-linux-ppc64-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-T7l409NhUE552RcAOcmJHj3xyZ2h7vMWzcwQI0hvn5tqHh3oSoclf9WgTl+0QqffWFG8MEVZZP1/OBglKZx52Q==",
|
"integrity": "sha512-sw1o3tfyk12k3OEpRddF68a1unZ5VCN7zoTNtSn2KndUE+ea3m3ROOKRCZxEpmT9nsGnogpFP9x6mnLTCaoLkA==",
|
||||||
|
"cpu": [
|
||||||
|
"ppc64"
|
||||||
|
],
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"linux"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"node_modules/@rollup/rollup-linux-ppc64-musl": {
|
||||||
|
"version": "4.59.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-ppc64-musl/-/rollup-linux-ppc64-musl-4.59.0.tgz",
|
||||||
|
"integrity": "sha512-+2kLtQ4xT3AiIxkzFVFXfsmlZiG5FXYW7ZyIIvGA7Bdeuh9Z0aN4hVyXS/G1E9bTP/vqszNIN/pUKCk/BTHsKA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"ppc64"
|
"ppc64"
|
||||||
],
|
],
|
||||||
@@ -1415,9 +1443,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-riscv64-gnu": {
|
"node_modules/@rollup/rollup-linux-riscv64-gnu": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-gnu/-/rollup-linux-riscv64-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-7OK5/GhxbnrMcxIFoYfhV/TkknarkYC1hqUw1wU2xUN3TVRLNT5FmBv4KkheSG2xZ6IEbRAhTooTV2+R5Tk0lQ==",
|
"integrity": "sha512-NDYMpsXYJJaj+I7UdwIuHHNxXZ/b/N2hR15NyH3m2qAtb/hHPA4g4SuuvrdxetTdndfj9b1WOmy73kcPRoERUg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"riscv64"
|
"riscv64"
|
||||||
],
|
],
|
||||||
@@ -1429,9 +1457,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-riscv64-musl": {
|
"node_modules/@rollup/rollup-linux-riscv64-musl": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-riscv64-musl/-/rollup-linux-riscv64-musl-4.59.0.tgz",
|
||||||
"integrity": "sha512-GwuDBE/PsXaTa76lO5eLJTyr2k8QkPipAyOrs4V/KJufHCZBJ495VCGJol35grx9xryk4V+2zd3Ri+3v7NPh+w==",
|
"integrity": "sha512-nLckB8WOqHIf1bhymk+oHxvM9D3tyPndZH8i8+35p/1YiVoVswPid2yLzgX7ZJP0KQvnkhM4H6QZ5m0LzbyIAg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"riscv64"
|
"riscv64"
|
||||||
],
|
],
|
||||||
@@ -1443,9 +1471,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-s390x-gnu": {
|
"node_modules/@rollup/rollup-linux-s390x-gnu": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-s390x-gnu/-/rollup-linux-s390x-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-IAE1Ziyr1qNfnmiQLHBURAD+eh/zH1pIeJjeShleII7Vj8kyEm2PF77o+lf3WTHDpNJcu4IXJxNO0Zluro8bOw==",
|
"integrity": "sha512-oF87Ie3uAIvORFBpwnCvUzdeYUqi2wY6jRFWJAy1qus/udHFYIkplYRW+wo+GRUP4sKzYdmE1Y3+rY5Gc4ZO+w==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"s390x"
|
"s390x"
|
||||||
],
|
],
|
||||||
@@ -1457,9 +1485,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-x64-gnu": {
|
"node_modules/@rollup/rollup-linux-x64-gnu": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-gnu/-/rollup-linux-x64-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-Pg6E+oP7GvZ4XwgRJBuSXZjcqpIW3yCBhK4BcsANvb47qMvAbCjR6E+1a/U2WXz1JJxp9/4Dno3/iSJLcm5auw==",
|
"integrity": "sha512-3AHmtQq/ppNuUspKAlvA8HtLybkDflkMuLK4DPo77DfthRb71V84/c4MlWJXixZz4uruIH4uaa07IqoAkG64fg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -1471,9 +1499,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-linux-x64-musl": {
|
"node_modules/@rollup/rollup-linux-x64-musl": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-linux-x64-musl/-/rollup-linux-x64-musl-4.59.0.tgz",
|
||||||
"integrity": "sha512-txGtluxDKTxaMDzUduGP0wdfng24y1rygUMnmlUJ88fzCCULCLn7oE5kb2+tRB+MWq1QDZT6ObT5RrR8HFRKqg==",
|
"integrity": "sha512-2UdiwS/9cTAx7qIUZB/fWtToJwvt0Vbo0zmnYt7ED35KPg13Q0ym1g442THLC7VyI6JfYTP4PiSOWyoMdV2/xg==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -1484,10 +1512,24 @@
|
|||||||
"linux"
|
"linux"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
"node_modules/@rollup/rollup-openbsd-x64": {
|
||||||
|
"version": "4.59.0",
|
||||||
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-openbsd-x64/-/rollup-openbsd-x64-4.59.0.tgz",
|
||||||
|
"integrity": "sha512-M3bLRAVk6GOwFlPTIxVBSYKUaqfLrn8l0psKinkCFxl4lQvOSz8ZrKDz2gxcBwHFpci0B6rttydI4IpS4IS/jQ==",
|
||||||
|
"cpu": [
|
||||||
|
"x64"
|
||||||
|
],
|
||||||
|
"dev": true,
|
||||||
|
"license": "MIT",
|
||||||
|
"optional": true,
|
||||||
|
"os": [
|
||||||
|
"openbsd"
|
||||||
|
]
|
||||||
|
},
|
||||||
"node_modules/@rollup/rollup-openharmony-arm64": {
|
"node_modules/@rollup/rollup-openharmony-arm64": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-openharmony-arm64/-/rollup-openharmony-arm64-4.59.0.tgz",
|
||||||
"integrity": "sha512-3DFiLPnTxiOQV993fMc+KO8zXHTcIjgaInrqlG8zDp1TlhYl6WgrOHuJkJQ6M8zHEcntSJsUp1XFZSY8C1DYbg==",
|
"integrity": "sha512-tt9KBJqaqp5i5HUZzoafHZX8b5Q2Fe7UjYERADll83O4fGqJ49O1FsL6LpdzVFQcpwvnyd0i+K/VSwu/o/nWlA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -1499,9 +1541,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-win32-arm64-msvc": {
|
"node_modules/@rollup/rollup-win32-arm64-msvc": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-arm64-msvc/-/rollup-win32-arm64-msvc-4.59.0.tgz",
|
||||||
"integrity": "sha512-nggc/wPpNTgjGg75hu+Q/3i32R00Lq1B6N1DO7MCU340MRKL3WZJMjA9U4K4gzy3dkZPXm9E1Nc81FItBVGRlA==",
|
"integrity": "sha512-V5B6mG7OrGTwnxaNUzZTDTjDS7F75PO1ae6MJYdiMu60sq0CqN5CVeVsbhPxalupvTX8gXVSU9gq+Rx1/hvu6A==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"arm64"
|
"arm64"
|
||||||
],
|
],
|
||||||
@@ -1513,9 +1555,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-win32-ia32-msvc": {
|
"node_modules/@rollup/rollup-win32-ia32-msvc": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-ia32-msvc/-/rollup-win32-ia32-msvc-4.59.0.tgz",
|
||||||
"integrity": "sha512-U/54pTbdQpPLBdEzCT6NBCFAfSZMvmjr0twhnD9f4EIvlm9wy3jjQ38yQj1AGznrNO65EWQMgm/QUjuIVrYF9w==",
|
"integrity": "sha512-UKFMHPuM9R0iBegwzKF4y0C4J9u8C6MEJgFuXTBerMk7EJ92GFVFYBfOZaSGLu6COf7FxpQNqhNS4c4icUPqxA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"ia32"
|
"ia32"
|
||||||
],
|
],
|
||||||
@@ -1527,9 +1569,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-win32-x64-gnu": {
|
"node_modules/@rollup/rollup-win32-x64-gnu": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-gnu/-/rollup-win32-x64-gnu-4.59.0.tgz",
|
||||||
"integrity": "sha512-2NqKgZSuLH9SXBBV2dWNRCZmocgSOx8OJSdpRaEcRlIfX8YrKxUT6z0F1NpvDVhOsl190UFTRh2F2WDWWCYp3A==",
|
"integrity": "sha512-laBkYlSS1n2L8fSo1thDNGrCTQMmxjYY5G0WFWjFFYZkKPjsMBsgJfGf4TLxXrF6RyhI60L8TMOjBMvXiTcxeA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -1541,9 +1583,9 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"node_modules/@rollup/rollup-win32-x64-msvc": {
|
"node_modules/@rollup/rollup-win32-x64-msvc": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/@rollup/rollup-win32-x64-msvc/-/rollup-win32-x64-msvc-4.59.0.tgz",
|
||||||
"integrity": "sha512-JRpZUhCfhZ4keB5v0fe02gQJy05GqboPOaxvjugW04RLSYYoB/9t2lx2u/tMs/Na/1NXfY8QYjgRljRpN+MjTQ==",
|
"integrity": "sha512-2HRCml6OztYXyJXAvdDXPKcawukWY2GpR5/nxKp4iBgiO3wcoEGkAaqctIbZcNB6KlUQBIqt8VYkNSj2397EfA==",
|
||||||
"cpu": [
|
"cpu": [
|
||||||
"x64"
|
"x64"
|
||||||
],
|
],
|
||||||
@@ -1737,41 +1779,33 @@
|
|||||||
"license": "MIT"
|
"license": "MIT"
|
||||||
},
|
},
|
||||||
"node_modules/@types/node": {
|
"node_modules/@types/node": {
|
||||||
"version": "25.3.0",
|
"version": "25.3.3",
|
||||||
"resolved": "https://registry.npmjs.org/@types/node/-/node-25.3.0.tgz",
|
"resolved": "https://registry.npmjs.org/@types/node/-/node-25.3.3.tgz",
|
||||||
"integrity": "sha512-4K3bqJpXpqfg2XKGK9bpDTc6xO/xoUP/RBWS7AtRMug6zZFaRekiLzjVtAoZMquxoAbzBvy5nxQ7veS5eYzf8A==",
|
"integrity": "sha512-DpzbrH7wIcBaJibpKo9nnSQL0MTRdnWttGyE5haGwK86xgMOkFLp7vEyfQPGLOJh5wNYiJ3V9PmUMDhV9u8kkQ==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"undici-types": "~7.18.0"
|
"undici-types": "~7.18.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@types/prop-types": {
|
|
||||||
"version": "15.7.15",
|
|
||||||
"resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.15.tgz",
|
|
||||||
"integrity": "sha512-F6bEyamV9jKGAFBEmlQnesRPGOQqS2+Uwi0Em15xenOxHaf2hv6L8YCVn3rPdPJOiJfPiCnLIRyvwVaqMY3MIw==",
|
|
||||||
"dev": true,
|
|
||||||
"license": "MIT"
|
|
||||||
},
|
|
||||||
"node_modules/@types/react": {
|
"node_modules/@types/react": {
|
||||||
"version": "18.3.27",
|
"version": "19.2.14",
|
||||||
"resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.27.tgz",
|
"resolved": "https://registry.npmjs.org/@types/react/-/react-19.2.14.tgz",
|
||||||
"integrity": "sha512-cisd7gxkzjBKU2GgdYrTdtQx1SORymWyaAFhaxQPK9bYO9ot3Y5OikQRvY0VYQtvwjeQnizCINJAenh/V7MK2w==",
|
"integrity": "sha512-ilcTH/UniCkMdtexkoCN0bI7pMcJDvmQFPvuPvmEaYA/NSfFTAgdUSLAoVjaRJm7+6PvcM+q1zYOwS4wTYMF9w==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@types/prop-types": "*",
|
|
||||||
"csstype": "^3.2.2"
|
"csstype": "^3.2.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@types/react-dom": {
|
"node_modules/@types/react-dom": {
|
||||||
"version": "18.3.7",
|
"version": "19.2.3",
|
||||||
"resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.3.7.tgz",
|
"resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-19.2.3.tgz",
|
||||||
"integrity": "sha512-MEe3UeoENYVFXzoXEWsvcpg6ZvlrFNlOQ7EOsvhI3CfAXwzPfO8Qwuxd40nepsYKqyyVQnTdEfv68q91yLcKrQ==",
|
"integrity": "sha512-jp2L/eY6fn+KgVVQAOqYItbF0VY/YApe5Mz2F0aykSO8gx31bYCZyvSeYxCHKvzHG5eZjc+zyaS5BrBWya2+kQ==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"@types/react": "^18.0.0"
|
"@types/react": "^19.2.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/@types/react-router": {
|
"node_modules/@types/react-router": {
|
||||||
@@ -2925,28 +2959,24 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/react": {
|
"node_modules/react": {
|
||||||
"version": "18.3.1",
|
"version": "19.2.4",
|
||||||
"resolved": "https://registry.npmjs.org/react/-/react-18.3.1.tgz",
|
"resolved": "https://registry.npmjs.org/react/-/react-19.2.4.tgz",
|
||||||
"integrity": "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==",
|
"integrity": "sha512-9nfp2hYpCwOjAN+8TZFGhtWEwgvWHXqESH8qT89AT/lWklpLON22Lc8pEtnpsZz7VmawabSU0gCjnj8aC0euHQ==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
|
||||||
"loose-envify": "^1.1.0"
|
|
||||||
},
|
|
||||||
"engines": {
|
"engines": {
|
||||||
"node": ">=0.10.0"
|
"node": ">=0.10.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/react-dom": {
|
"node_modules/react-dom": {
|
||||||
"version": "18.3.1",
|
"version": "19.2.4",
|
||||||
"resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.3.1.tgz",
|
"resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.4.tgz",
|
||||||
"integrity": "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==",
|
"integrity": "sha512-AXJdLo8kgMbimY95O2aKQqsz2iWi9jMgKJhRBAxECE4IFxfcazB2LmzloIoibJI3C12IlY20+KFaLv+71bUJeQ==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"loose-envify": "^1.1.0",
|
"scheduler": "^0.27.0"
|
||||||
"scheduler": "^0.23.2"
|
|
||||||
},
|
},
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"react": "^18.3.1"
|
"react": "^19.2.4"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/react-i18next": {
|
"node_modules/react-i18next": {
|
||||||
@@ -3056,9 +3086,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/rollup": {
|
"node_modules/rollup": {
|
||||||
"version": "4.53.5",
|
"version": "4.59.0",
|
||||||
"resolved": "https://registry.npmjs.org/rollup/-/rollup-4.53.5.tgz",
|
"resolved": "https://registry.npmjs.org/rollup/-/rollup-4.59.0.tgz",
|
||||||
"integrity": "sha512-iTNAbFSlRpcHeeWu73ywU/8KuU/LZmNCSxp6fjQkJBD3ivUb8tpDrXhIxEzA05HlYMEwmtaUnb3RP+YNv162OQ==",
|
"integrity": "sha512-2oMpl67a3zCH9H79LeMcbDhXW/UmWG/y2zuqnF2jQq5uq9TbM9TVyXvA4+t+ne2IIkBdrLpAaRQAvo7YI/Yyeg==",
|
||||||
"dev": true,
|
"dev": true,
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
@@ -3072,28 +3102,31 @@
|
|||||||
"npm": ">=8.0.0"
|
"npm": ">=8.0.0"
|
||||||
},
|
},
|
||||||
"optionalDependencies": {
|
"optionalDependencies": {
|
||||||
"@rollup/rollup-android-arm-eabi": "4.53.5",
|
"@rollup/rollup-android-arm-eabi": "4.59.0",
|
||||||
"@rollup/rollup-android-arm64": "4.53.5",
|
"@rollup/rollup-android-arm64": "4.59.0",
|
||||||
"@rollup/rollup-darwin-arm64": "4.53.5",
|
"@rollup/rollup-darwin-arm64": "4.59.0",
|
||||||
"@rollup/rollup-darwin-x64": "4.53.5",
|
"@rollup/rollup-darwin-x64": "4.59.0",
|
||||||
"@rollup/rollup-freebsd-arm64": "4.53.5",
|
"@rollup/rollup-freebsd-arm64": "4.59.0",
|
||||||
"@rollup/rollup-freebsd-x64": "4.53.5",
|
"@rollup/rollup-freebsd-x64": "4.59.0",
|
||||||
"@rollup/rollup-linux-arm-gnueabihf": "4.53.5",
|
"@rollup/rollup-linux-arm-gnueabihf": "4.59.0",
|
||||||
"@rollup/rollup-linux-arm-musleabihf": "4.53.5",
|
"@rollup/rollup-linux-arm-musleabihf": "4.59.0",
|
||||||
"@rollup/rollup-linux-arm64-gnu": "4.53.5",
|
"@rollup/rollup-linux-arm64-gnu": "4.59.0",
|
||||||
"@rollup/rollup-linux-arm64-musl": "4.53.5",
|
"@rollup/rollup-linux-arm64-musl": "4.59.0",
|
||||||
"@rollup/rollup-linux-loong64-gnu": "4.53.5",
|
"@rollup/rollup-linux-loong64-gnu": "4.59.0",
|
||||||
"@rollup/rollup-linux-ppc64-gnu": "4.53.5",
|
"@rollup/rollup-linux-loong64-musl": "4.59.0",
|
||||||
"@rollup/rollup-linux-riscv64-gnu": "4.53.5",
|
"@rollup/rollup-linux-ppc64-gnu": "4.59.0",
|
||||||
"@rollup/rollup-linux-riscv64-musl": "4.53.5",
|
"@rollup/rollup-linux-ppc64-musl": "4.59.0",
|
||||||
"@rollup/rollup-linux-s390x-gnu": "4.53.5",
|
"@rollup/rollup-linux-riscv64-gnu": "4.59.0",
|
||||||
"@rollup/rollup-linux-x64-gnu": "4.53.5",
|
"@rollup/rollup-linux-riscv64-musl": "4.59.0",
|
||||||
"@rollup/rollup-linux-x64-musl": "4.53.5",
|
"@rollup/rollup-linux-s390x-gnu": "4.59.0",
|
||||||
"@rollup/rollup-openharmony-arm64": "4.53.5",
|
"@rollup/rollup-linux-x64-gnu": "4.59.0",
|
||||||
"@rollup/rollup-win32-arm64-msvc": "4.53.5",
|
"@rollup/rollup-linux-x64-musl": "4.59.0",
|
||||||
"@rollup/rollup-win32-ia32-msvc": "4.53.5",
|
"@rollup/rollup-openbsd-x64": "4.59.0",
|
||||||
"@rollup/rollup-win32-x64-gnu": "4.53.5",
|
"@rollup/rollup-openharmony-arm64": "4.59.0",
|
||||||
"@rollup/rollup-win32-x64-msvc": "4.53.5",
|
"@rollup/rollup-win32-arm64-msvc": "4.59.0",
|
||||||
|
"@rollup/rollup-win32-ia32-msvc": "4.59.0",
|
||||||
|
"@rollup/rollup-win32-x64-gnu": "4.59.0",
|
||||||
|
"@rollup/rollup-win32-x64-msvc": "4.59.0",
|
||||||
"fsevents": "~2.3.2"
|
"fsevents": "~2.3.2"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@@ -3111,9 +3144,9 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"node_modules/scheduler": {
|
"node_modules/scheduler": {
|
||||||
"version": "0.23.2",
|
"version": "0.27.0",
|
||||||
"resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.2.tgz",
|
"resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.27.0.tgz",
|
||||||
"integrity": "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==",
|
"integrity": "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q==",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"loose-envify": "^1.1.0"
|
"loose-envify": "^1.1.0"
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "medassist-ng-frontend",
|
"name": "medassist-ng-frontend",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "1.16.0",
|
"version": "1.19.0",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
@@ -16,6 +16,8 @@
|
|||||||
"test:coverage": "vitest run --coverage",
|
"test:coverage": "vitest run --coverage",
|
||||||
"test:e2e": "rm -rf test-results && playwright test --config=playwright.stable.config.ts",
|
"test:e2e": "rm -rf test-results && playwright test --config=playwright.stable.config.ts",
|
||||||
"test:e2e:all": "rm -rf test-results && playwright test --config=playwright.all.config.ts",
|
"test:e2e:all": "rm -rf test-results && playwright test --config=playwright.all.config.ts",
|
||||||
|
"test:e2e:local": "PLAYWRIGHT_HTML_OPEN=never PLAYWRIGHT_WORKERS=4 npm run test:e2e",
|
||||||
|
"test:e2e:all:local": "PLAYWRIGHT_HTML_OPEN=never PLAYWRIGHT_WORKERS=4 npm run test:e2e:all",
|
||||||
"test:e2e:with-video": "npm run test:e2e && npm run test:e2e:video",
|
"test:e2e:with-video": "npm run test:e2e && npm run test:e2e:video",
|
||||||
"test:e2e:all:with-video": "npm run test:e2e:all && npm run test:e2e:video",
|
"test:e2e:all:with-video": "npm run test:e2e:all && npm run test:e2e:video",
|
||||||
"test:e2e:video": "find \"$PWD/test-results\" -name video.webm -not -path '*retry*' -print0 | xargs -0 ls -tr > /tmp/e2e-videos.list && if [ -s /tmp/e2e-videos.list ]; then sed \"s/^/file '/\" /tmp/e2e-videos.list | sed \"s/$/'/\" > /tmp/e2e-videos.txt && ffmpeg -y -f concat -safe 0 -i /tmp/e2e-videos.txt -c copy test-results/all-tests.webm; else echo 'No videos found to merge'; fi",
|
"test:e2e:video": "find \"$PWD/test-results\" -name video.webm -not -path '*retry*' -print0 | xargs -0 ls -tr > /tmp/e2e-videos.list && if [ -s /tmp/e2e-videos.list ]; then sed \"s/^/file '/\" /tmp/e2e-videos.list | sed \"s/$/'/\" > /tmp/e2e-videos.txt && ffmpeg -y -f concat -safe 0 -i /tmp/e2e-videos.txt -c copy test-results/all-tests.webm; else echo 'No videos found to merge'; fi",
|
||||||
@@ -28,8 +30,8 @@
|
|||||||
"i18next": "^25.8.13",
|
"i18next": "^25.8.13",
|
||||||
"i18next-browser-languagedetector": "^8.2.1",
|
"i18next-browser-languagedetector": "^8.2.1",
|
||||||
"lucide-react": "^0.575.0",
|
"lucide-react": "^0.575.0",
|
||||||
"react": "^18.3.1",
|
"react": "^19.2.4",
|
||||||
"react-dom": "^18.3.1",
|
"react-dom": "^19.2.4",
|
||||||
"react-i18next": "^15.4.1",
|
"react-i18next": "^15.4.1",
|
||||||
"react-router-dom": "^7.13.1",
|
"react-router-dom": "^7.13.1",
|
||||||
"zod": "^4.3.6"
|
"zod": "^4.3.6"
|
||||||
@@ -40,9 +42,9 @@
|
|||||||
"@testing-library/jest-dom": "^6.9.1",
|
"@testing-library/jest-dom": "^6.9.1",
|
||||||
"@testing-library/react": "^16.3.2",
|
"@testing-library/react": "^16.3.2",
|
||||||
"@testing-library/user-event": "^14.6.1",
|
"@testing-library/user-event": "^14.6.1",
|
||||||
"@types/node": "^25.3.0",
|
"@types/node": "^25.3.3",
|
||||||
"@types/react": "^18.3.4",
|
"@types/react": "^19.2.14",
|
||||||
"@types/react-dom": "^18.3.0",
|
"@types/react-dom": "^19.2.3",
|
||||||
"@types/react-router-dom": "^5.3.3",
|
"@types/react-router-dom": "^5.3.3",
|
||||||
"@vitejs/plugin-react": "^5.1.4",
|
"@vitejs/plugin-react": "^5.1.4",
|
||||||
"@vitest/coverage-v8": "^4.0.18",
|
"@vitest/coverage-v8": "^4.0.18",
|
||||||
|
|||||||
@@ -6,6 +6,10 @@ export function buildPlaywrightConfig(runAllBrowsers: boolean) {
|
|||||||
? ((globalThis as { process?: { env?: Record<string, string | undefined> } }).process?.env ?? {})
|
? ((globalThis as { process?: { env?: Record<string, string | undefined> } }).process?.env ?? {})
|
||||||
: {};
|
: {};
|
||||||
const baseURL = env.PLAYWRIGHT_BASE_URL || "http://localhost:5173";
|
const baseURL = env.PLAYWRIGHT_BASE_URL || "http://localhost:5173";
|
||||||
|
const parsedWorkers = Number.parseInt(env.PLAYWRIGHT_WORKERS ?? "", 10);
|
||||||
|
// Default to single-worker execution to keep API-seeded E2E suites deterministic.
|
||||||
|
// Still allow explicit local overrides via PLAYWRIGHT_WORKERS.
|
||||||
|
const workers = Number.isFinite(parsedWorkers) && parsedWorkers > 0 ? parsedWorkers : 1;
|
||||||
|
|
||||||
const projects: NonNullable<PlaywrightTestConfig["projects"]> = [
|
const projects: NonNullable<PlaywrightTestConfig["projects"]> = [
|
||||||
{
|
{
|
||||||
@@ -17,13 +21,13 @@ export function buildPlaywrightConfig(runAllBrowsers: boolean) {
|
|||||||
use: {
|
use: {
|
||||||
...devices["Desktop Chrome"],
|
...devices["Desktop Chrome"],
|
||||||
},
|
},
|
||||||
testIgnore: /.*-(?:data|crud|edit|status|schedule)\.spec\.ts/,
|
testIgnore: /.*-(?:data|crud|edit|status|schedule|lifecycle)\.spec\.ts|performance\.spec\.ts/,
|
||||||
dependencies: ["setup"],
|
dependencies: ["setup"],
|
||||||
retries: 1,
|
retries: 1,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
name: "chromium-data",
|
name: "chromium-data",
|
||||||
testMatch: /.*-(?:data|crud|edit|status|schedule)\.spec\.ts/,
|
testMatch: /.*-(?:data|crud|edit|status|schedule|lifecycle)\.spec\.ts|performance\.spec\.ts/,
|
||||||
use: {
|
use: {
|
||||||
...devices["Desktop Chrome"],
|
...devices["Desktop Chrome"],
|
||||||
},
|
},
|
||||||
@@ -40,7 +44,7 @@ export function buildPlaywrightConfig(runAllBrowsers: boolean) {
|
|||||||
use: {
|
use: {
|
||||||
...devices["Desktop Firefox"],
|
...devices["Desktop Firefox"],
|
||||||
},
|
},
|
||||||
testIgnore: /.*-(?:data|crud|edit|status|schedule)\.spec\.ts/,
|
testIgnore: /.*-(?:data|crud|edit|status|schedule|lifecycle)\.spec\.ts|performance\.spec\.ts/,
|
||||||
dependencies: ["setup"],
|
dependencies: ["setup"],
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -48,7 +52,7 @@ export function buildPlaywrightConfig(runAllBrowsers: boolean) {
|
|||||||
use: {
|
use: {
|
||||||
...devices["Desktop Safari"],
|
...devices["Desktop Safari"],
|
||||||
},
|
},
|
||||||
testIgnore: /.*-(?:data|crud|edit|status|schedule)\.spec\.ts/,
|
testIgnore: /.*-(?:data|crud|edit|status|schedule|lifecycle)\.spec\.ts|performance\.spec\.ts/,
|
||||||
dependencies: ["setup"],
|
dependencies: ["setup"],
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
@@ -64,7 +68,7 @@ export function buildPlaywrightConfig(runAllBrowsers: boolean) {
|
|||||||
fullyParallel: true,
|
fullyParallel: true,
|
||||||
forbidOnly: !!env.CI,
|
forbidOnly: !!env.CI,
|
||||||
retries: env.CI ? 2 : 0,
|
retries: env.CI ? 2 : 0,
|
||||||
workers: 1,
|
workers,
|
||||||
reporter: env.CI
|
reporter: env.CI
|
||||||
? [["html", { outputFolder: "playwright-report" }], ["github"]]
|
? [["html", { outputFolder: "playwright-report" }], ["github"]]
|
||||||
: [["html", { outputFolder: "playwright-report" }], ["list"]],
|
: [["html", { outputFolder: "playwright-report" }], ["list"]],
|
||||||
|
|||||||
+20
-3
@@ -37,13 +37,29 @@ export default function App() {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getInitialAuthTheme(): "light" | "dark" {
|
||||||
|
if (typeof window === "undefined") return "dark";
|
||||||
|
|
||||||
|
const stored = localStorage.getItem("theme");
|
||||||
|
if (stored === "light" || stored === "dark") {
|
||||||
|
return stored;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (stored === "system") {
|
||||||
|
return window.matchMedia?.("(prefers-color-scheme: light)").matches ? "light" : "dark";
|
||||||
|
}
|
||||||
|
|
||||||
|
return window.matchMedia?.("(prefers-color-scheme: light)").matches ? "light" : "dark";
|
||||||
|
}
|
||||||
|
|
||||||
function AppRouter() {
|
function AppRouter() {
|
||||||
const { user, authState, loading, authError } = useAuth();
|
const { user, authState, loading, authError } = useAuth();
|
||||||
|
const authTheme = getInitialAuthTheme();
|
||||||
|
|
||||||
// Show loading while checking auth state
|
// Show loading while checking auth state
|
||||||
if (loading) {
|
if (loading) {
|
||||||
return (
|
return (
|
||||||
<div className="auth-container">
|
<div className="auth-container" data-theme={authTheme}>
|
||||||
<div className="auth-card" style={{ textAlign: "center" }}>
|
<div className="auth-card" style={{ textAlign: "center" }}>
|
||||||
<h1 className="auth-title">💊 MedAssist-ng</h1>
|
<h1 className="auth-title">💊 MedAssist-ng</h1>
|
||||||
<p>Loading...</p>
|
<p>Loading...</p>
|
||||||
@@ -55,7 +71,7 @@ function AppRouter() {
|
|||||||
// Show error if we couldn't connect to the server
|
// Show error if we couldn't connect to the server
|
||||||
if (authError) {
|
if (authError) {
|
||||||
return (
|
return (
|
||||||
<div className="auth-container">
|
<div className="auth-container" data-theme={authTheme}>
|
||||||
<div className="auth-card" style={{ textAlign: "center" }}>
|
<div className="auth-card" style={{ textAlign: "center" }}>
|
||||||
<h1 className="auth-title">💊 MedAssist-ng</h1>
|
<h1 className="auth-title">💊 MedAssist-ng</h1>
|
||||||
<div className="auth-error" style={{ marginBottom: "1rem" }}>
|
<div className="auth-error" style={{ marginBottom: "1rem" }}>
|
||||||
@@ -77,7 +93,7 @@ function AppRouter() {
|
|||||||
// If auth state is null (shouldn't happen after loading, but be safe)
|
// If auth state is null (shouldn't happen after loading, but be safe)
|
||||||
if (!authState) {
|
if (!authState) {
|
||||||
return (
|
return (
|
||||||
<div className="auth-container">
|
<div className="auth-container" data-theme={authTheme}>
|
||||||
<div className="auth-card" style={{ textAlign: "center" }}>
|
<div className="auth-card" style={{ textAlign: "center" }}>
|
||||||
<h1 className="auth-title">💊 MedAssist-ng</h1>
|
<h1 className="auth-title">💊 MedAssist-ng</h1>
|
||||||
<p>Initializing...</p>
|
<p>Initializing...</p>
|
||||||
@@ -301,6 +317,7 @@ function AppContent() {
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const handleEscape = (e: KeyboardEvent) => {
|
const handleEscape = (e: KeyboardEvent) => {
|
||||||
if (e.key !== "Escape") return;
|
if (e.key !== "Escape") return;
|
||||||
|
if (e.defaultPrevented) return;
|
||||||
|
|
||||||
if (scheduleLightboxImage) {
|
if (scheduleLightboxImage) {
|
||||||
closeScheduleLightbox();
|
closeScheduleLightbox();
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { FRONTEND_VERSION, GITHUB_URL } from "../App";
|
import { FRONTEND_VERSION, GITHUB_URL } from "../App";
|
||||||
import { useEscapeKey } from "../hooks/useEscapeKey";
|
|
||||||
|
|
||||||
interface UpdateCheckResult {
|
interface UpdateCheckResult {
|
||||||
status: "up-to-date" | "update-available" | "error";
|
status: "up-to-date" | "update-available" | "error";
|
||||||
@@ -18,7 +17,7 @@ export default function AboutModal({ isOpen, onClose }: AboutModalProps) {
|
|||||||
const [isChecking, setIsChecking] = useState(false);
|
const [isChecking, setIsChecking] = useState(false);
|
||||||
const [updateCheckResult, setUpdateCheckResult] = useState<UpdateCheckResult | null>(null);
|
const [updateCheckResult, setUpdateCheckResult] = useState<UpdateCheckResult | null>(null);
|
||||||
|
|
||||||
useEscapeKey(isOpen, onClose);
|
// ESC is handled by the global handler in App.tsx to avoid double history.back()
|
||||||
|
|
||||||
// Reset check result when modal opens so stale results are never shown
|
// Reset check result when modal opens so stale results are never shown
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
|||||||
@@ -20,7 +20,7 @@ export interface User {
|
|||||||
export interface AuthState {
|
export interface AuthState {
|
||||||
authEnabled: boolean;
|
authEnabled: boolean;
|
||||||
registrationEnabled: boolean;
|
registrationEnabled: boolean;
|
||||||
localAuthEnabled: boolean;
|
formLoginEnabled: boolean;
|
||||||
oidcEnabled: boolean;
|
oidcEnabled: boolean;
|
||||||
oidcProviderName: string;
|
oidcProviderName: string;
|
||||||
hasUsers: boolean;
|
hasUsers: boolean;
|
||||||
@@ -157,7 +157,7 @@ export function AuthProvider({ children }: { children: ReactNode }) {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
log.warn("[Auth] Session refresh failed, clearing local user state", { correlationId });
|
log.debug("[Auth] Session refresh unavailable, clearing local user state", { correlationId });
|
||||||
setUser(null);
|
setUser(null);
|
||||||
} else {
|
} else {
|
||||||
log.warn("[Auth] Unexpected /auth/me response", { status: res.status, correlationId });
|
log.warn("[Auth] Unexpected /auth/me response", { status: res.status, correlationId });
|
||||||
@@ -181,7 +181,11 @@ export function AuthProvider({ children }: { children: ReactNode }) {
|
|||||||
);
|
);
|
||||||
const res = await fetch("/api/auth/refresh", init);
|
const res = await fetch("/api/auth/refresh", init);
|
||||||
if (!res.ok) {
|
if (!res.ok) {
|
||||||
log.warn("[Auth] Token refresh rejected", { status: res.status, correlationId });
|
if (res.status === 401) {
|
||||||
|
log.debug("[Auth] Token refresh rejected (unauthenticated)", { status: res.status, correlationId });
|
||||||
|
} else {
|
||||||
|
log.warn("[Auth] Token refresh rejected", { status: res.status, correlationId });
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return res.ok;
|
return res.ok;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@@ -425,7 +429,7 @@ export function LoginForm({
|
|||||||
</svg>
|
</svg>
|
||||||
{t("auth.loginWithSSO", "Login with {{provider}}", { provider: authState.oidcProviderName || "SSO" })}
|
{t("auth.loginWithSSO", "Login with {{provider}}", { provider: authState.oidcProviderName || "SSO" })}
|
||||||
</button>
|
</button>
|
||||||
{authState?.localAuthEnabled && (
|
{authState?.formLoginEnabled && (
|
||||||
<div className="auth-divider">
|
<div className="auth-divider">
|
||||||
<span>{t("auth.or", "or")}</span>
|
<span>{t("auth.or", "or")}</span>
|
||||||
</div>
|
</div>
|
||||||
@@ -433,8 +437,8 @@ export function LoginForm({
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{/* Local Login Form - only show if local auth is enabled */}
|
{/* Local login form - only show if form login is enabled */}
|
||||||
{authState?.localAuthEnabled && (
|
{authState?.formLoginEnabled && (
|
||||||
<form onSubmit={handleSubmit} className="auth-form">
|
<form onSubmit={handleSubmit} className="auth-form">
|
||||||
{error && <div className="auth-error">{error}</div>}
|
{error && <div className="auth-error">{error}</div>}
|
||||||
|
|
||||||
@@ -474,7 +478,7 @@ export function LoginForm({
|
|||||||
</form>
|
</form>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{authState?.registrationEnabled && authState?.localAuthEnabled && onSwitchToRegister && (
|
{authState?.registrationEnabled && authState?.formLoginEnabled && onSwitchToRegister && (
|
||||||
<div className="auth-links">
|
<div className="auth-links">
|
||||||
<button type="button" className="auth-link-btn" onClick={onSwitchToRegister}>
|
<button type="button" className="auth-link-btn" onClick={onSwitchToRegister}>
|
||||||
{t("auth.createAccount", "Create account")}
|
{t("auth.createAccount", "Create account")}
|
||||||
@@ -540,7 +544,7 @@ export function RegisterForm({ onSuccess, onSwitchToLogin }: { onSuccess?: () =>
|
|||||||
</svg>
|
</svg>
|
||||||
{t("auth.loginWithSSO", "Login with {{provider}}", { provider: authState.oidcProviderName || "SSO" })}
|
{t("auth.loginWithSSO", "Login with {{provider}}", { provider: authState.oidcProviderName || "SSO" })}
|
||||||
</button>
|
</button>
|
||||||
{authState?.localAuthEnabled && (
|
{authState?.formLoginEnabled && (
|
||||||
<div className="auth-divider">
|
<div className="auth-divider">
|
||||||
<span>{t("auth.or", "or")}</span>
|
<span>{t("auth.or", "or")}</span>
|
||||||
</div>
|
</div>
|
||||||
@@ -549,7 +553,7 @@ export function RegisterForm({ onSuccess, onSwitchToLogin }: { onSuccess?: () =>
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
{/* Local Registration Form - only show if local auth is enabled */}
|
{/* Local Registration Form - only show if local auth is enabled */}
|
||||||
{authState?.localAuthEnabled && (
|
{authState?.formLoginEnabled && (
|
||||||
<form onSubmit={handleSubmit} className="auth-form">
|
<form onSubmit={handleSubmit} className="auth-form">
|
||||||
{error && <div className="auth-error">{error}</div>}
|
{error && <div className="auth-error">{error}</div>}
|
||||||
|
|
||||||
|
|||||||
@@ -15,7 +15,14 @@ import { useTranslation } from "react-i18next";
|
|||||||
import { Lightbox, MedicationAvatar } from "../components";
|
import { Lightbox, MedicationAvatar } from "../components";
|
||||||
import { useEscapeKey } from "../hooks";
|
import { useEscapeKey } from "../hooks";
|
||||||
import type { Coverage, Medication, RefillEntry, StockThresholds } from "../types";
|
import type { Coverage, Medication, RefillEntry, StockThresholds } from "../types";
|
||||||
import { getMedTotal, getPackageSize } from "../types";
|
import {
|
||||||
|
getMedDisplayName,
|
||||||
|
getMedTotal,
|
||||||
|
getPackageSize,
|
||||||
|
isAmountBasedPackageType,
|
||||||
|
isLiquidContainerPackageType,
|
||||||
|
isTubePackageType,
|
||||||
|
} from "../types";
|
||||||
import { formatNumber, generateICS, getExpiryClass, getSystemLocale } from "../utils";
|
import { formatNumber, generateICS, getExpiryClass, getSystemLocale } from "../utils";
|
||||||
import { getStockStatus } from "../utils/schedule";
|
import { getStockStatus } from "../utils/schedule";
|
||||||
import { splitCurrentBlisterStock } from "../utils/stock";
|
import { splitCurrentBlisterStock } from "../utils/stock";
|
||||||
@@ -159,8 +166,8 @@ export function MedDetailModal({
|
|||||||
// Escape key: only one handler is active at a time (sub-modal states are mutually exclusive).
|
// Escape key: only one handler is active at a time (sub-modal states are mutually exclusive).
|
||||||
// Lightbox has its own useEscapeKey internally.
|
// Lightbox has its own useEscapeKey internally.
|
||||||
useEscapeKey(!showEditStockModal && !showImageLightbox && !showRefillModal, onClose);
|
useEscapeKey(!showEditStockModal && !showImageLightbox && !showRefillModal, onClose);
|
||||||
useEscapeKey(showEditStockModal, onCloseEditStockModal);
|
useEscapeKey(showEditStockModal, onCloseEditStockModal, { capture: true });
|
||||||
useEscapeKey(showRefillModal, onCloseRefillModal);
|
useEscapeKey(showRefillModal, onCloseRefillModal, { capture: true });
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (showEditStockModal) return;
|
if (showEditStockModal) return;
|
||||||
@@ -170,7 +177,7 @@ export function MedDetailModal({
|
|||||||
}, [showEditStockModal]);
|
}, [showEditStockModal]);
|
||||||
|
|
||||||
const remainingPrescriptionRefills = Math.max(0, Number(selectedMed?.prescriptionRemainingRefills) || 0);
|
const remainingPrescriptionRefills = Math.max(0, Number(selectedMed?.prescriptionRemainingRefills) || 0);
|
||||||
const prescriptionPackCapEnabled = selectedMed?.packageType === "blister" && usePrescriptionRefill;
|
const prescriptionPackCapEnabled = !isAmountBasedPackageType(selectedMed?.packageType) && usePrescriptionRefill;
|
||||||
const cappedRefillPacks = prescriptionPackCapEnabled
|
const cappedRefillPacks = prescriptionPackCapEnabled
|
||||||
? Math.min(refillPacks, remainingPrescriptionRefills)
|
? Math.min(refillPacks, remainingPrescriptionRefills)
|
||||||
: refillPacks;
|
: refillPacks;
|
||||||
@@ -179,7 +186,7 @@ export function MedDetailModal({
|
|||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!selectedMed) return;
|
if (!selectedMed) return;
|
||||||
if (!showRefillModal) return;
|
if (!showRefillModal) return;
|
||||||
if (selectedMed.packageType !== "blister" || !usePrescriptionRefill) return;
|
if (isAmountBasedPackageType(selectedMed.packageType) || !usePrescriptionRefill) return;
|
||||||
if (refillPacks <= remainingPrescriptionRefills) return;
|
if (refillPacks <= remainingPrescriptionRefills) return;
|
||||||
onRefillPacksChange(remainingPrescriptionRefills);
|
onRefillPacksChange(remainingPrescriptionRefills);
|
||||||
}, [
|
}, [
|
||||||
@@ -192,14 +199,20 @@ export function MedDetailModal({
|
|||||||
]);
|
]);
|
||||||
|
|
||||||
if (!selectedMed) return null;
|
if (!selectedMed) return null;
|
||||||
|
const isAmountPackage =
|
||||||
|
isTubePackageType(selectedMed.packageType) || isLiquidContainerPackageType(selectedMed.packageType);
|
||||||
|
const amountUnitLabel =
|
||||||
|
isLiquidContainerPackageType(selectedMed.packageType) || selectedMed.medicationForm === "liquid"
|
||||||
|
? t("form.packageAmountUnitMl")
|
||||||
|
: t("form.packageAmountUnitG");
|
||||||
|
const stockUnitLabel = isAmountPackage ? amountUnitLabel : null;
|
||||||
|
|
||||||
const medCoverage = coverage.all.find((c) => c.name === selectedMed.name);
|
const medCoverage = coverage.all.find((c) => c.name === getMedDisplayName(selectedMed));
|
||||||
const packageSize = getPackageSize(selectedMed);
|
const packageSize = getPackageSize(selectedMed);
|
||||||
// Structural max = sealed package capacity only (excludes pre-existing looseTablets).
|
// Structural max = sealed package capacity only (excludes pre-existing looseTablets).
|
||||||
const structuralMax =
|
const structuralMax = isAmountBasedPackageType(selectedMed.packageType)
|
||||||
selectedMed.packageType === "bottle"
|
? (selectedMed.totalPills ?? packageSize)
|
||||||
? (selectedMed.totalPills ?? packageSize)
|
: selectedMed.packCount * selectedMed.blistersPerPack * selectedMed.pillsPerBlister;
|
||||||
: selectedMed.packCount * selectedMed.blistersPerPack * selectedMed.pillsPerBlister;
|
|
||||||
const currentStock = medCoverage ? Math.round(medCoverage.medsLeft) : getMedTotal(selectedMed);
|
const currentStock = medCoverage ? Math.round(medCoverage.medsLeft) : getMedTotal(selectedMed);
|
||||||
const status = medCoverage ? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, settings) : null;
|
const status = medCoverage ? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, settings) : null;
|
||||||
const fallbackTextClass = status?.className === "warning" ? "warning-text" : "success-text";
|
const fallbackTextClass = status?.className === "warning" ? "warning-text" : "success-text";
|
||||||
@@ -208,8 +221,21 @@ export function MedDetailModal({
|
|||||||
const currentFullBlisters = Math.max(0, stock.fullBlisters);
|
const currentFullBlisters = Math.max(0, stock.fullBlisters);
|
||||||
const currentPartialPills = Math.max(0, stock.openBlisterPills);
|
const currentPartialPills = Math.max(0, stock.openBlisterPills);
|
||||||
const currentLoosePills = Math.max(0, stock.loosePills);
|
const currentLoosePills = Math.max(0, stock.loosePills);
|
||||||
const stockDisplayTotal =
|
const stockDisplayTotal = isAmountBasedPackageType(selectedMed.packageType)
|
||||||
selectedMed.packageType === "bottle" ? (selectedMed.totalPills ?? packageSize) : Math.max(0, structuralMax);
|
? (selectedMed.totalPills ?? packageSize)
|
||||||
|
: Math.max(0, structuralMax);
|
||||||
|
const packageCount = Math.max(1, Number(selectedMed.packCount) || 1);
|
||||||
|
const amountPerPackage = (() => {
|
||||||
|
const configured = Number(selectedMed.packageAmountValue ?? 0);
|
||||||
|
if (Number.isFinite(configured) && configured > 0) return configured;
|
||||||
|
|
||||||
|
const totalAmount = Number(stockDisplayTotal ?? 0);
|
||||||
|
if (Number.isFinite(totalAmount) && totalAmount > 0) {
|
||||||
|
return Math.max(0, totalAmount / packageCount);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
})();
|
||||||
const maxPartialPills = Math.min(
|
const maxPartialPills = Math.min(
|
||||||
Math.max(0, selectedMed.pillsPerBlister),
|
Math.max(0, selectedMed.pillsPerBlister),
|
||||||
Math.max(0, structuralMax - Math.max(0, editStockFullBlisters) * selectedMed.pillsPerBlister)
|
Math.max(0, structuralMax - Math.max(0, editStockFullBlisters) * selectedMed.pillsPerBlister)
|
||||||
@@ -219,6 +245,33 @@ export function MedDetailModal({
|
|||||||
const closeLabel = t("common.close");
|
const closeLabel = t("common.close");
|
||||||
const decrementLabel = t("editStock.decreaseValue");
|
const decrementLabel = t("editStock.decreaseValue");
|
||||||
const incrementLabel = t("editStock.increaseValue");
|
const incrementLabel = t("editStock.increaseValue");
|
||||||
|
const getScheduleUsageLabel = (usage: number, intakeUnit?: "ml" | "tsp" | "tbsp" | null) => {
|
||||||
|
if (isLiquidContainerPackageType(selectedMed.packageType)) {
|
||||||
|
if (intakeUnit === "tsp") {
|
||||||
|
return `${usage} ${t("form.blisters.teaspoons", { count: Math.abs(usage) })}`;
|
||||||
|
}
|
||||||
|
if (intakeUnit === "tbsp") {
|
||||||
|
return `${usage} ${t("form.blisters.tablespoons", { count: Math.abs(usage) })}`;
|
||||||
|
}
|
||||||
|
return `${usage} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
if (isTubePackageType(selectedMed.packageType)) {
|
||||||
|
return `${usage} ${t("form.blisters.applications", { count: Math.abs(usage) })}`;
|
||||||
|
}
|
||||||
|
return `${usage} ${usage !== 1 ? t("common.pills") : t("common.pill")}`;
|
||||||
|
};
|
||||||
|
const scheduleIntakes =
|
||||||
|
selectedMed.intakes && selectedMed.intakes.length > 0
|
||||||
|
? selectedMed.intakes
|
||||||
|
: selectedMed.blisters.map((blister) => ({
|
||||||
|
usage: blister.usage,
|
||||||
|
every: blister.every,
|
||||||
|
start: blister.start,
|
||||||
|
takenBy: null,
|
||||||
|
intakeRemindersEnabled: false,
|
||||||
|
intakeUnit: null,
|
||||||
|
}));
|
||||||
|
const hasAnyIntakeReminder = scheduleIntakes.some((intake) => intake.intakeRemindersEnabled === true);
|
||||||
const normalizeBlisterStock = (nextFull: number, nextPartial: number, nextLoose: number) => {
|
const normalizeBlisterStock = (nextFull: number, nextPartial: number, nextLoose: number) => {
|
||||||
let normalizedFull = Math.max(0, nextFull);
|
let normalizedFull = Math.max(0, nextFull);
|
||||||
let normalizedPartial = Math.max(0, nextPartial);
|
let normalizedPartial = Math.max(0, nextPartial);
|
||||||
@@ -347,6 +400,10 @@ export function MedDetailModal({
|
|||||||
|
|
||||||
const renderEditStockModal = () => {
|
const renderEditStockModal = () => {
|
||||||
if (!showEditStockModal) return null;
|
if (!showEditStockModal) return null;
|
||||||
|
const isLiquidPackage = isLiquidContainerPackageType(selectedMed.packageType);
|
||||||
|
const liquidBottleCount = Math.max(1, editStockFullBlisters);
|
||||||
|
const liquidAmountPerBottle = Math.max(1, Number.isFinite(amountPerPackage) ? amountPerPackage : 1);
|
||||||
|
const liquidCapacity = Math.max(1, Math.round(liquidBottleCount * liquidAmountPerBottle));
|
||||||
const fullInputMax = Math.min(
|
const fullInputMax = Math.min(
|
||||||
maxFullBlisters,
|
maxFullBlisters,
|
||||||
Math.floor(Math.max(0, structuralMax - Math.max(0, editStockPartialBlisterPills)) / selectedMed.pillsPerBlister)
|
Math.floor(Math.max(0, structuralMax - Math.max(0, editStockPartialBlisterPills)) / selectedMed.pillsPerBlister)
|
||||||
@@ -360,14 +417,14 @@ export function MedDetailModal({
|
|||||||
onCloseEditStockModal();
|
onCloseEditStockModal();
|
||||||
}}
|
}}
|
||||||
onKeyDown={(e) => {
|
onKeyDown={(e) => {
|
||||||
if (e.key !== "Escape") e.stopPropagation();
|
e.stopPropagation();
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<div
|
<div
|
||||||
className="modal-content edit-stock-modal"
|
className="modal-content edit-stock-modal"
|
||||||
onClick={(e) => e.stopPropagation()}
|
onClick={(e) => e.stopPropagation()}
|
||||||
onKeyDown={(e) => {
|
onKeyDown={(e) => {
|
||||||
if (e.key !== "Escape") e.stopPropagation();
|
e.stopPropagation();
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<button
|
<button
|
||||||
@@ -380,9 +437,9 @@ export function MedDetailModal({
|
|||||||
<X size={18} aria-hidden="true" />
|
<X size={18} aria-hidden="true" />
|
||||||
</button>
|
</button>
|
||||||
<h2>{t("editStock.title")}</h2>
|
<h2>{t("editStock.title")}</h2>
|
||||||
<p className="edit-stock-med-name">{selectedMed.name}</p>
|
<p className="edit-stock-med-name">{getMedDisplayName(selectedMed)}</p>
|
||||||
<p className="edit-stock-hint">{t("editStock.hint")}</p>
|
<p className="edit-stock-hint">{t("editStock.hint")}</p>
|
||||||
{selectedMed.packageType === "blister" && (
|
{!isAmountBasedPackageType(selectedMed.packageType) && (
|
||||||
<p className="edit-stock-cap-info edit-stock-live-breakdown">
|
<p className="edit-stock-cap-info edit-stock-live-breakdown">
|
||||||
{t("editStock.currentComposition", {
|
{t("editStock.currentComposition", {
|
||||||
fullBlisters: currentFullBlisters,
|
fullBlisters: currentFullBlisters,
|
||||||
@@ -392,9 +449,15 @@ export function MedDetailModal({
|
|||||||
})}
|
})}
|
||||||
</p>
|
</p>
|
||||||
)}
|
)}
|
||||||
{selectedMed.packageType === "bottle" && (
|
{isAmountBasedPackageType(selectedMed.packageType) && !isTubePackageType(selectedMed.packageType) && (
|
||||||
<p className="edit-stock-cap-info">{t("editStock.packageSize", { count: structuralMax })}</p>
|
<p className="edit-stock-cap-info">{t("editStock.packageSize", { count: structuralMax })}</p>
|
||||||
)}
|
)}
|
||||||
|
{(isTubePackageType(selectedMed.packageType) || isLiquidContainerPackageType(selectedMed.packageType)) && (
|
||||||
|
<p className="edit-stock-cap-info">
|
||||||
|
{t("form.totalAmount")}: {formatNumber(isLiquidPackage ? liquidCapacity : structuralMax)}{" "}
|
||||||
|
{amountUnitLabel}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
{showStockCapNotice && (
|
{showStockCapNotice && (
|
||||||
<p className="edit-stock-cap-warning">{t("editStock.maxExceeded", { count: structuralMax })}</p>
|
<p className="edit-stock-cap-warning">{t("editStock.maxExceeded", { count: structuralMax })}</p>
|
||||||
)}
|
)}
|
||||||
@@ -402,12 +465,14 @@ export function MedDetailModal({
|
|||||||
{(() => {
|
{(() => {
|
||||||
const dbTotal = getMedTotal(selectedMed);
|
const dbTotal = getMedTotal(selectedMed);
|
||||||
const currentTotal = medCoverage ? Math.round(medCoverage.medsLeft) : dbTotal;
|
const currentTotal = medCoverage ? Math.round(medCoverage.medsLeft) : dbTotal;
|
||||||
const isBottle = selectedMed.packageType === "bottle";
|
const isBottle = isAmountBasedPackageType(selectedMed.packageType);
|
||||||
const enteredTotal = isBottle
|
const enteredTotal = isLiquidPackage
|
||||||
? editStockPartialBlisterPills
|
? Math.min(liquidCapacity, editStockPartialBlisterPills)
|
||||||
: editStockFullBlisters * selectedMed.pillsPerBlister +
|
: isBottle
|
||||||
editStockPartialBlisterPills +
|
? editStockPartialBlisterPills
|
||||||
editStockLoosePills;
|
: editStockFullBlisters * selectedMed.pillsPerBlister +
|
||||||
|
editStockPartialBlisterPills +
|
||||||
|
editStockLoosePills;
|
||||||
const newTotal = Math.max(0, enteredTotal);
|
const newTotal = Math.max(0, enteredTotal);
|
||||||
const difference = newTotal - currentTotal;
|
const difference = newTotal - currentTotal;
|
||||||
const differenceClass = difference > 0 ? "positive" : difference < 0 ? "negative" : "";
|
const differenceClass = difference > 0 ? "positive" : difference < 0 ? "negative" : "";
|
||||||
@@ -417,36 +482,39 @@ export function MedDetailModal({
|
|||||||
<div className="edit-stock-form">
|
<div className="edit-stock-form">
|
||||||
{isBottle ? (
|
{isBottle ? (
|
||||||
<label>
|
<label>
|
||||||
{t("editStock.totalPills")}
|
{isAmountPackage ? t("form.currentAmount") : t("editStock.totalPills")}
|
||||||
{renderStepperInput({
|
{renderStepperInput({
|
||||||
value: editStockPartialInput,
|
value: editStockPartialInput,
|
||||||
min: 0,
|
min: 0,
|
||||||
max: structuralMax,
|
max: isLiquidPackage ? liquidCapacity : structuralMax,
|
||||||
onChange: (raw) => {
|
onChange: (raw) => {
|
||||||
const parsed = raw === "" ? 0 : Math.max(0, parseStockInput(raw));
|
const parsed = raw === "" ? 0 : Math.max(0, parseStockInput(raw));
|
||||||
setEditStockPartialInput(raw);
|
setEditStockPartialInput(raw);
|
||||||
onEditStockPartialBlisterPillsChange(raw === "" ? 0 : Math.min(structuralMax, parsed));
|
const maxTotal = isLiquidPackage ? liquidCapacity : structuralMax;
|
||||||
setShowStockCapNotice(parsed > structuralMax);
|
onEditStockPartialBlisterPillsChange(raw === "" ? 0 : Math.min(maxTotal, parsed));
|
||||||
|
setShowStockCapNotice(parsed > maxTotal);
|
||||||
},
|
},
|
||||||
onBlur: () => {
|
onBlur: () => {
|
||||||
const normalized = Math.min(
|
const maxTotal = isLiquidPackage ? liquidCapacity : structuralMax;
|
||||||
structuralMax,
|
const normalized = Math.min(maxTotal, Math.max(0, parseStockInput(editStockPartialInput)));
|
||||||
Math.max(0, parseStockInput(editStockPartialInput))
|
|
||||||
);
|
|
||||||
onEditStockPartialBlisterPillsChange(normalized);
|
onEditStockPartialBlisterPillsChange(normalized);
|
||||||
setEditStockPartialInput(String(normalized));
|
setEditStockPartialInput(String(normalized));
|
||||||
setShowStockCapNotice(false);
|
setShowStockCapNotice(false);
|
||||||
},
|
},
|
||||||
onStep: (delta) => {
|
onStep: (delta) => {
|
||||||
const next = Math.min(
|
const maxTotal = isLiquidPackage ? liquidCapacity : structuralMax;
|
||||||
structuralMax,
|
const next = Math.min(maxTotal, Math.max(0, parseStockInput(editStockPartialInput) + delta));
|
||||||
Math.max(0, parseStockInput(editStockPartialInput) + delta)
|
|
||||||
);
|
|
||||||
onEditStockPartialBlisterPillsChange(next);
|
onEditStockPartialBlisterPillsChange(next);
|
||||||
setEditStockPartialInput(String(next));
|
setEditStockPartialInput(String(next));
|
||||||
setShowStockCapNotice(false);
|
setShowStockCapNotice(false);
|
||||||
},
|
},
|
||||||
})}
|
})}
|
||||||
|
{isLiquidPackage && (
|
||||||
|
<p className="edit-stock-cap-info" style={{ marginTop: "0.35rem" }}>
|
||||||
|
{t("form.currentAmount")}: {Math.max(0, editStockPartialBlisterPills)} {amountUnitLabel} /{" "}
|
||||||
|
{liquidCapacity} {amountUnitLabel}
|
||||||
|
</p>
|
||||||
|
)}
|
||||||
</label>
|
</label>
|
||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
@@ -584,26 +652,72 @@ export function MedDetailModal({
|
|||||||
</label>
|
</label>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
{isLiquidPackage && (
|
||||||
|
<label>
|
||||||
|
{t("form.bottles")}
|
||||||
|
{renderStepperInput({
|
||||||
|
value: editStockFullInput,
|
||||||
|
min: 1,
|
||||||
|
max: Number.MAX_SAFE_INTEGER,
|
||||||
|
onChange: (raw) => {
|
||||||
|
const nextBottleCount = raw === "" ? 1 : Math.max(1, parseStockInput(raw));
|
||||||
|
setEditStockFullInput(raw === "" ? "1" : raw);
|
||||||
|
onEditStockFullBlistersChange(nextBottleCount);
|
||||||
|
const syncedTotal = Math.round(nextBottleCount * liquidAmountPerBottle);
|
||||||
|
onEditStockPartialBlisterPillsChange(syncedTotal);
|
||||||
|
setEditStockPartialInput(String(syncedTotal));
|
||||||
|
setShowStockCapNotice(false);
|
||||||
|
},
|
||||||
|
onBlur: () => {
|
||||||
|
const normalized = Math.max(1, parseStockInput(editStockFullInput));
|
||||||
|
onEditStockFullBlistersChange(normalized);
|
||||||
|
setEditStockFullInput(String(normalized));
|
||||||
|
const syncedTotal = Math.round(normalized * liquidAmountPerBottle);
|
||||||
|
onEditStockPartialBlisterPillsChange(syncedTotal);
|
||||||
|
setEditStockPartialInput(String(syncedTotal));
|
||||||
|
setShowStockCapNotice(false);
|
||||||
|
},
|
||||||
|
onStep: (delta) => {
|
||||||
|
const next = Math.max(1, parseStockInput(editStockFullInput) + delta);
|
||||||
|
onEditStockFullBlistersChange(next);
|
||||||
|
setEditStockFullInput(String(next));
|
||||||
|
const syncedTotal = Math.round(next * liquidAmountPerBottle);
|
||||||
|
onEditStockPartialBlisterPillsChange(syncedTotal);
|
||||||
|
setEditStockPartialInput(String(syncedTotal));
|
||||||
|
setShowStockCapNotice(false);
|
||||||
|
},
|
||||||
|
})}
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="edit-stock-summary">
|
<div className="edit-stock-summary">
|
||||||
<div className="summary-row">
|
<div className="summary-row">
|
||||||
<span>{t("editStock.currentTotal")}:</span>
|
<span>{t("editStock.currentTotal")}:</span>
|
||||||
<span>
|
<span>
|
||||||
{currentTotal} {currentTotal === 1 ? t("common.pill") : t("common.pills")}
|
{currentTotal}
|
||||||
|
{isAmountPackage
|
||||||
|
? ` ${stockUnitLabel}`
|
||||||
|
: ` ${currentTotal === 1 ? t("common.pill") : t("common.pills")}`}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div className="summary-row">
|
<div className="summary-row">
|
||||||
<span>{t("editStock.newTotal")}:</span>
|
<span>{t("editStock.newTotal")}:</span>
|
||||||
<span>
|
<span>
|
||||||
{newTotal} {newTotal === 1 ? t("common.pill") : t("common.pills")}
|
{newTotal}
|
||||||
|
{isAmountPackage
|
||||||
|
? ` ${stockUnitLabel}`
|
||||||
|
: ` ${newTotal === 1 ? t("common.pill") : t("common.pills")}`}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div className={`summary-row difference ${differenceClass}`}>
|
<div className={`summary-row difference ${differenceClass}`}>
|
||||||
<span>{t("editStock.difference")}:</span>
|
<span>{t("editStock.difference")}:</span>
|
||||||
<span>
|
<span>
|
||||||
{difference > 0 ? "+" : ""}
|
{difference > 0 ? "+" : ""}
|
||||||
{difference} {Math.abs(difference) === 1 ? t("common.pill") : t("common.pills")}
|
{difference}
|
||||||
|
{isAmountPackage
|
||||||
|
? ` ${stockUnitLabel}`
|
||||||
|
: ` ${Math.abs(difference) === 1 ? t("common.pill") : t("common.pills")}`}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -667,12 +781,14 @@ export function MedDetailModal({
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<MedicationAvatar name={selectedMed.name} imageUrl={selectedMed.imageUrl} size="lg" />
|
<MedicationAvatar name={getMedDisplayName(selectedMed)} imageUrl={selectedMed.imageUrl} size="lg" />
|
||||||
{selectedMed.imageUrl && <span className="expand-icon">🔍</span>}
|
{selectedMed.imageUrl && <span className="expand-icon">🔍</span>}
|
||||||
</div>
|
</div>
|
||||||
<div className="med-detail-titles">
|
<div className="med-detail-titles">
|
||||||
<h2>{selectedMed.name}</h2>
|
<h2>{getMedDisplayName(selectedMed)}</h2>
|
||||||
{selectedMed.genericName && <span className="med-generic-name">{selectedMed.genericName}</span>}
|
{selectedMed.name && selectedMed.genericName && (
|
||||||
|
<span className="med-generic-name">{selectedMed.genericName}</span>
|
||||||
|
)}
|
||||||
{selectedMed.takenBy && (selectedMed.takenBy || []).length > 0 && (
|
{selectedMed.takenBy && (selectedMed.takenBy || []).length > 0 && (
|
||||||
<span className="med-taken-by">
|
<span className="med-taken-by">
|
||||||
{t("modal.for")}{" "}
|
{t("modal.for")}{" "}
|
||||||
@@ -694,7 +810,7 @@ export function MedDetailModal({
|
|||||||
<div className="med-detail-section">
|
<div className="med-detail-section">
|
||||||
<h3>{t("modal.stockInfo")}</h3>
|
<h3>{t("modal.stockInfo")}</h3>
|
||||||
<div className="med-detail-grid">
|
<div className="med-detail-grid">
|
||||||
{selectedMed.packageType === "blister" && (
|
{!isAmountBasedPackageType(selectedMed.packageType) && (
|
||||||
<>
|
<>
|
||||||
<div className="med-detail-item">
|
<div className="med-detail-item">
|
||||||
<span className="med-detail-label">{t("table.fullBlisters")}</span>
|
<span className="med-detail-label">{t("table.fullBlisters")}</span>
|
||||||
@@ -713,10 +829,14 @@ export function MedDetailModal({
|
|||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
<div className={`med-detail-item ${selectedMed.packageType === "bottle" ? "full-width" : "full-width"}`}>
|
<div className="med-detail-item full-width">
|
||||||
<span className="med-detail-label">{t("modal.currentStock")}</span>
|
<span className="med-detail-label">
|
||||||
|
{isAmountPackage ? t("form.currentAmount") : t("modal.currentStock")}
|
||||||
|
</span>
|
||||||
<span className={`med-detail-value ${textClass}`}>
|
<span className={`med-detail-value ${textClass}`}>
|
||||||
{currentStock} / {stockDisplayTotal}
|
{isAmountPackage
|
||||||
|
? `${formatNumber(currentStock)} / ${formatNumber(stockDisplayTotal)} ${amountUnitLabel}`
|
||||||
|
: `${currentStock} / ${stockDisplayTotal}`}
|
||||||
{currentStock > stockDisplayTotal && (
|
{currentStock > stockDisplayTotal && (
|
||||||
<span
|
<span
|
||||||
className="info-tooltip tooltip-align-left warning-text"
|
className="info-tooltip tooltip-align-left warning-text"
|
||||||
@@ -735,10 +855,27 @@ export function MedDetailModal({
|
|||||||
<div className="med-detail-section">
|
<div className="med-detail-section">
|
||||||
<h3>
|
<h3>
|
||||||
{t("modal.packageDetails")} (
|
{t("modal.packageDetails")} (
|
||||||
{selectedMed.packageType === "bottle" ? t("form.packageTypeBottle") : t("form.packageTypeBlister")})
|
{isTubePackageType(selectedMed.packageType)
|
||||||
|
? t("form.packageTypeTube")
|
||||||
|
: isLiquidContainerPackageType(selectedMed.packageType)
|
||||||
|
? t("form.packageTypeLiquidContainer")
|
||||||
|
: isAmountBasedPackageType(selectedMed.packageType)
|
||||||
|
? t("form.packageTypeBottle")
|
||||||
|
: t("form.packageTypeBlister")}
|
||||||
|
)
|
||||||
|
{isTubePackageType(selectedMed.packageType) && (
|
||||||
|
<span className="info-tooltip small" data-tooltip={t("modal.packageTypeTubeHint")}>
|
||||||
|
ℹ️
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
{isLiquidContainerPackageType(selectedMed.packageType) && (
|
||||||
|
<span className="info-tooltip small" data-tooltip={t("modal.packageTypeLiquidHint")}>
|
||||||
|
ℹ️
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
</h3>
|
</h3>
|
||||||
<div className="med-detail-grid">
|
<div className="med-detail-grid">
|
||||||
{selectedMed.packageType === "blister" ? (
|
{!isAmountBasedPackageType(selectedMed.packageType) ? (
|
||||||
<>
|
<>
|
||||||
<div className="med-detail-item">
|
<div className="med-detail-item">
|
||||||
<span className="med-detail-label">{t("modal.packs")}</span>
|
<span className="med-detail-label">{t("modal.packs")}</span>
|
||||||
@@ -753,6 +890,44 @@ export function MedDetailModal({
|
|||||||
<span className="med-detail-value">{selectedMed.pillsPerBlister}</span>
|
<span className="med-detail-value">{selectedMed.pillsPerBlister}</span>
|
||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
|
) : isLiquidContainerPackageType(selectedMed.packageType) ? (
|
||||||
|
<>
|
||||||
|
<div className="med-detail-item">
|
||||||
|
<span className="med-detail-label">{t("form.bottles")}</span>
|
||||||
|
<span className="med-detail-value">{packageCount}</span>
|
||||||
|
</div>
|
||||||
|
<div className="med-detail-item">
|
||||||
|
<span className="med-detail-label">{t("form.packageAmountPerBottle")}</span>
|
||||||
|
<span className="med-detail-value">
|
||||||
|
{formatNumber(amountPerPackage)} {amountUnitLabel}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className="med-detail-item">
|
||||||
|
<span className="med-detail-label">{t("form.totalAmount")}</span>
|
||||||
|
<span className="med-detail-value">
|
||||||
|
{formatNumber(stockDisplayTotal)} {amountUnitLabel}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
) : isTubePackageType(selectedMed.packageType) ? (
|
||||||
|
<>
|
||||||
|
<div className="med-detail-item">
|
||||||
|
<span className="med-detail-label">{t("form.tubes")}</span>
|
||||||
|
<span className="med-detail-value">{packageCount}</span>
|
||||||
|
</div>
|
||||||
|
<div className="med-detail-item">
|
||||||
|
<span className="med-detail-label">{t("form.packageAmountPerTube")}</span>
|
||||||
|
<span className="med-detail-value">
|
||||||
|
{formatNumber(amountPerPackage)} {amountUnitLabel}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className="med-detail-item">
|
||||||
|
<span className="med-detail-label">{t("form.totalAmount")}</span>
|
||||||
|
<span className="med-detail-value">
|
||||||
|
{formatNumber(stockDisplayTotal)} {amountUnitLabel}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
) : (
|
) : (
|
||||||
<div className="med-detail-item">
|
<div className="med-detail-item">
|
||||||
<span className="med-detail-label">{t("form.totalCapacity")}</span>
|
<span className="med-detail-label">{t("form.totalCapacity")}</span>
|
||||||
@@ -789,53 +964,33 @@ export function MedDetailModal({
|
|||||||
<div className="med-detail-section">
|
<div className="med-detail-section">
|
||||||
<h3>
|
<h3>
|
||||||
{t("modal.intakeSchedule")}{" "}
|
{t("modal.intakeSchedule")}{" "}
|
||||||
{selectedMed.intakeRemindersEnabled && (
|
{hasAnyIntakeReminder && (
|
||||||
<span className="reminder-icon info-tooltip" data-tooltip={t("tooltips.intakeReminders")}>
|
<span className="reminder-icon info-tooltip" data-tooltip={t("tooltips.intakeReminders")}>
|
||||||
<Bell size={14} aria-hidden="true" />
|
<Bell size={14} aria-hidden="true" />
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
</h3>
|
</h3>
|
||||||
<div className="med-detail-schedules">
|
<div className="med-detail-schedules">
|
||||||
{(selectedMed.intakes && selectedMed.intakes.length > 0
|
{scheduleIntakes.map((intake, idx) => {
|
||||||
? selectedMed.intakes
|
|
||||||
: selectedMed.blisters.map((blister) => ({
|
|
||||||
usage: blister.usage,
|
|
||||||
every: blister.every,
|
|
||||||
start: blister.start,
|
|
||||||
takenBy: null,
|
|
||||||
intakeRemindersEnabled: selectedMed.intakeRemindersEnabled ?? false,
|
|
||||||
}))
|
|
||||||
).map((intake, idx) => {
|
|
||||||
const hasPerIntakeTakenBy = !!intake.takenBy;
|
const hasPerIntakeTakenBy = !!intake.takenBy;
|
||||||
const personCount = Math.max(1, selectedMed.takenBy?.length ?? 0);
|
const personCount = Math.max(1, selectedMed.takenBy?.length ?? 0);
|
||||||
const totalUsage = hasPerIntakeTakenBy ? intake.usage : intake.usage * personCount;
|
const totalUsage = hasPerIntakeTakenBy ? intake.usage : intake.usage * personCount;
|
||||||
const showIntakeBell = intake.intakeRemindersEnabled ?? selectedMed.intakeRemindersEnabled ?? false;
|
const showIntakeBell = intake.intakeRemindersEnabled === true;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div key={`${intake.start}-${intake.usage}-${intake.every}-${idx}`} className="med-schedule-item">
|
<div
|
||||||
|
key={`${intake.start}-${intake.usage}-${intake.every}-${idx}`}
|
||||||
|
className="med-schedule-row blister-row-simple"
|
||||||
|
>
|
||||||
<span className="med-schedule-usage">
|
<span className="med-schedule-usage">
|
||||||
{totalUsage} {totalUsage !== 1 ? t("common.pills") : t("common.pill")}
|
{getScheduleUsageLabel(totalUsage, intake.intakeUnit)}
|
||||||
{selectedMed.pillWeightMg &&
|
{selectedMed.pillWeightMg &&
|
||||||
` (${totalUsage * selectedMed.pillWeightMg} ${selectedMed.doseUnit ?? "mg"})`}
|
` (${totalUsage * selectedMed.pillWeightMg} ${selectedMed.doseUnit ?? "mg"})`}
|
||||||
</span>
|
</span>
|
||||||
<span className="med-schedule-freq">
|
<span className="med-schedule-freq">
|
||||||
{intake.every === 1 ? t("common.daily") : t("common.everyNDays", { count: intake.every })}
|
{intake.every === 1 ? t("common.daily") : t("common.everyNDays", { count: intake.every })}
|
||||||
</span>
|
</span>
|
||||||
{hasPerIntakeTakenBy && (
|
{hasPerIntakeTakenBy && <span className="med-schedule-person">{intake.takenBy}</span>}
|
||||||
<span className="med-schedule-person">
|
|
||||||
{intake.takenBy}
|
|
||||||
{showIntakeBell && (
|
|
||||||
<span className="med-schedule-bell" role="img" aria-label={t("tooltips.intakeReminders")}>
|
|
||||||
<Bell size={13} aria-hidden="true" />
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
{!hasPerIntakeTakenBy && showIntakeBell && (
|
|
||||||
<span className="med-schedule-bell" role="img" aria-label={t("tooltips.intakeReminders")}>
|
|
||||||
<Bell size={13} aria-hidden="true" />
|
|
||||||
</span>
|
|
||||||
)}
|
|
||||||
<span className="med-schedule-time">
|
<span className="med-schedule-time">
|
||||||
{t("modal.at")}{" "}
|
{t("modal.at")}{" "}
|
||||||
{new Date(intake.start).toLocaleTimeString(getSystemLocale(i18n.language), {
|
{new Date(intake.start).toLocaleTimeString(getSystemLocale(i18n.language), {
|
||||||
@@ -843,6 +998,11 @@ export function MedDetailModal({
|
|||||||
minute: "2-digit",
|
minute: "2-digit",
|
||||||
})}
|
})}
|
||||||
</span>
|
</span>
|
||||||
|
{showIntakeBell && (
|
||||||
|
<span className="med-schedule-bell" title={t("form.blisters.remindTooltip")}>
|
||||||
|
<Bell size={12} aria-hidden="true" />
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
@@ -952,12 +1112,11 @@ export function MedDetailModal({
|
|||||||
</span>
|
</span>
|
||||||
<span className="refill-amount">
|
<span className="refill-amount">
|
||||||
{(() => {
|
{(() => {
|
||||||
const total =
|
const total = isAmountBasedPackageType(selectedMed.packageType)
|
||||||
selectedMed.packageType === "bottle"
|
? entry.loosePillsAdded
|
||||||
? entry.loosePillsAdded
|
: entry.packsAdded * selectedMed.blistersPerPack * selectedMed.pillsPerBlister +
|
||||||
: entry.packsAdded * selectedMed.blistersPerPack * selectedMed.pillsPerBlister +
|
entry.loosePillsAdded;
|
||||||
entry.loosePillsAdded;
|
return `+${total}${isAmountPackage ? ` ${stockUnitLabel}` : ` ${total === 1 ? t("common.pill") : t("common.pills")}`}`;
|
||||||
return `+${total} ${total === 1 ? t("common.pill") : t("common.pills")}`;
|
|
||||||
})()}
|
})()}
|
||||||
{entry.usedPrescription && (
|
{entry.usedPrescription && (
|
||||||
<span className="refill-prescription-badge" title={t("refill.viaPrescription")}>
|
<span className="refill-prescription-badge" title={t("refill.viaPrescription")}>
|
||||||
@@ -1017,7 +1176,11 @@ export function MedDetailModal({
|
|||||||
|
|
||||||
{/* Image Lightbox */}
|
{/* Image Lightbox */}
|
||||||
{showImageLightbox && selectedMed.imageUrl && (
|
{showImageLightbox && selectedMed.imageUrl && (
|
||||||
<Lightbox src={`/api/images/${selectedMed.imageUrl}`} alt={selectedMed.name} onClose={onCloseImageLightbox} />
|
<Lightbox
|
||||||
|
src={`/api/images/${selectedMed.imageUrl}`}
|
||||||
|
alt={getMedDisplayName(selectedMed)}
|
||||||
|
onClose={onCloseImageLightbox}
|
||||||
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{/* Refill Modal */}
|
{/* Refill Modal */}
|
||||||
@@ -1049,10 +1212,10 @@ export function MedDetailModal({
|
|||||||
<X size={18} aria-hidden="true" />
|
<X size={18} aria-hidden="true" />
|
||||||
</button>
|
</button>
|
||||||
<h2>{t("refill.title")}</h2>
|
<h2>{t("refill.title")}</h2>
|
||||||
<p className="refill-med-name">{selectedMed.name}</p>
|
<p className="refill-med-name">{getMedDisplayName(selectedMed)}</p>
|
||||||
|
|
||||||
<div className="refill-form">
|
<div className="refill-form">
|
||||||
{selectedMed.packageType === "blister" ? (
|
{!isAmountBasedPackageType(selectedMed.packageType) ? (
|
||||||
<>
|
<>
|
||||||
<label>
|
<label>
|
||||||
{t("refill.packs")}
|
{t("refill.packs")}
|
||||||
@@ -1096,7 +1259,7 @@ export function MedDetailModal({
|
|||||||
onUsePrescriptionRefillChange(checked);
|
onUsePrescriptionRefillChange(checked);
|
||||||
if (
|
if (
|
||||||
checked &&
|
checked &&
|
||||||
selectedMed.packageType === "blister" &&
|
!isAmountBasedPackageType(selectedMed.packageType) &&
|
||||||
refillPacks > remainingPrescriptionRefills
|
refillPacks > remainingPrescriptionRefills
|
||||||
) {
|
) {
|
||||||
onRefillPacksChange(remainingPrescriptionRefills);
|
onRefillPacksChange(remainingPrescriptionRefills);
|
||||||
@@ -1122,7 +1285,7 @@ export function MedDetailModal({
|
|||||||
className="success"
|
className="success"
|
||||||
onClick={() => onSubmitRefill(selectedMed.id, usePrescriptionRefill)}
|
onClick={() => onSubmitRefill(selectedMed.id, usePrescriptionRefill)}
|
||||||
disabled={
|
disabled={
|
||||||
(selectedMed.packageType === "bottle"
|
(isAmountBasedPackageType(selectedMed.packageType)
|
||||||
? refillLoose < 1
|
? refillLoose < 1
|
||||||
: cappedRefillPacks < 1 && refillLoose < 1) ||
|
: cappedRefillPacks < 1 && refillLoose < 1) ||
|
||||||
exceedsPrescriptionPackLimit ||
|
exceedsPrescriptionPackLimit ||
|
||||||
@@ -1132,13 +1295,15 @@ export function MedDetailModal({
|
|||||||
{refillSaving ? t("common.saving") : t("refill.button")}
|
{refillSaving ? t("common.saving") : t("refill.button")}
|
||||||
</button>
|
</button>
|
||||||
{(() => {
|
{(() => {
|
||||||
const totalRefill =
|
const totalRefill = !isAmountBasedPackageType(selectedMed.packageType)
|
||||||
selectedMed.packageType === "blister"
|
? cappedRefillPacks * selectedMed.blistersPerPack * selectedMed.pillsPerBlister + refillLoose
|
||||||
? cappedRefillPacks * selectedMed.blistersPerPack * selectedMed.pillsPerBlister + refillLoose
|
: refillLoose;
|
||||||
: refillLoose;
|
|
||||||
return totalRefill > 0 ? (
|
return totalRefill > 0 ? (
|
||||||
<span className="refill-preview">
|
<span className="refill-preview">
|
||||||
+{totalRefill} {totalRefill === 1 ? t("common.pill") : t("common.pills")}
|
+{totalRefill}
|
||||||
|
{isAmountPackage
|
||||||
|
? ` ${stockUnitLabel}`
|
||||||
|
: ` ${totalRefill === 1 ? t("common.pill") : t("common.pills")}`}
|
||||||
</span>
|
</span>
|
||||||
) : null;
|
) : null;
|
||||||
})()}
|
})()}
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
// MedicationAvatar Component
|
// MedicationAvatar Component
|
||||||
// =============================================================================
|
// =============================================================================
|
||||||
|
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useRef, useState } from "react";
|
||||||
|
|
||||||
export type MedicationAvatarProps = {
|
export type MedicationAvatarProps = {
|
||||||
name: string;
|
name: string;
|
||||||
@@ -12,8 +12,11 @@ export type MedicationAvatarProps = {
|
|||||||
|
|
||||||
export function MedicationAvatar({ name, imageUrl, size = "sm" }: MedicationAvatarProps) {
|
export function MedicationAvatar({ name, imageUrl, size = "sm" }: MedicationAvatarProps) {
|
||||||
const [thumbFailed, setThumbFailed] = useState(false);
|
const [thumbFailed, setThumbFailed] = useState(false);
|
||||||
|
const previousImageUrlRef = useRef(imageUrl);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
if (previousImageUrlRef.current === imageUrl) return;
|
||||||
|
previousImageUrlRef.current = imageUrl;
|
||||||
setThumbFailed(false);
|
setThumbFailed(false);
|
||||||
}, [imageUrl]);
|
}, [imageUrl]);
|
||||||
|
|
||||||
|
|||||||
@@ -5,12 +5,19 @@
|
|||||||
|
|
||||||
/* biome-ignore-all lint/a11y/noLabelWithoutControl: modal uses custom DateInput and static value fields */
|
/* biome-ignore-all lint/a11y/noLabelWithoutControl: modal uses custom DateInput and static value fields */
|
||||||
import { Bell, Minus, Plus, Trash2 } from "lucide-react";
|
import { Bell, Minus, Plus, Trash2 } from "lucide-react";
|
||||||
import { useEffect, useRef, useState } from "react";
|
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { useEscapeKey } from "../hooks/useEscapeKey";
|
import { useEscapeKey } from "../hooks/useEscapeKey";
|
||||||
import { useScrollLock } from "../hooks/useScrollLock";
|
import { useScrollLock } from "../hooks/useScrollLock";
|
||||||
import type { DoseUnit, FieldErrors, FormBlister, FormIntake, FormState, Medication } from "../types";
|
import type { DoseUnit, FieldErrors, FormBlister, FormIntake, FormState, Medication } from "../types";
|
||||||
import { DOSE_UNITS } from "../types";
|
import {
|
||||||
|
allowsPillFormSelection,
|
||||||
|
DOSE_UNITS,
|
||||||
|
isAmountBasedPackageType,
|
||||||
|
isLiquidContainerPackageType,
|
||||||
|
isTubePackageType,
|
||||||
|
PACKAGE_PROFILES,
|
||||||
|
} from "../types";
|
||||||
import { deriveTotal } from "../utils";
|
import { deriveTotal } from "../utils";
|
||||||
import { DateInput } from "./DateInput";
|
import { DateInput } from "./DateInput";
|
||||||
import { FormNumberStepper } from "./FormNumberStepper";
|
import { FormNumberStepper } from "./FormNumberStepper";
|
||||||
@@ -68,7 +75,7 @@ export interface MobileEditModalProps {
|
|||||||
|
|
||||||
/** Calculate total pills from form state */
|
/** Calculate total pills from form state */
|
||||||
function deriveTotalFromForm(form: FormState) {
|
function deriveTotalFromForm(form: FormState) {
|
||||||
if (form.packageType === "bottle") {
|
if (isAmountBasedPackageType(form.packageType)) {
|
||||||
// For bottle type, looseTablets is the current stock
|
// For bottle type, looseTablets is the current stock
|
||||||
return Number(form.looseTablets) || 0;
|
return Number(form.looseTablets) || 0;
|
||||||
}
|
}
|
||||||
@@ -96,9 +103,9 @@ export function MobileEditModal({
|
|||||||
onAddTakenByPerson,
|
onAddTakenByPerson,
|
||||||
onRemoveTakenByPerson,
|
onRemoveTakenByPerson,
|
||||||
onTakenByKeyDown,
|
onTakenByKeyDown,
|
||||||
_onSetBlisterValue,
|
onSetBlisterValue: _onSetBlisterValue,
|
||||||
_onAddBlister,
|
onAddBlister: _onAddBlister,
|
||||||
_onRemoveBlister,
|
onRemoveBlister: _onRemoveBlister,
|
||||||
onSetIntakeValue,
|
onSetIntakeValue,
|
||||||
onAddIntake,
|
onAddIntake,
|
||||||
onRemoveIntake,
|
onRemoveIntake,
|
||||||
@@ -108,7 +115,7 @@ export function MobileEditModal({
|
|||||||
onDeleteMedImage,
|
onDeleteMedImage,
|
||||||
imageUploadError,
|
imageUploadError,
|
||||||
onClose,
|
onClose,
|
||||||
_onResetForm,
|
onResetForm: _onResetForm,
|
||||||
onSaveMedication,
|
onSaveMedication,
|
||||||
}: MobileEditModalProps) {
|
}: MobileEditModalProps) {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
@@ -125,6 +132,33 @@ export function MobileEditModal({
|
|||||||
const [showNameValidation, setShowNameValidation] = useState(false);
|
const [showNameValidation, setShowNameValidation] = useState(false);
|
||||||
const activeTabIndexRef = useRef(0);
|
const activeTabIndexRef = useRef(0);
|
||||||
|
|
||||||
|
const allowFractionalIntake = useMemo(() => {
|
||||||
|
if (isLiquidContainerPackageType(form.packageType)) return true;
|
||||||
|
if (isTubePackageType(form.packageType)) return form.medicationForm === "liquid";
|
||||||
|
return form.pillForm === "tablet";
|
||||||
|
}, [form.packageType, form.medicationForm, form.pillForm]);
|
||||||
|
|
||||||
|
const getUsageLabel = useCallback(
|
||||||
|
(intake: (typeof form.intakes)[number]) => {
|
||||||
|
if (isLiquidContainerPackageType(form.packageType)) {
|
||||||
|
if (intake.intakeUnit === "tsp") return t("form.blisters.usageTsp");
|
||||||
|
if (intake.intakeUnit === "tbsp") return t("form.blisters.usageTbsp");
|
||||||
|
return t("form.blisters.usageMl");
|
||||||
|
}
|
||||||
|
if (isTubePackageType(form.packageType)) {
|
||||||
|
return form.medicationForm === "liquid" ? t("form.blisters.usageMl") : t("form.blisters.usageApplication");
|
||||||
|
}
|
||||||
|
if (form.pillForm === "capsule") return t("form.blisters.usageCapsules");
|
||||||
|
return t("form.blisters.usageTablets");
|
||||||
|
},
|
||||||
|
[form.packageType, form.medicationForm, form.pillForm, t]
|
||||||
|
);
|
||||||
|
|
||||||
|
const usesAmountLabels = isTubePackageType(form.packageType) || isLiquidContainerPackageType(form.packageType);
|
||||||
|
const totalCapacityLabel = usesAmountLabels ? t("form.totalAmount") : t("form.totalCapacity");
|
||||||
|
const currentStockLabel = usesAmountLabels ? t("form.currentAmount") : t("form.currentPills");
|
||||||
|
const totalLabel = usesAmountLabels ? t("form.totalAmountLabel") : t("form.total");
|
||||||
|
|
||||||
// Reset tab when modal opens
|
// Reset tab when modal opens
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (show) {
|
if (show) {
|
||||||
@@ -253,7 +287,10 @@ export function MobileEditModal({
|
|||||||
const mobileTitle = (() => {
|
const mobileTitle = (() => {
|
||||||
if (!editingId) return t("form.newEntry");
|
if (!editingId) return t("form.newEntry");
|
||||||
if (readOnlyMode) return t("form.viewEntry");
|
if (readOnlyMode) return t("form.viewEntry");
|
||||||
const medicationName = currentMed?.name?.trim() || form.name.trim();
|
const medicationName =
|
||||||
|
(currentMed ? currentMed.name?.trim() || currentMed.genericName?.trim() : null) ||
|
||||||
|
form.name.trim() ||
|
||||||
|
form.genericName.trim();
|
||||||
if (!medicationName) return t("form.editEntry");
|
if (!medicationName) return t("form.editEntry");
|
||||||
return t("form.editEntryWithName", { name: medicationName });
|
return t("form.editEntryWithName", { name: medicationName });
|
||||||
})();
|
})();
|
||||||
@@ -361,27 +398,35 @@ export function MobileEditModal({
|
|||||||
onBlur={() => setShowNameValidation(true)}
|
onBlur={() => setShowNameValidation(true)}
|
||||||
placeholder={t("form.placeholders.commercial")}
|
placeholder={t("form.placeholders.commercial")}
|
||||||
maxLength={FIELD_LIMITS.name.max}
|
maxLength={FIELD_LIMITS.name.max}
|
||||||
required={!readOnlyMode}
|
|
||||||
/>
|
/>
|
||||||
{!readOnlyMode && showNameValidation && fieldErrors.name && (
|
{!readOnlyMode && showNameValidation && fieldErrors.name && (
|
||||||
<span className="field-error">{fieldErrors.name}</span>
|
<span className="field-error">{fieldErrors.name}</span>
|
||||||
)}
|
)}
|
||||||
</label>
|
</label>
|
||||||
<label className={`full ${fieldErrors.genericName ? "has-error" : ""}`}>
|
<label
|
||||||
|
className={`full ${!readOnlyMode && showNameValidation && fieldErrors.genericName ? "has-error" : ""}`}
|
||||||
|
>
|
||||||
{t("form.genericName")}
|
{t("form.genericName")}
|
||||||
<input
|
<input
|
||||||
value={form.genericName}
|
value={form.genericName}
|
||||||
onChange={(e) => onFormChange({ ...form, genericName: e.target.value })}
|
onChange={(e) => {
|
||||||
|
setShowNameValidation(true);
|
||||||
|
onFormChange({ ...form, genericName: e.target.value });
|
||||||
|
}}
|
||||||
|
onBlur={() => setShowNameValidation(true)}
|
||||||
placeholder={t("form.placeholders.generic")}
|
placeholder={t("form.placeholders.generic")}
|
||||||
maxLength={FIELD_LIMITS.genericName.max}
|
maxLength={FIELD_LIMITS.genericName.max}
|
||||||
/>
|
/>
|
||||||
{fieldErrors.genericName && <span className="field-error">{fieldErrors.genericName}</span>}
|
{!readOnlyMode && showNameValidation && fieldErrors.genericName && (
|
||||||
|
<span className="field-error">{fieldErrors.genericName}</span>
|
||||||
|
)}
|
||||||
</label>
|
</label>
|
||||||
<label className="full">
|
<label className="full">
|
||||||
{t("form.medicationStartDate")}
|
{t("form.medicationStartDate")}
|
||||||
<DateInput
|
<DateInput
|
||||||
value={form.medicationStartDate}
|
value={form.medicationStartDate}
|
||||||
onChange={(e) => onHandleValueChange("medicationStartDate", e.target.value)}
|
onChange={(e) => onHandleValueChange("medicationStartDate", e.target.value)}
|
||||||
|
placeholder={t("common.optional")}
|
||||||
/>
|
/>
|
||||||
{!readOnlyMode && dateConsistencyError && (
|
{!readOnlyMode && dateConsistencyError && (
|
||||||
<span className="field-error">{dateConsistencyError}</span>
|
<span className="field-error">{dateConsistencyError}</span>
|
||||||
@@ -392,12 +437,64 @@ export function MobileEditModal({
|
|||||||
<select
|
<select
|
||||||
className="package-type-select"
|
className="package-type-select"
|
||||||
value={form.packageType}
|
value={form.packageType}
|
||||||
onChange={(e) => onHandleValueChange("packageType", e.target.value)}
|
onChange={(e) => onHandleValueChange("packageType", e.target.value as FormState["packageType"])}
|
||||||
>
|
>
|
||||||
<option value="blister">{t("form.packageTypeBlister")}</option>
|
{PACKAGE_PROFILES.map((profile) => (
|
||||||
<option value="bottle">{t("form.packageTypeBottle")}</option>
|
<option key={profile.value} value={profile.value}>
|
||||||
|
{t(profile.labelKey)}
|
||||||
|
</option>
|
||||||
|
))}
|
||||||
</select>
|
</select>
|
||||||
</label>
|
</label>
|
||||||
|
<label className="full">
|
||||||
|
{t("form.medicationEndDate")}
|
||||||
|
<DateInput
|
||||||
|
value={form.medicationEndDate}
|
||||||
|
onChange={(e) => onHandleValueChange("medicationEndDate", e.target.value)}
|
||||||
|
placeholder={t("common.optional")}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
{allowsPillFormSelection(form.packageType) && (
|
||||||
|
<label className="full">
|
||||||
|
{t("form.pillForm")}
|
||||||
|
<select
|
||||||
|
value={form.pillForm}
|
||||||
|
onChange={(e) => onHandleValueChange("pillForm", e.target.value as FormState["pillForm"])}
|
||||||
|
>
|
||||||
|
<option value="tablet">{t("form.medicationFormTablet")}</option>
|
||||||
|
<option value="capsule">{t("form.medicationFormCapsule")}</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
|
{isTubePackageType(form.packageType) && (
|
||||||
|
<label className="full">
|
||||||
|
{t("form.medicationForm")}
|
||||||
|
<select value={"topical"} onChange={() => onHandleValueChange("medicationForm", "topical")}>
|
||||||
|
<option value="topical">{t("form.medicationFormTopical")}</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
|
{isLiquidContainerPackageType(form.packageType) && (
|
||||||
|
<label className="full">
|
||||||
|
{t("form.medicationForm")}
|
||||||
|
<select value={"liquid"} onChange={() => onHandleValueChange("medicationForm", "liquid")}>
|
||||||
|
<option value="liquid">{t("form.medicationFormLiquid")}</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
|
{form.medicationEndDate && (
|
||||||
|
<label className="full">
|
||||||
|
{t("form.autoMarkObsoleteAfterEndDate")}
|
||||||
|
<span className="toggle-switch small">
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
checked={form.autoMarkObsoleteAfterEndDate}
|
||||||
|
onChange={(e) => onHandleValueChange("autoMarkObsoleteAfterEndDate", e.target.checked)}
|
||||||
|
/>
|
||||||
|
<span className="toggle-slider"></span>
|
||||||
|
</span>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
<label className={`full ${fieldErrors.takenBy ? "has-error" : ""}`}>
|
<label className={`full ${fieldErrors.takenBy ? "has-error" : ""}`}>
|
||||||
{t("form.takenBy")}
|
{t("form.takenBy")}
|
||||||
<div className="tag-input-container">
|
<div className="tag-input-container">
|
||||||
@@ -470,101 +567,193 @@ export function MobileEditModal({
|
|||||||
<div className={`form-tab-panel${activeTab === "stock" ? " active" : ""}`}>
|
<div className={`form-tab-panel${activeTab === "stock" ? " active" : ""}`}>
|
||||||
<div className="full form-category">
|
<div className="full form-category">
|
||||||
<h4 className="form-category-title">{t("form.sections.stock")}</h4>
|
<h4 className="form-category-title">{t("form.sections.stock")}</h4>
|
||||||
{form.packageType === "blister" ? (
|
{(() => {
|
||||||
<>
|
if (!isAmountBasedPackageType(form.packageType)) {
|
||||||
<label>
|
return (
|
||||||
{t("form.packs")}
|
<>
|
||||||
<FormNumberStepper
|
<label>
|
||||||
value={form.packCount}
|
{t("form.packs")}
|
||||||
onChange={(nextValue) => onHandleValueChange("packCount", nextValue)}
|
<FormNumberStepper
|
||||||
min={0}
|
value={form.packCount}
|
||||||
decrementLabel={decrementValueLabel}
|
onChange={(nextValue) => onHandleValueChange("packCount", nextValue)}
|
||||||
incrementLabel={incrementValueLabel}
|
min={0}
|
||||||
/>
|
decrementLabel={decrementValueLabel}
|
||||||
</label>
|
incrementLabel={incrementValueLabel}
|
||||||
<label>
|
/>
|
||||||
{t("form.blistersPerPack")}
|
</label>
|
||||||
<FormNumberStepper
|
<label>
|
||||||
value={form.blistersPerPack}
|
{t("form.blistersPerPack")}
|
||||||
onChange={(nextValue) => onHandleValueChange("blistersPerPack", nextValue)}
|
<FormNumberStepper
|
||||||
min={1}
|
value={form.blistersPerPack}
|
||||||
decrementLabel={decrementValueLabel}
|
onChange={(nextValue) => onHandleValueChange("blistersPerPack", nextValue)}
|
||||||
incrementLabel={incrementValueLabel}
|
min={1}
|
||||||
/>
|
decrementLabel={decrementValueLabel}
|
||||||
</label>
|
incrementLabel={incrementValueLabel}
|
||||||
<label>
|
/>
|
||||||
{t("form.pillsPerBlister")}
|
</label>
|
||||||
<FormNumberStepper
|
<label>
|
||||||
value={form.pillsPerBlister}
|
{t("form.pillsPerBlister")}
|
||||||
onChange={(nextValue) => onHandleValueChange("pillsPerBlister", nextValue)}
|
<FormNumberStepper
|
||||||
min={1}
|
value={form.pillsPerBlister}
|
||||||
decrementLabel={decrementValueLabel}
|
onChange={(nextValue) => onHandleValueChange("pillsPerBlister", nextValue)}
|
||||||
incrementLabel={incrementValueLabel}
|
min={1}
|
||||||
/>
|
decrementLabel={decrementValueLabel}
|
||||||
</label>
|
incrementLabel={incrementValueLabel}
|
||||||
<label>
|
/>
|
||||||
{t("form.total")}
|
</label>
|
||||||
<div className="static-value">{deriveTotalFromForm(form)}</div>
|
<label>
|
||||||
</label>
|
{t("form.total")}
|
||||||
</>
|
<div className="static-value">{deriveTotalFromForm(form)}</div>
|
||||||
) : (
|
</label>
|
||||||
<>
|
</>
|
||||||
<label>
|
);
|
||||||
{t("form.totalCapacity")}
|
}
|
||||||
<FormNumberStepper
|
|
||||||
value={form.totalPills}
|
if (isTubePackageType(form.packageType)) {
|
||||||
onChange={(nextValue) => onHandleValueChange("totalPills", nextValue)}
|
return (
|
||||||
min={0}
|
<>
|
||||||
decrementLabel={decrementValueLabel}
|
<label>
|
||||||
incrementLabel={incrementValueLabel}
|
{t("form.tubes")}
|
||||||
/>
|
<div className="static-value">1</div>
|
||||||
</label>
|
</label>
|
||||||
<label>
|
<label className="full">
|
||||||
{t("form.currentPills")}
|
{t("form.packageAmountPerTube")}
|
||||||
<FormNumberStepper
|
<div className="dose-input-group">
|
||||||
value={form.looseTablets}
|
<input
|
||||||
onChange={(nextValue) => onHandleValueChange("looseTablets", nextValue)}
|
type="text"
|
||||||
min={0}
|
inputMode="decimal"
|
||||||
decrementLabel={decrementValueLabel}
|
pattern="[0-9]*\.?[0-9]*"
|
||||||
incrementLabel={incrementValueLabel}
|
value={form.packageAmountValue ?? "0"}
|
||||||
/>
|
onChange={(e) => onHandleValueChange("packageAmountValue", e.target.value)}
|
||||||
</label>
|
placeholder="0"
|
||||||
</>
|
/>
|
||||||
)}
|
<select
|
||||||
{form.packageType === "bottle" && (
|
value="g"
|
||||||
|
disabled
|
||||||
|
className="dose-unit-select"
|
||||||
|
aria-label={t("form.packageAmountUnitG")}
|
||||||
|
>
|
||||||
|
<option value="g">{t("form.packageAmountUnitG")}</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
<label>
|
||||||
|
{t("form.totalAmount")}
|
||||||
|
<div className="static-value">
|
||||||
|
{(Number(form.packCount) || 0) * (Number(form.packageAmountValue ?? 0) || 0)}{" "}
|
||||||
|
{t("form.packageAmountUnitG")}
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isLiquidContainerPackageType(form.packageType)) {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<label>
|
||||||
|
{t("form.bottles")}
|
||||||
|
<FormNumberStepper
|
||||||
|
value={form.packCount}
|
||||||
|
onChange={(nextValue) => onHandleValueChange("packCount", nextValue)}
|
||||||
|
min={1}
|
||||||
|
decrementLabel={decrementValueLabel}
|
||||||
|
incrementLabel={incrementValueLabel}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
<label className="full">
|
||||||
|
{t("form.packageAmountPerBottle")}
|
||||||
|
<div className="dose-input-group">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
inputMode="decimal"
|
||||||
|
pattern="[0-9]*\.?[0-9]*"
|
||||||
|
value={form.packageAmountValue ?? "0"}
|
||||||
|
onChange={(e) => onHandleValueChange("packageAmountValue", e.target.value)}
|
||||||
|
placeholder="0"
|
||||||
|
/>
|
||||||
|
<select
|
||||||
|
value="ml"
|
||||||
|
disabled
|
||||||
|
className="dose-unit-select"
|
||||||
|
aria-label={t("form.packageAmountUnitMl")}
|
||||||
|
>
|
||||||
|
<option value="ml">{t("form.packageAmountUnitMl")}</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
<label>
|
||||||
|
{t("form.totalAmount")}
|
||||||
|
<div className="static-value">
|
||||||
|
{(Number(form.packCount) || 0) * (Number(form.packageAmountValue ?? 0) || 0)}{" "}
|
||||||
|
{t("form.packageAmountUnitMl")}
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<label>
|
||||||
|
{totalCapacityLabel}
|
||||||
|
<FormNumberStepper
|
||||||
|
value={form.totalPills}
|
||||||
|
onChange={(nextValue) => onHandleValueChange("totalPills", nextValue)}
|
||||||
|
min={0}
|
||||||
|
decrementLabel={decrementValueLabel}
|
||||||
|
incrementLabel={incrementValueLabel}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
<label>
|
||||||
|
{currentStockLabel}
|
||||||
|
<FormNumberStepper
|
||||||
|
value={form.looseTablets}
|
||||||
|
onChange={(nextValue) => onHandleValueChange("looseTablets", nextValue)}
|
||||||
|
min={0}
|
||||||
|
decrementLabel={decrementValueLabel}
|
||||||
|
incrementLabel={incrementValueLabel}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
})()}
|
||||||
|
{isAmountBasedPackageType(form.packageType) && !isTubePackageType(form.packageType) && (
|
||||||
<div className="full stock-total-row">
|
<div className="full stock-total-row">
|
||||||
<div className="stock-total-field">
|
<div className="stock-total-field">
|
||||||
<p className="sub">
|
<p className="sub">
|
||||||
<strong>{t("form.total")}:</strong> {deriveTotalFromForm(form)}{" "}
|
<strong>{totalLabel}:</strong> {deriveTotalFromForm(form)}
|
||||||
{deriveTotalFromForm(form) === 1 ? t("common.pill") : t("common.pills")}
|
{` ${deriveTotalFromForm(form) === 1 ? t("common.pill") : t("common.pills")}`}
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
<label className="full">
|
{allowsPillFormSelection(form.packageType) && (
|
||||||
{t("form.pillWeight")} ({form.doseUnit})
|
<label className="full">
|
||||||
<div className="dose-input-group">
|
{t("form.pillWeight")} ({form.doseUnit})
|
||||||
<input
|
<div className="dose-input-group">
|
||||||
type="text"
|
<input
|
||||||
inputMode="decimal"
|
type="text"
|
||||||
pattern="[0-9]*\.?[0-9]*"
|
inputMode="decimal"
|
||||||
value={form.pillWeightMg}
|
pattern="[0-9]*\.?[0-9]*"
|
||||||
onChange={(e) => onFormChange({ ...form, pillWeightMg: e.target.value })}
|
value={form.pillWeightMg}
|
||||||
placeholder={t("form.placeholders.weight")}
|
onChange={(e) => onFormChange({ ...form, pillWeightMg: e.target.value })}
|
||||||
/>
|
placeholder={t("form.placeholders.weight")}
|
||||||
<select
|
/>
|
||||||
value={form.doseUnit}
|
<select
|
||||||
onChange={(e) => onFormChange({ ...form, doseUnit: e.target.value as DoseUnit })}
|
value={form.doseUnit}
|
||||||
className="dose-unit-select"
|
onChange={(e) => onFormChange({ ...form, doseUnit: e.target.value as DoseUnit })}
|
||||||
>
|
className="dose-unit-select"
|
||||||
{DOSE_UNITS.map((unit) => (
|
>
|
||||||
<option key={unit.value} value={unit.value}>
|
{DOSE_UNITS.map((unit) => (
|
||||||
{unit.label}
|
<option key={unit.value} value={unit.value}>
|
||||||
</option>
|
{unit.label}
|
||||||
))}
|
</option>
|
||||||
</select>
|
))}
|
||||||
</div>
|
</select>
|
||||||
</label>
|
</div>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
<label className="full">
|
<label className="full">
|
||||||
{t("form.expiryDate")}
|
{t("form.expiryDate")}
|
||||||
<DateInput
|
<DateInput
|
||||||
@@ -616,17 +805,17 @@ export function MobileEditModal({
|
|||||||
</div>
|
</div>
|
||||||
{form.intakes.map((intake, idx) => (
|
{form.intakes.map((intake, idx) => (
|
||||||
<div
|
<div
|
||||||
key={`${intake.startDate}-${intake.startTime}-${intake.usage}-${intake.every}-${intake.takenBy ?? ""}`}
|
key={`${intake.startDate}-${intake.startTime}-${intake.usage}-${intake.every}-${intake.takenBy ?? ""}-${idx}`}
|
||||||
className="blister-row"
|
className="blister-row"
|
||||||
>
|
>
|
||||||
<label className="compact">
|
<label className="compact">
|
||||||
<span>{t("form.blisters.usage")}</span>
|
<span>{getUsageLabel(intake)}</span>
|
||||||
<FormNumberStepper
|
<FormNumberStepper
|
||||||
value={intake.usage}
|
value={intake.usage}
|
||||||
onChange={(nextValue) => onSetIntakeValue(idx, "usage", nextValue)}
|
onChange={(nextValue) => onSetIntakeValue(idx, "usage", nextValue)}
|
||||||
min={0.5}
|
min={allowFractionalIntake ? 0.5 : 1}
|
||||||
step={0.5}
|
step={allowFractionalIntake ? 0.5 : 1}
|
||||||
allowDecimal={true}
|
allowDecimal={allowFractionalIntake}
|
||||||
decrementLabel={decrementValueLabel}
|
decrementLabel={decrementValueLabel}
|
||||||
incrementLabel={incrementValueLabel}
|
incrementLabel={incrementValueLabel}
|
||||||
/>
|
/>
|
||||||
@@ -656,6 +845,21 @@ export function MobileEditModal({
|
|||||||
onChange={(e) => onSetIntakeValue(idx, "startTime", e.target.value)}
|
onChange={(e) => onSetIntakeValue(idx, "startTime", e.target.value)}
|
||||||
/>
|
/>
|
||||||
</label>
|
</label>
|
||||||
|
{isLiquidContainerPackageType(form.packageType) && (
|
||||||
|
<label className="compact full-row">
|
||||||
|
<span>{t("form.blisters.intakeUnit")}</span>
|
||||||
|
<select
|
||||||
|
value={intake.intakeUnit}
|
||||||
|
onChange={(e) =>
|
||||||
|
onSetIntakeValue(idx, "intakeUnit", e.target.value as "ml" | "tsp" | "tbsp")
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<option value="ml">{t("form.blisters.intakeUnitMl")}</option>
|
||||||
|
<option value="tsp">{t("form.blisters.intakeUnitTsp")}</option>
|
||||||
|
<option value="tbsp">{t("form.blisters.intakeUnitTbsp")}</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
{form.takenBy.length === 0 ? null : (
|
{form.takenBy.length === 0 ? null : (
|
||||||
<label className="compact full-row taken-by-field">
|
<label className="compact full-row taken-by-field">
|
||||||
<span>{t("form.blisters.takenByIntake")}</span>
|
<span>{t("form.blisters.takenByIntake")}</span>
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
import { useEscapeKey } from "../hooks/useEscapeKey";
|
|
||||||
import { UserProfile } from "./Auth";
|
import { UserProfile } from "./Auth";
|
||||||
|
|
||||||
interface ProfileModalProps {
|
interface ProfileModalProps {
|
||||||
@@ -7,7 +6,7 @@ interface ProfileModalProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export default function ProfileModal({ isOpen, onClose }: ProfileModalProps) {
|
export default function ProfileModal({ isOpen, onClose }: ProfileModalProps) {
|
||||||
useEscapeKey(isOpen, onClose);
|
// ESC is handled by the global handler in App.tsx to avoid double history.back()
|
||||||
|
|
||||||
if (!isOpen) return null;
|
if (!isOpen) return null;
|
||||||
|
|
||||||
|
|||||||
@@ -3,7 +3,13 @@ import { useTranslation } from "react-i18next";
|
|||||||
import { useEscapeKey } from "../hooks/useEscapeKey";
|
import { useEscapeKey } from "../hooks/useEscapeKey";
|
||||||
import { useScrollLock } from "../hooks/useScrollLock";
|
import { useScrollLock } from "../hooks/useScrollLock";
|
||||||
import type { Medication } from "../types";
|
import type { Medication } from "../types";
|
||||||
import { getPackageSize } from "../types";
|
import {
|
||||||
|
getMedDisplayName,
|
||||||
|
getPackageSize,
|
||||||
|
isAmountBasedPackageType,
|
||||||
|
isLiquidContainerPackageType,
|
||||||
|
isTubePackageType,
|
||||||
|
} from "../types";
|
||||||
import { MedicationAvatar } from "./MedicationAvatar";
|
import { MedicationAvatar } from "./MedicationAvatar";
|
||||||
|
|
||||||
type ReportFormat = "txt" | "md" | "pdf";
|
type ReportFormat = "txt" | "md" | "pdf";
|
||||||
@@ -200,10 +206,10 @@ export function ReportModal({ isOpen, onClose, medications }: ReportModalProps)
|
|||||||
{activeMeds.map((med) => (
|
{activeMeds.map((med) => (
|
||||||
<label key={med.id} className="report-med-item">
|
<label key={med.id} className="report-med-item">
|
||||||
<input type="checkbox" checked={selectedIds.has(med.id)} onChange={() => toggleMed(med.id)} />
|
<input type="checkbox" checked={selectedIds.has(med.id)} onChange={() => toggleMed(med.id)} />
|
||||||
<MedicationAvatar name={med.name} imageUrl={med.imageUrl} size="sm" />
|
<MedicationAvatar name={getMedDisplayName(med)} imageUrl={med.imageUrl} size="sm" />
|
||||||
<span className="report-med-name">
|
<span className="report-med-name">
|
||||||
{med.name}
|
{getMedDisplayName(med)}
|
||||||
{med.genericName && <span className="report-med-generic"> ({med.genericName})</span>}
|
{med.name && med.genericName && <span className="report-med-generic"> ({med.genericName})</span>}
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
))}
|
))}
|
||||||
@@ -218,10 +224,10 @@ export function ReportModal({ isOpen, onClose, medications }: ReportModalProps)
|
|||||||
{obsoleteMeds.map((med) => (
|
{obsoleteMeds.map((med) => (
|
||||||
<label key={med.id} className="report-med-item">
|
<label key={med.id} className="report-med-item">
|
||||||
<input type="checkbox" checked={selectedIds.has(med.id)} onChange={() => toggleMed(med.id)} />
|
<input type="checkbox" checked={selectedIds.has(med.id)} onChange={() => toggleMed(med.id)} />
|
||||||
<MedicationAvatar name={med.name} imageUrl={med.imageUrl} size="sm" />
|
<MedicationAvatar name={getMedDisplayName(med)} imageUrl={med.imageUrl} size="sm" />
|
||||||
<span className="report-med-name obsolete-name">
|
<span className="report-med-name obsolete-name">
|
||||||
{med.name}
|
{getMedDisplayName(med)}
|
||||||
{med.genericName && <span className="report-med-generic"> ({med.genericName})</span>}
|
{med.name && med.genericName && <span className="report-med-generic"> ({med.genericName})</span>}
|
||||||
</span>
|
</span>
|
||||||
</label>
|
</label>
|
||||||
))}
|
))}
|
||||||
@@ -298,6 +304,39 @@ function fmtDateTime(iso: string | null | undefined): string {
|
|||||||
return `${m[3]}.${m[2]}.${m[1]} ${m[4]}:${m[5]}`;
|
return `${m[3]}.${m[2]}.${m[1]} ${m[4]}:${m[5]}`;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getTubeUnitKey(med: Medication): "form.ml" | "blisters.applications" {
|
||||||
|
if (isLiquidContainerPackageType(med.packageType)) return "form.ml";
|
||||||
|
return med.medicationForm === "liquid" ? "form.ml" : "blisters.applications";
|
||||||
|
}
|
||||||
|
|
||||||
|
function getUsageText(med: Medication, usage: number, t: TFn): string {
|
||||||
|
if (isTubePackageType(med.packageType) || isLiquidContainerPackageType(med.packageType)) {
|
||||||
|
return `${usage} ${t(getTubeUnitKey(med))}`;
|
||||||
|
}
|
||||||
|
return `${usage} ${usage === 1 ? t("common.pill") : t("common.pills")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getTotalCapacityLabel(med: Medication, t: TFn): string {
|
||||||
|
if (isTubePackageType(med.packageType) || isLiquidContainerPackageType(med.packageType)) {
|
||||||
|
return t("form.totalAmountLabel", { unit: t(getTubeUnitKey(med)) });
|
||||||
|
}
|
||||||
|
return t("report.docTotalCapacity");
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCurrentStockText(med: Medication, t: TFn): string {
|
||||||
|
if (isTubePackageType(med.packageType) || isLiquidContainerPackageType(med.packageType)) {
|
||||||
|
return `${getPackageSize(med)} ${t(getTubeUnitKey(med))}`;
|
||||||
|
}
|
||||||
|
return `${getPackageSize(med)} ${t("common.pills")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
function getReportPackageTypeLabel(med: Medication, t: TFn): string {
|
||||||
|
if (isTubePackageType(med.packageType)) return t("report.docTube");
|
||||||
|
if (isLiquidContainerPackageType(med.packageType)) return t("form.packageTypeLiquidContainer");
|
||||||
|
if (isAmountBasedPackageType(med.packageType)) return t("report.docBottle");
|
||||||
|
return t("report.docBlister");
|
||||||
|
}
|
||||||
|
|
||||||
function generateTextReport(
|
function generateTextReport(
|
||||||
meds: Medication[],
|
meds: Medication[],
|
||||||
reportData: ReportData,
|
reportData: ReportData,
|
||||||
@@ -320,13 +359,15 @@ function generateTextReport(
|
|||||||
for (const med of meds) {
|
for (const med of meds) {
|
||||||
lines.push(sep);
|
lines.push(sep);
|
||||||
lines.push("");
|
lines.push("");
|
||||||
const title = med.isObsolete ? `${med.name} (${t("report.docStatusObsolete")})` : med.name;
|
const title = med.isObsolete
|
||||||
|
? `${getMedDisplayName(med)} (${t("report.docStatusObsolete")})`
|
||||||
|
: getMedDisplayName(med);
|
||||||
lines.push(h2(title));
|
lines.push(h2(title));
|
||||||
lines.push("");
|
lines.push("");
|
||||||
|
|
||||||
// General
|
// General
|
||||||
lines.push(h3(t("report.docGeneral")));
|
lines.push(h3(t("report.docGeneral")));
|
||||||
lines.push(item(t("report.docCommercialName"), med.name));
|
if (med.name) lines.push(item(t("report.docCommercialName"), med.name));
|
||||||
if (med.genericName) lines.push(item(t("report.docGenericName"), med.genericName));
|
if (med.genericName) lines.push(item(t("report.docGenericName"), med.genericName));
|
||||||
if (med.takenBy?.length) lines.push(item(t("report.docTakenBy"), med.takenBy.join(", ")));
|
if (med.takenBy?.length) lines.push(item(t("report.docTakenBy"), med.takenBy.join(", ")));
|
||||||
lines.push(
|
lines.push(
|
||||||
@@ -338,19 +379,18 @@ function generateTextReport(
|
|||||||
|
|
||||||
// Package / Stock
|
// Package / Stock
|
||||||
lines.push(h3(t("report.docPackage")));
|
lines.push(h3(t("report.docPackage")));
|
||||||
lines.push(
|
lines.push(item(t("report.docPackageType"), getReportPackageTypeLabel(med, t)));
|
||||||
item(t("report.docPackageType"), med.packageType === "bottle" ? t("report.docBottle") : t("report.docBlister"))
|
if (!isAmountBasedPackageType(med.packageType)) {
|
||||||
);
|
|
||||||
if (med.packageType === "blister") {
|
|
||||||
lines.push(item(t("report.docPacks"), String(med.packCount)));
|
lines.push(item(t("report.docPacks"), String(med.packCount)));
|
||||||
lines.push(item(t("report.docBlistersPerPack"), String(med.blistersPerPack)));
|
lines.push(item(t("report.docBlistersPerPack"), String(med.blistersPerPack)));
|
||||||
lines.push(item(t("report.docPillsPerBlister"), String(med.pillsPerBlister)));
|
lines.push(item(t("report.docPillsPerBlister"), String(med.pillsPerBlister)));
|
||||||
if (med.looseTablets > 0) lines.push(item(t("report.docLoosePills"), String(med.looseTablets)));
|
if (med.looseTablets > 0) lines.push(item(t("report.docLoosePills"), String(med.looseTablets)));
|
||||||
} else {
|
} else {
|
||||||
lines.push(item(t("report.docTotalCapacity"), String(med.totalPills ?? med.looseTablets)));
|
lines.push(item(getTotalCapacityLabel(med, t), String(med.totalPills ?? med.looseTablets)));
|
||||||
}
|
}
|
||||||
lines.push(item(t("report.docCurrentStock"), `${getPackageSize(med)} ${t("common.pills")}`));
|
lines.push(item(t("report.docCurrentStock"), getCurrentStockText(med, t)));
|
||||||
if (med.pillWeightMg) lines.push(item(t("report.docDosePerPill"), `${med.pillWeightMg} ${med.doseUnit ?? "mg"}`));
|
if (!isTubePackageType(med.packageType) && !isLiquidContainerPackageType(med.packageType) && med.pillWeightMg)
|
||||||
|
lines.push(item(t("report.docDosePerPill"), `${med.pillWeightMg} ${med.doseUnit ?? "mg"}`));
|
||||||
if (med.expiryDate) lines.push(item(t("report.docExpiryDate"), fmtDate(med.expiryDate)));
|
if (med.expiryDate) lines.push(item(t("report.docExpiryDate"), fmtDate(med.expiryDate)));
|
||||||
if (med.notes) lines.push(item(t("report.docNotes"), med.notes));
|
if (med.notes) lines.push(item(t("report.docNotes"), med.notes));
|
||||||
lines.push("");
|
lines.push("");
|
||||||
@@ -363,7 +403,7 @@ function generateTextReport(
|
|||||||
if (intakes?.length) {
|
if (intakes?.length) {
|
||||||
lines.push(h3(t("report.docIntakeSchedule")));
|
lines.push(h3(t("report.docIntakeSchedule")));
|
||||||
for (const intake of intakes) {
|
for (const intake of intakes) {
|
||||||
let entry = `${intake.usage} ${intake.usage === 1 ? t("common.pill") : t("common.pills")}`;
|
let entry = getUsageText(med, intake.usage, t);
|
||||||
entry += ` ${intake.every === 1 ? t("common.daily") : t("common.everyNDays", { count: intake.every })}`;
|
entry += ` ${intake.every === 1 ? t("common.daily") : t("common.everyNDays", { count: intake.every })}`;
|
||||||
entry += ` ${t("form.blisters.from")} ${fmtDateTime(intake.start)}`;
|
entry += ` ${t("form.blisters.from")} ${fmtDateTime(intake.start)}`;
|
||||||
if ("takenBy" in intake && intake.takenBy)
|
if ("takenBy" in intake && intake.takenBy)
|
||||||
@@ -405,7 +445,7 @@ function generateTextReport(
|
|||||||
if (data.refills.length > 0) {
|
if (data.refills.length > 0) {
|
||||||
lines.push(h3(t("report.docRefillHistory")));
|
lines.push(h3(t("report.docRefillHistory")));
|
||||||
for (const r of data.refills) {
|
for (const r of data.refills) {
|
||||||
let entry = `${fmtDate(r.refillDate)}: +${r.packsAdded} ${t("report.docPacks")}, +${r.loosePillsAdded} ${t("common.pills")}`;
|
let entry = `${fmtDate(r.refillDate)}: +${r.packsAdded} ${t("report.docPacks")}, +${r.loosePillsAdded} ${isTubePackageType(med.packageType) || isLiquidContainerPackageType(med.packageType) ? t(getTubeUnitKey(med)) : t("common.pills")}`;
|
||||||
if (r.usedPrescription) entry += ` ${t("report.docRefillPrescription")}`;
|
if (r.usedPrescription) entry += ` ${t("report.docRefillPrescription")}`;
|
||||||
lines.push(fmt === "md" ? `- ${entry}` : ` • ${entry}`);
|
lines.push(fmt === "md" ? `- ${entry}` : ` • ${entry}`);
|
||||||
}
|
}
|
||||||
@@ -489,22 +529,24 @@ function buildPrintHtml(
|
|||||||
for (const med of meds) {
|
for (const med of meds) {
|
||||||
const data = reportData[med.id];
|
const data = reportData[med.id];
|
||||||
const intakes = med.intakes ?? med.blisters;
|
const intakes = med.intakes ?? med.blisters;
|
||||||
|
const displayName = getMedDisplayName(med);
|
||||||
const title = med.isObsolete
|
const title = med.isObsolete
|
||||||
? `${escHtml(med.name)} <span class="obsolete-badge">${escHtml(t("report.docStatusObsolete"))}</span>`
|
? `${escHtml(displayName)} <span class="obsolete-badge">${escHtml(t("report.docStatusObsolete"))}</span>`
|
||||||
: escHtml(med.name);
|
: escHtml(displayName);
|
||||||
|
|
||||||
let s = `<div class="med-section">`;
|
let s = `<div class="med-section">`;
|
||||||
const imgDataUrl = imageMap[med.id];
|
const imgDataUrl = imageMap[med.id];
|
||||||
|
|
||||||
// Title with generic name subtitle
|
// Title with generic name subtitle
|
||||||
s += `<h2>${title}</h2>`;
|
s += `<h2>${title}</h2>`;
|
||||||
if (med.genericName) s += `<p class="generic-subtitle">${escHtml(med.genericName)}</p>`;
|
if (med.name && med.genericName) s += `<p class="generic-subtitle">${escHtml(med.genericName)}</p>`;
|
||||||
|
|
||||||
// Build general info table rows
|
// Build general info table rows
|
||||||
const generalRows: string[] = [];
|
const generalRows: string[] = [];
|
||||||
generalRows.push(
|
if (med.name)
|
||||||
`<tr><td class="label">${escHtml(t("report.docCommercialName"))}</td><td>${escHtml(med.name)}</td></tr>`
|
generalRows.push(
|
||||||
);
|
`<tr><td class="label">${escHtml(t("report.docCommercialName"))}</td><td>${escHtml(med.name)}</td></tr>`
|
||||||
|
);
|
||||||
if (med.genericName)
|
if (med.genericName)
|
||||||
generalRows.push(
|
generalRows.push(
|
||||||
`<tr><td class="label">${escHtml(t("report.docGenericName"))}</td><td>${escHtml(med.genericName)}</td></tr>`
|
`<tr><td class="label">${escHtml(t("report.docGenericName"))}</td><td>${escHtml(med.genericName)}</td></tr>`
|
||||||
@@ -527,7 +569,7 @@ function buildPrintHtml(
|
|||||||
const generalTable = `<h3>${escHtml(t("report.docGeneral"))}</h3><table><tbody>${generalRows.join("")}</tbody></table>`;
|
const generalTable = `<h3>${escHtml(t("report.docGeneral"))}</h3><table><tbody>${generalRows.join("")}</tbody></table>`;
|
||||||
|
|
||||||
if (imgDataUrl) {
|
if (imgDataUrl) {
|
||||||
s += `<div class="med-overview"><img class="med-img" src="${imgDataUrl}" alt="${escHtml(med.name)}" /><div class="med-overview-info">${generalTable}</div></div>`;
|
s += `<div class="med-overview"><img class="med-img" src="${imgDataUrl}" alt="${escHtml(displayName)}" /><div class="med-overview-info">${generalTable}</div></div>`;
|
||||||
} else {
|
} else {
|
||||||
s += generalTable;
|
s += generalTable;
|
||||||
}
|
}
|
||||||
@@ -535,18 +577,18 @@ function buildPrintHtml(
|
|||||||
// Package / Stock
|
// Package / Stock
|
||||||
s += `<h3>${escHtml(t("report.docPackage"))}</h3>`;
|
s += `<h3>${escHtml(t("report.docPackage"))}</h3>`;
|
||||||
s += `<table><tbody>`;
|
s += `<table><tbody>`;
|
||||||
s += `<tr><td class="label">${escHtml(t("report.docPackageType"))}</td><td>${escHtml(med.packageType === "bottle" ? t("report.docBottle") : t("report.docBlister"))}</td></tr>`;
|
s += `<tr><td class="label">${escHtml(t("report.docPackageType"))}</td><td>${escHtml(getReportPackageTypeLabel(med, t))}</td></tr>`;
|
||||||
if (med.packageType === "blister") {
|
if (!isAmountBasedPackageType(med.packageType)) {
|
||||||
s += `<tr><td class="label">${escHtml(t("report.docPacks"))}</td><td>${med.packCount}</td></tr>`;
|
s += `<tr><td class="label">${escHtml(t("report.docPacks"))}</td><td>${med.packCount}</td></tr>`;
|
||||||
s += `<tr><td class="label">${escHtml(t("report.docBlistersPerPack"))}</td><td>${med.blistersPerPack}</td></tr>`;
|
s += `<tr><td class="label">${escHtml(t("report.docBlistersPerPack"))}</td><td>${med.blistersPerPack}</td></tr>`;
|
||||||
s += `<tr><td class="label">${escHtml(t("report.docPillsPerBlister"))}</td><td>${med.pillsPerBlister}</td></tr>`;
|
s += `<tr><td class="label">${escHtml(t("report.docPillsPerBlister"))}</td><td>${med.pillsPerBlister}</td></tr>`;
|
||||||
if (med.looseTablets > 0)
|
if (med.looseTablets > 0)
|
||||||
s += `<tr><td class="label">${escHtml(t("report.docLoosePills"))}</td><td>${med.looseTablets}</td></tr>`;
|
s += `<tr><td class="label">${escHtml(t("report.docLoosePills"))}</td><td>${med.looseTablets}</td></tr>`;
|
||||||
} else {
|
} else {
|
||||||
s += `<tr><td class="label">${escHtml(t("report.docTotalCapacity"))}</td><td>${med.totalPills ?? med.looseTablets}</td></tr>`;
|
s += `<tr><td class="label">${escHtml(getTotalCapacityLabel(med, t))}</td><td>${med.totalPills ?? med.looseTablets}</td></tr>`;
|
||||||
}
|
}
|
||||||
s += `<tr><td class="label">${escHtml(t("report.docCurrentStock"))}</td><td>${getPackageSize(med)} ${escHtml(t("common.pills"))}</td></tr>`;
|
s += `<tr><td class="label">${escHtml(t("report.docCurrentStock"))}</td><td>${escHtml(getCurrentStockText(med, t))}</td></tr>`;
|
||||||
if (med.pillWeightMg)
|
if (!isTubePackageType(med.packageType) && !isLiquidContainerPackageType(med.packageType) && med.pillWeightMg)
|
||||||
s += `<tr><td class="label">${escHtml(t("report.docDosePerPill"))}</td><td>${med.pillWeightMg} ${escHtml(med.doseUnit ?? "mg")}</td></tr>`;
|
s += `<tr><td class="label">${escHtml(t("report.docDosePerPill"))}</td><td>${med.pillWeightMg} ${escHtml(med.doseUnit ?? "mg")}</td></tr>`;
|
||||||
if (med.expiryDate)
|
if (med.expiryDate)
|
||||||
s += `<tr><td class="label">${escHtml(t("report.docExpiryDate"))}</td><td>${fmtDate(med.expiryDate)}</td></tr>`;
|
s += `<tr><td class="label">${escHtml(t("report.docExpiryDate"))}</td><td>${fmtDate(med.expiryDate)}</td></tr>`;
|
||||||
@@ -563,7 +605,7 @@ function buildPrintHtml(
|
|||||||
s += `<h3>${escHtml(t("report.docIntakeSchedule"))}</h3>`;
|
s += `<h3>${escHtml(t("report.docIntakeSchedule"))}</h3>`;
|
||||||
s += `<ul>`;
|
s += `<ul>`;
|
||||||
for (const intake of filteredPrintIntakes) {
|
for (const intake of filteredPrintIntakes) {
|
||||||
let entry = `${intake.usage} ${escHtml(intake.usage === 1 ? t("common.pill") : t("common.pills"))}`;
|
let entry = escHtml(getUsageText(med, intake.usage, t));
|
||||||
entry += ` ${escHtml(intake.every === 1 ? t("common.daily") : t("common.everyNDays", { count: intake.every }))}`;
|
entry += ` ${escHtml(intake.every === 1 ? t("common.daily") : t("common.everyNDays", { count: intake.every }))}`;
|
||||||
entry += ` ${escHtml(t("form.blisters.from"))} ${fmtDateTime(intake.start)}`;
|
entry += ` ${escHtml(t("form.blisters.from"))} ${fmtDateTime(intake.start)}`;
|
||||||
if ("takenBy" in intake && intake.takenBy)
|
if ("takenBy" in intake && intake.takenBy)
|
||||||
@@ -610,7 +652,7 @@ function buildPrintHtml(
|
|||||||
s += `<h3>${escHtml(t("report.docRefillHistory"))}</h3>`;
|
s += `<h3>${escHtml(t("report.docRefillHistory"))}</h3>`;
|
||||||
s += `<ul>`;
|
s += `<ul>`;
|
||||||
for (const r of data.refills) {
|
for (const r of data.refills) {
|
||||||
let entry = `${fmtDate(r.refillDate)}: +${r.packsAdded} ${escHtml(t("report.docPacks"))}, +${r.loosePillsAdded} ${escHtml(t("common.pills"))}`;
|
let entry = `${fmtDate(r.refillDate)}: +${r.packsAdded} ${escHtml(t("report.docPacks"))}, +${r.loosePillsAdded} ${escHtml(isTubePackageType(med.packageType) || isLiquidContainerPackageType(med.packageType) ? t(getTubeUnitKey(med)) : t("common.pills"))}`;
|
||||||
if (r.usedPrescription) entry += ` <em>${escHtml(t("report.docRefillPrescription"))}</em>`;
|
if (r.usedPrescription) entry += ` <em>${escHtml(t("report.docRefillPrescription"))}</em>`;
|
||||||
s += `<li>${entry}</li>`;
|
s += `<li>${entry}</li>`;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,7 +5,6 @@
|
|||||||
|
|
||||||
import { Check, Copy, Link2, X } from "lucide-react";
|
import { Check, Copy, Link2, X } from "lucide-react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { useEscapeKey } from "../hooks/useEscapeKey";
|
|
||||||
|
|
||||||
export interface ShareDialogProps {
|
export interface ShareDialogProps {
|
||||||
show: boolean;
|
show: boolean;
|
||||||
@@ -44,7 +43,7 @@ export function ShareDialog({
|
|||||||
const closeLabel = t("common.close");
|
const closeLabel = t("common.close");
|
||||||
const copyLabel = shareCopied ? t("share.copied") : t("share.copyLink");
|
const copyLabel = shareCopied ? t("share.copied") : t("share.copyLink");
|
||||||
|
|
||||||
useEscapeKey(show, onClose);
|
// ESC is handled by the global handler in App.tsx to avoid double history.back()
|
||||||
|
|
||||||
if (!show) return null;
|
if (!show) return null;
|
||||||
|
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ import { useTranslation } from "react-i18next";
|
|||||||
import { useParams } from "react-router-dom";
|
import { useParams } from "react-router-dom";
|
||||||
import { useEscapeKey } from "../hooks";
|
import { useEscapeKey } from "../hooks";
|
||||||
import type { ExpiredLinkData, SharedScheduleData } from "../types";
|
import type { ExpiredLinkData, SharedScheduleData } from "../types";
|
||||||
import { getMedTotal } from "../types";
|
import { getMedDisplayName, getMedTotal, isLiquidContainerPackageType, isTubePackageType } from "../types";
|
||||||
import { getSystemLocale } from "../utils/formatters";
|
import { getSystemLocale } from "../utils/formatters";
|
||||||
import { isDoseDismissed } from "../utils/schedule";
|
import { isDoseDismissed } from "../utils/schedule";
|
||||||
import { loadCollapsedDaysFromStorage } from "../utils/storage";
|
import { loadCollapsedDaysFromStorage } from "../utils/storage";
|
||||||
@@ -21,11 +21,20 @@ import { MedicationAvatar } from "./MedicationAvatar";
|
|||||||
function getStockStatus(
|
function getStockStatus(
|
||||||
daysLeft: number | null,
|
daysLeft: number | null,
|
||||||
medsLeft: number,
|
medsLeft: number,
|
||||||
thresholds: { lowStockDays: number; normalStockDays: number; highStockDays: number; reminderDaysBefore: number }
|
thresholds: { lowStockDays: number; normalStockDays: number; highStockDays: number; criticalStockDays: number },
|
||||||
|
packageType?: string
|
||||||
) {
|
) {
|
||||||
|
if (isTubePackageType(packageType)) return { className: "success", label: "status.noSchedule" };
|
||||||
if (medsLeft <= 0 || daysLeft === 0) return { className: "danger", label: "status.outOfStock" };
|
if (medsLeft <= 0 || daysLeft === 0) return { className: "danger", label: "status.outOfStock" };
|
||||||
if (daysLeft === null) return { className: "success", label: "status.noSchedule" };
|
if (daysLeft === null) return { className: "success", label: "status.noSchedule" };
|
||||||
if (daysLeft <= thresholds.reminderDaysBefore) return { className: "danger", label: "status.criticalStock" };
|
if (isLiquidContainerPackageType(packageType)) {
|
||||||
|
const lowDays = Math.max(1, Math.floor(thresholds.criticalStockDays));
|
||||||
|
const criticalDays = Math.max(1, Math.ceil(lowDays / 2));
|
||||||
|
if (daysLeft <= criticalDays) return { className: "danger", label: "status.criticalStock" };
|
||||||
|
if (daysLeft <= lowDays) return { className: "warning", label: "status.lowStock" };
|
||||||
|
return { className: "success", label: "status.normal" };
|
||||||
|
}
|
||||||
|
if (daysLeft <= thresholds.criticalStockDays) return { className: "danger", label: "status.criticalStock" };
|
||||||
if (daysLeft < thresholds.lowStockDays) return { className: "warning", label: "status.lowStock" };
|
if (daysLeft < thresholds.lowStockDays) return { className: "warning", label: "status.lowStock" };
|
||||||
if (daysLeft >= thresholds.highStockDays) return { className: "high", label: "status.highStock" };
|
if (daysLeft >= thresholds.highStockDays) return { className: "high", label: "status.highStock" };
|
||||||
return { className: "success", label: "status.normal" };
|
return { className: "success", label: "status.normal" };
|
||||||
@@ -44,6 +53,100 @@ export function SharedSchedule() {
|
|||||||
const [showPastDays, setShowPastDays] = useState(false);
|
const [showPastDays, setShowPastDays] = useState(false);
|
||||||
const [showFutureDays, setShowFutureDays] = useState(false);
|
const [showFutureDays, setShowFutureDays] = useState(false);
|
||||||
|
|
||||||
|
const isLiquidContainerMed = (med: SharedScheduleData["medications"][number] | undefined) =>
|
||||||
|
isLiquidContainerPackageType(med?.packageType);
|
||||||
|
|
||||||
|
const convertLiquidUsageToMl = (usage: number, unit: "ml" | "tsp" | "tbsp" | null | undefined): number => {
|
||||||
|
if (unit === "tsp") return usage * 5;
|
||||||
|
if (unit === "tbsp") return usage * 15;
|
||||||
|
return usage;
|
||||||
|
};
|
||||||
|
|
||||||
|
const convertUsageForStock = (
|
||||||
|
usage: number,
|
||||||
|
med: SharedScheduleData["medications"][number] | undefined,
|
||||||
|
unit: "ml" | "tsp" | "tbsp" | null | undefined
|
||||||
|
): number => {
|
||||||
|
if (isTubePackageType(med?.packageType)) return 0;
|
||||||
|
if (!isLiquidContainerMed(med)) return usage;
|
||||||
|
return convertLiquidUsageToMl(usage, unit);
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatAmount = (value: number) => {
|
||||||
|
const rounded = Math.round(value * 100) / 100;
|
||||||
|
return String(rounded);
|
||||||
|
};
|
||||||
|
|
||||||
|
const getLiquidCountUnitLabel = (unit: "ml" | "tsp" | "tbsp" | null | undefined, usage: number): string => {
|
||||||
|
if (unit === "tsp") return t("form.blisters.teaspoons", { count: Math.abs(usage) });
|
||||||
|
if (unit === "tbsp") return t("form.blisters.tablespoons", { count: Math.abs(usage) });
|
||||||
|
return t("form.packageAmountUnitMl");
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatLiquidUsageLabel = (usage: number, unit: "ml" | "tsp" | "tbsp" | null | undefined): string => {
|
||||||
|
const normalizedUsage = Number(usage);
|
||||||
|
if (!Number.isFinite(normalizedUsage) || normalizedUsage <= 0) {
|
||||||
|
return `0 ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (unit === "ml" || unit == null) {
|
||||||
|
return `${formatAmount(normalizedUsage)} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const mlTotal = convertLiquidUsageToMl(normalizedUsage, unit);
|
||||||
|
return `${formatAmount(normalizedUsage)} ${getLiquidCountUnitLabel(unit, normalizedUsage)} ${formatAmount(mlTotal)} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatDoseUsageLabel = (
|
||||||
|
med: SharedScheduleData["medications"][number] | undefined,
|
||||||
|
usage: number,
|
||||||
|
intakeUnit?: "ml" | "tsp" | "tbsp" | null
|
||||||
|
) => {
|
||||||
|
if (isLiquidContainerMed(med)) {
|
||||||
|
return formatLiquidUsageLabel(usage, intakeUnit);
|
||||||
|
}
|
||||||
|
return `${usage} ${usage !== 1 ? t("common.pills") : t("common.pill")}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatTotalUsageLabel = (
|
||||||
|
med: SharedScheduleData["medications"][number] | undefined,
|
||||||
|
total: number,
|
||||||
|
doses?: Array<{ usage: number; intakeUnit?: "ml" | "tsp" | "tbsp" | null }>
|
||||||
|
) => {
|
||||||
|
if (isLiquidContainerMed(med)) {
|
||||||
|
if (doses && doses.length > 0) {
|
||||||
|
const normalizedDoses = doses.filter((dose) => Number.isFinite(Number(dose.usage)) && Number(dose.usage) > 0);
|
||||||
|
if (normalizedDoses.length > 0) {
|
||||||
|
const allUnits = new Set(normalizedDoses.map((dose) => dose.intakeUnit ?? "ml"));
|
||||||
|
if (allUnits.size === 1) {
|
||||||
|
const onlyUnit = normalizedDoses[0]?.intakeUnit ?? "ml";
|
||||||
|
const totalUsageInUnit = normalizedDoses.reduce((sum, dose) => sum + Number(dose.usage), 0);
|
||||||
|
return formatLiquidUsageLabel(totalUsageInUnit, onlyUnit);
|
||||||
|
}
|
||||||
|
|
||||||
|
const totalMl = normalizedDoses.reduce(
|
||||||
|
(sum, dose) => sum + convertLiquidUsageToMl(Number(dose.usage), dose.intakeUnit ?? "ml"),
|
||||||
|
0
|
||||||
|
);
|
||||||
|
return `${formatAmount(totalMl)} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return `${formatAmount(total)} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return t("common.pillsTotal", { count: total });
|
||||||
|
};
|
||||||
|
|
||||||
|
const shouldHideNoScheduleStatusForTube = (
|
||||||
|
med: SharedScheduleData["medications"][number] | undefined,
|
||||||
|
status: { className: string; label: string } | null
|
||||||
|
) => isTubePackageType(med?.packageType) && status?.label === "status.noSchedule";
|
||||||
|
|
||||||
|
const getVisibleStockStatus = (
|
||||||
|
med: SharedScheduleData["medications"][number] | undefined,
|
||||||
|
status: { className: string; label: string } | null
|
||||||
|
) => (shouldHideNoScheduleStatusForTube(med, status) ? null : status);
|
||||||
|
|
||||||
// Theme preference: light, dark, or system
|
// Theme preference: light, dark, or system
|
||||||
type ThemePreference = "light" | "dark" | "system";
|
type ThemePreference = "light" | "dark" | "system";
|
||||||
const [themePreference, setThemePreference] = useState<ThemePreference>(() => {
|
const [themePreference, setThemePreference] = useState<ThemePreference>(() => {
|
||||||
@@ -309,6 +412,7 @@ export function SharedSchedule() {
|
|||||||
when: number;
|
when: number;
|
||||||
medName: string;
|
medName: string;
|
||||||
usage: number;
|
usage: number;
|
||||||
|
intakeUnit?: "ml" | "tsp" | "tbsp" | null;
|
||||||
timeStr: string;
|
timeStr: string;
|
||||||
isPast: boolean;
|
isPast: boolean;
|
||||||
takenBy: string | null; // Per-intake takenBy (single person or null)
|
takenBy: string | null; // Per-intake takenBy (single person or null)
|
||||||
@@ -319,7 +423,12 @@ export function SharedSchedule() {
|
|||||||
// Use intakes (with per-intake takenBy) if available, fallback to blisters (legacy)
|
// Use intakes (with per-intake takenBy) if available, fallback to blisters (legacy)
|
||||||
const intakes =
|
const intakes =
|
||||||
med.intakes ||
|
med.intakes ||
|
||||||
med.blisters.map((b) => ({ ...b, takenBy: null as string | null, intakeRemindersEnabled: false }));
|
med.blisters.map((b) => ({
|
||||||
|
...b,
|
||||||
|
intakeUnit: null,
|
||||||
|
takenBy: null as string | null,
|
||||||
|
intakeRemindersEnabled: false,
|
||||||
|
}));
|
||||||
|
|
||||||
intakes.forEach((intake, intakeIdx) => {
|
intakes.forEach((intake, intakeIdx) => {
|
||||||
// Filter: only include intakes for this person (null = everyone, or matches share's takenBy)
|
// Filter: only include intakes for this person (null = everyone, or matches share's takenBy)
|
||||||
@@ -343,8 +452,9 @@ export function SharedSchedule() {
|
|||||||
doses.push({
|
doses.push({
|
||||||
id: doseId,
|
id: doseId,
|
||||||
when: t,
|
when: t,
|
||||||
medName: med.name,
|
medName: getMedDisplayName(med),
|
||||||
usage: intake.usage,
|
usage: intake.usage,
|
||||||
|
intakeUnit: intake.intakeUnit ?? null,
|
||||||
isPast,
|
isPast,
|
||||||
takenBy: intake.takenBy, // Per-intake takenBy (string | null)
|
takenBy: intake.takenBy, // Per-intake takenBy (string | null)
|
||||||
timeStr: d.toLocaleTimeString(getSystemLocale(i18n.language), { hour: "2-digit", minute: "2-digit" }),
|
timeStr: d.toLocaleTimeString(getSystemLocale(i18n.language), { hour: "2-digit", minute: "2-digit" }),
|
||||||
@@ -430,8 +540,14 @@ export function SharedSchedule() {
|
|||||||
const depletion: Record<string, number | null> = {};
|
const depletion: Record<string, number | null> = {};
|
||||||
|
|
||||||
for (const med of data.medications) {
|
for (const med of data.medications) {
|
||||||
const intakes = med.intakes || med.blisters.map((b) => ({ ...b, takenBy: null as string | null }));
|
const intakes =
|
||||||
const blisters = med.blisters;
|
med.intakes ||
|
||||||
|
med.blisters.map((b) => ({
|
||||||
|
...b,
|
||||||
|
intakeUnit: null,
|
||||||
|
takenBy: null as string | null,
|
||||||
|
intakeRemindersEnabled: false,
|
||||||
|
}));
|
||||||
|
|
||||||
// Count unique people from all intakes (for per-intake takenBy)
|
// Count unique people from all intakes (for per-intake takenBy)
|
||||||
const uniquePeople = new Set<string>();
|
const uniquePeople = new Set<string>();
|
||||||
@@ -443,9 +559,9 @@ export function SharedSchedule() {
|
|||||||
|
|
||||||
// Calculate daily consumption rate accounting for per-intake takenBy
|
// Calculate daily consumption rate accounting for per-intake takenBy
|
||||||
let dailyRate = 0;
|
let dailyRate = 0;
|
||||||
blisters.forEach((s, idx) => {
|
intakes.forEach((intake) => {
|
||||||
const baseRate = s.every > 0 ? s.usage / s.every : 0;
|
const usageForStock = convertUsageForStock(intake.usage, med, intake.intakeUnit ?? "ml");
|
||||||
const intake = intakes[idx];
|
const baseRate = intake.every > 0 ? usageForStock / intake.every : 0;
|
||||||
if (intake?.takenBy) {
|
if (intake?.takenBy) {
|
||||||
dailyRate += baseRate; // Per-intake takenBy: 1 person
|
dailyRate += baseRate; // Per-intake takenBy: 1 person
|
||||||
} else {
|
} else {
|
||||||
@@ -458,9 +574,10 @@ export function SharedSchedule() {
|
|||||||
|
|
||||||
if (calcMode === "automatic") {
|
if (calcMode === "automatic") {
|
||||||
// Time-based: every scheduled dose counts as consumed once its time has passed
|
// Time-based: every scheduled dose counts as consumed once its time has passed
|
||||||
blisters.forEach((s, blisterIdx) => {
|
intakes.forEach((intake, blisterIdx) => {
|
||||||
const blisterStart = new Date(s.start).getTime();
|
const usageForStock = convertUsageForStock(intake.usage, med, intake.intakeUnit ?? "ml");
|
||||||
const period = Math.max(1, s.every) * MS_PER_DAY;
|
const blisterStart = new Date(intake.start).getTime();
|
||||||
|
const period = Math.max(1, intake.every) * MS_PER_DAY;
|
||||||
|
|
||||||
let effectiveStart: number;
|
let effectiveStart: number;
|
||||||
if (stockCorrectionCutoff > 0 && stockCorrectionCutoff >= blisterStart) {
|
if (stockCorrectionCutoff > 0 && stockCorrectionCutoff >= blisterStart) {
|
||||||
@@ -472,7 +589,6 @@ export function SharedSchedule() {
|
|||||||
}
|
}
|
||||||
if (Number.isNaN(effectiveStart)) return;
|
if (Number.isNaN(effectiveStart)) return;
|
||||||
|
|
||||||
const intake = intakes[blisterIdx];
|
|
||||||
const intakePerson = intake?.takenBy;
|
const intakePerson = intake?.takenBy;
|
||||||
const fallbackPeople = med.takenBy?.length > 0 ? med.takenBy : [null];
|
const fallbackPeople = med.takenBy?.length > 0 ? med.takenBy : [null];
|
||||||
const peopleForThisIntake = intakePerson ? [intakePerson] : fallbackPeople;
|
const peopleForThisIntake = intakePerson ? [intakePerson] : fallbackPeople;
|
||||||
@@ -482,7 +598,7 @@ export function SharedSchedule() {
|
|||||||
|
|
||||||
if (effectiveStart <= now) {
|
if (effectiveStart <= now) {
|
||||||
const occurrences = Math.floor((now - effectiveStart) / period) + 1;
|
const occurrences = Math.floor((now - effectiveStart) / period) + 1;
|
||||||
timeBasedConsumed = occurrences * s.usage * peopleForThisIntake.length;
|
timeBasedConsumed = occurrences * usageForStock * peopleForThisIntake.length;
|
||||||
const lastDoseTime = new Date(effectiveStart + (occurrences - 1) * period);
|
const lastDoseTime = new Date(effectiveStart + (occurrences - 1) * period);
|
||||||
lastAutoConsumedDateMs = new Date(
|
lastAutoConsumedDateMs = new Date(
|
||||||
lastDoseTime.getFullYear(),
|
lastDoseTime.getFullYear(),
|
||||||
@@ -510,7 +626,7 @@ export function SharedSchedule() {
|
|||||||
const bIdx = parseInt(parts[1], 10);
|
const bIdx = parseInt(parts[1], 10);
|
||||||
const timestamp = parseInt(parts[2], 10);
|
const timestamp = parseInt(parts[2], 10);
|
||||||
if (medId === med.id && bIdx === blisterIdx && timestamp > earlyCutoff) {
|
if (medId === med.id && bIdx === blisterIdx && timestamp > earlyCutoff) {
|
||||||
earlyTakenConsumed += s.usage;
|
earlyTakenConsumed += usageForStock;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -525,8 +641,8 @@ export function SharedSchedule() {
|
|||||||
const medId = parseInt(parts[0], 10);
|
const medId = parseInt(parts[0], 10);
|
||||||
const blisterIdx = parseInt(parts[1], 10);
|
const blisterIdx = parseInt(parts[1], 10);
|
||||||
const doseTimestamp = parseInt(parts[2], 10);
|
const doseTimestamp = parseInt(parts[2], 10);
|
||||||
if (medId === med.id && blisters[blisterIdx]) {
|
if (medId === med.id && intakes[blisterIdx]) {
|
||||||
const blisterStartDate = new Date(blisters[blisterIdx].start);
|
const blisterStartDate = new Date(intakes[blisterIdx].start);
|
||||||
const blisterStartDateOnly = new Date(
|
const blisterStartDateOnly = new Date(
|
||||||
blisterStartDate.getFullYear(),
|
blisterStartDate.getFullYear(),
|
||||||
blisterStartDate.getMonth(),
|
blisterStartDate.getMonth(),
|
||||||
@@ -534,7 +650,11 @@ export function SharedSchedule() {
|
|||||||
).getTime();
|
).getTime();
|
||||||
const afterCorrection = stockCorrectionCutoff === 0 || doseTimestamp > stockCorrectionCutoff;
|
const afterCorrection = stockCorrectionCutoff === 0 || doseTimestamp > stockCorrectionCutoff;
|
||||||
if (!Number.isNaN(blisterStartDateOnly) && doseTimestamp >= blisterStartDateOnly && afterCorrection) {
|
if (!Number.isNaN(blisterStartDateOnly) && doseTimestamp >= blisterStartDateOnly && afterCorrection) {
|
||||||
consumed += blisters[blisterIdx].usage;
|
consumed += convertUsageForStock(
|
||||||
|
intakes[blisterIdx].usage,
|
||||||
|
med,
|
||||||
|
intakes[blisterIdx].intakeUnit ?? "ml"
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -547,8 +667,8 @@ export function SharedSchedule() {
|
|||||||
const daysLeft = rawDaysLeft !== null ? Math.max(0, Math.floor(rawDaysLeft)) : null;
|
const daysLeft = rawDaysLeft !== null ? Math.max(0, Math.floor(rawDaysLeft)) : null;
|
||||||
const depletionMs = daysLeft !== null ? now + daysLeft * MS_PER_DAY : null;
|
const depletionMs = daysLeft !== null ? now + daysLeft * MS_PER_DAY : null;
|
||||||
|
|
||||||
coverage[med.name] = { daysLeft, medsLeft: Number(medsLeft.toFixed(1)), dailyUsage: dailyRate };
|
coverage[getMedDisplayName(med)] = { daysLeft, medsLeft: Number(medsLeft.toFixed(1)), dailyUsage: dailyRate };
|
||||||
depletion[med.name] = depletionMs;
|
depletion[getMedDisplayName(med)] = depletionMs;
|
||||||
}
|
}
|
||||||
return { coverageByMed: coverage, depletionByMed: depletion };
|
return { coverageByMed: coverage, depletionByMed: depletion };
|
||||||
}, [data, takenDoses]);
|
}, [data, takenDoses]);
|
||||||
@@ -569,11 +689,13 @@ export function SharedSchedule() {
|
|||||||
function getDayStockStatus(meds: { medName: string; lastWhen: number }[]) {
|
function getDayStockStatus(meds: { medName: string; lastWhen: number }[]) {
|
||||||
const statuses = meds.map((item) => {
|
const statuses = meds.map((item) => {
|
||||||
const coverage = coverageByMed[item.medName];
|
const coverage = coverageByMed[item.medName];
|
||||||
|
const med = data?.medications.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
const depletionTime = depletionByMed[item.medName];
|
const depletionTime = depletionByMed[item.medName];
|
||||||
if (typeof depletionTime === "number" && item.lastWhen > depletionTime) return "danger";
|
if (typeof depletionTime === "number" && item.lastWhen > depletionTime) return "danger";
|
||||||
if (!coverage) return "success";
|
if (!coverage) return "success";
|
||||||
const status = getStockStatus(coverage.daysLeft, coverage.medsLeft, stockThresholds);
|
const rawStatus = getStockStatus(coverage.daysLeft, coverage.medsLeft, stockThresholds, med?.packageType);
|
||||||
return status.className;
|
const status = getVisibleStockStatus(med, rawStatus);
|
||||||
|
return status?.className ?? "success";
|
||||||
});
|
});
|
||||||
const fallbackStatus = statuses.includes("warning") ? "warning" : "success";
|
const fallbackStatus = statuses.includes("warning") ? "warning" : "success";
|
||||||
return statuses.includes("danger") ? "danger" : fallbackStatus;
|
return statuses.includes("danger") ? "danger" : fallbackStatus;
|
||||||
@@ -583,6 +705,11 @@ export function SharedSchedule() {
|
|||||||
const showStock = data?.shareStockStatus !== false;
|
const showStock = data?.shareStockStatus !== false;
|
||||||
const showOnlyToday = data?.shareScheduleTodayOnly === true && (data?.upcomingTodayOnly ?? true);
|
const showOnlyToday = data?.shareScheduleTodayOnly === true && (data?.upcomingTodayOnly ?? true);
|
||||||
|
|
||||||
|
const renderDoseUsage = (
|
||||||
|
med: SharedScheduleData["medications"][number] | undefined,
|
||||||
|
dose: { usage: number; intakeUnit?: "ml" | "tsp" | "tbsp" | null }
|
||||||
|
) => formatDoseUsageLabel(med, dose.usage, dose.intakeUnit);
|
||||||
|
|
||||||
// Helper: check if a dose is "done" (taken, per-dose dismissed, or med-level dismissed)
|
// Helper: check if a dose is "done" (taken, per-dose dismissed, or med-level dismissed)
|
||||||
function isDoseIdDone(doseId: string): boolean {
|
function isDoseIdDone(doseId: string): boolean {
|
||||||
if (takenDoses.has(doseId)) return true;
|
if (takenDoses.has(doseId)) return true;
|
||||||
@@ -746,7 +873,7 @@ export function SharedSchedule() {
|
|||||||
|
|
||||||
// Count missed doses that are NOT dismissed (for warning icon)
|
// Count missed doses that are NOT dismissed (for warning icon)
|
||||||
const missedNotDismissedCount = day.meds.reduce((count, item) => {
|
const missedNotDismissedCount = day.meds.reduce((count, item) => {
|
||||||
const med = data.medications.find((m) => m.name === item.medName);
|
const med = data.medications.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
const dismissedUntilDate = med?.dismissedUntil ?? undefined;
|
const dismissedUntilDate = med?.dismissedUntil ?? undefined;
|
||||||
return (
|
return (
|
||||||
count +
|
count +
|
||||||
@@ -800,7 +927,7 @@ export function SharedSchedule() {
|
|||||||
</div>
|
</div>
|
||||||
{!isCollapsed &&
|
{!isCollapsed &&
|
||||||
day.meds.map((item) => {
|
day.meds.map((item) => {
|
||||||
const med = data.medications.find((m) => m.name === item.medName);
|
const med = data.medications.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
const medCoverage = coverageByMed[item.medName];
|
const medCoverage = coverageByMed[item.medName];
|
||||||
const isEmpty = showStock && medCoverage ? medCoverage.medsLeft <= 0 : false;
|
const isEmpty = showStock && medCoverage ? medCoverage.medsLeft <= 0 : false;
|
||||||
const depletionTime = depletionByMed[item.medName];
|
const depletionTime = depletionByMed[item.medName];
|
||||||
@@ -809,9 +936,15 @@ export function SharedSchedule() {
|
|||||||
? willBeOutOfStock
|
? willBeOutOfStock
|
||||||
? { className: "danger", label: "status.outOfStock" }
|
? { className: "danger", label: "status.outOfStock" }
|
||||||
: medCoverage
|
: medCoverage
|
||||||
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds)
|
? getStockStatus(
|
||||||
|
medCoverage.daysLeft,
|
||||||
|
medCoverage.medsLeft,
|
||||||
|
stockThresholds,
|
||||||
|
med?.packageType
|
||||||
|
)
|
||||||
: null
|
: null
|
||||||
: null;
|
: null;
|
||||||
|
const visibleStatus = getVisibleStockStatus(med, status);
|
||||||
|
|
||||||
const itemDoseIds = item.doses.map((d) => d.id);
|
const itemDoseIds = item.doses.map((d) => d.id);
|
||||||
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
||||||
@@ -825,10 +958,10 @@ export function SharedSchedule() {
|
|||||||
<div className="med-name">
|
<div className="med-name">
|
||||||
<div
|
<div
|
||||||
className={med?.imageUrl ? "med-avatar clickable" : ""}
|
className={med?.imageUrl ? "med-avatar clickable" : ""}
|
||||||
onClick={() => med?.imageUrl && openLightbox(med.imageUrl, med.name)}
|
onClick={() => med?.imageUrl && openLightbox(med.imageUrl, getMedDisplayName(med))}
|
||||||
onKeyDown={(e) => {
|
onKeyDown={(e) => {
|
||||||
if (e.key === "Enter" || e.key === " ") {
|
if (e.key === "Enter" || e.key === " ") {
|
||||||
if (med?.imageUrl) openLightbox(med.imageUrl, med.name);
|
if (med?.imageUrl) openLightbox(med.imageUrl, getMedDisplayName(med));
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
@@ -840,9 +973,13 @@ export function SharedSchedule() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="tag-row">
|
<div className="tag-row">
|
||||||
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span>
|
<span className="tag subtle">
|
||||||
{status && (
|
{formatTotalUsageLabel(med, item.total, item.doses)}
|
||||||
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span>
|
</span>
|
||||||
|
{visibleStatus && (
|
||||||
|
<span className={`status-chip small ${visibleStatus.className}`}>
|
||||||
|
{t(visibleStatus.label)}
|
||||||
|
</span>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -853,9 +990,7 @@ export function SharedSchedule() {
|
|||||||
<div key={dose.id} className="dose-item past">
|
<div key={dose.id} className="dose-item past">
|
||||||
<span className="dose-time">{dose.timeStr}</span>
|
<span className="dose-time">{dose.timeStr}</span>
|
||||||
<span className="dose-usage">
|
<span className="dose-usage">
|
||||||
<span className="dose-usage-main">
|
<span className="dose-usage-main">{renderDoseUsage(med, dose)}</span>
|
||||||
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")}
|
|
||||||
</span>
|
|
||||||
{med?.pillWeightMg && (
|
{med?.pillWeightMg && (
|
||||||
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
||||||
)}
|
)}
|
||||||
@@ -984,7 +1119,7 @@ export function SharedSchedule() {
|
|||||||
</div>
|
</div>
|
||||||
{!isCollapsed &&
|
{!isCollapsed &&
|
||||||
day.meds.map((item) => {
|
day.meds.map((item) => {
|
||||||
const med = data.medications.find((m) => m.name === item.medName);
|
const med = data.medications.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
const medCoverage = coverageByMed[item.medName];
|
const medCoverage = coverageByMed[item.medName];
|
||||||
const isEmpty = showStock && medCoverage ? medCoverage.medsLeft <= 0 : false;
|
const isEmpty = showStock && medCoverage ? medCoverage.medsLeft <= 0 : false;
|
||||||
const depletionTime = depletionByMed[item.medName];
|
const depletionTime = depletionByMed[item.medName];
|
||||||
@@ -993,9 +1128,15 @@ export function SharedSchedule() {
|
|||||||
? willBeOutOfStock
|
? willBeOutOfStock
|
||||||
? { className: "danger", label: "status.outOfStock" }
|
? { className: "danger", label: "status.outOfStock" }
|
||||||
: medCoverage
|
: medCoverage
|
||||||
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds)
|
? getStockStatus(
|
||||||
|
medCoverage.daysLeft,
|
||||||
|
medCoverage.medsLeft,
|
||||||
|
stockThresholds,
|
||||||
|
med?.packageType
|
||||||
|
)
|
||||||
: null
|
: null
|
||||||
: null;
|
: null;
|
||||||
|
const visibleStatus = getVisibleStockStatus(med, status);
|
||||||
|
|
||||||
const itemDoseIds = item.doses.map((d) => d.id);
|
const itemDoseIds = item.doses.map((d) => d.id);
|
||||||
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
||||||
@@ -1008,10 +1149,10 @@ export function SharedSchedule() {
|
|||||||
<div className="med-name">
|
<div className="med-name">
|
||||||
<div
|
<div
|
||||||
className={med?.imageUrl ? "med-avatar clickable" : ""}
|
className={med?.imageUrl ? "med-avatar clickable" : ""}
|
||||||
onClick={() => med?.imageUrl && openLightbox(med.imageUrl, med.name)}
|
onClick={() => med?.imageUrl && openLightbox(med.imageUrl, getMedDisplayName(med))}
|
||||||
onKeyDown={(e) => {
|
onKeyDown={(e) => {
|
||||||
if (e.key === "Enter" || e.key === " ") {
|
if (e.key === "Enter" || e.key === " ") {
|
||||||
if (med?.imageUrl) openLightbox(med.imageUrl, med.name);
|
if (med?.imageUrl) openLightbox(med.imageUrl, getMedDisplayName(med));
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
@@ -1023,9 +1164,13 @@ export function SharedSchedule() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="tag-row">
|
<div className="tag-row">
|
||||||
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span>
|
<span className="tag subtle">
|
||||||
{status && (
|
{formatTotalUsageLabel(med, item.total, item.doses)}
|
||||||
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span>
|
</span>
|
||||||
|
{visibleStatus && (
|
||||||
|
<span className={`status-chip small ${visibleStatus.className}`}>
|
||||||
|
{t(visibleStatus.label)}
|
||||||
|
</span>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -1040,9 +1185,7 @@ export function SharedSchedule() {
|
|||||||
>
|
>
|
||||||
<span className="dose-time">{dose.timeStr}</span>
|
<span className="dose-time">{dose.timeStr}</span>
|
||||||
<span className="dose-usage">
|
<span className="dose-usage">
|
||||||
<span className="dose-usage-main">
|
<span className="dose-usage-main">{renderDoseUsage(med, dose)}</span>
|
||||||
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")}
|
|
||||||
</span>
|
|
||||||
{med?.pillWeightMg && (
|
{med?.pillWeightMg && (
|
||||||
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
||||||
)}
|
)}
|
||||||
@@ -1161,7 +1304,7 @@ export function SharedSchedule() {
|
|||||||
</div>
|
</div>
|
||||||
{!isCollapsed &&
|
{!isCollapsed &&
|
||||||
day.meds.map((item) => {
|
day.meds.map((item) => {
|
||||||
const med = data.medications.find((m) => m.name === item.medName);
|
const med = data.medications.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
const medCoverage = coverageByMed[item.medName];
|
const medCoverage = coverageByMed[item.medName];
|
||||||
const depletionTime = depletionByMed[item.medName];
|
const depletionTime = depletionByMed[item.medName];
|
||||||
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
|
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
|
||||||
@@ -1169,9 +1312,15 @@ export function SharedSchedule() {
|
|||||||
? willBeOutOfStock
|
? willBeOutOfStock
|
||||||
? { className: "danger", label: "status.outOfStock" }
|
? { className: "danger", label: "status.outOfStock" }
|
||||||
: medCoverage
|
: medCoverage
|
||||||
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds)
|
? getStockStatus(
|
||||||
|
medCoverage.daysLeft,
|
||||||
|
medCoverage.medsLeft,
|
||||||
|
stockThresholds,
|
||||||
|
med?.packageType
|
||||||
|
)
|
||||||
: null
|
: null
|
||||||
: null;
|
: null;
|
||||||
|
const visibleStatus = getVisibleStockStatus(med, status);
|
||||||
|
|
||||||
const itemDoseIds = item.doses.map((d) => d.id);
|
const itemDoseIds = item.doses.map((d) => d.id);
|
||||||
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
||||||
@@ -1184,10 +1333,10 @@ export function SharedSchedule() {
|
|||||||
<div className="med-name">
|
<div className="med-name">
|
||||||
<div
|
<div
|
||||||
className={med?.imageUrl ? "med-avatar clickable" : ""}
|
className={med?.imageUrl ? "med-avatar clickable" : ""}
|
||||||
onClick={() => med?.imageUrl && openLightbox(med.imageUrl, med.name)}
|
onClick={() => med?.imageUrl && openLightbox(med.imageUrl, getMedDisplayName(med))}
|
||||||
onKeyDown={(e) => {
|
onKeyDown={(e) => {
|
||||||
if (e.key === "Enter" || e.key === " ") {
|
if (e.key === "Enter" || e.key === " ") {
|
||||||
if (med?.imageUrl) openLightbox(med.imageUrl, med.name);
|
if (med?.imageUrl) openLightbox(med.imageUrl, getMedDisplayName(med));
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
@@ -1199,9 +1348,13 @@ export function SharedSchedule() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="tag-row">
|
<div className="tag-row">
|
||||||
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span>
|
<span className="tag subtle">
|
||||||
{status && (
|
{formatTotalUsageLabel(med, item.total, item.doses)}
|
||||||
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span>
|
</span>
|
||||||
|
{visibleStatus && (
|
||||||
|
<span className={`status-chip small ${visibleStatus.className}`}>
|
||||||
|
{t(visibleStatus.label)}
|
||||||
|
</span>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -1212,9 +1365,7 @@ export function SharedSchedule() {
|
|||||||
<div key={dose.id} className={`dose-item future ${isTaken ? "all-taken" : ""}`}>
|
<div key={dose.id} className={`dose-item future ${isTaken ? "all-taken" : ""}`}>
|
||||||
<span className="dose-time">{dose.timeStr}</span>
|
<span className="dose-time">{dose.timeStr}</span>
|
||||||
<span className="dose-usage">
|
<span className="dose-usage">
|
||||||
<span className="dose-usage-main">
|
<span className="dose-usage-main">{renderDoseUsage(med, dose)}</span>
|
||||||
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")}
|
|
||||||
</span>
|
|
||||||
{med?.pillWeightMg && (
|
{med?.pillWeightMg && (
|
||||||
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ import { useTranslation } from "react-i18next";
|
|||||||
import { MedicationAvatar } from "../components";
|
import { MedicationAvatar } from "../components";
|
||||||
import { useEscapeKey } from "../hooks/useEscapeKey";
|
import { useEscapeKey } from "../hooks/useEscapeKey";
|
||||||
import type { Coverage, Medication, StockThresholds } from "../types";
|
import type { Coverage, Medication, StockThresholds } from "../types";
|
||||||
import { getMedTotal, getPackageSize } from "../types";
|
import { getMedDisplayName, getMedTotal, getPackageSize } from "../types";
|
||||||
import { formatNumber } from "../utils";
|
import { formatNumber } from "../utils";
|
||||||
import { getSystemLocale } from "../utils/formatters";
|
import { getSystemLocale } from "../utils/formatters";
|
||||||
import { getStockStatus } from "../utils/schedule";
|
import { getStockStatus } from "../utils/schedule";
|
||||||
@@ -64,11 +64,11 @@ export function UserFilterModal({
|
|||||||
|
|
||||||
<div className="user-meds-list">
|
<div className="user-meds-list">
|
||||||
{userMeds.map((med) => {
|
{userMeds.map((med) => {
|
||||||
const medCoverage = coverage.all.find((c) => c.name === med.name);
|
const medCoverage = coverage.all.find((c) => c.name === getMedDisplayName(med));
|
||||||
// Fallback: if no coverage data (e.g. obsolete med), compute basic status from total pills
|
// Fallback: if no coverage data (e.g. obsolete med), compute basic status from total pills
|
||||||
const status = medCoverage
|
const status = medCoverage
|
||||||
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, settings)
|
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, settings, med.packageType)
|
||||||
: getStockStatus(null, getMedTotal(med), settings);
|
: getStockStatus(null, getMedTotal(med), settings, med.packageType);
|
||||||
const packageSize = getPackageSize(med);
|
const packageSize = getPackageSize(med);
|
||||||
const currentStock = medCoverage ? formatNumber(medCoverage.medsLeft) : formatNumber(getMedTotal(med));
|
const currentStock = medCoverage ? formatNumber(medCoverage.medsLeft) : formatNumber(getMedTotal(med));
|
||||||
|
|
||||||
@@ -97,10 +97,10 @@ export function UserFilterModal({
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<MedicationAvatar name={med.name} imageUrl={med.imageUrl} size="sm" />
|
<MedicationAvatar name={getMedDisplayName(med)} imageUrl={med.imageUrl} size="sm" />
|
||||||
<div className="user-med-info">
|
<div className="user-med-info">
|
||||||
<span className="user-med-name">{med.name}</span>
|
<span className="user-med-name">{getMedDisplayName(med)}</span>
|
||||||
{med.genericName && <span className="user-med-generic">{med.genericName}</span>}
|
{med.name && med.genericName && <span className="user-med-generic">{med.genericName}</span>}
|
||||||
{personIntakes.length > 0 && (
|
{personIntakes.length > 0 && (
|
||||||
<div className="user-med-intakes">
|
<div className="user-med-intakes">
|
||||||
{personIntakes.map((intake) => {
|
{personIntakes.map((intake) => {
|
||||||
|
|||||||
@@ -6,7 +6,7 @@ import { useCollapsedDays, useDoses, useMedications, useRefill, useSettings, use
|
|||||||
import type { Coverage, FormState, Medication, ScheduleEvent, StockThresholds } from "../types";
|
import type { Coverage, FormState, Medication, ScheduleEvent, StockThresholds } from "../types";
|
||||||
import { getSystemLocale } from "../utils/formatters";
|
import { getSystemLocale } from "../utils/formatters";
|
||||||
import { log } from "../utils/logger";
|
import { log } from "../utils/logger";
|
||||||
import { buildSchedulePreview, calculateCoverage, computeMissedPastDoseIds } from "../utils/schedule";
|
import { buildSchedulePreview, calculateCoverage, computeMissedPastDoseIds, getStockStatus } from "../utils/schedule";
|
||||||
|
|
||||||
// =============================================================================
|
// =============================================================================
|
||||||
// Types
|
// Types
|
||||||
@@ -17,6 +17,7 @@ export type DoseInfo = {
|
|||||||
timeStr: string;
|
timeStr: string;
|
||||||
when: number;
|
when: number;
|
||||||
usage: number;
|
usage: number;
|
||||||
|
intakeUnit?: "ml" | "tsp" | "tbsp" | null;
|
||||||
takenBy: string[];
|
takenBy: string[];
|
||||||
intakeRemindersEnabled: boolean;
|
intakeRemindersEnabled: boolean;
|
||||||
};
|
};
|
||||||
@@ -384,6 +385,7 @@ export function AppProvider({ children }: { children: React.ReactNode }) {
|
|||||||
(dayMeds: { medName: string; lastWhen: number }[]): "success" | "warning" | "danger" => {
|
(dayMeds: { medName: string; lastWhen: number }[]): "success" | "warning" | "danger" => {
|
||||||
const statuses = dayMeds.map((item) => {
|
const statuses = dayMeds.map((item) => {
|
||||||
const cov = coverageByMed[item.medName];
|
const cov = coverageByMed[item.medName];
|
||||||
|
const med = activeMeds.find((m) => m.name === item.medName || m.genericName === item.medName);
|
||||||
const depletionTime = depletionByMed[item.medName];
|
const depletionTime = depletionByMed[item.medName];
|
||||||
|
|
||||||
// Will be out of stock by this day?
|
// Will be out of stock by this day?
|
||||||
@@ -392,21 +394,15 @@ export function AppProvider({ children }: { children: React.ReactNode }) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!cov) return "success";
|
if (!cov) return "success";
|
||||||
const { daysLeft, medsLeft } = cov;
|
const status = getStockStatus(cov.daysLeft, cov.medsLeft, stockThresholds, med?.packageType);
|
||||||
|
if (status.className === "danger") return "danger";
|
||||||
// Currently out of stock
|
if (status.className === "warning") return "warning";
|
||||||
if (medsLeft <= 0 || daysLeft === 0) return "danger";
|
|
||||||
// No schedule (can't calculate)
|
|
||||||
if (daysLeft === null) return "success";
|
|
||||||
// Low stock: < lowStockDays (warning)
|
|
||||||
if (daysLeft < settingsHook.settings.lowStockDays) return "warning";
|
|
||||||
// Normal/High stock
|
|
||||||
return "success";
|
return "success";
|
||||||
});
|
});
|
||||||
const fallbackStatus = statuses.includes("warning") ? "warning" : "success";
|
const fallbackStatus = statuses.includes("warning") ? "warning" : "success";
|
||||||
return statuses.includes("danger") ? "danger" : fallbackStatus;
|
return statuses.includes("danger") ? "danger" : fallbackStatus;
|
||||||
},
|
},
|
||||||
[coverageByMed, depletionByMed, settingsHook.settings.lowStockDays]
|
[coverageByMed, depletionByMed, activeMeds, stockThresholds]
|
||||||
);
|
);
|
||||||
|
|
||||||
const groupedSchedule = useMemo(() => {
|
const groupedSchedule = useMemo(() => {
|
||||||
@@ -439,6 +435,7 @@ export function AppProvider({ children }: { children: React.ReactNode }) {
|
|||||||
timeStr: event.timeStr,
|
timeStr: event.timeStr,
|
||||||
when: event.when,
|
when: event.when,
|
||||||
usage: event.usage,
|
usage: event.usage,
|
||||||
|
intakeUnit: event.intakeUnit ?? null,
|
||||||
takenBy: event.takenBy ? [event.takenBy] : [],
|
takenBy: event.takenBy ? [event.takenBy] : [],
|
||||||
intakeRemindersEnabled: event.intakeRemindersEnabled,
|
intakeRemindersEnabled: event.intakeRemindersEnabled,
|
||||||
});
|
});
|
||||||
@@ -665,7 +662,18 @@ export function AppProvider({ children }: { children: React.ReactNode }) {
|
|||||||
|
|
||||||
// Get the response text first to handle non-JSON responses
|
// Get the response text first to handle non-JSON responses
|
||||||
const text = await res.text();
|
const text = await res.text();
|
||||||
let data: { error?: string; message?: string; imported?: number } = {};
|
let data: {
|
||||||
|
error?: string;
|
||||||
|
message?: string;
|
||||||
|
imported?:
|
||||||
|
| {
|
||||||
|
medications?: number;
|
||||||
|
doseHistory?: number;
|
||||||
|
refillHistory?: number;
|
||||||
|
shareLinks?: number;
|
||||||
|
}
|
||||||
|
| number;
|
||||||
|
} = {};
|
||||||
try {
|
try {
|
||||||
data = text ? JSON.parse(text) : {};
|
data = text ? JSON.parse(text) : {};
|
||||||
} catch {
|
} catch {
|
||||||
@@ -680,11 +688,12 @@ export function AppProvider({ children }: { children: React.ReactNode }) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Show success message in UI instead of browser alert
|
// Show success message in UI instead of browser alert
|
||||||
|
const importedCounts = typeof data.imported === "object" && data.imported !== null ? data.imported : null;
|
||||||
setImportResult({
|
setImportResult({
|
||||||
medications: data.imported?.medications || 0,
|
medications: importedCounts?.medications || 0,
|
||||||
doses: data.imported?.doseHistory || 0,
|
doses: importedCounts?.doseHistory || 0,
|
||||||
refills: data.imported?.refillHistory || 0,
|
refills: importedCounts?.refillHistory || 0,
|
||||||
shares: data.imported?.shareLinks || 0,
|
shares: importedCounts?.shareLinks || 0,
|
||||||
});
|
});
|
||||||
|
|
||||||
// Reload all data
|
// Reload all data
|
||||||
|
|||||||
@@ -27,6 +27,12 @@ export function useEscapeKey(active: boolean, onClose: () => void, options?: { c
|
|||||||
if (!active) return;
|
if (!active) return;
|
||||||
const handleKeyDown = (e: KeyboardEvent) => {
|
const handleKeyDown = (e: KeyboardEvent) => {
|
||||||
if (e.key === "Escape" && activeRef.current) {
|
if (e.key === "Escape" && activeRef.current) {
|
||||||
|
if (capture) {
|
||||||
|
// In nested modals, consume Escape so parent/global handlers
|
||||||
|
// do not process the same key press again.
|
||||||
|
e.preventDefault();
|
||||||
|
e.stopPropagation();
|
||||||
|
}
|
||||||
onCloseRef.current();
|
onCloseRef.current();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,7 +1,13 @@
|
|||||||
import { useCallback, useEffect, useMemo, useState } from "react";
|
import { useCallback, useEffect, useMemo, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import type { FieldErrors, FormBlister, FormIntake, FormState, Medication } from "../types";
|
import type { FieldErrors, FormBlister, FormIntake, FormState, Medication } from "../types";
|
||||||
import { FIELD_LIMITS } from "../types";
|
import {
|
||||||
|
FIELD_LIMITS,
|
||||||
|
isAmountBasedPackageType,
|
||||||
|
isLiquidContainerPackageType,
|
||||||
|
isTubePackageType,
|
||||||
|
normalizePackageType,
|
||||||
|
} from "../types";
|
||||||
import { toDateValue, toTimeValue } from "../utils/formatters";
|
import { toDateValue, toTimeValue } from "../utils/formatters";
|
||||||
|
|
||||||
export const defaultBlister = (): FormBlister => {
|
export const defaultBlister = (): FormBlister => {
|
||||||
@@ -24,6 +30,7 @@ export const defaultIntake = (takenBy: string = ""): FormIntake => {
|
|||||||
every: "1",
|
every: "1",
|
||||||
startDate: toDateValue(now),
|
startDate: toDateValue(now),
|
||||||
startTime: toTimeValue(now),
|
startTime: toTimeValue(now),
|
||||||
|
intakeUnit: "ml",
|
||||||
takenBy, // Per-intake user assignment (empty string = null/everyone)
|
takenBy, // Per-intake user assignment (empty string = null/everyone)
|
||||||
intakeRemindersEnabled: false,
|
intakeRemindersEnabled: false,
|
||||||
};
|
};
|
||||||
@@ -33,15 +40,22 @@ export const defaultForm = (): FormState => ({
|
|||||||
name: "",
|
name: "",
|
||||||
genericName: "",
|
genericName: "",
|
||||||
takenBy: [],
|
takenBy: [],
|
||||||
|
medicationForm: "tablet",
|
||||||
|
pillForm: "tablet",
|
||||||
|
lifecycleCategory: "refill_when_empty",
|
||||||
packageType: "blister",
|
packageType: "blister",
|
||||||
packCount: "1",
|
packCount: "1",
|
||||||
blistersPerPack: "1",
|
blistersPerPack: "1",
|
||||||
pillsPerBlister: "1",
|
pillsPerBlister: "1",
|
||||||
|
packageAmountValue: "0",
|
||||||
|
packageAmountUnit: "ml",
|
||||||
totalPills: "",
|
totalPills: "",
|
||||||
looseTablets: "0",
|
looseTablets: "0",
|
||||||
pillWeightMg: "",
|
pillWeightMg: "",
|
||||||
doseUnit: "mg",
|
doseUnit: "mg",
|
||||||
medicationStartDate: "",
|
medicationStartDate: "",
|
||||||
|
medicationEndDate: "",
|
||||||
|
autoMarkObsoleteAfterEndDate: true,
|
||||||
expiryDate: "",
|
expiryDate: "",
|
||||||
notes: "",
|
notes: "",
|
||||||
prescriptionEnabled: false,
|
prescriptionEnabled: false,
|
||||||
@@ -115,9 +129,6 @@ export function useMedicationForm(): UseMedicationFormReturn {
|
|||||||
// Skip validation for takenBy array (individual items validated on add)
|
// Skip validation for takenBy array (individual items validated on add)
|
||||||
if (field === "takenBy") return undefined;
|
if (field === "takenBy") return undefined;
|
||||||
const strValue = typeof value === "string" ? value : "";
|
const strValue = typeof value === "string" ? value : "";
|
||||||
if (field === "name" && (!strValue || strValue.trim().length === 0)) {
|
|
||||||
return t("common.validation.required");
|
|
||||||
}
|
|
||||||
if ("max" in limits && strValue.length > limits.max) {
|
if ("max" in limits && strValue.length > limits.max) {
|
||||||
return t("common.validation.maxLength", { max: limits.max, current: strValue.length });
|
return t("common.validation.maxLength", { max: limits.max, current: strValue.length });
|
||||||
}
|
}
|
||||||
@@ -150,8 +161,16 @@ export function useMedicationForm(): UseMedicationFormReturn {
|
|||||||
const error = validateField(f, form[f]);
|
const error = validateField(f, form[f]);
|
||||||
if (error) errors[f] = error;
|
if (error) errors[f] = error;
|
||||||
});
|
});
|
||||||
|
// Cross-field validation: at least one of name or genericName is required
|
||||||
|
const hasName = form.name && form.name.trim().length > 0;
|
||||||
|
const hasGenericName = form.genericName && form.genericName.trim().length > 0;
|
||||||
|
if (!hasName && !hasGenericName) {
|
||||||
|
const msg = t("common.validation.nameOrGenericRequired");
|
||||||
|
errors.name = errors.name || msg;
|
||||||
|
errors.genericName = errors.genericName || msg;
|
||||||
|
}
|
||||||
setFieldErrors(errors);
|
setFieldErrors(errors);
|
||||||
}, [form.name, form.genericName, form.notes, validateField, form]);
|
}, [form.name, form.genericName, form.notes, validateField, form, t]);
|
||||||
|
|
||||||
const setBlisterValue = useCallback((idx: number, field: keyof FormBlister, value: string) => {
|
const setBlisterValue = useCallback((idx: number, field: keyof FormBlister, value: string) => {
|
||||||
setForm((prev) => {
|
setForm((prev) => {
|
||||||
@@ -200,6 +219,7 @@ export function useMedicationForm(): UseMedicationFormReturn {
|
|||||||
every: String(i.every),
|
every: String(i.every),
|
||||||
startDate: toDateValue(i.start),
|
startDate: toDateValue(i.start),
|
||||||
startTime: toTimeValue(i.start),
|
startTime: toTimeValue(i.start),
|
||||||
|
intakeUnit: (i.intakeUnit ?? "ml") as FormIntake["intakeUnit"],
|
||||||
takenBy: i.takenBy ?? "", // Convert null to empty string for form
|
takenBy: i.takenBy ?? "", // Convert null to empty string for form
|
||||||
intakeRemindersEnabled: i.intakeRemindersEnabled,
|
intakeRemindersEnabled: i.intakeRemindersEnabled,
|
||||||
}))
|
}))
|
||||||
@@ -208,6 +228,7 @@ export function useMedicationForm(): UseMedicationFormReturn {
|
|||||||
every: String(s.every),
|
every: String(s.every),
|
||||||
startDate: toDateValue(s.start),
|
startDate: toDateValue(s.start),
|
||||||
startTime: toTimeValue(s.start),
|
startTime: toTimeValue(s.start),
|
||||||
|
intakeUnit: "ml" as const,
|
||||||
takenBy: "", // Legacy blisters have no per-intake takenBy
|
takenBy: "", // Legacy blisters have no per-intake takenBy
|
||||||
intakeRemindersEnabled: med.intakeRemindersEnabled ?? false,
|
intakeRemindersEnabled: med.intakeRemindersEnabled ?? false,
|
||||||
}));
|
}));
|
||||||
@@ -215,21 +236,77 @@ export function useMedicationForm(): UseMedicationFormReturn {
|
|||||||
const authorizedRefills = Math.max(0, med.prescriptionAuthorizedRefills ?? 0);
|
const authorizedRefills = Math.max(0, med.prescriptionAuthorizedRefills ?? 0);
|
||||||
const remainingRefills = Math.min(Math.max(0, med.prescriptionRemainingRefills ?? 0), authorizedRefills);
|
const remainingRefills = Math.min(Math.max(0, med.prescriptionRemainingRefills ?? 0), authorizedRefills);
|
||||||
const lowRefillThreshold = Math.min(Math.max(0, med.prescriptionLowRefillThreshold ?? 1), authorizedRefills);
|
const lowRefillThreshold = Math.min(Math.max(0, med.prescriptionLowRefillThreshold ?? 1), authorizedRefills);
|
||||||
|
const packageType = normalizePackageType(med.packageType);
|
||||||
|
const isTubeOrLiquidPackage = isTubePackageType(packageType) || isLiquidContainerPackageType(packageType);
|
||||||
|
let normalizedPackCount = String(med.packCount);
|
||||||
|
let normalizedPackageAmountValue = String(med.packageAmountValue ?? 0);
|
||||||
|
|
||||||
const bottleTotalPills = med.packageType === "bottle" && med.looseTablets ? String(med.looseTablets) : "";
|
if (isTubeOrLiquidPackage) {
|
||||||
|
const safePackCount = isTubePackageType(packageType) ? 1 : Math.max(1, med.packCount || 1);
|
||||||
|
normalizedPackCount = String(safePackCount);
|
||||||
|
|
||||||
|
const rawPackageAmount = Number(med.packageAmountValue ?? 0);
|
||||||
|
const legacyKnownAmount = Math.max(0, Number(med.totalPills ?? 0), Number(med.looseTablets ?? 0));
|
||||||
|
|
||||||
|
if (isTubePackageType(packageType)) {
|
||||||
|
normalizedPackageAmountValue = String(
|
||||||
|
legacyKnownAmount > 0 ? legacyKnownAmount : Math.max(1, rawPackageAmount)
|
||||||
|
);
|
||||||
|
} else if (rawPackageAmount > 0) {
|
||||||
|
normalizedPackageAmountValue = String(rawPackageAmount);
|
||||||
|
} else {
|
||||||
|
normalizedPackageAmountValue = String(legacyKnownAmount);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const normalizedDerivedTotal = isTubeOrLiquidPackage
|
||||||
|
? Math.max(0, (Number(normalizedPackCount) || 0) * (Number(normalizedPackageAmountValue) || 0))
|
||||||
|
: null;
|
||||||
|
|
||||||
|
const bottleTotalPills = isAmountBasedPackageType(packageType) && med.looseTablets ? String(med.looseTablets) : "";
|
||||||
|
let resolvedForm = med.medicationForm;
|
||||||
|
if (!resolvedForm) {
|
||||||
|
if (isTubePackageType(packageType)) {
|
||||||
|
resolvedForm = "topical";
|
||||||
|
} else if (isLiquidContainerPackageType(packageType)) {
|
||||||
|
resolvedForm = "liquid";
|
||||||
|
} else {
|
||||||
|
resolvedForm = med.pillForm ?? "tablet";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const resolvedPillForm = med.pillForm ?? (resolvedForm === "capsule" ? "capsule" : "tablet");
|
||||||
|
let normalizedPackageAmountUnit = med.packageAmountUnit ?? "ml";
|
||||||
|
if (isTubePackageType(packageType)) {
|
||||||
|
normalizedPackageAmountUnit = "g";
|
||||||
|
} else if (isLiquidContainerPackageType(packageType)) {
|
||||||
|
normalizedPackageAmountUnit = "ml";
|
||||||
|
}
|
||||||
|
let resolvedTotalPills = bottleTotalPills;
|
||||||
|
if (normalizedDerivedTotal != null) {
|
||||||
|
resolvedTotalPills = String(normalizedDerivedTotal);
|
||||||
|
} else if (med.totalPills) {
|
||||||
|
resolvedTotalPills = String(med.totalPills);
|
||||||
|
}
|
||||||
const editForm: FormState = {
|
const editForm: FormState = {
|
||||||
name: med.name,
|
name: med.name,
|
||||||
genericName: med.genericName ?? "",
|
genericName: med.genericName ?? "",
|
||||||
takenBy: med.takenBy || [], // Already an array from API
|
takenBy: med.takenBy || [], // Already an array from API
|
||||||
packageType: med.packageType ?? "blister",
|
medicationForm: resolvedForm,
|
||||||
packCount: String(med.packCount),
|
pillForm: resolvedPillForm,
|
||||||
|
lifecycleCategory: med.lifecycleCategory ?? "refill_when_empty",
|
||||||
|
packageType,
|
||||||
|
packCount: normalizedPackCount,
|
||||||
blistersPerPack: String(med.blistersPerPack),
|
blistersPerPack: String(med.blistersPerPack),
|
||||||
pillsPerBlister: String(med.pillsPerBlister),
|
pillsPerBlister: String(med.pillsPerBlister),
|
||||||
totalPills: med.totalPills ? String(med.totalPills) : bottleTotalPills,
|
packageAmountValue: normalizedPackageAmountValue,
|
||||||
looseTablets: String(med.looseTablets),
|
packageAmountUnit: normalizedPackageAmountUnit,
|
||||||
|
totalPills: resolvedTotalPills,
|
||||||
|
looseTablets: normalizedDerivedTotal != null ? String(normalizedDerivedTotal) : String(med.looseTablets),
|
||||||
pillWeightMg: med.pillWeightMg ? String(med.pillWeightMg) : "",
|
pillWeightMg: med.pillWeightMg ? String(med.pillWeightMg) : "",
|
||||||
doseUnit: med.doseUnit ?? "mg",
|
doseUnit: med.doseUnit ?? "mg",
|
||||||
medicationStartDate: med.medicationStartDate ?? "",
|
medicationStartDate: med.medicationStartDate ?? "",
|
||||||
|
medicationEndDate: med.medicationEndDate ?? "",
|
||||||
|
autoMarkObsoleteAfterEndDate: med.autoMarkObsoleteAfterEndDate ?? true,
|
||||||
expiryDate: med.expiryDate ? med.expiryDate.slice(0, 10) : "",
|
expiryDate: med.expiryDate ? med.expiryDate.slice(0, 10) : "",
|
||||||
notes: med.notes ?? "",
|
notes: med.notes ?? "",
|
||||||
prescriptionEnabled: med.prescriptionEnabled ?? false,
|
prescriptionEnabled: med.prescriptionEnabled ?? false,
|
||||||
@@ -272,6 +349,63 @@ export function useMedicationForm(): UseMedicationFormReturn {
|
|||||||
setForm((prev) => {
|
setForm((prev) => {
|
||||||
const next = { ...prev, [key]: value } as FormState;
|
const next = { ...prev, [key]: value } as FormState;
|
||||||
|
|
||||||
|
if (key === "packageType") {
|
||||||
|
if (isTubePackageType(value)) {
|
||||||
|
next.packCount = "1";
|
||||||
|
next.packageAmountValue = String(Math.max(1, Number(next.packageAmountValue) || 0));
|
||||||
|
next.medicationForm = "topical";
|
||||||
|
next.lifecycleCategory = "treatment_period";
|
||||||
|
next.doseUnit = "units";
|
||||||
|
next.packageAmountUnit = "g";
|
||||||
|
} else if (isLiquidContainerPackageType(value)) {
|
||||||
|
next.packCount = String(Math.max(1, Number(next.packCount) || 1));
|
||||||
|
next.packageAmountValue = String(Math.max(1, Number(next.packageAmountValue) || 0));
|
||||||
|
next.medicationForm = "liquid";
|
||||||
|
next.lifecycleCategory = "refill_when_empty";
|
||||||
|
next.doseUnit = "ml";
|
||||||
|
next.packageAmountUnit = "ml";
|
||||||
|
next.intakes = next.intakes.map((intake) => ({ ...intake, intakeUnit: intake.intakeUnit || "ml" }));
|
||||||
|
} else {
|
||||||
|
next.medicationForm = next.pillForm;
|
||||||
|
next.lifecycleCategory = "refill_when_empty";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (key === "medicationForm") {
|
||||||
|
if (isTubePackageType(next.packageType)) {
|
||||||
|
next.medicationForm = "topical";
|
||||||
|
next.lifecycleCategory = "treatment_period";
|
||||||
|
next.doseUnit = "units";
|
||||||
|
next.packageAmountUnit = "g";
|
||||||
|
} else if (isLiquidContainerPackageType(next.packageType)) {
|
||||||
|
next.medicationForm = "liquid";
|
||||||
|
next.lifecycleCategory = "refill_when_empty";
|
||||||
|
next.doseUnit = "ml";
|
||||||
|
next.packageAmountUnit = "ml";
|
||||||
|
next.intakes = next.intakes.map((intake) => ({ ...intake, intakeUnit: intake.intakeUnit || "ml" }));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isTubePackageType(next.packageType)) {
|
||||||
|
next.packCount = "1";
|
||||||
|
next.packageAmountUnit = "g";
|
||||||
|
} else if (isLiquidContainerPackageType(next.packageType)) {
|
||||||
|
next.packageAmountUnit = "ml";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (key === "pillForm" && value === "capsule") {
|
||||||
|
next.medicationForm = "capsule";
|
||||||
|
next.intakes = next.intakes.map((intake) => {
|
||||||
|
const parsedUsage = Number.parseFloat(intake.usage);
|
||||||
|
const rounded = Number.isFinite(parsedUsage) ? Math.max(0, Math.round(parsedUsage)) : 1;
|
||||||
|
return { ...intake, usage: String(rounded || 1) };
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (key === "pillForm" && value === "tablet") {
|
||||||
|
next.medicationForm = "tablet";
|
||||||
|
}
|
||||||
|
|
||||||
if (key === "prescriptionAuthorizedRefills") {
|
if (key === "prescriptionAuthorizedRefills") {
|
||||||
const raw = String(value);
|
const raw = String(value);
|
||||||
next.prescriptionAuthorizedRefills = raw === "" ? "" : String(parseNonNegativeInt(raw));
|
next.prescriptionAuthorizedRefills = raw === "" ? "" : String(parseNonNegativeInt(raw));
|
||||||
|
|||||||
@@ -1,6 +1,12 @@
|
|||||||
import { useCallback, useEffect, useState } from "react";
|
import { useCallback, useEffect, useState } from "react";
|
||||||
import type { Coverage, FormState, Medication, RefillEntry } from "../types";
|
import type { Coverage, FormState, Medication, RefillEntry } from "../types";
|
||||||
import { getMedTotal, getPackageSize } from "../types";
|
import {
|
||||||
|
getMedTotal,
|
||||||
|
getPackageSize,
|
||||||
|
isAmountBasedPackageType,
|
||||||
|
isLiquidContainerPackageType,
|
||||||
|
isTubePackageType,
|
||||||
|
} from "../types";
|
||||||
|
|
||||||
export interface UseRefillReturn {
|
export interface UseRefillReturn {
|
||||||
// Refill state
|
// Refill state
|
||||||
@@ -137,51 +143,96 @@ export function useRefill(): UseRefillReturn {
|
|||||||
if (!selectedMed) return;
|
if (!selectedMed) return;
|
||||||
setEditStockSaving(true);
|
setEditStockSaving(true);
|
||||||
try {
|
try {
|
||||||
|
const isTubePackage = isTubePackageType(selectedMed.packageType);
|
||||||
|
const isLiquidPackage = isLiquidContainerPackageType(selectedMed.packageType);
|
||||||
|
const isAmountPackage = isAmountBasedPackageType(selectedMed.packageType);
|
||||||
|
const liquidAmountPerBottle = Math.max(
|
||||||
|
1,
|
||||||
|
Number.isFinite(Number(selectedMed.packageAmountValue)) && Number(selectedMed.packageAmountValue) > 0
|
||||||
|
? Number(selectedMed.packageAmountValue)
|
||||||
|
: Math.max(
|
||||||
|
1,
|
||||||
|
Math.round(Number(getPackageSize(selectedMed) || 0) / Math.max(1, Number(selectedMed.packCount || 1)))
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
// Clamp all fields to non-negative values.
|
// Clamp all fields to non-negative values.
|
||||||
let finalFullBlisters = Math.max(0, editStockFullBlisters);
|
let finalFullBlisters = Math.max(0, editStockFullBlisters);
|
||||||
let finalPartialPills =
|
let finalPartialPills = isAmountPackage
|
||||||
selectedMed.packageType === "bottle"
|
? Math.max(0, editStockPartialBlisterPills)
|
||||||
? Math.max(0, editStockPartialBlisterPills)
|
: Math.max(0, editStockPartialBlisterPills);
|
||||||
: Math.max(0, editStockPartialBlisterPills);
|
|
||||||
const finalLoosePills = Math.max(0, editStockLoosePills);
|
const finalLoosePills = Math.max(0, editStockLoosePills);
|
||||||
|
|
||||||
// Canonicalize blister values: partial overflow becomes additional full blisters.
|
// Canonicalize blister values: partial overflow becomes additional full blisters.
|
||||||
if (selectedMed.packageType !== "bottle" && selectedMed.pillsPerBlister > 0) {
|
if (!isAmountPackage && selectedMed.pillsPerBlister > 0) {
|
||||||
finalFullBlisters += Math.floor(finalPartialPills / selectedMed.pillsPerBlister);
|
finalFullBlisters += Math.floor(finalPartialPills / selectedMed.pillsPerBlister);
|
||||||
finalPartialPills %= selectedMed.pillsPerBlister;
|
finalPartialPills %= selectedMed.pillsPerBlister;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Structural max = sealed package capacity only (no looseTablets offset).
|
// Structural max = sealed package capacity only (no looseTablets offset).
|
||||||
const structuralMax =
|
const structuralMax = isAmountPackage
|
||||||
selectedMed.packageType === "bottle"
|
? (selectedMed.totalPills ?? getPackageSize(selectedMed))
|
||||||
? (selectedMed.totalPills ?? getPackageSize(selectedMed))
|
: selectedMed.packCount * selectedMed.blistersPerPack * selectedMed.pillsPerBlister;
|
||||||
: selectedMed.packCount * selectedMed.blistersPerPack * selectedMed.pillsPerBlister;
|
const correctedLiquidBottleCount = isLiquidPackage
|
||||||
|
? Math.max(1, finalFullBlisters)
|
||||||
|
: Math.max(1, selectedMed.packCount);
|
||||||
|
const liquidStructuralMax = isLiquidPackage
|
||||||
|
? correctedLiquidBottleCount * liquidAmountPerBottle
|
||||||
|
: structuralMax;
|
||||||
|
|
||||||
// For blister meds, only sealed pills are capped to package size.
|
// For blister meds, only sealed pills are capped to package size.
|
||||||
// Loose pills are extra and can be above package size.
|
// Loose pills are extra and can be above package size.
|
||||||
const desiredTotal =
|
let desiredTotal: number;
|
||||||
selectedMed.packageType === "bottle"
|
if (isTubePackage) {
|
||||||
? Math.min(structuralMax, Math.max(0, finalPartialPills))
|
desiredTotal = Math.max(0, finalPartialPills);
|
||||||
: Math.min(structuralMax, finalFullBlisters * selectedMed.pillsPerBlister + finalPartialPills) +
|
} else if (isAmountPackage) {
|
||||||
finalLoosePills;
|
desiredTotal = Math.min(liquidStructuralMax, Math.max(0, finalPartialPills));
|
||||||
|
} else {
|
||||||
|
desiredTotal =
|
||||||
|
Math.min(structuralMax, finalFullBlisters * selectedMed.pillsPerBlister + finalPartialPills) +
|
||||||
|
finalLoosePills;
|
||||||
|
}
|
||||||
|
|
||||||
// The "base" from DB structure used to compute stockAdjustment differs by type:
|
// The "base" from DB structure used to compute stockAdjustment differs by type:
|
||||||
// - Bottle: looseTablets is the base (not changed during correction)
|
// - Bottle: looseTablets is the base (not changed during correction)
|
||||||
// - Blister: use structuralMax + finalLoosePills as the new base so that
|
// - Blister: use structuralMax + finalLoosePills as the new base so that
|
||||||
// updating looseTablets in the DB doesn't cause a stale-split display bug.
|
// updating looseTablets in the DB doesn't cause a stale-split display bug.
|
||||||
const baseTotal =
|
let baseTotal: number;
|
||||||
selectedMed.packageType === "bottle"
|
if (isLiquidPackage) {
|
||||||
? getPackageSize(selectedMed) // bottle: stockAdjustment relative to fixed looseTablets base
|
baseTotal = liquidStructuralMax;
|
||||||
: structuralMax + finalLoosePills; // blister: base = sealed capacity + NEW loose pills
|
} else if (isAmountPackage) {
|
||||||
|
baseTotal = getPackageSize(selectedMed); // bottle: stockAdjustment relative to fixed looseTablets base
|
||||||
|
} else {
|
||||||
|
baseTotal = structuralMax + finalLoosePills; // blister: base = sealed capacity + NEW loose pills
|
||||||
|
}
|
||||||
// stockAdjustment = what we need to make getMedTotal() return desiredTotal
|
// stockAdjustment = what we need to make getMedTotal() return desiredTotal
|
||||||
const newStockAdjustment = desiredTotal - baseTotal;
|
const newStockAdjustment = desiredTotal - baseTotal;
|
||||||
|
|
||||||
// For blister corrections also send the new looseTablets value so the DB
|
// For blister corrections also send the new looseTablets value so the DB
|
||||||
// reflects the actual loose count (avoids stale-split display on reload).
|
// reflects the actual loose count (avoids stale-split display on reload).
|
||||||
const patchBody: { stockAdjustment: number; looseTablets?: number } = {
|
const patchBody: {
|
||||||
|
stockAdjustment: number;
|
||||||
|
looseTablets?: number;
|
||||||
|
totalPills?: number;
|
||||||
|
packageAmountValue?: number;
|
||||||
|
packCount?: number;
|
||||||
|
} = {
|
||||||
stockAdjustment: newStockAdjustment,
|
stockAdjustment: newStockAdjustment,
|
||||||
};
|
};
|
||||||
if (selectedMed.packageType !== "bottle") {
|
if (isTubePackage) {
|
||||||
|
// Tube has fixed count=1 and no automatic depletion.
|
||||||
|
// Correction must update the base amount fields directly.
|
||||||
|
patchBody.stockAdjustment = 0;
|
||||||
|
patchBody.packCount = 1;
|
||||||
|
patchBody.totalPills = desiredTotal;
|
||||||
|
patchBody.looseTablets = desiredTotal;
|
||||||
|
patchBody.packageAmountValue = desiredTotal;
|
||||||
|
} else if (isLiquidPackage) {
|
||||||
|
// Liquid correction supports bottle-count updates.
|
||||||
|
// Keep packageAmountValue (ml per bottle) and update capacity base by bottle count.
|
||||||
|
patchBody.packCount = correctedLiquidBottleCount;
|
||||||
|
patchBody.totalPills = liquidStructuralMax;
|
||||||
|
} else if (!isAmountPackage) {
|
||||||
patchBody.looseTablets = finalLoosePills;
|
patchBody.looseTablets = finalLoosePills;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -222,6 +273,7 @@ export function useRefill(): UseRefillReturn {
|
|||||||
const openEditStockModal = useCallback((selectedMed: Medication, coverage: { all: Coverage[] }) => {
|
const openEditStockModal = useCallback((selectedMed: Medication, coverage: { all: Coverage[] }) => {
|
||||||
if (!selectedMed) return;
|
if (!selectedMed) return;
|
||||||
setEditStockMedication(selectedMed);
|
setEditStockMedication(selectedMed);
|
||||||
|
const isAmountPackage = isAmountBasedPackageType(selectedMed.packageType);
|
||||||
// Get current stock from coverage (after consumption)
|
// Get current stock from coverage (after consumption)
|
||||||
const medCoverage = coverage.all.find((c) => c.name === selectedMed.name);
|
const medCoverage = coverage.all.find((c) => c.name === selectedMed.name);
|
||||||
const dbTotal = getMedTotal(selectedMed);
|
const dbTotal = getMedTotal(selectedMed);
|
||||||
@@ -231,15 +283,20 @@ export function useRefill(): UseRefillReturn {
|
|||||||
// For blister, keep loose pills separated from sealed blister/partial counts.
|
// For blister, keep loose pills separated from sealed blister/partial counts.
|
||||||
const knownLoose = Math.min(currentStock, Math.max(0, selectedMed.looseTablets));
|
const knownLoose = Math.min(currentStock, Math.max(0, selectedMed.looseTablets));
|
||||||
const sealedPills = Math.max(0, currentStock - knownLoose);
|
const sealedPills = Math.max(0, currentStock - knownLoose);
|
||||||
const fullBlisters =
|
let fullBlisters: number;
|
||||||
selectedMed.packageType === "bottle" ? 0 : Math.floor(sealedPills / selectedMed.pillsPerBlister);
|
if (isLiquidContainerPackageType(selectedMed.packageType)) {
|
||||||
const partialPills =
|
fullBlisters = Math.max(1, selectedMed.packCount);
|
||||||
selectedMed.packageType === "bottle" ? Math.max(0, currentStock) : sealedPills % selectedMed.pillsPerBlister;
|
} else if (isAmountPackage) {
|
||||||
|
fullBlisters = 0;
|
||||||
|
} else {
|
||||||
|
fullBlisters = Math.floor(sealedPills / selectedMed.pillsPerBlister);
|
||||||
|
}
|
||||||
|
const partialPills = isAmountPackage ? Math.max(0, currentStock) : sealedPills % selectedMed.pillsPerBlister;
|
||||||
|
|
||||||
// Pre-fill with current values
|
// Pre-fill with current values
|
||||||
setEditStockFullBlisters(fullBlisters);
|
setEditStockFullBlisters(fullBlisters);
|
||||||
setEditStockPartialBlisterPills(partialPills);
|
setEditStockPartialBlisterPills(partialPills);
|
||||||
setEditStockLoosePills(selectedMed.packageType === "bottle" ? 0 : knownLoose);
|
setEditStockLoosePills(isAmountPackage ? 0 : knownLoose);
|
||||||
setShowEditStockModal(true);
|
setShowEditStockModal(true);
|
||||||
window.history.pushState({ modal: "editStock" }, "");
|
window.history.pushState({ modal: "editStock" }, "");
|
||||||
}, []);
|
}, []);
|
||||||
|
|||||||
@@ -49,6 +49,8 @@ export interface Settings {
|
|||||||
upcomingTodayOnly: boolean;
|
upcomingTodayOnly: boolean;
|
||||||
shareScheduleTodayOnly: boolean;
|
shareScheduleTodayOnly: boolean;
|
||||||
swapDashboardMainSections: boolean;
|
swapDashboardMainSections: boolean;
|
||||||
|
reminderHour: number;
|
||||||
|
reminderMinutesBefore: number;
|
||||||
expiryWarningDays: number;
|
expiryWarningDays: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -96,6 +98,8 @@ const defaultSettings: Settings = {
|
|||||||
upcomingTodayOnly: false,
|
upcomingTodayOnly: false,
|
||||||
shareScheduleTodayOnly: false,
|
shareScheduleTodayOnly: false,
|
||||||
swapDashboardMainSections: false,
|
swapDashboardMainSections: false,
|
||||||
|
reminderHour: 6,
|
||||||
|
reminderMinutesBefore: 15,
|
||||||
expiryWarningDays: 30,
|
expiryWarningDays: 30,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
+68
-11
@@ -110,6 +110,7 @@
|
|||||||
"fullBlisters": "Volle Blister",
|
"fullBlisters": "Volle Blister",
|
||||||
"openBlister": "Offener Blister",
|
"openBlister": "Offener Blister",
|
||||||
"stock": "Bestand",
|
"stock": "Bestand",
|
||||||
|
"dailyConsumption": "Taeglicher Verbrauch",
|
||||||
"stockDetails": "Details",
|
"stockDetails": "Details",
|
||||||
"daysLeft": "Tage übrig",
|
"daysLeft": "Tage übrig",
|
||||||
"status": "Status",
|
"status": "Status",
|
||||||
@@ -118,7 +119,8 @@
|
|||||||
"expiry": "Ablaufdatum",
|
"expiry": "Ablaufdatum",
|
||||||
"pillsCount": "{{count}} Tabletten",
|
"pillsCount": "{{count}} Tabletten",
|
||||||
"pillsCount_one": "{{count}} Tablette",
|
"pillsCount_one": "{{count}} Tablette",
|
||||||
"pillsCount_other": "{{count}} Tabletten"
|
"pillsCount_other": "{{count}} Tabletten",
|
||||||
|
"perDayWithUnit": "{{value}} {{unit}}"
|
||||||
},
|
},
|
||||||
"medications": {
|
"medications": {
|
||||||
"list": {
|
"list": {
|
||||||
@@ -130,7 +132,8 @@
|
|||||||
"reactivate": "Reaktivieren",
|
"reactivate": "Reaktivieren",
|
||||||
"obsoleteTitle": "Obsolet ({{count}})",
|
"obsoleteTitle": "Obsolet ({{count}})",
|
||||||
"obsoleteSince": "Beendet",
|
"obsoleteSince": "Beendet",
|
||||||
"started": "Gestartet"
|
"started": "Gestartet",
|
||||||
|
"emptyState": "Noch keine Medikamente. Fuege dein erstes Medikament hinzu."
|
||||||
},
|
},
|
||||||
"details": {
|
"details": {
|
||||||
"packs": "Packungen",
|
"packs": "Packungen",
|
||||||
@@ -139,6 +142,7 @@
|
|||||||
"loose": "Lose",
|
"loose": "Lose",
|
||||||
"total": "Gesamt",
|
"total": "Gesamt",
|
||||||
"stock": "Bestand",
|
"stock": "Bestand",
|
||||||
|
"capacityPerPackage": "Kapazitaet pro Packung",
|
||||||
"totalCapacity": "Kapazität",
|
"totalCapacity": "Kapazität",
|
||||||
"type": "Typ"
|
"type": "Typ"
|
||||||
},
|
},
|
||||||
@@ -167,18 +171,41 @@
|
|||||||
"commercialName": "Handelsname",
|
"commercialName": "Handelsname",
|
||||||
"genericName": "Wirkstoff",
|
"genericName": "Wirkstoff",
|
||||||
"takenBy": "Eingenommen von",
|
"takenBy": "Eingenommen von",
|
||||||
|
"medicationForm": "Medikationsform",
|
||||||
|
"medicationFormCapsule": "Kapsel",
|
||||||
|
"medicationFormTablet": "Tablette",
|
||||||
|
"medicationFormLiquid": "Fluessigkeit",
|
||||||
|
"medicationFormTopical": "Topisch",
|
||||||
|
"pillForm": "Pillenform",
|
||||||
|
"lifecycleCategory": "Lebenszyklus",
|
||||||
|
"lifecycleRefillWhenEmpty": "Nachfuellen wenn leer",
|
||||||
|
"lifecycleTreatmentPeriod": "Behandlungszeitraum",
|
||||||
"packageType": "Verpackungsart",
|
"packageType": "Verpackungsart",
|
||||||
"packageTypeBlister": "Blisterpackung",
|
"packageTypeBlister": "Blisterpackung",
|
||||||
"packageTypeBottle": "Pillendose",
|
"packageTypeBottle": "Pillendose",
|
||||||
|
"packageTypeTube": "Tube",
|
||||||
|
"packageTypeLiquidContainer": "Fluessigbehaeltnis",
|
||||||
"packs": "Packungen",
|
"packs": "Packungen",
|
||||||
|
"bottles": "Flaschen",
|
||||||
|
"tubes": "Tuben",
|
||||||
"blistersPerPack": "Blister pro Packung",
|
"blistersPerPack": "Blister pro Packung",
|
||||||
"pillsPerBlister": "Tabletten pro Blister",
|
"pillsPerBlister": "Tabletten pro Blister",
|
||||||
"totalCapacity": "Gesamtkapazität",
|
"totalCapacity": "Gesamtkapazität",
|
||||||
"currentPills": "Aktuelle Tabletten",
|
"currentPills": "Aktuelle Tabletten",
|
||||||
|
"totalAmount": "Gesamtmenge",
|
||||||
|
"currentAmount": "Aktuelle Menge",
|
||||||
|
"totalAmountLabel": "Gesamt (Menge)",
|
||||||
|
"packageAmount": "Packungsinhalt",
|
||||||
|
"packageAmountPerBottle": "Inhalt pro Flasche",
|
||||||
|
"packageAmountPerTube": "Inhalt pro Tube",
|
||||||
|
"packageAmountUnitMl": "ml",
|
||||||
|
"packageAmountUnitG": "g",
|
||||||
"loosePills": "Lose Tabletten",
|
"loosePills": "Lose Tabletten",
|
||||||
"pillWeight": "Dosis pro Tablette",
|
"pillWeight": "Dosis pro Tablette",
|
||||||
"total": "Gesamt (Tabletten)",
|
"total": "Gesamt (Tabletten)",
|
||||||
"medicationStartDate": "Startdatum der Medikation",
|
"medicationStartDate": "Startdatum der Medikation",
|
||||||
|
"medicationEndDate": "Enddatum der Medikation",
|
||||||
|
"autoMarkObsoleteAfterEndDate": "Nach Enddatum automatisch als obsolet markieren",
|
||||||
"expiryDate": "Ablaufdatum",
|
"expiryDate": "Ablaufdatum",
|
||||||
"notes": "Notizen",
|
"notes": "Notizen",
|
||||||
"medicationImage": "Medikamentenbild",
|
"medicationImage": "Medikamentenbild",
|
||||||
@@ -192,21 +219,44 @@
|
|||||||
},
|
},
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
"commercial": "z.B. Ozempic",
|
"commercial": "z.B. Ozempic",
|
||||||
"generic": "z.B. Semaglutid (optional)",
|
"generic": "z.B. Semaglutid",
|
||||||
"takenBy": "Name eingeben und Enter drücken",
|
"takenBy": "Name eingeben und Enter drücken",
|
||||||
"addPerson": "Weitere Person hinzufügen...",
|
"addPerson": "Weitere Person hinzufügen...",
|
||||||
"weight": "z.B. 240",
|
"weight": "z.B. 240",
|
||||||
"notes": "z.B. Mit Essen einnehmen, Alkohol vermeiden... (optional)"
|
"notes": "z.B. Mit Essen einnehmen, Alkohol vermeiden... (optional)"
|
||||||
},
|
},
|
||||||
"validation": {
|
"validation": {
|
||||||
"startDateAfterIntake": "Das Medikations-Startdatum ({{medicationStartDate}}) darf nicht nach dem Einnahmedatum ({{intakeDate}}) liegen."
|
"startDateAfterIntake": "Das Medikations-Startdatum ({{medicationStartDate}}) darf nicht nach dem Einnahmedatum ({{intakeDate}}) liegen.",
|
||||||
},
|
"endDateBeforeStart": "Das Medikations-Enddatum ({{medicationEndDate}}) darf nicht vor dem Startdatum ({{medicationStartDate}}) liegen."
|
||||||
|
},
|
||||||
"blisters": {
|
"blisters": {
|
||||||
"title": "Einnahmeplan",
|
"title": "Einnahmeplan",
|
||||||
"remind": "Erinnern",
|
"remind": "Erinnern",
|
||||||
"remindTooltip": "Erhalte eine Benachrichtigung 15 Minuten vor jeder geplanten Einnahme",
|
"remindTooltip": "Erhalte eine Benachrichtigung 15 Minuten vor jeder geplanten Einnahme",
|
||||||
"addIntake": "Einnahme",
|
"addIntake": "Einnahme",
|
||||||
"usage": "Dosis (Tabletten)",
|
"usage": "Dosis",
|
||||||
|
"usageTablets": "Dosis (Tabletten)",
|
||||||
|
"usageCapsules": "Dosis (Kapseln)",
|
||||||
|
"usageMl": "Dosis (ml)",
|
||||||
|
"usageTsp": "Dosis (tsp)",
|
||||||
|
"usageTbsp": "Dosis (tbsp)",
|
||||||
|
"usageApplication": "Dosis (Anwendungen)",
|
||||||
|
"intakeUnit": "Einnahmeeinheit",
|
||||||
|
"intakeUnitMl": "Milliliter (ml)",
|
||||||
|
"intakeUnitTsp": "Teeloeffel (5 ml)",
|
||||||
|
"intakeUnitTbsp": "Essloeffel (15 ml)",
|
||||||
|
"intakes": "Einnahmen",
|
||||||
|
"intakes_one": "Einnahme",
|
||||||
|
"intakes_other": "Einnahmen",
|
||||||
|
"teaspoons": "Teeloeffel",
|
||||||
|
"teaspoons_one": "Teeloeffel",
|
||||||
|
"teaspoons_other": "Teeloeffel",
|
||||||
|
"tablespoons": "Essloeffel",
|
||||||
|
"tablespoons_one": "Essloeffel",
|
||||||
|
"tablespoons_other": "Essloeffel",
|
||||||
|
"applications": "Anwendungen",
|
||||||
|
"applications_one": "Anwendung",
|
||||||
|
"applications_other": "Anwendungen",
|
||||||
"everyDays": "Alle (Tage)",
|
"everyDays": "Alle (Tage)",
|
||||||
"every": "alle",
|
"every": "alle",
|
||||||
"from": "ab",
|
"from": "ab",
|
||||||
@@ -273,9 +323,9 @@
|
|||||||
"schedule": {
|
"schedule": {
|
||||||
"title": "Erinnerungsplan",
|
"title": "Erinnerungsplan",
|
||||||
"stockCheck": "Bestands- & Rezeptprüfung",
|
"stockCheck": "Bestands- & Rezeptprüfung",
|
||||||
"dailyAt6": "Täglich um 6:00 Uhr",
|
"dailyAtHour": "Täglich um {{hour}}:00 Uhr",
|
||||||
"intakeCheck": "Einnahmeprüfung",
|
"intakeCheck": "Einnahmeprüfung",
|
||||||
"15minBefore": "15 Min. vor geplanter Zeit",
|
"minutesBefore": "{{minutes}} Min. vor geplanter Zeit",
|
||||||
"nextCheck": "Nächste Bestandsprüfung",
|
"nextCheck": "Nächste Bestandsprüfung",
|
||||||
"lastSent": "Letzte Benachrichtigung",
|
"lastSent": "Letzte Benachrichtigung",
|
||||||
"lastStockSent": "Letzte Bestands-Erinnerung",
|
"lastStockSent": "Letzte Bestands-Erinnerung",
|
||||||
@@ -299,7 +349,8 @@
|
|||||||
"highStockTooltip": "Bestand über diesem Wert bedeutet, dass du gut versorgt bist",
|
"highStockTooltip": "Bestand über diesem Wert bedeutet, dass du gut versorgt bist",
|
||||||
"thresholdValidation": "Werte müssen wie folgt sein: Kritisch < Niedrig < Hoch",
|
"thresholdValidation": "Werte müssen wie folgt sein: Kritisch < Niedrig < Hoch",
|
||||||
"shareStockStatus": "Bestand auf geteilten Links anzeigen",
|
"shareStockStatus": "Bestand auf geteilten Links anzeigen",
|
||||||
"shareStockStatusDesc": "Bestandsstatus (Normal/Niedrig/Kritisch) und farbige Rahmen auf geteilten Zeitplan-Links für Einnahme-Nutzer anzeigen"
|
"shareStockStatusDesc": "Bestandsstatus (Normal/Niedrig/Kritisch) und farbige Rahmen auf geteilten Zeitplan-Links für Einnahme-Nutzer anzeigen",
|
||||||
|
"packageTypesNote": "Hinweis: Tubenmedikamente sind von Bestands-Erinnerungen ausgeschlossen. Flüssigbehälter-Medikamente verwenden einen einzelnen Reminder-Basiswert (Niedrig und Kritisch werden automatisch von diesem Wert abgeleitet)."
|
||||||
},
|
},
|
||||||
"timeline": {
|
"timeline": {
|
||||||
"title": "Allgemeine UI",
|
"title": "Allgemeine UI",
|
||||||
@@ -316,7 +367,7 @@
|
|||||||
"stockReminder": {
|
"stockReminder": {
|
||||||
"title": "Bestands-Erinnerung",
|
"title": "Bestands-Erinnerung",
|
||||||
"description": "Bestands-Erinnerungen aktivieren",
|
"description": "Bestands-Erinnerungen aktivieren",
|
||||||
"infoTooltip": "Benachrichtigungen umfassen alle Medikamente mit Niedrig- oder Kritisch-Status. Niedrig: Bestand wird knapp. Kritisch: Bestand ist kritisch niedrig — bald nachbestellen.",
|
"infoTooltip": "Benachrichtigungen umfassen alle Medikamente mit Niedrig- oder Kritisch-Status. Hinweis: Tubenmedikamente sind ausgeschlossen; Flüssigbehälter verwenden einen einzelnen Basiswert (Niedrig und Kritisch werden abgeleitet).",
|
||||||
"repeatDaily": "Täglich wiederholen",
|
"repeatDaily": "Täglich wiederholen",
|
||||||
"repeatTooltip": "Wenn aktiviert, wird täglich eine Erinnerung gesendet solange der Bestand kritisch ist. Andernfalls nur einmal pro Medikament bis zum Auffüllen."
|
"repeatTooltip": "Wenn aktiviert, wird täglich eine Erinnerung gesendet solange der Bestand kritisch ist. Andernfalls nur einmal pro Medikament bis zum Auffüllen."
|
||||||
},
|
},
|
||||||
@@ -327,6 +378,8 @@
|
|||||||
"at": "um",
|
"at": "um",
|
||||||
"stockInfo": "Aktueller Bestand",
|
"stockInfo": "Aktueller Bestand",
|
||||||
"packageDetails": "Packungsdetails",
|
"packageDetails": "Packungsdetails",
|
||||||
|
"packageTypeTubeHint": "Tubenmedikamente enthalten feste Mengen (z. B. Cremes, Gele). Der Bestand wird nicht verfolgt und Erinnerungen werden nicht gesendet.",
|
||||||
|
"packageTypeLiquidHint": "Flüssigbehälter verwenden ein vereinfachtes Erinnerungsmodell. Niedrig- und Kritisch-Stufen werden automatisch von einem einzelnen Basiswert abgeleitet.",
|
||||||
"currentStock": "Tabletten",
|
"currentStock": "Tabletten",
|
||||||
"packs": "Packungen",
|
"packs": "Packungen",
|
||||||
"blistersPerPack": "Blister/Packung",
|
"blistersPerPack": "Blister/Packung",
|
||||||
@@ -436,6 +489,7 @@
|
|||||||
},
|
},
|
||||||
"validation": {
|
"validation": {
|
||||||
"required": "Dieses Feld ist erforderlich",
|
"required": "Dieses Feld ist erforderlich",
|
||||||
|
"nameOrGenericRequired": "Handelsname oder Wirkstoff ist erforderlich",
|
||||||
"maxLength": "Maximal {{max}} Zeichen ({{current}}/{{max}})",
|
"maxLength": "Maximal {{max}} Zeichen ({{current}}/{{max}})",
|
||||||
"tooLong": "{{current}}/{{max}} Zeichen"
|
"tooLong": "{{current}}/{{max}} Zeichen"
|
||||||
},
|
},
|
||||||
@@ -576,9 +630,11 @@
|
|||||||
"loosePills": "Lose Tabletten",
|
"loosePills": "Lose Tabletten",
|
||||||
"pillsPerBlister": "(je {{count}} Tabletten)",
|
"pillsPerBlister": "(je {{count}} Tabletten)",
|
||||||
"packageSize": "Packungsgröße: {{count}} Tabletten",
|
"packageSize": "Packungsgröße: {{count}} Tabletten",
|
||||||
|
"packageSizeAmount": "Packungsgroesse: {{count}} {{unit}}",
|
||||||
"packageSizeBreakdown": "{{packCount}} x {{sizePerPack}} Tabletten Packung = {{total}} Tabletten",
|
"packageSizeBreakdown": "{{packCount}} x {{sizePerPack}} Tabletten Packung = {{total}} Tabletten",
|
||||||
"currentComposition": "Aktueller Bestand: {{fullBlisters}} volle Blister + {{partialPills}} angebrochen + {{loosePills}} lose = {{total}} Tabletten",
|
"currentComposition": "Aktueller Bestand: {{fullBlisters}} volle Blister + {{partialPills}} angebrochen + {{loosePills}} lose = {{total}} Tabletten",
|
||||||
"maxExceeded": "Die maximale Packungsgröße beträgt {{count}} Tabletten. Werte wurden begrenzt.",
|
"maxExceeded": "Die maximale Packungsgröße beträgt {{count}} Tabletten. Werte wurden begrenzt.",
|
||||||
|
"maxExceededAmount": "Die maximale Packungsgroesse betraegt {{count}} {{unit}}. Werte wurden begrenzt.",
|
||||||
"decreaseValue": "Wert verringern",
|
"decreaseValue": "Wert verringern",
|
||||||
"increaseValue": "Wert erhöhen",
|
"increaseValue": "Wert erhöhen",
|
||||||
"currentTotal": "Aktueller Bestand",
|
"currentTotal": "Aktueller Bestand",
|
||||||
@@ -638,6 +694,7 @@
|
|||||||
"docPackageType": "Verpackungsart",
|
"docPackageType": "Verpackungsart",
|
||||||
"docBlister": "Blisterpackung",
|
"docBlister": "Blisterpackung",
|
||||||
"docBottle": "Pillendose",
|
"docBottle": "Pillendose",
|
||||||
|
"docTube": "Tube",
|
||||||
"docPacks": "Packungen",
|
"docPacks": "Packungen",
|
||||||
"docBlistersPerPack": "Blister pro Packung",
|
"docBlistersPerPack": "Blister pro Packung",
|
||||||
"docPillsPerBlister": "Tabletten pro Blister",
|
"docPillsPerBlister": "Tabletten pro Blister",
|
||||||
|
|||||||
@@ -110,6 +110,7 @@
|
|||||||
"fullBlisters": "Full blisters",
|
"fullBlisters": "Full blisters",
|
||||||
"openBlister": "Open blister",
|
"openBlister": "Open blister",
|
||||||
"stock": "Stock",
|
"stock": "Stock",
|
||||||
|
"dailyConsumption": "Daily consumption",
|
||||||
"stockDetails": "Details",
|
"stockDetails": "Details",
|
||||||
"daysLeft": "Days left",
|
"daysLeft": "Days left",
|
||||||
"status": "Status",
|
"status": "Status",
|
||||||
@@ -118,7 +119,8 @@
|
|||||||
"expiry": "Expiry",
|
"expiry": "Expiry",
|
||||||
"pillsCount": "{{count}} pills",
|
"pillsCount": "{{count}} pills",
|
||||||
"pillsCount_one": "{{count}} pill",
|
"pillsCount_one": "{{count}} pill",
|
||||||
"pillsCount_other": "{{count}} pills"
|
"pillsCount_other": "{{count}} pills",
|
||||||
|
"perDayWithUnit": "{{value}} {{unit}}"
|
||||||
},
|
},
|
||||||
"medications": {
|
"medications": {
|
||||||
"list": {
|
"list": {
|
||||||
@@ -130,7 +132,8 @@
|
|||||||
"reactivate": "Reactivate",
|
"reactivate": "Reactivate",
|
||||||
"obsoleteTitle": "Obsolete ({{count}})",
|
"obsoleteTitle": "Obsolete ({{count}})",
|
||||||
"obsoleteSince": "Stopped",
|
"obsoleteSince": "Stopped",
|
||||||
"started": "Started"
|
"started": "Started",
|
||||||
|
"emptyState": "No medications yet. Add your first medication to get started."
|
||||||
},
|
},
|
||||||
"details": {
|
"details": {
|
||||||
"packs": "Packs",
|
"packs": "Packs",
|
||||||
@@ -139,6 +142,7 @@
|
|||||||
"loose": "Loose",
|
"loose": "Loose",
|
||||||
"total": "Total",
|
"total": "Total",
|
||||||
"stock": "Stock",
|
"stock": "Stock",
|
||||||
|
"capacityPerPackage": "Capacity per package",
|
||||||
"totalCapacity": "Capacity",
|
"totalCapacity": "Capacity",
|
||||||
"type": "Type"
|
"type": "Type"
|
||||||
},
|
},
|
||||||
@@ -167,18 +171,41 @@
|
|||||||
"commercialName": "Commercial Name",
|
"commercialName": "Commercial Name",
|
||||||
"genericName": "Generic Name",
|
"genericName": "Generic Name",
|
||||||
"takenBy": "Taken by",
|
"takenBy": "Taken by",
|
||||||
|
"medicationForm": "Medication Form",
|
||||||
|
"medicationFormCapsule": "Capsule",
|
||||||
|
"medicationFormTablet": "Tablet",
|
||||||
|
"medicationFormLiquid": "Liquid",
|
||||||
|
"medicationFormTopical": "Topical",
|
||||||
|
"pillForm": "Pill Form",
|
||||||
|
"lifecycleCategory": "Lifecycle",
|
||||||
|
"lifecycleRefillWhenEmpty": "Refill when empty",
|
||||||
|
"lifecycleTreatmentPeriod": "Treatment period",
|
||||||
"packageType": "Package Type",
|
"packageType": "Package Type",
|
||||||
"packageTypeBlister": "Blister Pack",
|
"packageTypeBlister": "Blister Pack",
|
||||||
"packageTypeBottle": "Pill Bottle",
|
"packageTypeBottle": "Pill Bottle",
|
||||||
|
"packageTypeTube": "Tube",
|
||||||
|
"packageTypeLiquidContainer": "Liquid Container",
|
||||||
"packs": "Packs",
|
"packs": "Packs",
|
||||||
|
"bottles": "Bottles",
|
||||||
|
"tubes": "Tubes",
|
||||||
"blistersPerPack": "Blisters per pack",
|
"blistersPerPack": "Blisters per pack",
|
||||||
"pillsPerBlister": "Pills per blister",
|
"pillsPerBlister": "Pills per blister",
|
||||||
"totalCapacity": "Total Capacity",
|
"totalCapacity": "Total Capacity",
|
||||||
"currentPills": "Current Pills",
|
"currentPills": "Current Pills",
|
||||||
|
"totalAmount": "Total Amount",
|
||||||
|
"currentAmount": "Current Amount",
|
||||||
|
"totalAmountLabel": "Total (amount)",
|
||||||
|
"packageAmount": "Package amount",
|
||||||
|
"packageAmountPerBottle": "Amount per bottle",
|
||||||
|
"packageAmountPerTube": "Amount per tube",
|
||||||
|
"packageAmountUnitMl": "ml",
|
||||||
|
"packageAmountUnitG": "g",
|
||||||
"loosePills": "Loose pills",
|
"loosePills": "Loose pills",
|
||||||
"pillWeight": "Dose per pill",
|
"pillWeight": "Dose per pill",
|
||||||
"total": "Total (pills)",
|
"total": "Total (pills)",
|
||||||
"medicationStartDate": "Medication Start Date",
|
"medicationStartDate": "Medication Start Date",
|
||||||
|
"medicationEndDate": "Medication End Date",
|
||||||
|
"autoMarkObsoleteAfterEndDate": "Automatically mark obsolete after end date",
|
||||||
"expiryDate": "Expiry Date",
|
"expiryDate": "Expiry Date",
|
||||||
"notes": "Notes",
|
"notes": "Notes",
|
||||||
"medicationImage": "Medication Image",
|
"medicationImage": "Medication Image",
|
||||||
@@ -192,21 +219,44 @@
|
|||||||
},
|
},
|
||||||
"placeholders": {
|
"placeholders": {
|
||||||
"commercial": "e.g. Ozempic",
|
"commercial": "e.g. Ozempic",
|
||||||
"generic": "e.g. Semaglutide (optional)",
|
"generic": "e.g. Semaglutide",
|
||||||
"takenBy": "Type name and press Enter",
|
"takenBy": "Type name and press Enter",
|
||||||
"addPerson": "Add another person...",
|
"addPerson": "Add another person...",
|
||||||
"weight": "e.g. 240",
|
"weight": "e.g. 240",
|
||||||
"notes": "e.g. Take with food, avoid alcohol... (optional)"
|
"notes": "e.g. Take with food, avoid alcohol... (optional)"
|
||||||
},
|
},
|
||||||
"validation": {
|
"validation": {
|
||||||
"startDateAfterIntake": "Medication start date ({{medicationStartDate}}) cannot be after intake date ({{intakeDate}})."
|
"startDateAfterIntake": "Medication start date ({{medicationStartDate}}) cannot be after intake date ({{intakeDate}}).",
|
||||||
|
"endDateBeforeStart": "Medication end date ({{medicationEndDate}}) cannot be before medication start date ({{medicationStartDate}})."
|
||||||
},
|
},
|
||||||
"blisters": {
|
"blisters": {
|
||||||
"title": "Intake schedule",
|
"title": "Intake schedule",
|
||||||
"remind": "Remind",
|
"remind": "Remind",
|
||||||
"remindTooltip": "Receive a notification 15 minutes before each scheduled intake",
|
"remindTooltip": "Receive a notification 15 minutes before each scheduled intake",
|
||||||
"addIntake": "Intake",
|
"addIntake": "Intake",
|
||||||
"usage": "Usage (pills)",
|
"usage": "Usage",
|
||||||
|
"usageTablets": "Usage (tablets)",
|
||||||
|
"usageCapsules": "Usage (capsules)",
|
||||||
|
"usageMl": "Usage (ml)",
|
||||||
|
"usageTsp": "Usage (tsp)",
|
||||||
|
"usageTbsp": "Usage (tbsp)",
|
||||||
|
"usageApplication": "Usage (applications)",
|
||||||
|
"intakeUnit": "Intake unit",
|
||||||
|
"intakeUnitMl": "Milliliters (ml)",
|
||||||
|
"intakeUnitTsp": "Teaspoon (5 ml)",
|
||||||
|
"intakeUnitTbsp": "Tablespoon (15 ml)",
|
||||||
|
"intakes": "intakes",
|
||||||
|
"intakes_one": "intake",
|
||||||
|
"intakes_other": "intakes",
|
||||||
|
"teaspoons": "teaspoons",
|
||||||
|
"teaspoons_one": "teaspoon",
|
||||||
|
"teaspoons_other": "teaspoons",
|
||||||
|
"tablespoons": "tablespoons",
|
||||||
|
"tablespoons_one": "tablespoon",
|
||||||
|
"tablespoons_other": "tablespoons",
|
||||||
|
"applications": "applications",
|
||||||
|
"applications_one": "application",
|
||||||
|
"applications_other": "applications",
|
||||||
"everyDays": "Every (days)",
|
"everyDays": "Every (days)",
|
||||||
"every": "every",
|
"every": "every",
|
||||||
"from": "from",
|
"from": "from",
|
||||||
@@ -273,9 +323,9 @@
|
|||||||
"schedule": {
|
"schedule": {
|
||||||
"title": "Reminder Schedule",
|
"title": "Reminder Schedule",
|
||||||
"stockCheck": "Stock & prescription check",
|
"stockCheck": "Stock & prescription check",
|
||||||
"dailyAt6": "Daily at 6:00 AM",
|
"dailyAtHour": "Daily at {{hour}}:00",
|
||||||
"intakeCheck": "Intake check",
|
"intakeCheck": "Intake check",
|
||||||
"15minBefore": "15 min before scheduled time",
|
"minutesBefore": "{{minutes}} min before scheduled time",
|
||||||
"nextCheck": "Next stock check",
|
"nextCheck": "Next stock check",
|
||||||
"lastSent": "Last notification sent",
|
"lastSent": "Last notification sent",
|
||||||
"lastStockSent": "Last stock reminder",
|
"lastStockSent": "Last stock reminder",
|
||||||
@@ -299,7 +349,8 @@
|
|||||||
"highStockTooltip": "Stock above this value means you are well supplied",
|
"highStockTooltip": "Stock above this value means you are well supplied",
|
||||||
"thresholdValidation": "Values must be: Critical < Low < High",
|
"thresholdValidation": "Values must be: Critical < Low < High",
|
||||||
"shareStockStatus": "Show Stock on Shared Links",
|
"shareStockStatus": "Show Stock on Shared Links",
|
||||||
"shareStockStatusDesc": "Show stock status (Normal/Low/Critical) and colored borders on shared schedule links for intake users"
|
"shareStockStatusDesc": "Show stock status (Normal/Low/Critical) and colored borders on shared schedule links for intake users",
|
||||||
|
"packageTypesNote": "Note: Tube medications are excluded from stock reminders. Liquid container medications use a single reminder baseline (Low and Critical are automatically derived from this value)."
|
||||||
},
|
},
|
||||||
"timeline": {
|
"timeline": {
|
||||||
"title": "General UI",
|
"title": "General UI",
|
||||||
@@ -316,7 +367,7 @@
|
|||||||
"stockReminder": {
|
"stockReminder": {
|
||||||
"title": "Stock Reminder",
|
"title": "Stock Reminder",
|
||||||
"description": "Enable stock reminders",
|
"description": "Enable stock reminders",
|
||||||
"infoTooltip": "Notifications include all medications with Low or Critical stock status. Low: stock is running low. Critical: stock is critically low — reorder soon.",
|
"infoTooltip": "Notifications include all medications with Low or Critical stock status. Note: Tube medications are excluded; Liquid containers use a single baseline threshold (Low and Critical are derived).",
|
||||||
"repeatDaily": "Repeat daily",
|
"repeatDaily": "Repeat daily",
|
||||||
"repeatTooltip": "When enabled, sends reminders every day while stock is critical. Otherwise, only notifies once per medication until restocked."
|
"repeatTooltip": "When enabled, sends reminders every day while stock is critical. Otherwise, only notifies once per medication until restocked."
|
||||||
},
|
},
|
||||||
@@ -327,6 +378,8 @@
|
|||||||
"at": "at",
|
"at": "at",
|
||||||
"stockInfo": "Current Stock",
|
"stockInfo": "Current Stock",
|
||||||
"packageDetails": "Package Details",
|
"packageDetails": "Package Details",
|
||||||
|
"packageTypeTubeHint": "Tube medications contain fixed amounts (e.g., creams, gels). Stock is not tracked and reminders are not sent.",
|
||||||
|
"packageTypeLiquidHint": "Liquid containers use a simplified reminder model. Low and Critical levels are automatically derived from a single baseline threshold for simplicity.",
|
||||||
"currentStock": "Pills",
|
"currentStock": "Pills",
|
||||||
"packs": "Packs",
|
"packs": "Packs",
|
||||||
"blistersPerPack": "Blisters/Pack",
|
"blistersPerPack": "Blisters/Pack",
|
||||||
@@ -436,6 +489,7 @@
|
|||||||
},
|
},
|
||||||
"validation": {
|
"validation": {
|
||||||
"required": "This field is required",
|
"required": "This field is required",
|
||||||
|
"nameOrGenericRequired": "Either commercial name or generic name is required",
|
||||||
"maxLength": "Maximum {{max}} characters ({{current}}/{{max}})",
|
"maxLength": "Maximum {{max}} characters ({{current}}/{{max}})",
|
||||||
"tooLong": "{{current}}/{{max}} characters"
|
"tooLong": "{{current}}/{{max}} characters"
|
||||||
},
|
},
|
||||||
@@ -576,9 +630,11 @@
|
|||||||
"loosePills": "Loose pills",
|
"loosePills": "Loose pills",
|
||||||
"pillsPerBlister": "({{count}} pills each)",
|
"pillsPerBlister": "({{count}} pills each)",
|
||||||
"packageSize": "Package size: {{count}} pills",
|
"packageSize": "Package size: {{count}} pills",
|
||||||
|
"packageSizeAmount": "Package size: {{count}} {{unit}}",
|
||||||
"packageSizeBreakdown": "{{packCount}} x {{sizePerPack}} pills Pack = {{total}} pills",
|
"packageSizeBreakdown": "{{packCount}} x {{sizePerPack}} pills Pack = {{total}} pills",
|
||||||
"currentComposition": "Current stock: {{fullBlisters}} full blisters + {{partialPills}} partial + {{loosePills}} loose = {{total}} pills",
|
"currentComposition": "Current stock: {{fullBlisters}} full blisters + {{partialPills}} partial + {{loosePills}} loose = {{total}} pills",
|
||||||
"maxExceeded": "Maximum package size is {{count}} pills. Values were capped.",
|
"maxExceeded": "Maximum package size is {{count}} pills. Values were capped.",
|
||||||
|
"maxExceededAmount": "Maximum package size is {{count}} {{unit}}. Values were capped.",
|
||||||
"decreaseValue": "Decrease value",
|
"decreaseValue": "Decrease value",
|
||||||
"increaseValue": "Increase value",
|
"increaseValue": "Increase value",
|
||||||
"currentTotal": "Current total",
|
"currentTotal": "Current total",
|
||||||
@@ -638,6 +694,7 @@
|
|||||||
"docPackageType": "Package Type",
|
"docPackageType": "Package Type",
|
||||||
"docBlister": "Blister Pack",
|
"docBlister": "Blister Pack",
|
||||||
"docBottle": "Pill Bottle",
|
"docBottle": "Pill Bottle",
|
||||||
|
"docTube": "Tube",
|
||||||
"docPacks": "Packs",
|
"docPacks": "Packs",
|
||||||
"docBlistersPerPack": "Blisters per pack",
|
"docBlistersPerPack": "Blisters per pack",
|
||||||
"docPillsPerBlister": "Pills per blister",
|
"docPillsPerBlister": "Pills per blister",
|
||||||
|
|||||||
@@ -6,6 +6,14 @@ import { ConfirmModal, MedicationAvatar } from "../components";
|
|||||||
import { useAuth } from "../components/Auth";
|
import { useAuth } from "../components/Auth";
|
||||||
import { useAppContext } from "../context";
|
import { useAppContext } from "../context";
|
||||||
import { useModalHistory } from "../hooks";
|
import { useModalHistory } from "../hooks";
|
||||||
|
import {
|
||||||
|
allowsPillFormSelection,
|
||||||
|
type Coverage,
|
||||||
|
getMedDisplayName,
|
||||||
|
isAmountBasedPackageType,
|
||||||
|
isLiquidContainerPackageType,
|
||||||
|
isTubePackageType,
|
||||||
|
} from "../types";
|
||||||
import { formatNumber, getExpiryClass, getSystemLocale } from "../utils/formatters";
|
import { formatNumber, getExpiryClass, getSystemLocale } from "../utils/formatters";
|
||||||
import { expandDoseIds, getStockStatus, isDoseDismissed } from "../utils/schedule";
|
import { expandDoseIds, getStockStatus, isDoseDismissed } from "../utils/schedule";
|
||||||
import {
|
import {
|
||||||
@@ -86,6 +94,7 @@ export function DashboardPage() {
|
|||||||
settings.lowStockDays,
|
settings.lowStockDays,
|
||||||
coverage.low,
|
coverage.low,
|
||||||
coverage.all,
|
coverage.all,
|
||||||
|
meds,
|
||||||
settings.lastAutoEmailSent,
|
settings.lastAutoEmailSent,
|
||||||
settings.lastNotificationType,
|
settings.lastNotificationType,
|
||||||
settings.lastNotificationChannel,
|
settings.lastNotificationChannel,
|
||||||
@@ -118,7 +127,7 @@ export function DashboardPage() {
|
|||||||
})
|
})
|
||||||
.map((med) => ({
|
.map((med) => ({
|
||||||
id: med.id,
|
id: med.id,
|
||||||
name: med.name,
|
name: getMedDisplayName(med),
|
||||||
remainingRefills: med.prescriptionRemainingRefills ?? 0,
|
remainingRefills: med.prescriptionRemainingRefills ?? 0,
|
||||||
threshold: med.prescriptionLowRefillThreshold ?? 1,
|
threshold: med.prescriptionLowRefillThreshold ?? 1,
|
||||||
}))
|
}))
|
||||||
@@ -128,6 +137,158 @@ export function DashboardPage() {
|
|||||||
const showOnlyToday = settings.upcomingTodayOnly;
|
const showOnlyToday = settings.upcomingTodayOnly;
|
||||||
|
|
||||||
const prescriptionEmptyCount = prescriptionLowMeds.filter((med) => med.remainingRefills <= 0).length;
|
const prescriptionEmptyCount = prescriptionLowMeds.filter((med) => med.remainingRefills <= 0).length;
|
||||||
|
|
||||||
|
const getTubeUnitLabel = (med: (typeof meds)[number] | undefined, value: number) =>
|
||||||
|
isLiquidContainerPackageType(med?.packageType) || med?.medicationForm === "liquid"
|
||||||
|
? t("form.packageAmountUnitMl")
|
||||||
|
: t("form.blisters.applications", { count: Math.abs(value) });
|
||||||
|
|
||||||
|
const formatStockLabel = (med: (typeof meds)[number] | undefined, medsLeft: number) => {
|
||||||
|
if (isLiquidContainerPackageType(med?.packageType)) {
|
||||||
|
return `${formatNumber(medsLeft)} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
if (isTubePackageType(med?.packageType)) {
|
||||||
|
return `${formatNumber(medsLeft)} ${getTubeUnitLabel(med, medsLeft)}`;
|
||||||
|
}
|
||||||
|
return t("table.pillsCount", { count: Math.round(medsLeft) });
|
||||||
|
};
|
||||||
|
|
||||||
|
const convertLiquidUsageToMl = (usage: number, unit: "ml" | "tsp" | "tbsp" | null | undefined): number => {
|
||||||
|
if (unit === "tsp") return usage * 5;
|
||||||
|
if (unit === "tbsp") return usage * 15;
|
||||||
|
return usage;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getLiquidCountUnitLabel = (unit: "ml" | "tsp" | "tbsp" | null | undefined, usage: number): string => {
|
||||||
|
if (unit === "tsp") return t("form.blisters.teaspoons", { count: Math.abs(usage) });
|
||||||
|
if (unit === "tbsp") return t("form.blisters.tablespoons", { count: Math.abs(usage) });
|
||||||
|
return t("form.packageAmountUnitMl");
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatLiquidUsageLabel = (usage: number, unit: "ml" | "tsp" | "tbsp" | null | undefined): string => {
|
||||||
|
const normalizedUsage = Number(usage);
|
||||||
|
if (!Number.isFinite(normalizedUsage) || normalizedUsage <= 0) {
|
||||||
|
return `0 ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (unit === "ml" || unit == null) {
|
||||||
|
return `${formatNumber(normalizedUsage)} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const mlTotal = convertLiquidUsageToMl(normalizedUsage, unit);
|
||||||
|
return `${formatNumber(normalizedUsage)} ${getLiquidCountUnitLabel(unit, normalizedUsage)} ${formatNumber(mlTotal)} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatDoseUsageLabel = (
|
||||||
|
med: (typeof meds)[number] | undefined,
|
||||||
|
usage: number,
|
||||||
|
intakeUnit?: "ml" | "tsp" | "tbsp" | null
|
||||||
|
) => {
|
||||||
|
if (isLiquidContainerPackageType(med?.packageType)) {
|
||||||
|
return formatLiquidUsageLabel(usage, intakeUnit);
|
||||||
|
}
|
||||||
|
if (isTubePackageType(med?.packageType)) {
|
||||||
|
return `${usage} ${getTubeUnitLabel(med, usage)}`;
|
||||||
|
}
|
||||||
|
return `${usage} ${usage !== 1 ? t("common.pills") : t("common.pill")}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatTotalUsageLabel = (
|
||||||
|
med: (typeof meds)[number] | undefined,
|
||||||
|
total: number,
|
||||||
|
intakeUnit?: "ml" | "tsp" | "tbsp" | null,
|
||||||
|
doses?: Array<{ usage: number; intakeUnit?: "ml" | "tsp" | "tbsp" | null }>
|
||||||
|
) => {
|
||||||
|
if (isLiquidContainerPackageType(med?.packageType)) {
|
||||||
|
if (doses && doses.length > 0) {
|
||||||
|
const normalizedDoses = doses.filter((dose) => Number.isFinite(Number(dose.usage)) && Number(dose.usage) > 0);
|
||||||
|
if (normalizedDoses.length > 0) {
|
||||||
|
const allUnits = new Set(normalizedDoses.map((dose) => dose.intakeUnit ?? "ml"));
|
||||||
|
const totalMl = normalizedDoses.reduce(
|
||||||
|
(sum, dose) => sum + convertLiquidUsageToMl(Number(dose.usage), dose.intakeUnit ?? "ml"),
|
||||||
|
0
|
||||||
|
);
|
||||||
|
|
||||||
|
if (allUnits.size === 1) {
|
||||||
|
const onlyUnit = normalizedDoses[0]?.intakeUnit ?? "ml";
|
||||||
|
const totalUsageInUnit = normalizedDoses.reduce((sum, dose) => sum + Number(dose.usage), 0);
|
||||||
|
return formatLiquidUsageLabel(totalUsageInUnit, onlyUnit);
|
||||||
|
}
|
||||||
|
|
||||||
|
return `${formatNumber(totalMl)} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return formatLiquidUsageLabel(total, intakeUnit);
|
||||||
|
}
|
||||||
|
if (isTubePackageType(med?.packageType)) {
|
||||||
|
return `${total} ${getTubeUnitLabel(med, total)}`;
|
||||||
|
}
|
||||||
|
return t("common.pillsTotal", { count: total });
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatDailyConsumption = (med: (typeof meds)[number] | undefined) => {
|
||||||
|
if (!med) return "-";
|
||||||
|
|
||||||
|
const intakes =
|
||||||
|
med.intakes && med.intakes.length > 0
|
||||||
|
? med.intakes
|
||||||
|
: med.blisters.map((blister) => ({
|
||||||
|
usage: blister.usage,
|
||||||
|
every: blister.every,
|
||||||
|
intakeUnit: null as "ml" | "tsp" | "tbsp" | null,
|
||||||
|
takenBy: null as string | null,
|
||||||
|
}));
|
||||||
|
|
||||||
|
if (intakes.length === 0) return "-";
|
||||||
|
|
||||||
|
let dailyTotal = 0;
|
||||||
|
for (const intake of intakes) {
|
||||||
|
const usage = Number(intake.usage);
|
||||||
|
const every = Math.max(1, Number(intake.every) || 1);
|
||||||
|
if (!Number.isFinite(usage) || usage <= 0) continue;
|
||||||
|
|
||||||
|
const hasPerIntakeTakenBy = typeof intake.takenBy === "string" && intake.takenBy.trim().length > 0;
|
||||||
|
const personMultiplier = hasPerIntakeTakenBy ? 1 : Math.max(1, med.takenBy?.length ?? 0);
|
||||||
|
const normalizedUsage = (usage * personMultiplier) / every;
|
||||||
|
|
||||||
|
if (isLiquidContainerPackageType(med.packageType)) {
|
||||||
|
dailyTotal += convertLiquidUsageToMl(normalizedUsage, intake.intakeUnit ?? "ml");
|
||||||
|
} else {
|
||||||
|
dailyTotal += normalizedUsage;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (dailyTotal <= 0) return "-";
|
||||||
|
|
||||||
|
if (isLiquidContainerPackageType(med.packageType)) {
|
||||||
|
return t("table.perDayWithUnit", { value: formatNumber(dailyTotal), unit: t("form.packageAmountUnitMl") });
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isTubePackageType(med.packageType)) {
|
||||||
|
const tubeUnit =
|
||||||
|
med.medicationForm === "liquid"
|
||||||
|
? t("form.packageAmountUnitMl")
|
||||||
|
: t("form.blisters.applications", { count: Math.abs(dailyTotal) });
|
||||||
|
return t("table.perDayWithUnit", { value: formatNumber(dailyTotal), unit: tubeUnit });
|
||||||
|
}
|
||||||
|
|
||||||
|
const pillUnit = dailyTotal === 1 ? t("common.pill") : t("common.pills");
|
||||||
|
return t("table.perDayWithUnit", { value: formatNumber(dailyTotal), unit: pillUnit });
|
||||||
|
};
|
||||||
|
|
||||||
|
const shouldHideNoScheduleStatusForTube = (
|
||||||
|
med: (typeof meds)[number] | undefined,
|
||||||
|
status: { className: string; label: string } | null
|
||||||
|
) => isTubePackageType(med?.packageType) && status?.label === "status.noSchedule";
|
||||||
|
|
||||||
|
const getVisibleStockStatus = (
|
||||||
|
med: (typeof meds)[number] | undefined,
|
||||||
|
status: { className: string; label: string } | null
|
||||||
|
) => (shouldHideNoScheduleStatusForTube(med, status) ? null : status);
|
||||||
|
|
||||||
|
const getMedByName = (name: string) => meds.find((m) => getMedDisplayName(m) === name);
|
||||||
|
|
||||||
const prescriptionStatus =
|
const prescriptionStatus =
|
||||||
prescriptionRemindersEnabled && prescriptionLowMeds.length > 0
|
prescriptionRemindersEnabled && prescriptionLowMeds.length > 0
|
||||||
? {
|
? {
|
||||||
@@ -250,9 +411,11 @@ export function DashboardPage() {
|
|||||||
<span className="reminder-status-label">{t("dashboard.reminders.needsRefill")}:</span>
|
<span className="reminder-status-label">{t("dashboard.reminders.needsRefill")}:</span>
|
||||||
<span className="reminder-status-value">
|
<span className="reminder-status-value">
|
||||||
{reminderData.lowStockMeds.map((med, idx) => {
|
{reminderData.lowStockMeds.map((med, idx) => {
|
||||||
const medication = meds.find((m) => m.name === med.name);
|
const medication = meds.find((m) => getMedDisplayName(m) === med.name);
|
||||||
const cov = coverage.all.find((c) => c.name === med.name);
|
const cov = coverage.all.find((c) => c.name === med.name);
|
||||||
const status = cov ? getStockStatus(cov.daysLeft, cov.medsLeft, stockThresholds) : null;
|
const status = cov
|
||||||
|
? getStockStatus(cov.daysLeft, cov.medsLeft, stockThresholds, medication?.packageType)
|
||||||
|
: null;
|
||||||
const textClass =
|
const textClass =
|
||||||
status?.className === "danger"
|
status?.className === "danger"
|
||||||
? "danger-text"
|
? "danger-text"
|
||||||
@@ -322,7 +485,7 @@ export function DashboardPage() {
|
|||||||
(() => {
|
(() => {
|
||||||
const names = reminderData.lastStockSent!.medNames!.split(", ");
|
const names = reminderData.lastStockSent!.medNames!.split(", ");
|
||||||
return names.map((name, idx) => {
|
return names.map((name, idx) => {
|
||||||
const medication = meds.find((m) => m.name === name);
|
const medication = meds.find((m) => getMedDisplayName(m) === name);
|
||||||
return (
|
return (
|
||||||
<span key={name}>
|
<span key={name}>
|
||||||
{idx > 0 && ", "}
|
{idx > 0 && ", "}
|
||||||
@@ -353,7 +516,9 @@ export function DashboardPage() {
|
|||||||
<span className="reminder-status-value">
|
<span className="reminder-status-value">
|
||||||
{reminderData.lastIntakeSent.medName &&
|
{reminderData.lastIntakeSent.medName &&
|
||||||
(() => {
|
(() => {
|
||||||
const medication = meds.find((m) => m.name === reminderData.lastIntakeSent!.medName);
|
const medication = meds.find(
|
||||||
|
(m) => getMedDisplayName(m) === reminderData.lastIntakeSent!.medName
|
||||||
|
);
|
||||||
return medication ? (
|
return medication ? (
|
||||||
<span
|
<span
|
||||||
className="med-link clickable"
|
className="med-link clickable"
|
||||||
@@ -408,7 +573,9 @@ export function DashboardPage() {
|
|||||||
const lowStockMap = new Map<string, Coverage>();
|
const lowStockMap = new Map<string, Coverage>();
|
||||||
for (const c of coverage.all) {
|
for (const c of coverage.all) {
|
||||||
if (c.daysLeft === null && c.medsLeft > 0) continue; // no schedule, has stock
|
if (c.daysLeft === null && c.medsLeft > 0) continue; // no schedule, has stock
|
||||||
if (c.medsLeft <= 0 || c.daysLeft === null || c.daysLeft < settings.lowStockDays) {
|
const med = getMedByName(c.name);
|
||||||
|
const status = getStockStatus(c.daysLeft, c.medsLeft, stockThresholds, med?.packageType);
|
||||||
|
if (status.className === "danger" || status.className === "warning") {
|
||||||
const existing = lowStockMap.get(c.name);
|
const existing = lowStockMap.get(c.name);
|
||||||
if (!existing || (c.daysLeft ?? 0) < (existing.daysLeft ?? 0)) {
|
if (!existing || (c.daysLeft ?? 0) < (existing.daysLeft ?? 0)) {
|
||||||
lowStockMap.set(c.name, c);
|
lowStockMap.set(c.name, c);
|
||||||
@@ -428,8 +595,8 @@ export function DashboardPage() {
|
|||||||
<p>
|
<p>
|
||||||
{t("dashboard.reorder.lowWarningPrefix")}{" "}
|
{t("dashboard.reorder.lowWarningPrefix")}{" "}
|
||||||
{lowStockMeds.map((c, idx) => {
|
{lowStockMeds.map((c, idx) => {
|
||||||
const med = meds.find((m) => m.name === c.name);
|
const med = meds.find((m) => getMedDisplayName(m) === c.name);
|
||||||
const status = getStockStatus(c.daysLeft, c.medsLeft, stockThresholds);
|
const status = getStockStatus(c.daysLeft, c.medsLeft, stockThresholds, med?.packageType);
|
||||||
const textClass =
|
const textClass =
|
||||||
status.className === "danger"
|
status.className === "danger"
|
||||||
? "danger-text"
|
? "danger-text"
|
||||||
@@ -473,10 +640,11 @@ export function DashboardPage() {
|
|||||||
<div className="card-head">
|
<div className="card-head">
|
||||||
<h2>{t("dashboard.overview.title")}</h2>
|
<h2>{t("dashboard.overview.title")}</h2>
|
||||||
</div>
|
</div>
|
||||||
<div className="table table-7">
|
<div className="table table-8">
|
||||||
<div className="table-head">
|
<div className="table-head">
|
||||||
<span>{t("table.name")}</span>
|
<span>{t("table.name")}</span>
|
||||||
<span>{t("table.stock")}</span>
|
<span>{t("table.stock")}</span>
|
||||||
|
<span>{t("table.dailyConsumption")}</span>
|
||||||
<span>{t("table.stockDetails")}</span>
|
<span>{t("table.stockDetails")}</span>
|
||||||
<span>{t("table.daysLeft")}</span>
|
<span>{t("table.daysLeft")}</span>
|
||||||
<span>{t("table.runsOut")}</span>
|
<span>{t("table.runsOut")}</span>
|
||||||
@@ -484,13 +652,14 @@ export function DashboardPage() {
|
|||||||
<span>{t("table.status")}</span>
|
<span>{t("table.status")}</span>
|
||||||
</div>
|
</div>
|
||||||
{coverage.all.map((row) => {
|
{coverage.all.map((row) => {
|
||||||
const status = getStockStatus(row.daysLeft, row.medsLeft, stockThresholds);
|
const med = meds.find((m) => getMedDisplayName(m) === row.name);
|
||||||
const med = meds.find((m) => m.name === row.name);
|
const rawStatus = getStockStatus(row.daysLeft, row.medsLeft, stockThresholds, med?.packageType);
|
||||||
|
const status = getVisibleStockStatus(med, rawStatus);
|
||||||
const expiryClass = getExpiryClass(med?.expiryDate, settings.expiryWarningDays);
|
const expiryClass = getExpiryClass(med?.expiryDate, settings.expiryWarningDays);
|
||||||
const textClass =
|
const textClass =
|
||||||
status.className === "danger"
|
rawStatus.className === "danger"
|
||||||
? "danger-text"
|
? "danger-text"
|
||||||
: status.className === "warning"
|
: rawStatus.className === "warning"
|
||||||
? "warning-text"
|
? "warning-text"
|
||||||
: "success-text";
|
: "success-text";
|
||||||
const stock = getBlisterStock(
|
const stock = getBlisterStock(
|
||||||
@@ -584,15 +753,18 @@ export function DashboardPage() {
|
|||||||
</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
<span data-label={t("table.stock")} className={textClass}>
|
<span data-label={t("table.stock")} className={textClass}>
|
||||||
{med?.packageType === "bottle"
|
{isAmountBasedPackageType(med?.packageType)
|
||||||
? t("table.pillsCount", { count: Math.round(row.medsLeft) })
|
? formatStockLabel(med, row.medsLeft)
|
||||||
: formatFullBlisters(stock.fullBlisters, t)}
|
: formatFullBlisters(stock.fullBlisters, t)}
|
||||||
</span>
|
</span>
|
||||||
|
<span data-label={t("table.dailyConsumption")} className={textClass}>
|
||||||
|
{formatDailyConsumption(med)}
|
||||||
|
</span>
|
||||||
<span
|
<span
|
||||||
data-label={t("table.stockDetails")}
|
data-label={t("table.stockDetails")}
|
||||||
className={`${textClass}${med?.packageType === "bottle" ? " hide-on-card" : ""}`}
|
className={`${textClass}${isAmountBasedPackageType(med?.packageType) ? " hide-on-card" : ""}`}
|
||||||
>
|
>
|
||||||
{med?.packageType === "bottle"
|
{isAmountBasedPackageType(med?.packageType)
|
||||||
? "—"
|
? "—"
|
||||||
: formatOpenBlisterAndLoose(
|
: formatOpenBlisterAndLoose(
|
||||||
stock.openBlisterPills,
|
stock.openBlisterPills,
|
||||||
@@ -614,8 +786,8 @@ export function DashboardPage() {
|
|||||||
})
|
})
|
||||||
: "-"}
|
: "-"}
|
||||||
</span>
|
</span>
|
||||||
<span data-label={t("table.status")} className={`status-chip ${status.className}`}>
|
<span data-label={t("table.status")} className={status ? `status-chip ${status.className}` : ""}>
|
||||||
{t(status.label)}
|
{status ? t(status.label) : "-"}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
@@ -673,7 +845,7 @@ export function DashboardPage() {
|
|||||||
|
|
||||||
// Count missed doses that are NOT dismissed (for warning icon)
|
// Count missed doses that are NOT dismissed (for warning icon)
|
||||||
const missedNotDismissedCount = day.meds.reduce((count, item) => {
|
const missedNotDismissedCount = day.meds.reduce((count, item) => {
|
||||||
const med = meds.find((m) => m.name === item.medName);
|
const med = meds.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
const dismissedUntilDate = med?.dismissedUntil ?? undefined;
|
const dismissedUntilDate = med?.dismissedUntil ?? undefined;
|
||||||
return (
|
return (
|
||||||
count +
|
count +
|
||||||
@@ -729,12 +901,13 @@ export function DashboardPage() {
|
|||||||
</div>
|
</div>
|
||||||
{!isCollapsed &&
|
{!isCollapsed &&
|
||||||
day.meds.map((item) => {
|
day.meds.map((item) => {
|
||||||
const med = meds.find((m) => m.name === item.medName);
|
const med = meds.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
const medCov = coverageByMed[item.medName];
|
const medCov = coverageByMed[item.medName];
|
||||||
const isEmpty = medCov ? medCov.medsLeft <= 0 : false;
|
const isEmpty = medCov ? medCov.medsLeft <= 0 : false;
|
||||||
const status = medCov
|
const rawStatus = medCov
|
||||||
? getStockStatus(medCov.daysLeft, medCov.medsLeft, stockThresholds)
|
? getStockStatus(medCov.daysLeft, medCov.medsLeft, stockThresholds, med?.packageType)
|
||||||
: null;
|
: null;
|
||||||
|
const status = getVisibleStockStatus(med, rawStatus);
|
||||||
const itemDoseIds = expandDoseIds(item.doses);
|
const itemDoseIds = expandDoseIds(item.doses);
|
||||||
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
||||||
return (
|
return (
|
||||||
@@ -769,7 +942,9 @@ export function DashboardPage() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="tag-row">
|
<div className="tag-row">
|
||||||
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span>
|
<span className="tag subtle">
|
||||||
|
{formatTotalUsageLabel(med, item.total, item.doses[0]?.intakeUnit, item.doses)}
|
||||||
|
</span>
|
||||||
{status && (
|
{status && (
|
||||||
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span>
|
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span>
|
||||||
)}
|
)}
|
||||||
@@ -784,9 +959,9 @@ export function DashboardPage() {
|
|||||||
<span className="dose-time">{dose.timeStr}</span>
|
<span className="dose-time">{dose.timeStr}</span>
|
||||||
<span className="dose-usage">
|
<span className="dose-usage">
|
||||||
<span className="dose-usage-main">
|
<span className="dose-usage-main">
|
||||||
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")}
|
{formatDoseUsageLabel(med, dose.usage, dose.intakeUnit)}
|
||||||
</span>
|
</span>
|
||||||
{med?.pillWeightMg && (
|
{allowsPillFormSelection(med?.packageType) && med?.pillWeightMg && (
|
||||||
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
||||||
)}
|
)}
|
||||||
</span>
|
</span>
|
||||||
@@ -831,7 +1006,8 @@ export function DashboardPage() {
|
|||||||
🤖
|
🤖
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
↩
|
<span className="dose-btn-label">{t("common.undo")}</span>
|
||||||
|
<span aria-hidden="true">↩</span>
|
||||||
</button>
|
</button>
|
||||||
) : (
|
) : (
|
||||||
<button
|
<button
|
||||||
@@ -943,7 +1119,13 @@ export function DashboardPage() {
|
|||||||
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
|
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
|
||||||
if (willBeOutOfStock) return "danger";
|
if (willBeOutOfStock) return "danger";
|
||||||
if (!medCoverage) return "success";
|
if (!medCoverage) return "success";
|
||||||
const status = getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds);
|
const med = getMedByName(item.medName);
|
||||||
|
const status = getStockStatus(
|
||||||
|
medCoverage.daysLeft,
|
||||||
|
medCoverage.medsLeft,
|
||||||
|
stockThresholds,
|
||||||
|
med?.packageType
|
||||||
|
);
|
||||||
return status.className;
|
return status.className;
|
||||||
});
|
});
|
||||||
const worstStatus = dayStockStatuses.includes("danger")
|
const worstStatus = dayStockStatuses.includes("danger")
|
||||||
@@ -986,15 +1168,21 @@ export function DashboardPage() {
|
|||||||
{!isCollapsed &&
|
{!isCollapsed &&
|
||||||
day.meds.map((item) => {
|
day.meds.map((item) => {
|
||||||
const medCoverage = coverageByMed[item.medName];
|
const medCoverage = coverageByMed[item.medName];
|
||||||
const med = meds.find((m) => m.name === item.medName);
|
const med = meds.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
const depletionTime = depletionByMed[item.medName];
|
const depletionTime = depletionByMed[item.medName];
|
||||||
const isEmpty = medCoverage ? medCoverage.medsLeft <= 0 : false;
|
const isEmpty = medCoverage ? medCoverage.medsLeft <= 0 : false;
|
||||||
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
|
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
|
||||||
const status = willBeOutOfStock
|
const status = willBeOutOfStock
|
||||||
? { className: "danger", label: "status.outOfStock" }
|
? { className: "danger", label: "status.outOfStock" }
|
||||||
: medCoverage
|
: medCoverage
|
||||||
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds)
|
? getStockStatus(
|
||||||
|
medCoverage.daysLeft,
|
||||||
|
medCoverage.medsLeft,
|
||||||
|
stockThresholds,
|
||||||
|
med?.packageType
|
||||||
|
)
|
||||||
: null;
|
: null;
|
||||||
|
const visibleStatus = getVisibleStockStatus(med, status);
|
||||||
const itemDoseIds = expandDoseIds(item.doses);
|
const itemDoseIds = expandDoseIds(item.doses);
|
||||||
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
||||||
return (
|
return (
|
||||||
@@ -1029,9 +1217,13 @@ export function DashboardPage() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="tag-row">
|
<div className="tag-row">
|
||||||
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span>
|
<span className="tag subtle">
|
||||||
{status && (
|
{formatTotalUsageLabel(med, item.total, item.doses[0]?.intakeUnit, item.doses)}
|
||||||
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span>
|
</span>
|
||||||
|
{visibleStatus && (
|
||||||
|
<span className={`status-chip small ${visibleStatus.className}`}>
|
||||||
|
{t(visibleStatus.label)}
|
||||||
|
</span>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -1048,9 +1240,9 @@ export function DashboardPage() {
|
|||||||
<span className="dose-time">{dose.timeStr}</span>
|
<span className="dose-time">{dose.timeStr}</span>
|
||||||
<span className="dose-usage">
|
<span className="dose-usage">
|
||||||
<span className="dose-usage-main">
|
<span className="dose-usage-main">
|
||||||
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")}
|
{formatDoseUsageLabel(med, dose.usage, dose.intakeUnit)}
|
||||||
</span>
|
</span>
|
||||||
{med?.pillWeightMg && (
|
{allowsPillFormSelection(med?.packageType) && med?.pillWeightMg && (
|
||||||
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
||||||
)}
|
)}
|
||||||
</span>
|
</span>
|
||||||
@@ -1095,7 +1287,8 @@ export function DashboardPage() {
|
|||||||
🤖
|
🤖
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
↩
|
<span className="dose-btn-label">{t("common.undo")}</span>
|
||||||
|
<span aria-hidden="true">↩</span>
|
||||||
</button>
|
</button>
|
||||||
) : (
|
) : (
|
||||||
<button
|
<button
|
||||||
@@ -1175,7 +1368,13 @@ export function DashboardPage() {
|
|||||||
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
|
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
|
||||||
if (willBeOutOfStock) return "danger";
|
if (willBeOutOfStock) return "danger";
|
||||||
if (!medCoverage) return "success";
|
if (!medCoverage) return "success";
|
||||||
const status = getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds);
|
const med = getMedByName(item.medName);
|
||||||
|
const status = getStockStatus(
|
||||||
|
medCoverage.daysLeft,
|
||||||
|
medCoverage.medsLeft,
|
||||||
|
stockThresholds,
|
||||||
|
med?.packageType
|
||||||
|
);
|
||||||
return status.className;
|
return status.className;
|
||||||
});
|
});
|
||||||
const worstStatus = dayStockStatuses.includes("danger")
|
const worstStatus = dayStockStatuses.includes("danger")
|
||||||
@@ -1217,15 +1416,21 @@ export function DashboardPage() {
|
|||||||
{!isCollapsed &&
|
{!isCollapsed &&
|
||||||
day.meds.map((item) => {
|
day.meds.map((item) => {
|
||||||
const medCoverage = coverageByMed[item.medName];
|
const medCoverage = coverageByMed[item.medName];
|
||||||
const med = meds.find((m) => m.name === item.medName);
|
const med = meds.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
const depletionTime = depletionByMed[item.medName];
|
const depletionTime = depletionByMed[item.medName];
|
||||||
const _isEmpty = medCoverage ? medCoverage.medsLeft <= 0 : false;
|
const _isEmpty = medCoverage ? medCoverage.medsLeft <= 0 : false;
|
||||||
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
|
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
|
||||||
const status = willBeOutOfStock
|
const status = willBeOutOfStock
|
||||||
? { className: "danger", label: "status.outOfStock" }
|
? { className: "danger", label: "status.outOfStock" }
|
||||||
: medCoverage
|
: medCoverage
|
||||||
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, stockThresholds)
|
? getStockStatus(
|
||||||
|
medCoverage.daysLeft,
|
||||||
|
medCoverage.medsLeft,
|
||||||
|
stockThresholds,
|
||||||
|
med?.packageType
|
||||||
|
)
|
||||||
: null;
|
: null;
|
||||||
|
const visibleStatus = getVisibleStockStatus(med, status);
|
||||||
const itemDoseIds = expandDoseIds(item.doses);
|
const itemDoseIds = expandDoseIds(item.doses);
|
||||||
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
||||||
return (
|
return (
|
||||||
@@ -1260,9 +1465,13 @@ export function DashboardPage() {
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="tag-row">
|
<div className="tag-row">
|
||||||
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span>
|
<span className="tag subtle">
|
||||||
{status && (
|
{formatTotalUsageLabel(med, item.total, item.doses[0]?.intakeUnit, item.doses)}
|
||||||
<span className={`status-chip small ${status.className}`}>{t(status.label)}</span>
|
</span>
|
||||||
|
{visibleStatus && (
|
||||||
|
<span className={`status-chip small ${visibleStatus.className}`}>
|
||||||
|
{t(visibleStatus.label)}
|
||||||
|
</span>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -1275,9 +1484,9 @@ export function DashboardPage() {
|
|||||||
<span className="dose-time">{dose.timeStr}</span>
|
<span className="dose-time">{dose.timeStr}</span>
|
||||||
<span className="dose-usage">
|
<span className="dose-usage">
|
||||||
<span className="dose-usage-main">
|
<span className="dose-usage-main">
|
||||||
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")}
|
{formatDoseUsageLabel(med, dose.usage, dose.intakeUnit)}
|
||||||
</span>
|
</span>
|
||||||
{med?.pillWeightMg && (
|
{allowsPillFormSelection(med?.packageType) && med?.pillWeightMg && (
|
||||||
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
||||||
)}
|
)}
|
||||||
</span>
|
</span>
|
||||||
@@ -1322,7 +1531,8 @@ export function DashboardPage() {
|
|||||||
🤖
|
🤖
|
||||||
</span>
|
</span>
|
||||||
)}
|
)}
|
||||||
↩
|
<span className="dose-btn-label">{t("common.undo")}</span>
|
||||||
|
<span aria-hidden="true">↩</span>
|
||||||
</button>
|
</button>
|
||||||
) : (
|
) : (
|
||||||
<button
|
<button
|
||||||
|
|||||||
@@ -17,8 +17,20 @@ import {
|
|||||||
import { useAuth } from "../components/Auth";
|
import { useAuth } from "../components/Auth";
|
||||||
import { useAppContext, useUnsavedChanges } from "../context";
|
import { useAppContext, useUnsavedChanges } from "../context";
|
||||||
import { useMedicationForm, useModalHistory, useUnsavedChangesWarning } from "../hooks";
|
import { useMedicationForm, useModalHistory, useUnsavedChangesWarning } from "../hooks";
|
||||||
import type { DoseUnit, Medication } from "../types";
|
import type { DoseUnit, FormState, Medication, PackageType } from "../types";
|
||||||
import { DOSE_UNITS, FIELD_LIMITS, getPackageSize } from "../types";
|
import {
|
||||||
|
allowsPillFormSelection,
|
||||||
|
DOSE_UNITS,
|
||||||
|
FIELD_LIMITS,
|
||||||
|
getMedDisplayName,
|
||||||
|
getPackageProfile,
|
||||||
|
getPackageSize,
|
||||||
|
isAmountBasedPackageType,
|
||||||
|
isLiquidContainerPackageType,
|
||||||
|
isTubePackageType,
|
||||||
|
normalizePackageType,
|
||||||
|
PACKAGE_PROFILES,
|
||||||
|
} from "../types";
|
||||||
import { combineDateAndTime, formatDate, formatDateTime, formatNumber } from "../utils/formatters";
|
import { combineDateAndTime, formatDate, formatDateTime, formatNumber } from "../utils/formatters";
|
||||||
import { MAX_IMAGE_UPLOAD_BYTES, resolveImageUploadError } from "../utils/image-upload";
|
import { MAX_IMAGE_UPLOAD_BYTES, resolveImageUploadError } from "../utils/image-upload";
|
||||||
import { log } from "../utils/logger";
|
import { log } from "../utils/logger";
|
||||||
@@ -239,7 +251,7 @@ export function MedicationsPage() {
|
|||||||
|
|
||||||
// Calculate total tablets
|
// Calculate total tablets
|
||||||
const totalTablets = useMemo(() => {
|
const totalTablets = useMemo(() => {
|
||||||
if (form.packageType === "bottle") {
|
if (isAmountBasedPackageType(form.packageType)) {
|
||||||
// For bottle type, looseTablets is the current stock
|
// For bottle type, looseTablets is the current stock
|
||||||
return Number(form.looseTablets) || 0;
|
return Number(form.looseTablets) || 0;
|
||||||
}
|
}
|
||||||
@@ -254,6 +266,14 @@ export function MedicationsPage() {
|
|||||||
|
|
||||||
const dateConsistencyError = useMemo(() => {
|
const dateConsistencyError = useMemo(() => {
|
||||||
const medicationStartDate = form.medicationStartDate;
|
const medicationStartDate = form.medicationStartDate;
|
||||||
|
const medicationEndDate = form.medicationEndDate;
|
||||||
|
if (medicationStartDate && medicationEndDate && medicationEndDate < medicationStartDate) {
|
||||||
|
return t("form.validation.endDateBeforeStart", {
|
||||||
|
medicationStartDate,
|
||||||
|
medicationEndDate,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
if (!medicationStartDate) return null;
|
if (!medicationStartDate) return null;
|
||||||
|
|
||||||
const conflictingIntake = form.intakes.find((intake) => intake.startDate && intake.startDate < medicationStartDate);
|
const conflictingIntake = form.intakes.find((intake) => intake.startDate && intake.startDate < medicationStartDate);
|
||||||
@@ -263,7 +283,62 @@ export function MedicationsPage() {
|
|||||||
medicationStartDate,
|
medicationStartDate,
|
||||||
intakeDate: conflictingIntake.startDate,
|
intakeDate: conflictingIntake.startDate,
|
||||||
});
|
});
|
||||||
}, [form.medicationStartDate, form.intakes, t]);
|
}, [form.medicationStartDate, form.medicationEndDate, form.intakes, t]);
|
||||||
|
|
||||||
|
const allowFractionalIntake = useMemo(() => {
|
||||||
|
if (isLiquidContainerPackageType(form.packageType)) return true;
|
||||||
|
if (isTubePackageType(form.packageType)) return form.medicationForm === "liquid";
|
||||||
|
return form.pillForm === "tablet";
|
||||||
|
}, [form.packageType, form.medicationForm, form.pillForm]);
|
||||||
|
|
||||||
|
const getUsageLabel = useCallback(
|
||||||
|
(intakeUnit: "ml" | "tsp" | "tbsp") => {
|
||||||
|
if (isLiquidContainerPackageType(form.packageType)) {
|
||||||
|
if (intakeUnit === "tsp") return t("form.blisters.usageTsp");
|
||||||
|
if (intakeUnit === "tbsp") return t("form.blisters.usageTbsp");
|
||||||
|
return t("form.blisters.usageMl");
|
||||||
|
}
|
||||||
|
if (isTubePackageType(form.packageType)) {
|
||||||
|
return form.medicationForm === "liquid" ? t("form.blisters.usageMl") : t("form.blisters.usageApplication");
|
||||||
|
}
|
||||||
|
if (form.pillForm === "capsule") return t("form.blisters.usageCapsules");
|
||||||
|
return t("form.blisters.usageTablets");
|
||||||
|
},
|
||||||
|
[form.packageType, form.medicationForm, form.pillForm, t]
|
||||||
|
);
|
||||||
|
|
||||||
|
const usesAmountLabels = isTubePackageType(form.packageType) || isLiquidContainerPackageType(form.packageType);
|
||||||
|
const totalCapacityLabel = usesAmountLabels ? t("form.totalAmount") : t("form.totalCapacity");
|
||||||
|
const currentStockLabel = usesAmountLabels ? t("form.currentAmount") : t("form.currentPills");
|
||||||
|
const totalLabel = usesAmountLabels ? t("form.totalAmountLabel") : t("form.total");
|
||||||
|
|
||||||
|
const getMedicationPackageTypeLabel = useCallback(
|
||||||
|
(med: Medication) => {
|
||||||
|
return t(getPackageProfile(med.packageType).labelKey);
|
||||||
|
},
|
||||||
|
[t]
|
||||||
|
);
|
||||||
|
|
||||||
|
const getMedicationStockSuffix = useCallback(
|
||||||
|
(med: Medication) => {
|
||||||
|
if (isTubePackageType(med.packageType)) return "";
|
||||||
|
if (isLiquidContainerPackageType(med.packageType)) return " ml";
|
||||||
|
return ` ${getPackageSize(med) === 1 ? t("common.pill") : t("common.pills")}`;
|
||||||
|
},
|
||||||
|
[t]
|
||||||
|
);
|
||||||
|
|
||||||
|
const getMedicationUsageUnitLabel = useCallback(
|
||||||
|
(med: Medication, usage: number) => {
|
||||||
|
if (isTubePackageType(med.packageType)) {
|
||||||
|
return med.medicationForm === "liquid" ? "ml" : t("form.blisters.usageApplication");
|
||||||
|
}
|
||||||
|
if (isLiquidContainerPackageType(med.packageType)) return "ml";
|
||||||
|
if (usage === 1) return t("common.pill");
|
||||||
|
return t("common.pills");
|
||||||
|
},
|
||||||
|
[t]
|
||||||
|
);
|
||||||
|
|
||||||
const clearEditMedIdParam = useCallback(() => {
|
const clearEditMedIdParam = useCallback(() => {
|
||||||
setSearchParams(
|
setSearchParams(
|
||||||
@@ -450,6 +525,10 @@ export function MedicationsPage() {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (saving) return;
|
if (saving) return;
|
||||||
|
if (form.pillForm === "capsule" && form.intakes.some((i) => !Number.isInteger(Number(i.usage)))) {
|
||||||
|
setShowNameValidation(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
setSaving(true);
|
setSaving(true);
|
||||||
|
|
||||||
// Prepare intakes data with per-intake takenBy
|
// Prepare intakes data with per-intake takenBy
|
||||||
@@ -457,6 +536,7 @@ export function MedicationsPage() {
|
|||||||
usage: Number(intake.usage) || 1,
|
usage: Number(intake.usage) || 1,
|
||||||
every: Number(intake.every) || 1,
|
every: Number(intake.every) || 1,
|
||||||
start: combineDateAndTime(intake.startDate, intake.startTime),
|
start: combineDateAndTime(intake.startDate, intake.startTime),
|
||||||
|
intakeUnit: isLiquidContainerPackageType(form.packageType) ? intake.intakeUnit : null,
|
||||||
takenBy: intake.takenBy.trim() || null, // Empty string becomes null
|
takenBy: intake.takenBy.trim() || null, // Empty string becomes null
|
||||||
intakeRemindersEnabled: intake.intakeRemindersEnabled,
|
intakeRemindersEnabled: intake.intakeRemindersEnabled,
|
||||||
}));
|
}));
|
||||||
@@ -472,19 +552,50 @@ export function MedicationsPage() {
|
|||||||
const remainingRefills = Math.min(Number(form.prescriptionRemainingRefills || 0), authorizedRefills);
|
const remainingRefills = Math.min(Number(form.prescriptionRemainingRefills || 0), authorizedRefills);
|
||||||
const lowRefillThreshold = Math.min(Number(form.prescriptionLowRefillThreshold || 1), authorizedRefills);
|
const lowRefillThreshold = Math.min(Number(form.prescriptionLowRefillThreshold || 1), authorizedRefills);
|
||||||
|
|
||||||
|
let derivedMedicationForm: string;
|
||||||
|
if (isTubePackageType(form.packageType)) {
|
||||||
|
derivedMedicationForm =
|
||||||
|
form.medicationForm === "liquid" || form.medicationForm === "topical" ? form.medicationForm : "topical";
|
||||||
|
} else if (isLiquidContainerPackageType(form.packageType)) {
|
||||||
|
derivedMedicationForm = "liquid";
|
||||||
|
} else {
|
||||||
|
derivedMedicationForm = form.pillForm;
|
||||||
|
}
|
||||||
|
|
||||||
|
const tubeTotalAmount = isTubePackageType(form.packageType)
|
||||||
|
? (Number(form.packCount) || 0) * (Number(form.packageAmountValue ?? 0) || 0)
|
||||||
|
: null;
|
||||||
|
|
||||||
|
let packageAmountUnit = form.packageAmountUnit ?? "ml";
|
||||||
|
if (isTubePackageType(form.packageType)) {
|
||||||
|
packageAmountUnit = "g";
|
||||||
|
} else if (isLiquidContainerPackageType(form.packageType)) {
|
||||||
|
packageAmountUnit = "ml";
|
||||||
|
}
|
||||||
|
|
||||||
const body = {
|
const body = {
|
||||||
name: form.name.trim(),
|
name: form.name.trim(),
|
||||||
genericName: form.genericName.trim() || null,
|
genericName: form.genericName.trim() || null,
|
||||||
takenBy: form.takenBy.length > 0 ? form.takenBy : [],
|
takenBy: form.takenBy.length > 0 ? form.takenBy : [],
|
||||||
packageType: form.packageType,
|
medicationForm: derivedMedicationForm,
|
||||||
packCount: Number(form.packCount) || 0,
|
pillForm:
|
||||||
blistersPerPack: Number(form.blistersPerPack) || 1,
|
isTubePackageType(form.packageType) || isLiquidContainerPackageType(form.packageType) ? null : form.pillForm,
|
||||||
pillsPerBlister: Number(form.pillsPerBlister) || 1,
|
lifecycleCategory: form.lifecycleCategory,
|
||||||
totalPills: Number(form.totalPills) || null,
|
packageType: normalizePackageType(form.packageType),
|
||||||
looseTablets: Number(form.looseTablets) || 0,
|
packCount: isTubePackageType(form.packageType)
|
||||||
|
? Math.max(1, Number(form.packCount) || 1)
|
||||||
|
: Number(form.packCount) || 0,
|
||||||
|
blistersPerPack: isTubePackageType(form.packageType) ? 1 : Number(form.blistersPerPack) || 1,
|
||||||
|
pillsPerBlister: isTubePackageType(form.packageType) ? 1 : Number(form.pillsPerBlister) || 1,
|
||||||
|
packageAmountValue: Number(form.packageAmountValue ?? 0) || 0,
|
||||||
|
packageAmountUnit,
|
||||||
|
totalPills: isTubePackageType(form.packageType) ? tubeTotalAmount : Number(form.totalPills) || null,
|
||||||
|
looseTablets: isTubePackageType(form.packageType) ? tubeTotalAmount || 0 : Number(form.looseTablets) || 0,
|
||||||
pillWeightMg: Number(form.pillWeightMg) || null,
|
pillWeightMg: Number(form.pillWeightMg) || null,
|
||||||
doseUnit: form.doseUnit,
|
doseUnit: form.doseUnit,
|
||||||
medicationStartDate: form.medicationStartDate || null,
|
medicationStartDate: form.medicationStartDate || null,
|
||||||
|
medicationEndDate: form.medicationEndDate || null,
|
||||||
|
autoMarkObsoleteAfterEndDate: form.autoMarkObsoleteAfterEndDate,
|
||||||
expiryDate: form.expiryDate || null,
|
expiryDate: form.expiryDate || null,
|
||||||
notes: form.notes.trim() || null,
|
notes: form.notes.trim() || null,
|
||||||
intakeRemindersEnabled: form.intakeRemindersEnabled,
|
intakeRemindersEnabled: form.intakeRemindersEnabled,
|
||||||
@@ -719,6 +830,7 @@ export function MedicationsPage() {
|
|||||||
setReadOnlyView(true);
|
setReadOnlyView(true);
|
||||||
startEdit(med, openEditModal);
|
startEdit(med, openEditModal);
|
||||||
setViewMode("form");
|
setViewMode("form");
|
||||||
|
scrollToTopForDesktopEdit();
|
||||||
};
|
};
|
||||||
setUnsavedConfirmSource(showEditModal ? "mobile-edit" : "desktop-form");
|
setUnsavedConfirmSource(showEditModal ? "mobile-edit" : "desktop-form");
|
||||||
setShowUnsavedConfirm(true);
|
setShowUnsavedConfirm(true);
|
||||||
@@ -729,6 +841,7 @@ export function MedicationsPage() {
|
|||||||
setActiveTab("general");
|
setActiveTab("general");
|
||||||
startEdit(med, openEditModal);
|
startEdit(med, openEditModal);
|
||||||
setViewMode("form");
|
setViewMode("form");
|
||||||
|
scrollToTopForDesktopEdit();
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleNewEntryClick() {
|
function handleNewEntryClick() {
|
||||||
@@ -792,6 +905,7 @@ export function MedicationsPage() {
|
|||||||
setActiveTab("general");
|
setActiveTab("general");
|
||||||
startEdit(medicationToEdit, openEditModal);
|
startEdit(medicationToEdit, openEditModal);
|
||||||
setViewMode("form");
|
setViewMode("form");
|
||||||
|
scrollToTopForDesktopEdit();
|
||||||
setPendingEditTransition(false);
|
setPendingEditTransition(false);
|
||||||
window.dispatchEvent(new Event("medassist:edit-transition-ready"));
|
window.dispatchEvent(new Event("medassist:edit-transition-ready"));
|
||||||
|
|
||||||
@@ -836,19 +950,21 @@ export function MedicationsPage() {
|
|||||||
<span
|
<span
|
||||||
className={med.imageUrl ? "med-avatar-clickable" : undefined}
|
className={med.imageUrl ? "med-avatar-clickable" : undefined}
|
||||||
onClick={() =>
|
onClick={() =>
|
||||||
med.imageUrl && setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: med.name })
|
med.imageUrl &&
|
||||||
|
setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: getMedDisplayName(med) })
|
||||||
}
|
}
|
||||||
onKeyDown={(e) => {
|
onKeyDown={(e) => {
|
||||||
if (e.key === "Enter" || e.key === " ") {
|
if (e.key === "Enter" || e.key === " ") {
|
||||||
if (med.imageUrl) setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: med.name });
|
if (med.imageUrl)
|
||||||
|
setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: getMedDisplayName(med) });
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<MedicationAvatar name={med.name} imageUrl={med.imageUrl} size="lg" />
|
<MedicationAvatar name={getMedDisplayName(med)} imageUrl={med.imageUrl} size="lg" />
|
||||||
</span>
|
</span>
|
||||||
<div className="med-name-block">
|
<div className="med-name-block">
|
||||||
<div className="med-name">{med.name}</div>
|
<div className="med-name">{getMedDisplayName(med)}</div>
|
||||||
{med.genericName && <div className="med-generic-name">{med.genericName}</div>}
|
{med.name && med.genericName && <div className="med-generic-name">{med.genericName}</div>}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="med-actions">
|
<div className="med-actions">
|
||||||
@@ -876,12 +992,9 @@ export function MedicationsPage() {
|
|||||||
</div>
|
</div>
|
||||||
<div className="med-details">
|
<div className="med-details">
|
||||||
<span>
|
<span>
|
||||||
{t("medications.details.type")}:{" "}
|
{t("medications.details.type")}: <strong>{getMedicationPackageTypeLabel(med)}</strong>
|
||||||
<strong>
|
|
||||||
{med.packageType === "bottle" ? t("form.packageTypeBottle") : t("form.packageTypeBlister")}
|
|
||||||
</strong>
|
|
||||||
</span>
|
</span>
|
||||||
{med.packageType === "blister" ? (
|
{!isAmountBasedPackageType(med.packageType) ? (
|
||||||
<>
|
<>
|
||||||
<span>
|
<span>
|
||||||
{t("medications.details.packs")}: <strong>{med.packCount}</strong>
|
{t("medications.details.packs")}: <strong>{med.packCount}</strong>
|
||||||
@@ -910,10 +1023,13 @@ export function MedicationsPage() {
|
|||||||
)}
|
)}
|
||||||
<div className="med-total">
|
<div className="med-total">
|
||||||
{t("medications.details.stock")}:{" "}
|
{t("medications.details.stock")}:{" "}
|
||||||
{coverageByMed[med.name] ? Math.round(coverageByMed[med.name].medsLeft) : getPackageSize(med)} /{" "}
|
{coverageByMed[getMedDisplayName(med)]
|
||||||
{getPackageSize(med)} {getPackageSize(med) === 1 ? t("common.pill") : t("common.pills")}
|
? Math.round(coverageByMed[getMedDisplayName(med)].medsLeft)
|
||||||
{(coverageByMed[med.name]
|
: getPackageSize(med)}{" "}
|
||||||
? Math.round(coverageByMed[med.name].medsLeft)
|
/ {getPackageSize(med)}
|
||||||
|
{getMedicationStockSuffix(med)}
|
||||||
|
{(coverageByMed[getMedDisplayName(med)]
|
||||||
|
? Math.round(coverageByMed[getMedDisplayName(med)].medsLeft)
|
||||||
: getPackageSize(med)) > getPackageSize(med) && (
|
: getPackageSize(med)) > getPackageSize(med) && (
|
||||||
<span
|
<span
|
||||||
className="info-tooltip tooltip-align-left warning-text"
|
className="info-tooltip tooltip-align-left warning-text"
|
||||||
@@ -929,7 +1045,7 @@ export function MedicationsPage() {
|
|||||||
<div className="blister-list">
|
<div className="blister-list">
|
||||||
{(med.intakes ?? med.blisters).map((s, idx) => (
|
{(med.intakes ?? med.blisters).map((s, idx) => (
|
||||||
<div key={`${med.id}-${idx}`} className="blister-row-simple">
|
<div key={`${med.id}-${idx}`} className="blister-row-simple">
|
||||||
{s.usage} {s.usage === 1 ? t("common.pill") : t("common.pills")} ·{" "}
|
{s.usage} {getMedicationUsageUnitLabel(med, s.usage)} ·{" "}
|
||||||
{s.every === 1 ? t("common.daily") : t("common.everyNDays", { count: s.every })} ·{" "}
|
{s.every === 1 ? t("common.daily") : t("common.everyNDays", { count: s.every })} ·{" "}
|
||||||
{t("form.blisters.from")} {formatDateTime(s.start)}
|
{t("form.blisters.from")} {formatDateTime(s.start)}
|
||||||
{"takenBy" in s && (s as import("../types").Intake).takenBy && (
|
{"takenBy" in s && (s as import("../types").Intake).takenBy && (
|
||||||
@@ -970,20 +1086,24 @@ export function MedicationsPage() {
|
|||||||
<span
|
<span
|
||||||
className={med.imageUrl ? "med-avatar-clickable" : undefined}
|
className={med.imageUrl ? "med-avatar-clickable" : undefined}
|
||||||
onClick={() =>
|
onClick={() =>
|
||||||
med.imageUrl && setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: med.name })
|
med.imageUrl &&
|
||||||
|
setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: getMedDisplayName(med) })
|
||||||
}
|
}
|
||||||
onKeyDown={(e) => {
|
onKeyDown={(e) => {
|
||||||
if (e.key === "Enter" || e.key === " ") {
|
if (e.key === "Enter" || e.key === " ") {
|
||||||
if (med.imageUrl)
|
if (med.imageUrl)
|
||||||
setLightboxImage({ src: `/api/images/${med.imageUrl}`, alt: med.name });
|
setLightboxImage({
|
||||||
|
src: `/api/images/${med.imageUrl}`,
|
||||||
|
alt: getMedDisplayName(med),
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<MedicationAvatar name={med.name} imageUrl={med.imageUrl} size="lg" />
|
<MedicationAvatar name={getMedDisplayName(med)} imageUrl={med.imageUrl} size="lg" />
|
||||||
</span>
|
</span>
|
||||||
<div className="med-name-block">
|
<div className="med-name-block">
|
||||||
<div className="med-name">{med.name}</div>
|
<div className="med-name">{getMedDisplayName(med)}</div>
|
||||||
{med.genericName && <div className="med-generic-name">{med.genericName}</div>}
|
{med.name && med.genericName && <div className="med-generic-name">{med.genericName}</div>}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="med-actions">
|
<div className="med-actions">
|
||||||
@@ -1106,27 +1226,33 @@ export function MedicationsPage() {
|
|||||||
onBlur={() => setShowNameValidation(true)}
|
onBlur={() => setShowNameValidation(true)}
|
||||||
placeholder={t("form.placeholders.commercial")}
|
placeholder={t("form.placeholders.commercial")}
|
||||||
maxLength={FIELD_LIMITS.name.max}
|
maxLength={FIELD_LIMITS.name.max}
|
||||||
required={!readOnlyView}
|
|
||||||
/>
|
/>
|
||||||
{!readOnlyView && showNameValidation && fieldErrors.name && (
|
{!readOnlyView && showNameValidation && fieldErrors.name && (
|
||||||
<span className="field-error">{fieldErrors.name}</span>
|
<span className="field-error">{fieldErrors.name}</span>
|
||||||
)}
|
)}
|
||||||
</label>
|
</label>
|
||||||
<label className={fieldErrors.genericName ? "has-error" : ""}>
|
<label className={!readOnlyView && showNameValidation && fieldErrors.genericName ? "has-error" : ""}>
|
||||||
{t("form.genericName")}
|
{t("form.genericName")}
|
||||||
<input
|
<input
|
||||||
value={form.genericName}
|
value={form.genericName}
|
||||||
onChange={(e) => setForm({ ...form, genericName: e.target.value })}
|
onChange={(e) => {
|
||||||
|
setShowNameValidation(true);
|
||||||
|
setForm({ ...form, genericName: e.target.value });
|
||||||
|
}}
|
||||||
|
onBlur={() => setShowNameValidation(true)}
|
||||||
placeholder={t("form.placeholders.generic")}
|
placeholder={t("form.placeholders.generic")}
|
||||||
maxLength={FIELD_LIMITS.genericName.max}
|
maxLength={FIELD_LIMITS.genericName.max}
|
||||||
/>
|
/>
|
||||||
{fieldErrors.genericName && <span className="field-error">{fieldErrors.genericName}</span>}
|
{!readOnlyView && showNameValidation && fieldErrors.genericName && (
|
||||||
|
<span className="field-error">{fieldErrors.genericName}</span>
|
||||||
|
)}
|
||||||
</label>
|
</label>
|
||||||
<label>
|
<label>
|
||||||
{t("form.medicationStartDate")}
|
{t("form.medicationStartDate")}
|
||||||
<DateInput
|
<DateInput
|
||||||
value={form.medicationStartDate}
|
value={form.medicationStartDate}
|
||||||
onChange={(e) => handleValueChange("medicationStartDate", e.target.value)}
|
onChange={(e) => handleValueChange("medicationStartDate", e.target.value)}
|
||||||
|
placeholder={t("common.optional")}
|
||||||
/>
|
/>
|
||||||
{!readOnlyView && dateConsistencyError && (
|
{!readOnlyView && dateConsistencyError && (
|
||||||
<span className="field-error">{dateConsistencyError}</span>
|
<span className="field-error">{dateConsistencyError}</span>
|
||||||
@@ -1137,14 +1263,64 @@ export function MedicationsPage() {
|
|||||||
<select
|
<select
|
||||||
className="package-type-select"
|
className="package-type-select"
|
||||||
value={form.packageType}
|
value={form.packageType}
|
||||||
onChange={(e) =>
|
onChange={(e) => handleValueChange("packageType", e.target.value as PackageType)}
|
||||||
handleValueChange("packageType", e.target.value as import("../types").PackageType)
|
|
||||||
}
|
|
||||||
>
|
>
|
||||||
<option value="blister">{t("form.packageTypeBlister")}</option>
|
{PACKAGE_PROFILES.map((profile) => (
|
||||||
<option value="bottle">{t("form.packageTypeBottle")}</option>
|
<option key={profile.value} value={profile.value}>
|
||||||
|
{t(profile.labelKey)}
|
||||||
|
</option>
|
||||||
|
))}
|
||||||
</select>
|
</select>
|
||||||
</label>
|
</label>
|
||||||
|
<label>
|
||||||
|
{t("form.medicationEndDate")}
|
||||||
|
<DateInput
|
||||||
|
value={form.medicationEndDate}
|
||||||
|
onChange={(e) => handleValueChange("medicationEndDate", e.target.value)}
|
||||||
|
placeholder={t("common.optional")}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
{allowsPillFormSelection(form.packageType) && (
|
||||||
|
<label>
|
||||||
|
{t("form.pillForm")}
|
||||||
|
<select
|
||||||
|
value={form.pillForm}
|
||||||
|
onChange={(e) => handleValueChange("pillForm", e.target.value as FormState["pillForm"])}
|
||||||
|
>
|
||||||
|
<option value="tablet">{t("form.medicationFormTablet")}</option>
|
||||||
|
<option value="capsule">{t("form.medicationFormCapsule")}</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
|
{isTubePackageType(form.packageType) && (
|
||||||
|
<label>
|
||||||
|
{t("form.medicationForm")}
|
||||||
|
<select value={"topical"} onChange={() => handleValueChange("medicationForm", "topical")}>
|
||||||
|
<option value="topical">{t("form.medicationFormTopical")}</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
|
{isLiquidContainerPackageType(form.packageType) && (
|
||||||
|
<label>
|
||||||
|
{t("form.medicationForm")}
|
||||||
|
<select value={"liquid"} onChange={() => handleValueChange("medicationForm", "liquid")}>
|
||||||
|
<option value="liquid">{t("form.medicationFormLiquid")}</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
|
{form.medicationEndDate && (
|
||||||
|
<label className="full">
|
||||||
|
{t("form.autoMarkObsoleteAfterEndDate")}
|
||||||
|
<label className="toggle-switch small">
|
||||||
|
<input
|
||||||
|
type="checkbox"
|
||||||
|
checked={form.autoMarkObsoleteAfterEndDate}
|
||||||
|
onChange={(e) => handleValueChange("autoMarkObsoleteAfterEndDate", e.target.checked)}
|
||||||
|
/>
|
||||||
|
<span className="toggle-slider"></span>
|
||||||
|
</label>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
<label className={`full ${fieldErrors.takenBy ? "has-error" : ""}`}>
|
<label className={`full ${fieldErrors.takenBy ? "has-error" : ""}`}>
|
||||||
{t("form.takenBy")}
|
{t("form.takenBy")}
|
||||||
<div className="tag-input-container">
|
<div className="tag-input-container">
|
||||||
@@ -1258,99 +1434,177 @@ export function MedicationsPage() {
|
|||||||
<div className={`form-tab-panel${activeTab === "stock" ? " active" : ""}`}>
|
<div className={`form-tab-panel${activeTab === "stock" ? " active" : ""}`}>
|
||||||
<div className="full form-category">
|
<div className="full form-category">
|
||||||
<h4 className="form-category-title">{t("form.sections.stock")}</h4>
|
<h4 className="form-category-title">{t("form.sections.stock")}</h4>
|
||||||
{form.packageType === "blister" ? (
|
{(() => {
|
||||||
<>
|
if (!isAmountBasedPackageType(form.packageType)) {
|
||||||
<label>
|
return (
|
||||||
{t("form.packs")}
|
<>
|
||||||
<FormNumberStepper
|
<label>
|
||||||
value={form.packCount}
|
{t("form.packs")}
|
||||||
onChange={(nextValue) => handleValueChange("packCount", nextValue)}
|
<FormNumberStepper
|
||||||
min={0}
|
value={form.packCount}
|
||||||
decrementLabel={decrementValueLabel}
|
onChange={(nextValue) => handleValueChange("packCount", nextValue)}
|
||||||
incrementLabel={incrementValueLabel}
|
min={0}
|
||||||
|
decrementLabel={decrementValueLabel}
|
||||||
|
incrementLabel={incrementValueLabel}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
<label>
|
||||||
|
{t("form.blistersPerPack")}
|
||||||
|
<FormNumberStepper
|
||||||
|
value={form.blistersPerPack}
|
||||||
|
onChange={(nextValue) => handleValueChange("blistersPerPack", nextValue)}
|
||||||
|
min={1}
|
||||||
|
decrementLabel={decrementValueLabel}
|
||||||
|
incrementLabel={incrementValueLabel}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
<label>
|
||||||
|
{t("form.pillsPerBlister")}
|
||||||
|
<FormNumberStepper
|
||||||
|
value={form.pillsPerBlister}
|
||||||
|
onChange={(nextValue) => handleValueChange("pillsPerBlister", nextValue)}
|
||||||
|
min={1}
|
||||||
|
decrementLabel={decrementValueLabel}
|
||||||
|
incrementLabel={incrementValueLabel}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
<label>
|
||||||
|
{t("form.total")}
|
||||||
|
<div className="static-value">{formatNumber(totalTablets)}</div>
|
||||||
|
</label>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isTubePackageType(form.packageType)) {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<label>
|
||||||
|
{t("form.tubes")}
|
||||||
|
<FormNumberStepper
|
||||||
|
value={form.packCount}
|
||||||
|
onChange={(nextValue) => handleValueChange("packCount", nextValue)}
|
||||||
|
min={1}
|
||||||
|
decrementLabel={decrementValueLabel}
|
||||||
|
incrementLabel={incrementValueLabel}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
<label className="full">
|
||||||
|
{t("form.packageAmountPerTube")}
|
||||||
|
<div className="dose-input-group">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
inputMode="decimal"
|
||||||
|
pattern="[0-9]*\.?[0-9]*"
|
||||||
|
value={form.packageAmountValue ?? "0"}
|
||||||
|
onChange={(e) => handleValueChange("packageAmountValue", e.target.value)}
|
||||||
|
placeholder="0"
|
||||||
|
/>
|
||||||
|
<select
|
||||||
|
value="g"
|
||||||
|
disabled
|
||||||
|
className="dose-unit-select"
|
||||||
|
aria-label={t("form.packageAmountUnitG")}
|
||||||
|
>
|
||||||
|
<option value="g">{t("form.packageAmountUnitG")}</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
<label>
|
||||||
|
{t("form.totalAmount")}
|
||||||
|
<div className="static-value">
|
||||||
|
{formatNumber(
|
||||||
|
(Number(form.packCount) || 0) * (Number(form.packageAmountValue ?? 0) || 0)
|
||||||
|
)}
|
||||||
|
{t("form.packageAmountUnitG")}
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<label>
|
||||||
|
{totalCapacityLabel}
|
||||||
|
<FormNumberStepper
|
||||||
|
value={form.totalPills}
|
||||||
|
onChange={(nextValue) => handleValueChange("totalPills", nextValue)}
|
||||||
|
min={0}
|
||||||
|
decrementLabel={decrementValueLabel}
|
||||||
|
incrementLabel={incrementValueLabel}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
<label>
|
||||||
|
{currentStockLabel}
|
||||||
|
<FormNumberStepper
|
||||||
|
value={form.looseTablets}
|
||||||
|
onChange={(nextValue) => handleValueChange("looseTablets", nextValue)}
|
||||||
|
min={0}
|
||||||
|
decrementLabel={decrementValueLabel}
|
||||||
|
incrementLabel={incrementValueLabel}
|
||||||
|
/>
|
||||||
|
</label>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
})()}
|
||||||
|
{allowsPillFormSelection(form.packageType) && (
|
||||||
|
<label className="full">
|
||||||
|
{t("form.pillWeight")} ({form.doseUnit})
|
||||||
|
<div className="dose-input-group">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
inputMode="decimal"
|
||||||
|
pattern="[0-9]*\.?[0-9]*"
|
||||||
|
value={form.pillWeightMg}
|
||||||
|
onChange={(e) => handleValueChange("pillWeightMg", e.target.value)}
|
||||||
|
placeholder={t("form.placeholders.weight")}
|
||||||
/>
|
/>
|
||||||
</label>
|
<select
|
||||||
<label>
|
value={form.doseUnit}
|
||||||
{t("form.blistersPerPack")}
|
onChange={(e) => handleValueChange("doseUnit", e.target.value as DoseUnit)}
|
||||||
<FormNumberStepper
|
className="dose-unit-select"
|
||||||
value={form.blistersPerPack}
|
>
|
||||||
onChange={(nextValue) => handleValueChange("blistersPerPack", nextValue)}
|
{DOSE_UNITS.map((unit) => (
|
||||||
min={1}
|
<option key={unit.value} value={unit.value}>
|
||||||
decrementLabel={decrementValueLabel}
|
{unit.label}
|
||||||
incrementLabel={incrementValueLabel}
|
</option>
|
||||||
/>
|
))}
|
||||||
</label>
|
</select>
|
||||||
<label>
|
</div>
|
||||||
{t("form.pillsPerBlister")}
|
</label>
|
||||||
<FormNumberStepper
|
|
||||||
value={form.pillsPerBlister}
|
|
||||||
onChange={(nextValue) => handleValueChange("pillsPerBlister", nextValue)}
|
|
||||||
min={1}
|
|
||||||
decrementLabel={decrementValueLabel}
|
|
||||||
incrementLabel={incrementValueLabel}
|
|
||||||
/>
|
|
||||||
</label>
|
|
||||||
<label>
|
|
||||||
{t("form.total")}
|
|
||||||
<div className="static-value">{formatNumber(totalTablets)}</div>
|
|
||||||
</label>
|
|
||||||
</>
|
|
||||||
) : (
|
|
||||||
<>
|
|
||||||
<label>
|
|
||||||
{t("form.totalCapacity")}
|
|
||||||
<FormNumberStepper
|
|
||||||
value={form.totalPills}
|
|
||||||
onChange={(nextValue) => handleValueChange("totalPills", nextValue)}
|
|
||||||
min={0}
|
|
||||||
decrementLabel={decrementValueLabel}
|
|
||||||
incrementLabel={incrementValueLabel}
|
|
||||||
/>
|
|
||||||
</label>
|
|
||||||
<label>
|
|
||||||
{t("form.currentPills")}
|
|
||||||
<FormNumberStepper
|
|
||||||
value={form.looseTablets}
|
|
||||||
onChange={(nextValue) => handleValueChange("looseTablets", nextValue)}
|
|
||||||
min={0}
|
|
||||||
decrementLabel={decrementValueLabel}
|
|
||||||
incrementLabel={incrementValueLabel}
|
|
||||||
/>
|
|
||||||
</label>
|
|
||||||
</>
|
|
||||||
)}
|
)}
|
||||||
<label className="full">
|
{isAmountBasedPackageType(form.packageType) && !isTubePackageType(form.packageType) && (
|
||||||
{t("form.pillWeight")} ({form.doseUnit})
|
|
||||||
<div className="dose-input-group">
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
inputMode="decimal"
|
|
||||||
pattern="[0-9]*\.?[0-9]*"
|
|
||||||
value={form.pillWeightMg}
|
|
||||||
onChange={(e) => handleValueChange("pillWeightMg", e.target.value)}
|
|
||||||
placeholder={t("form.placeholders.weight")}
|
|
||||||
/>
|
|
||||||
<select
|
|
||||||
value={form.doseUnit}
|
|
||||||
onChange={(e) => handleValueChange("doseUnit", e.target.value as DoseUnit)}
|
|
||||||
className="dose-unit-select"
|
|
||||||
>
|
|
||||||
{DOSE_UNITS.map((unit) => (
|
|
||||||
<option key={unit.value} value={unit.value}>
|
|
||||||
{unit.label}
|
|
||||||
</option>
|
|
||||||
))}
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
</label>
|
|
||||||
{form.packageType === "bottle" && (
|
|
||||||
<div className="full stock-total-row">
|
<div className="full stock-total-row">
|
||||||
<label className="stock-total-field">
|
<label className="stock-total-field">
|
||||||
{t("form.total")}
|
{totalLabel}
|
||||||
<div className="static-value">{formatNumber(totalTablets)}</div>
|
<div className="static-value">{formatNumber(totalTablets)}</div>
|
||||||
</label>
|
</label>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
{isLiquidContainerPackageType(form.packageType) && (
|
||||||
|
<label className="full">
|
||||||
|
{t("form.packageAmount")}
|
||||||
|
<div className="dose-input-group">
|
||||||
|
<input
|
||||||
|
type="text"
|
||||||
|
inputMode="decimal"
|
||||||
|
pattern="[0-9]*\.?[0-9]*"
|
||||||
|
value={form.packageAmountValue ?? "0"}
|
||||||
|
onChange={(e) => handleValueChange("packageAmountValue", e.target.value)}
|
||||||
|
placeholder="0"
|
||||||
|
/>
|
||||||
|
<select
|
||||||
|
value="ml"
|
||||||
|
disabled
|
||||||
|
className="dose-unit-select"
|
||||||
|
aria-label={t("form.packageAmountUnitMl")}
|
||||||
|
>
|
||||||
|
<option value="ml">{t("form.packageAmountUnitMl")}</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
<label>
|
<label>
|
||||||
{t("form.expiryDate")}
|
{t("form.expiryDate")}
|
||||||
<DateInput
|
<DateInput
|
||||||
@@ -1466,13 +1720,13 @@ export function MedicationsPage() {
|
|||||||
<div key={idx} className="blister-row">
|
<div key={idx} className="blister-row">
|
||||||
<div className="blister-inputs">
|
<div className="blister-inputs">
|
||||||
<label>
|
<label>
|
||||||
{t("form.blisters.usage")}
|
{getUsageLabel(intake.intakeUnit ?? "ml")}
|
||||||
<FormNumberStepper
|
<FormNumberStepper
|
||||||
value={intake.usage}
|
value={intake.usage}
|
||||||
onChange={(nextValue) => setIntakeValue(idx, "usage", nextValue)}
|
onChange={(nextValue) => setIntakeValue(idx, "usage", nextValue)}
|
||||||
min={0.5}
|
min={allowFractionalIntake ? 0.5 : 1}
|
||||||
step={0.5}
|
step={allowFractionalIntake ? 0.5 : 1}
|
||||||
allowDecimal={true}
|
allowDecimal={allowFractionalIntake}
|
||||||
decrementLabel={decrementValueLabel}
|
decrementLabel={decrementValueLabel}
|
||||||
incrementLabel={incrementValueLabel}
|
incrementLabel={incrementValueLabel}
|
||||||
/>
|
/>
|
||||||
@@ -1502,6 +1756,21 @@ export function MedicationsPage() {
|
|||||||
onChange={(e) => setIntakeValue(idx, "startTime", e.target.value)}
|
onChange={(e) => setIntakeValue(idx, "startTime", e.target.value)}
|
||||||
/>
|
/>
|
||||||
</label>
|
</label>
|
||||||
|
{isLiquidContainerPackageType(form.packageType) && (
|
||||||
|
<label>
|
||||||
|
{t("form.blisters.intakeUnit")}
|
||||||
|
<select
|
||||||
|
value={intake.intakeUnit}
|
||||||
|
onChange={(e) =>
|
||||||
|
setIntakeValue(idx, "intakeUnit", e.target.value as "ml" | "tsp" | "tbsp")
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<option value="ml">{t("form.blisters.intakeUnitMl")}</option>
|
||||||
|
<option value="tsp">{t("form.blisters.intakeUnitTsp")}</option>
|
||||||
|
<option value="tbsp">{t("form.blisters.intakeUnitTbsp")}</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
)}
|
||||||
{form.takenBy.length === 0 ? null : (
|
{form.takenBy.length === 0 ? null : (
|
||||||
<label className="taken-by-field" title={t("form.blisters.takenByTooltip")}>
|
<label className="taken-by-field" title={t("form.blisters.takenByTooltip")}>
|
||||||
{t("form.blisters.takenByIntake")}
|
{t("form.blisters.takenByIntake")}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import { DateTimeInput, MedicationAvatar } from "../components";
|
|||||||
import { useAuth } from "../components/Auth";
|
import { useAuth } from "../components/Auth";
|
||||||
import { useAppContext } from "../context";
|
import { useAppContext } from "../context";
|
||||||
import type { PlannerRow } from "../types";
|
import type { PlannerRow } from "../types";
|
||||||
|
import { getMedDisplayName, isAmountBasedPackageType, isLiquidContainerPackageType, isTubePackageType } from "../types";
|
||||||
import { toInputValue } from "../utils/formatters";
|
import { toInputValue } from "../utils/formatters";
|
||||||
|
|
||||||
// Date helpers
|
// Date helpers
|
||||||
@@ -121,6 +122,30 @@ export function PlannerPage() {
|
|||||||
const canSendNotification =
|
const canSendNotification =
|
||||||
(settings.emailEnabled && settings.notificationEmail) || (settings.shoutrrrEnabled && settings.shoutrrrUrl);
|
(settings.emailEnabled && settings.notificationEmail) || (settings.shoutrrrEnabled && settings.shoutrrrUrl);
|
||||||
|
|
||||||
|
const getUsageUnitLabel = (medicationId: number, count: number): string => {
|
||||||
|
const med = meds.find((m) => m.id === medicationId);
|
||||||
|
if (isLiquidContainerPackageType(med?.packageType)) {
|
||||||
|
return t("form.ml");
|
||||||
|
}
|
||||||
|
if (isTubePackageType(med?.packageType)) {
|
||||||
|
return med.medicationForm === "liquid" ? t("form.ml") : t("blisters.applications");
|
||||||
|
}
|
||||||
|
return count === 1 ? t("common.pill") : t("common.pills");
|
||||||
|
};
|
||||||
|
|
||||||
|
const getAvailableLabel = (medicationId: number, loosePills: number): string => {
|
||||||
|
const med = meds.find((m) => m.id === medicationId);
|
||||||
|
const roundedLoose = Math.round(loosePills * 10) / 10;
|
||||||
|
if (isLiquidContainerPackageType(med?.packageType)) {
|
||||||
|
return `${roundedLoose} ${t("form.ml")}`;
|
||||||
|
}
|
||||||
|
if (isTubePackageType(med?.packageType)) {
|
||||||
|
const unit = med.medicationForm === "liquid" ? t("form.ml") : t("blisters.applications");
|
||||||
|
return `${roundedLoose} ${unit}`;
|
||||||
|
}
|
||||||
|
return `${roundedLoose} ${roundedLoose === 1 ? t("common.pill") : t("common.pills")}`;
|
||||||
|
};
|
||||||
|
|
||||||
async function sendPlannerNotification() {
|
async function sendPlannerNotification() {
|
||||||
if (!canSendNotification || plannerRows.length === 0) return;
|
if (!canSendNotification || plannerRows.length === 0) return;
|
||||||
setSendingPlannerEmail(true);
|
setSendingPlannerEmail(true);
|
||||||
@@ -204,7 +229,8 @@ export function PlannerPage() {
|
|||||||
</div>
|
</div>
|
||||||
{plannerRows.map((row) => {
|
{plannerRows.map((row) => {
|
||||||
const med =
|
const med =
|
||||||
meds.find((m) => m.id === row.medicationId) || meds.find((m) => m.name === row.medicationName);
|
meds.find((m) => m.id === row.medicationId) ||
|
||||||
|
meds.find((m) => getMedDisplayName(m) === row.medicationName);
|
||||||
const remainingRefills = med?.prescriptionEnabled ? (med.prescriptionRemainingRefills ?? 0) : null;
|
const remainingRefills = med?.prescriptionEnabled ? (med.prescriptionRemainingRefills ?? 0) : null;
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
@@ -224,16 +250,16 @@ export function PlannerPage() {
|
|||||||
<span data-label={t("planner.table.usage")}>
|
<span data-label={t("planner.table.usage")}>
|
||||||
<span>
|
<span>
|
||||||
<strong>{row.plannerUsage}</strong>
|
<strong>{row.plannerUsage}</strong>
|
||||||
{row.plannerUsage === 1 ? t("common.pill") : t("common.pills")}
|
{getUsageUnitLabel(row.medicationId, row.plannerUsage)}
|
||||||
</span>
|
</span>
|
||||||
</span>
|
</span>
|
||||||
<span data-label={t("planner.table.blisters")}>
|
<span data-label={t("planner.table.blisters")}>
|
||||||
{row.packageType === "bottle" ? "–" : `${row.blistersNeeded} × ${row.blisterSize}`}
|
{isAmountBasedPackageType(row.packageType) ? "–" : `${row.blistersNeeded} × ${row.blisterSize}`}
|
||||||
</span>
|
</span>
|
||||||
<span data-label={t("planner.table.prescriptionRefills")}>{remainingRefills ?? "–"}</span>
|
<span data-label={t("planner.table.prescriptionRefills")}>{remainingRefills ?? "–"}</span>
|
||||||
<span data-label={t("planner.table.available")}>
|
<span data-label={t("planner.table.available")}>
|
||||||
{row.packageType === "bottle" ? (
|
{isAmountBasedPackageType(row.packageType) ? (
|
||||||
`${Math.round(row.loosePills * 10) / 10} ${Math.round(row.loosePills * 10) / 10 === 1 ? t("common.pill") : t("common.pills")}`
|
getAvailableLabel(row.medicationId, row.loosePills)
|
||||||
) : (
|
) : (
|
||||||
<>
|
<>
|
||||||
{row.fullBlisters} {t("common.blisters")}
|
{row.fullBlisters} {t("common.blisters")}
|
||||||
|
|||||||
@@ -5,6 +5,8 @@ import { MedicationAvatar } from "../components";
|
|||||||
import { useAuth } from "../components/Auth";
|
import { useAuth } from "../components/Auth";
|
||||||
import { useAppContext } from "../context";
|
import { useAppContext } from "../context";
|
||||||
import type { Coverage } from "../types";
|
import type { Coverage } from "../types";
|
||||||
|
import { getMedDisplayName, isLiquidContainerPackageType, isTubePackageType } from "../types";
|
||||||
|
import { formatNumber } from "../utils/formatters";
|
||||||
import { expandDoseIds, isDoseDismissed } from "../utils/schedule";
|
import { expandDoseIds, isDoseDismissed } from "../utils/schedule";
|
||||||
|
|
||||||
// Helper for user-specific localStorage keys
|
// Helper for user-specific localStorage keys
|
||||||
@@ -16,12 +18,21 @@ function userStorageKey(userId: number | undefined, key: string): string {
|
|||||||
function getStockStatus(
|
function getStockStatus(
|
||||||
daysLeft: number | null,
|
daysLeft: number | null,
|
||||||
medsLeft: number,
|
medsLeft: number,
|
||||||
settings: { lowStockDays: number; normalStockDays: number; highStockDays: number; reminderDaysBefore: number }
|
settings: { lowStockDays: number; normalStockDays: number; highStockDays: number; reminderDaysBefore: number },
|
||||||
|
packageType?: string
|
||||||
) {
|
) {
|
||||||
|
if (isTubePackageType(packageType)) return { className: "success", label: "status.noSchedule" };
|
||||||
// Out of stock or completely depleted = danger (red)
|
// Out of stock or completely depleted = danger (red)
|
||||||
if (medsLeft <= 0 || daysLeft === 0) return { className: "danger", label: "status.outOfStock" };
|
if (medsLeft <= 0 || daysLeft === 0) return { className: "danger", label: "status.outOfStock" };
|
||||||
// No schedule, but has stock = normal
|
// No schedule, but has stock = normal
|
||||||
if (daysLeft === null) return { className: "success", label: "status.noSchedule" };
|
if (daysLeft === null) return { className: "success", label: "status.noSchedule" };
|
||||||
|
if (isLiquidContainerPackageType(packageType)) {
|
||||||
|
const lowDays = Math.max(1, Math.floor(settings.reminderDaysBefore));
|
||||||
|
const criticalDays = Math.max(1, Math.ceil(lowDays / 2));
|
||||||
|
if (daysLeft <= criticalDays) return { className: "danger", label: "status.criticalStock" };
|
||||||
|
if (daysLeft <= lowDays) return { className: "warning", label: "status.lowStock" };
|
||||||
|
return { className: "success", label: "status.normal" };
|
||||||
|
}
|
||||||
// Critical: at or below reminder threshold = danger (red)
|
// Critical: at or below reminder threshold = danger (red)
|
||||||
if (daysLeft <= settings.reminderDaysBefore) return { className: "danger", label: "status.criticalStock" };
|
if (daysLeft <= settings.reminderDaysBefore) return { className: "danger", label: "status.criticalStock" };
|
||||||
// Low: below low stock threshold = warning (yellow)
|
// Low: below low stock threshold = warning (yellow)
|
||||||
@@ -36,13 +47,15 @@ function getStockStatus(
|
|||||||
function getDayStockStatus(
|
function getDayStockStatus(
|
||||||
dayMeds: Array<{ medName: string }>,
|
dayMeds: Array<{ medName: string }>,
|
||||||
coverageByMed: Record<string, Coverage>,
|
coverageByMed: Record<string, Coverage>,
|
||||||
settings: { lowStockDays: number; normalStockDays: number; highStockDays: number; reminderDaysBefore: number }
|
settings: { lowStockDays: number; normalStockDays: number; highStockDays: number; reminderDaysBefore: number },
|
||||||
|
meds: Array<{ name: string; genericName?: string | null; packageType?: string }>
|
||||||
): string {
|
): string {
|
||||||
let worstLevel = 3; // 3=success, 2=warning, 1=danger
|
let worstLevel = 3; // 3=success, 2=warning, 1=danger
|
||||||
for (const item of dayMeds) {
|
for (const item of dayMeds) {
|
||||||
const cov = coverageByMed[item.medName];
|
const cov = coverageByMed[item.medName];
|
||||||
if (!cov) continue;
|
if (!cov) continue;
|
||||||
const status = getStockStatus(cov.daysLeft, cov.medsLeft, settings);
|
const med = meds.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
|
const status = getStockStatus(cov.daysLeft, cov.medsLeft, settings, med?.packageType);
|
||||||
if (status.className === "danger") worstLevel = Math.min(worstLevel, 1);
|
if (status.className === "danger") worstLevel = Math.min(worstLevel, 1);
|
||||||
else if (status.className === "warning") worstLevel = Math.min(worstLevel, 2);
|
else if (status.className === "warning") worstLevel = Math.min(worstLevel, 2);
|
||||||
}
|
}
|
||||||
@@ -79,6 +92,87 @@ export function SchedulePage() {
|
|||||||
missedPastDoseIds,
|
missedPastDoseIds,
|
||||||
} = useAppContext();
|
} = useAppContext();
|
||||||
|
|
||||||
|
const shouldHideNoScheduleStatusForTube = (
|
||||||
|
med: (typeof meds)[number] | undefined,
|
||||||
|
status: { className: string; label: string } | null
|
||||||
|
) => isTubePackageType(med?.packageType) && status?.label === "status.noSchedule";
|
||||||
|
|
||||||
|
const getTubeUnitLabel = (med: (typeof meds)[number] | undefined, value: number) =>
|
||||||
|
isLiquidContainerPackageType(med?.packageType) || med?.medicationForm === "liquid"
|
||||||
|
? t("form.packageAmountUnitMl")
|
||||||
|
: t("form.blisters.applications", { count: Math.abs(value) });
|
||||||
|
|
||||||
|
const convertLiquidUsageToMl = (usage: number, unit: "ml" | "tsp" | "tbsp" | null | undefined): number => {
|
||||||
|
if (unit === "tsp") return usage * 5;
|
||||||
|
if (unit === "tbsp") return usage * 15;
|
||||||
|
return usage;
|
||||||
|
};
|
||||||
|
|
||||||
|
const getLiquidCountUnitLabel = (unit: "ml" | "tsp" | "tbsp" | null | undefined, usage: number): string => {
|
||||||
|
if (unit === "tsp") return t("form.blisters.teaspoons", { count: Math.abs(usage) });
|
||||||
|
if (unit === "tbsp") return t("form.blisters.tablespoons", { count: Math.abs(usage) });
|
||||||
|
return t("form.packageAmountUnitMl");
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatLiquidUsageLabel = (usage: number, unit: "ml" | "tsp" | "tbsp" | null | undefined): string => {
|
||||||
|
const normalizedUsage = Number(usage);
|
||||||
|
if (!Number.isFinite(normalizedUsage) || normalizedUsage <= 0) {
|
||||||
|
return `0 ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (unit === "ml" || unit == null) {
|
||||||
|
return `${formatNumber(normalizedUsage)} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
const mlTotal = convertLiquidUsageToMl(normalizedUsage, unit);
|
||||||
|
return `${formatNumber(normalizedUsage)} ${getLiquidCountUnitLabel(unit, normalizedUsage)} ${formatNumber(mlTotal)} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatDoseUsageLabel = (
|
||||||
|
med: (typeof meds)[number] | undefined,
|
||||||
|
usage: number,
|
||||||
|
intakeUnit?: "ml" | "tsp" | "tbsp" | null
|
||||||
|
) => {
|
||||||
|
if (isLiquidContainerPackageType(med?.packageType)) {
|
||||||
|
return formatLiquidUsageLabel(usage, intakeUnit);
|
||||||
|
}
|
||||||
|
if (isTubePackageType(med?.packageType)) {
|
||||||
|
return `${usage} ${getTubeUnitLabel(med, usage)}`;
|
||||||
|
}
|
||||||
|
return `${usage} ${usage !== 1 ? t("common.pills") : t("common.pill")}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
const formatTotalUsageLabel = (
|
||||||
|
med: (typeof meds)[number] | undefined,
|
||||||
|
total: number,
|
||||||
|
doses?: Array<{ usage: number; intakeUnit?: "ml" | "tsp" | "tbsp" | null }>
|
||||||
|
) => {
|
||||||
|
if (isLiquidContainerPackageType(med?.packageType)) {
|
||||||
|
if (doses && doses.length > 0) {
|
||||||
|
const normalizedDoses = doses.filter((dose) => Number.isFinite(Number(dose.usage)) && Number(dose.usage) > 0);
|
||||||
|
if (normalizedDoses.length > 0) {
|
||||||
|
const allUnits = new Set(normalizedDoses.map((dose) => dose.intakeUnit ?? "ml"));
|
||||||
|
if (allUnits.size === 1) {
|
||||||
|
const onlyUnit = normalizedDoses[0]?.intakeUnit ?? "ml";
|
||||||
|
const totalUsageInUnit = normalizedDoses.reduce((sum, dose) => sum + Number(dose.usage), 0);
|
||||||
|
return formatLiquidUsageLabel(totalUsageInUnit, onlyUnit);
|
||||||
|
}
|
||||||
|
|
||||||
|
const totalMl = normalizedDoses.reduce(
|
||||||
|
(sum, dose) => sum + convertLiquidUsageToMl(Number(dose.usage), dose.intakeUnit ?? "ml"),
|
||||||
|
0
|
||||||
|
);
|
||||||
|
return `${formatNumber(totalMl)} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return `${formatNumber(total)} ${t("form.packageAmountUnitMl")}`;
|
||||||
|
}
|
||||||
|
if (isTubePackageType(med?.packageType)) {
|
||||||
|
return `${total} ${getTubeUnitLabel(med, total)}`;
|
||||||
|
}
|
||||||
|
return t("common.pillsTotal", { count: total });
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<section className="grid">
|
<section className="grid">
|
||||||
<article className="card schedule-full">
|
<article className="card schedule-full">
|
||||||
@@ -116,7 +210,7 @@ export function SchedulePage() {
|
|||||||
|
|
||||||
// Count missed doses that are NOT dismissed (for warning icon)
|
// Count missed doses that are NOT dismissed (for warning icon)
|
||||||
const missedNotDismissedCount = day.meds.reduce((count, item) => {
|
const missedNotDismissedCount = day.meds.reduce((count, item) => {
|
||||||
const med = meds.find((m) => m.name === item.medName);
|
const med = meds.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
const dismissedUntilDate = med?.dismissedUntil ?? undefined;
|
const dismissedUntilDate = med?.dismissedUntil ?? undefined;
|
||||||
return (
|
return (
|
||||||
count +
|
count +
|
||||||
@@ -132,7 +226,7 @@ export function SchedulePage() {
|
|||||||
|
|
||||||
const isManuallyExpanded = manuallyExpandedDays.has(day.dateStr);
|
const isManuallyExpanded = manuallyExpandedDays.has(day.dateStr);
|
||||||
const isCollapsed = !isManuallyExpanded;
|
const isCollapsed = !isManuallyExpanded;
|
||||||
const _worstStatus = getDayStockStatus(day.meds, coverageByMed, settings);
|
const _worstStatus = getDayStockStatus(day.meds, coverageByMed, settings, meds);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
@@ -171,7 +265,7 @@ export function SchedulePage() {
|
|||||||
</div>
|
</div>
|
||||||
{!isCollapsed &&
|
{!isCollapsed &&
|
||||||
day.meds.map((item) => {
|
day.meds.map((item) => {
|
||||||
const med = meds.find((m) => m.name === item.medName);
|
const med = meds.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
const medCov = coverageByMed[item.medName];
|
const medCov = coverageByMed[item.medName];
|
||||||
const isEmpty = medCov ? medCov.medsLeft <= 0 : false;
|
const isEmpty = medCov ? medCov.medsLeft <= 0 : false;
|
||||||
const itemDoseIds = expandDoseIds(item.doses);
|
const itemDoseIds = expandDoseIds(item.doses);
|
||||||
@@ -184,7 +278,7 @@ export function SchedulePage() {
|
|||||||
<span className="med-name-text">{item.medName}</span>
|
<span className="med-name-text">{item.medName}</span>
|
||||||
</div>
|
</div>
|
||||||
<div className="tag-row">
|
<div className="tag-row">
|
||||||
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span>
|
<span className="tag subtle">{formatTotalUsageLabel(med, item.total, item.doses)}</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="doses-col">
|
<div className="doses-col">
|
||||||
@@ -196,7 +290,7 @@ export function SchedulePage() {
|
|||||||
<span className="dose-time">{dose.timeStr}</span>
|
<span className="dose-time">{dose.timeStr}</span>
|
||||||
<span className="dose-usage">
|
<span className="dose-usage">
|
||||||
<span className="dose-usage-main">
|
<span className="dose-usage-main">
|
||||||
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")}
|
{formatDoseUsageLabel(med, dose.usage, dose.intakeUnit)}
|
||||||
</span>
|
</span>
|
||||||
{med?.pillWeightMg && (
|
{med?.pillWeightMg && (
|
||||||
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
||||||
@@ -333,15 +427,16 @@ export function SchedulePage() {
|
|||||||
{day.meds.map((item) => {
|
{day.meds.map((item) => {
|
||||||
const medCoverage = coverageByMed[item.medName];
|
const medCoverage = coverageByMed[item.medName];
|
||||||
const isEmpty = medCoverage ? medCoverage.medsLeft <= 0 : false;
|
const isEmpty = medCoverage ? medCoverage.medsLeft <= 0 : false;
|
||||||
const med = meds.find((m) => m.name === item.medName);
|
const med = meds.find((m) => getMedDisplayName(m) === item.medName);
|
||||||
const depletionTime = depletionByMed[item.medName];
|
const depletionTime = depletionByMed[item.medName];
|
||||||
// Check if this dose is scheduled after medication runs out
|
// Check if this dose is scheduled after medication runs out
|
||||||
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
|
const willBeOutOfStock = typeof depletionTime === "number" && item.lastWhen > depletionTime;
|
||||||
const status = willBeOutOfStock
|
const status = willBeOutOfStock
|
||||||
? { className: "danger", label: "status.outOfStock" }
|
? { className: "danger", label: "status.outOfStock" }
|
||||||
: medCoverage
|
: medCoverage
|
||||||
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, settings)
|
? getStockStatus(medCoverage.daysLeft, medCoverage.medsLeft, settings, med?.packageType)
|
||||||
: null;
|
: null;
|
||||||
|
const visibleStatus = shouldHideNoScheduleStatusForTube(med, status) ? null : status;
|
||||||
const itemDoseIds = expandDoseIds(item.doses);
|
const itemDoseIds = expandDoseIds(item.doses);
|
||||||
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
const allTaken = itemDoseIds.every((id) => takenDoses.has(id));
|
||||||
return (
|
return (
|
||||||
@@ -352,8 +447,10 @@ export function SchedulePage() {
|
|||||||
<span className="med-name-text">{item.medName}</span>
|
<span className="med-name-text">{item.medName}</span>
|
||||||
</div>
|
</div>
|
||||||
<div className="tag-row">
|
<div className="tag-row">
|
||||||
<span className="tag subtle">{t("common.pillsTotal", { count: item.total })}</span>
|
<span className="tag subtle">{formatTotalUsageLabel(med, item.total, item.doses)}</span>
|
||||||
{status && <span className={`tag ${status.className}`}>{t(status.label)}</span>}
|
{visibleStatus && (
|
||||||
|
<span className={`tag ${visibleStatus.className}`}>{t(visibleStatus.label)}</span>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="doses-col">
|
<div className="doses-col">
|
||||||
@@ -367,7 +464,7 @@ export function SchedulePage() {
|
|||||||
<span className="dose-time">{dose.timeStr}</span>
|
<span className="dose-time">{dose.timeStr}</span>
|
||||||
<span className="dose-usage">
|
<span className="dose-usage">
|
||||||
<span className="dose-usage-main">
|
<span className="dose-usage-main">
|
||||||
{dose.usage} {dose.usage !== 1 ? t("common.pills") : t("common.pill")}
|
{formatDoseUsageLabel(med, dose.usage, dose.intakeUnit)}
|
||||||
</span>
|
</span>
|
||||||
{med?.pillWeightMg && (
|
{med?.pillWeightMg && (
|
||||||
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
<span className="dose-usage-weight">{`${dose.usage * med.pillWeightMg} ${med.doseUnit ?? "mg"}`}</span>
|
||||||
|
|||||||
@@ -479,11 +479,15 @@ export function SettingsPage() {
|
|||||||
</div>
|
</div>
|
||||||
<div className="schedule-row">
|
<div className="schedule-row">
|
||||||
<span className="schedule-label">{t("settings.schedule.stockCheck")}</span>
|
<span className="schedule-label">{t("settings.schedule.stockCheck")}</span>
|
||||||
<span className="schedule-value">{t("settings.schedule.dailyAt6")}</span>
|
<span className="schedule-value">
|
||||||
|
{t("settings.schedule.dailyAtHour", { hour: settings.reminderHour })}
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
<div className="schedule-row">
|
<div className="schedule-row">
|
||||||
<span className="schedule-label">{t("settings.schedule.intakeCheck")}</span>
|
<span className="schedule-label">{t("settings.schedule.intakeCheck")}</span>
|
||||||
<span className="schedule-value">{t("settings.schedule.15minBefore")}</span>
|
<span className="schedule-value">
|
||||||
|
{t("settings.schedule.minutesBefore", { minutes: settings.reminderMinutesBefore })}
|
||||||
|
</span>
|
||||||
</div>
|
</div>
|
||||||
{settings.nextScheduledCheck && (
|
{settings.nextScheduledCheck && (
|
||||||
<div className="schedule-row">
|
<div className="schedule-row">
|
||||||
@@ -663,6 +667,9 @@ export function SettingsPage() {
|
|||||||
settings.lowStockDays >= settings.highStockDays) && (
|
settings.lowStockDays >= settings.highStockDays) && (
|
||||||
<p className="threshold-validation-error">{t("settings.stock.thresholdValidation")}</p>
|
<p className="threshold-validation-error">{t("settings.stock.thresholdValidation")}</p>
|
||||||
)}
|
)}
|
||||||
|
<p className="hint-text" style={{ marginTop: "12px" }}>
|
||||||
|
ℹ️ {t("settings.stock.packageTypesNote")}
|
||||||
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</article>
|
</article>
|
||||||
|
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import type { Coverage } from "../types";
|
import type { Coverage, Medication, PackageType } from "../types";
|
||||||
import { getMedTotal as getMedTotalFromTypes } from "../types";
|
import { getMedTotal as getMedTotalFromTypes, isLiquidContainerPackageType, isTubePackageType } from "../types";
|
||||||
import { splitCurrentBlisterStock } from "../utils/stock";
|
import { splitCurrentBlisterStock } from "../utils/stock";
|
||||||
|
|
||||||
export function userStorageKey(userId: number | undefined, key: string): string {
|
export function userStorageKey(userId: number | undefined, key: string): string {
|
||||||
@@ -43,9 +43,12 @@ export function getMedTotal(med: {
|
|||||||
pillsPerBlister: number;
|
pillsPerBlister: number;
|
||||||
looseTablets: number;
|
looseTablets: number;
|
||||||
stockAdjustment?: number | null;
|
stockAdjustment?: number | null;
|
||||||
packageType?: string;
|
packageType?: PackageType;
|
||||||
}): number {
|
}): number {
|
||||||
return getMedTotalFromTypes(med);
|
return getMedTotalFromTypes({
|
||||||
|
...med,
|
||||||
|
stockAdjustment: med.stockAdjustment ?? undefined,
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getReminderStatusData(
|
export function getReminderStatusData(
|
||||||
@@ -53,6 +56,7 @@ export function getReminderStatusData(
|
|||||||
lowStockDays: number,
|
lowStockDays: number,
|
||||||
_allLowCoverage: Coverage[],
|
_allLowCoverage: Coverage[],
|
||||||
allCoverage: Coverage[],
|
allCoverage: Coverage[],
|
||||||
|
meds: Medication[],
|
||||||
lastAutoEmailSent: string | null,
|
lastAutoEmailSent: string | null,
|
||||||
_lastNotificationType: string | null,
|
_lastNotificationType: string | null,
|
||||||
_lastNotificationChannel: string | null,
|
_lastNotificationChannel: string | null,
|
||||||
@@ -70,8 +74,12 @@ export function getReminderStatusData(
|
|||||||
lastIntakeSent: { date: string; medName: string | null; takenBy: string | null } | null;
|
lastIntakeSent: { date: string; medName: string | null; takenBy: string | null } | null;
|
||||||
} {
|
} {
|
||||||
const lowStockMap = new Map<string, { name: string; daysLeft: number; isCritical: boolean }>();
|
const lowStockMap = new Map<string, { name: string; daysLeft: number; isCritical: boolean }>();
|
||||||
|
const medByName = new Map(meds.map((med) => [med.name || med.genericName || "", med] as const));
|
||||||
|
|
||||||
for (const c of allCoverage) {
|
for (const c of allCoverage) {
|
||||||
|
const med = medByName.get(c.name);
|
||||||
|
if (isTubePackageType(med?.packageType)) continue;
|
||||||
|
|
||||||
if (c.medsLeft <= 0) {
|
if (c.medsLeft <= 0) {
|
||||||
lowStockMap.set(c.name, { name: c.name, daysLeft: 0, isCritical: true });
|
lowStockMap.set(c.name, { name: c.name, daysLeft: 0, isCritical: true });
|
||||||
continue;
|
continue;
|
||||||
@@ -80,8 +88,11 @@ export function getReminderStatusData(
|
|||||||
if (c.daysLeft === null) continue;
|
if (c.daysLeft === null) continue;
|
||||||
|
|
||||||
const roundedDaysLeft = Math.round(c.daysLeft);
|
const roundedDaysLeft = Math.round(c.daysLeft);
|
||||||
const isCritical = c.daysLeft <= reminderDaysBefore;
|
const isLiquid = isLiquidContainerPackageType(med?.packageType);
|
||||||
const isLow = c.daysLeft < lowStockDays;
|
const liquidLowDays = Math.max(1, Math.floor(reminderDaysBefore));
|
||||||
|
const liquidCriticalDays = Math.max(1, Math.ceil(liquidLowDays / 2));
|
||||||
|
const isCritical = isLiquid ? c.daysLeft <= liquidCriticalDays : c.daysLeft <= reminderDaysBefore;
|
||||||
|
const isLow = isLiquid ? c.daysLeft <= liquidLowDays : c.daysLeft < lowStockDays;
|
||||||
if (!isCritical && !isLow) continue;
|
if (!isCritical && !isLow) continue;
|
||||||
|
|
||||||
const existing = lowStockMap.get(c.name);
|
const existing = lowStockMap.get(c.name);
|
||||||
|
|||||||
+60
-22
@@ -104,7 +104,7 @@ body.modal-open {
|
|||||||
.page {
|
.page {
|
||||||
max-width: 1200px;
|
max-width: 1200px;
|
||||||
margin: 0 auto;
|
margin: 0 auto;
|
||||||
padding: 2.5rem 1.5rem 3rem;
|
padding: 2.5rem 1.5rem 1.5rem;
|
||||||
overflow-x: hidden;
|
overflow-x: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -121,7 +121,7 @@ body.modal-open {
|
|||||||
.route-transition-mask.active {
|
.route-transition-mask.active {
|
||||||
transition: none;
|
transition: none;
|
||||||
opacity: 1;
|
opacity: 1;
|
||||||
pointer-events: auto;
|
pointer-events: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.hero {
|
.hero {
|
||||||
@@ -669,6 +669,16 @@ body.modal-open {
|
|||||||
background: color-mix(in srgb, var(--bg-secondary) 75%, var(--bg-tertiary));
|
background: color-mix(in srgb, var(--bg-secondary) 75%, var(--bg-tertiary));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.med-grid-wrapper.is-empty .med-group-active {
|
||||||
|
padding: 0.7rem 0.85rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.med-empty-state {
|
||||||
|
color: var(--text-secondary);
|
||||||
|
font-size: 0.92rem;
|
||||||
|
padding: 0.35rem 0.1rem;
|
||||||
|
}
|
||||||
|
|
||||||
.med-group-head {
|
.med-group-head {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
@@ -2660,6 +2670,11 @@ button.has-validation-error {
|
|||||||
grid-template-columns: minmax(140px, 1.5fr) 90px 70px 100px 100px 90px 90px;
|
grid-template-columns: minmax(140px, 1.5fr) 90px 70px 100px 100px 90px 90px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.table-8 .table-head,
|
||||||
|
.table-8 .table-row {
|
||||||
|
grid-template-columns: minmax(130px, 1.4fr) 90px 130px 70px 95px 95px 90px 95px;
|
||||||
|
}
|
||||||
|
|
||||||
.email-sent-status {
|
.email-sent-status {
|
||||||
font-size: 0.8rem;
|
font-size: 0.8rem;
|
||||||
color: var(--success);
|
color: var(--success);
|
||||||
@@ -2842,7 +2857,7 @@ button.has-validation-error {
|
|||||||
|
|
||||||
@media (max-width: 600px) {
|
@media (max-width: 600px) {
|
||||||
.page {
|
.page {
|
||||||
padding: 0.75rem 0.4rem 2rem;
|
padding: 0.75rem 0.4rem 1rem;
|
||||||
}
|
}
|
||||||
|
|
||||||
.grid {
|
.grid {
|
||||||
@@ -4674,55 +4689,78 @@ button.has-validation-error {
|
|||||||
}
|
}
|
||||||
|
|
||||||
.med-detail-schedules {
|
.med-detail-schedules {
|
||||||
display: flex;
|
display: grid;
|
||||||
flex-direction: column;
|
grid-template-columns: auto auto 1fr auto auto auto;
|
||||||
gap: 0.5rem;
|
gap: 0.45rem 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.med-schedule-item {
|
.med-schedule-row {
|
||||||
display: flex;
|
display: grid;
|
||||||
|
grid-column: 1 / -1;
|
||||||
|
grid-template-columns: subgrid;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
flex-wrap: wrap;
|
column-gap: 0.75rem;
|
||||||
gap: 0.35rem 0.75rem;
|
|
||||||
background: var(--bg-secondary);
|
|
||||||
padding: 0.75rem 1rem;
|
|
||||||
border-radius: 8px;
|
|
||||||
font-size: 0.9rem;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.med-schedule-usage {
|
.med-schedule-usage {
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
color: var(--accent);
|
color: var(--accent);
|
||||||
|
white-space: nowrap;
|
||||||
|
grid-column: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.med-schedule-freq {
|
.med-schedule-freq {
|
||||||
color: var(--text-secondary);
|
color: var(--text-secondary);
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
}
|
grid-column: 2;
|
||||||
|
|
||||||
.med-schedule-time {
|
|
||||||
font-weight: 500;
|
|
||||||
margin-left: auto;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.med-schedule-person {
|
.med-schedule-person {
|
||||||
color: var(--text-secondary);
|
color: var(--text-secondary);
|
||||||
font-size: 0.85rem;
|
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
grid-column: 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
.med-schedule-time {
|
||||||
|
font-weight: 700;
|
||||||
|
white-space: nowrap;
|
||||||
|
color: var(--text-primary);
|
||||||
|
text-align: right;
|
||||||
|
grid-column: 5;
|
||||||
}
|
}
|
||||||
|
|
||||||
.med-schedule-bell {
|
.med-schedule-bell {
|
||||||
color: var(--warning);
|
color: var(--warning);
|
||||||
display: inline-flex;
|
display: inline-flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
margin-left: 0.35rem;
|
grid-column: 6;
|
||||||
}
|
}
|
||||||
|
|
||||||
[data-theme="light"] .med-schedule-bell {
|
[data-theme="light"] .med-schedule-bell {
|
||||||
color: #b45309;
|
color: #b45309;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@media (max-width: 700px) {
|
||||||
|
.med-detail-schedules {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
}
|
||||||
|
|
||||||
|
.med-schedule-row {
|
||||||
|
grid-template-columns: 1fr;
|
||||||
|
row-gap: 0.25rem;
|
||||||
|
}
|
||||||
|
|
||||||
|
.med-schedule-usage,
|
||||||
|
.med-schedule-freq,
|
||||||
|
.med-schedule-person,
|
||||||
|
.med-schedule-time,
|
||||||
|
.med-schedule-bell {
|
||||||
|
grid-column: auto;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.med-detail-footer {
|
.med-detail-footer {
|
||||||
padding: 1rem 2rem 1.5rem;
|
padding: 1rem 2rem 1.5rem;
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|||||||
@@ -500,6 +500,8 @@
|
|||||||
color: var(--text-primary);
|
color: var(--text-primary);
|
||||||
font-size: 0.95rem;
|
font-size: 0.95rem;
|
||||||
font-family: inherit;
|
font-family: inherit;
|
||||||
|
text-transform: none;
|
||||||
|
letter-spacing: normal;
|
||||||
white-space: nowrap;
|
white-space: nowrap;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
text-overflow: ellipsis;
|
text-overflow: ellipsis;
|
||||||
|
|||||||
@@ -36,7 +36,7 @@ describe("AppHeader", () => {
|
|||||||
json: () =>
|
json: () =>
|
||||||
Promise.resolve({
|
Promise.resolve({
|
||||||
authEnabled: false,
|
authEnabled: false,
|
||||||
localAuthEnabled: true,
|
formLoginEnabled: true,
|
||||||
hasUsers: false,
|
hasUsers: false,
|
||||||
needsSetup: false,
|
needsSetup: false,
|
||||||
}),
|
}),
|
||||||
@@ -171,7 +171,7 @@ describe("AppHeader", () => {
|
|||||||
json: () =>
|
json: () =>
|
||||||
Promise.resolve({
|
Promise.resolve({
|
||||||
authEnabled: false,
|
authEnabled: false,
|
||||||
localAuthEnabled: true,
|
formLoginEnabled: true,
|
||||||
hasUsers: false,
|
hasUsers: false,
|
||||||
needsSetup: false,
|
needsSetup: false,
|
||||||
}),
|
}),
|
||||||
@@ -205,7 +205,7 @@ describe("AppHeader", () => {
|
|||||||
json: () =>
|
json: () =>
|
||||||
Promise.resolve({
|
Promise.resolve({
|
||||||
authEnabled: false,
|
authEnabled: false,
|
||||||
localAuthEnabled: true,
|
formLoginEnabled: true,
|
||||||
hasUsers: false,
|
hasUsers: false,
|
||||||
needsSetup: false,
|
needsSetup: false,
|
||||||
}),
|
}),
|
||||||
@@ -239,7 +239,7 @@ describe("AppHeader", () => {
|
|||||||
json: () =>
|
json: () =>
|
||||||
Promise.resolve({
|
Promise.resolve({
|
||||||
authEnabled: false,
|
authEnabled: false,
|
||||||
localAuthEnabled: true,
|
formLoginEnabled: true,
|
||||||
hasUsers: false,
|
hasUsers: false,
|
||||||
needsSetup: false,
|
needsSetup: false,
|
||||||
}),
|
}),
|
||||||
@@ -322,7 +322,7 @@ describe("AppHeader", () => {
|
|||||||
Promise.resolve({
|
Promise.resolve({
|
||||||
authEnabled: true,
|
authEnabled: true,
|
||||||
registrationEnabled: true,
|
registrationEnabled: true,
|
||||||
localAuthEnabled: true,
|
formLoginEnabled: true,
|
||||||
oidcEnabled: false,
|
oidcEnabled: false,
|
||||||
oidcProviderName: "",
|
oidcProviderName: "",
|
||||||
hasUsers: true,
|
hasUsers: true,
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ describe("AuthProvider", () => {
|
|||||||
vi.resetAllMocks();
|
vi.resetAllMocks();
|
||||||
(global.fetch as ReturnType<typeof vi.fn>).mockResolvedValue({
|
(global.fetch as ReturnType<typeof vi.fn>).mockResolvedValue({
|
||||||
ok: true,
|
ok: true,
|
||||||
json: () => Promise.resolve({ authEnabled: true, localAuthEnabled: true }),
|
json: () => Promise.resolve({ authEnabled: true, formLoginEnabled: true }),
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -79,7 +79,7 @@ describe("AuthProvider", () => {
|
|||||||
(global.fetch as ReturnType<typeof vi.fn>)
|
(global.fetch as ReturnType<typeof vi.fn>)
|
||||||
.mockResolvedValueOnce({
|
.mockResolvedValueOnce({
|
||||||
ok: true,
|
ok: true,
|
||||||
json: () => Promise.resolve({ authEnabled: false, localAuthEnabled: true }),
|
json: () => Promise.resolve({ authEnabled: false, formLoginEnabled: true }),
|
||||||
})
|
})
|
||||||
.mockResolvedValueOnce({ ok: false, status: 401 })
|
.mockResolvedValueOnce({ ok: false, status: 401 })
|
||||||
.mockResolvedValueOnce({ ok: true, status: 200 })
|
.mockResolvedValueOnce({ ok: true, status: 200 })
|
||||||
@@ -116,7 +116,7 @@ describe("AuthProvider", () => {
|
|||||||
(global.fetch as ReturnType<typeof vi.fn>)
|
(global.fetch as ReturnType<typeof vi.fn>)
|
||||||
.mockResolvedValueOnce({
|
.mockResolvedValueOnce({
|
||||||
ok: true,
|
ok: true,
|
||||||
json: () => Promise.resolve({ authEnabled: true, localAuthEnabled: true }),
|
json: () => Promise.resolve({ authEnabled: true, formLoginEnabled: true }),
|
||||||
})
|
})
|
||||||
.mockResolvedValueOnce({ ok: true, status: 200, json: () => Promise.resolve({ id: 1, username: "tester" }) })
|
.mockResolvedValueOnce({ ok: true, status: 200, json: () => Promise.resolve({ id: 1, username: "tester" }) })
|
||||||
.mockResolvedValueOnce({ ok: false, status: 401 })
|
.mockResolvedValueOnce({ ok: false, status: 401 })
|
||||||
@@ -141,7 +141,7 @@ describe("AuthProvider", () => {
|
|||||||
(global.fetch as ReturnType<typeof vi.fn>)
|
(global.fetch as ReturnType<typeof vi.fn>)
|
||||||
.mockResolvedValueOnce({
|
.mockResolvedValueOnce({
|
||||||
ok: true,
|
ok: true,
|
||||||
json: () => Promise.resolve({ authEnabled: true, localAuthEnabled: true }),
|
json: () => Promise.resolve({ authEnabled: true, formLoginEnabled: true }),
|
||||||
})
|
})
|
||||||
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ id: 1, username: "timer-user" }) })
|
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ id: 1, username: "timer-user" }) })
|
||||||
.mockResolvedValueOnce({ ok: true, status: 200 });
|
.mockResolvedValueOnce({ ok: true, status: 200 });
|
||||||
@@ -167,7 +167,7 @@ describe("AuthProvider", () => {
|
|||||||
describe("LoginForm", () => {
|
describe("LoginForm", () => {
|
||||||
const mockAuthState = {
|
const mockAuthState = {
|
||||||
authEnabled: true,
|
authEnabled: true,
|
||||||
localAuthEnabled: true,
|
formLoginEnabled: true,
|
||||||
oidcEnabled: false,
|
oidcEnabled: false,
|
||||||
registrationEnabled: true,
|
registrationEnabled: true,
|
||||||
hasUsers: true,
|
hasUsers: true,
|
||||||
@@ -281,7 +281,7 @@ describe("LoginForm", () => {
|
|||||||
json: () =>
|
json: () =>
|
||||||
Promise.resolve({
|
Promise.resolve({
|
||||||
authEnabled: false,
|
authEnabled: false,
|
||||||
localAuthEnabled: true,
|
formLoginEnabled: true,
|
||||||
oidcEnabled: false,
|
oidcEnabled: false,
|
||||||
registrationEnabled: true,
|
registrationEnabled: true,
|
||||||
hasUsers: true,
|
hasUsers: true,
|
||||||
@@ -317,7 +317,7 @@ describe("LoginForm", () => {
|
|||||||
describe("RegisterForm", () => {
|
describe("RegisterForm", () => {
|
||||||
const mockAuthState = {
|
const mockAuthState = {
|
||||||
authEnabled: true,
|
authEnabled: true,
|
||||||
localAuthEnabled: true,
|
formLoginEnabled: true,
|
||||||
oidcEnabled: false,
|
oidcEnabled: false,
|
||||||
registrationEnabled: true,
|
registrationEnabled: true,
|
||||||
hasUsers: false,
|
hasUsers: false,
|
||||||
@@ -404,7 +404,7 @@ describe("RegisterForm", () => {
|
|||||||
json: () =>
|
json: () =>
|
||||||
Promise.resolve({
|
Promise.resolve({
|
||||||
authEnabled: true,
|
authEnabled: true,
|
||||||
localAuthEnabled: true,
|
formLoginEnabled: true,
|
||||||
oidcEnabled: false,
|
oidcEnabled: false,
|
||||||
registrationEnabled: true,
|
registrationEnabled: true,
|
||||||
hasUsers: false,
|
hasUsers: false,
|
||||||
@@ -439,7 +439,7 @@ describe("RegisterForm", () => {
|
|||||||
describe("AuthPage", () => {
|
describe("AuthPage", () => {
|
||||||
const mockAuthState = {
|
const mockAuthState = {
|
||||||
authEnabled: true,
|
authEnabled: true,
|
||||||
localAuthEnabled: true,
|
formLoginEnabled: true,
|
||||||
oidcEnabled: false,
|
oidcEnabled: false,
|
||||||
registrationEnabled: true,
|
registrationEnabled: true,
|
||||||
hasUsers: true,
|
hasUsers: true,
|
||||||
@@ -504,7 +504,7 @@ describe("UserProfile", () => {
|
|||||||
(global.fetch as ReturnType<typeof vi.fn>)
|
(global.fetch as ReturnType<typeof vi.fn>)
|
||||||
.mockResolvedValueOnce({
|
.mockResolvedValueOnce({
|
||||||
ok: true,
|
ok: true,
|
||||||
json: () => Promise.resolve({ authEnabled: true, localAuthEnabled: true }),
|
json: () => Promise.resolve({ authEnabled: true, formLoginEnabled: true }),
|
||||||
})
|
})
|
||||||
.mockResolvedValueOnce({
|
.mockResolvedValueOnce({
|
||||||
ok: true,
|
ok: true,
|
||||||
@@ -724,7 +724,7 @@ describe("AuthProvider methods", () => {
|
|||||||
it("refreshUser retries after token refresh on 401", async () => {
|
it("refreshUser retries after token refresh on 401", async () => {
|
||||||
vi.clearAllMocks();
|
vi.clearAllMocks();
|
||||||
(global.fetch as ReturnType<typeof vi.fn>)
|
(global.fetch as ReturnType<typeof vi.fn>)
|
||||||
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ authEnabled: false, localAuthEnabled: true }) })
|
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ authEnabled: false, formLoginEnabled: true }) })
|
||||||
.mockResolvedValueOnce({ ok: false, status: 401 })
|
.mockResolvedValueOnce({ ok: false, status: 401 })
|
||||||
.mockResolvedValueOnce({ ok: true })
|
.mockResolvedValueOnce({ ok: true })
|
||||||
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ id: 1, username: "refreshed-user" }) });
|
.mockResolvedValueOnce({ ok: true, json: () => Promise.resolve({ id: 1, username: "refreshed-user" }) });
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
import { fireEvent, render, screen } from "@testing-library/react";
|
import { fireEvent, render, screen } from "@testing-library/react";
|
||||||
import { describe, expect, it, vi } from "vitest";
|
import { beforeEach, describe, expect, it, vi } from "vitest";
|
||||||
import { Lightbox } from "../../components/Lightbox";
|
import { Lightbox } from "../../components/Lightbox";
|
||||||
|
|
||||||
describe("Lightbox", () => {
|
describe("Lightbox", () => {
|
||||||
|
|||||||
@@ -55,6 +55,8 @@ const defaultProps = {
|
|||||||
onRefillPacksChange: vi.fn(),
|
onRefillPacksChange: vi.fn(),
|
||||||
refillLoose: 0,
|
refillLoose: 0,
|
||||||
onRefillLooseChange: vi.fn(),
|
onRefillLooseChange: vi.fn(),
|
||||||
|
usePrescriptionRefill: false,
|
||||||
|
onUsePrescriptionRefillChange: vi.fn(),
|
||||||
refillSaving: false,
|
refillSaving: false,
|
||||||
refillHistory: [] as RefillEntry[],
|
refillHistory: [] as RefillEntry[],
|
||||||
refillHistoryExpanded: false,
|
refillHistoryExpanded: false,
|
||||||
@@ -324,7 +326,7 @@ describe("MedDetailModal with refill modal", () => {
|
|||||||
|
|
||||||
const submitBtn = document.querySelector(".refill-modal .modal-footer .success") as HTMLButtonElement;
|
const submitBtn = document.querySelector(".refill-modal .modal-footer .success") as HTMLButtonElement;
|
||||||
fireEvent.click(submitBtn);
|
fireEvent.click(submitBtn);
|
||||||
expect(onSubmitRefill).toHaveBeenCalledWith(mockMedication.id, undefined);
|
expect(onSubmitRefill).toHaveBeenCalledWith(mockMedication.id, false);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("disables refill submit button when no pills are entered", () => {
|
it("disables refill submit button when no pills are entered", () => {
|
||||||
@@ -589,7 +591,7 @@ describe("MedDetailModal with refill history", () => {
|
|||||||
|
|
||||||
it("shows refill history when expanded", () => {
|
it("shows refill history when expanded", () => {
|
||||||
const refillHistory: RefillEntry[] = [
|
const refillHistory: RefillEntry[] = [
|
||||||
{ id: 1, medicationId: 1, timestamp: new Date().toISOString(), packsAdded: 1, looseAdded: 0 },
|
{ id: 1, refillDate: new Date().toISOString(), packsAdded: 1, loosePillsAdded: 0 },
|
||||||
];
|
];
|
||||||
|
|
||||||
render(<MedDetailModal {...defaultProps} refillHistory={refillHistory} refillHistoryExpanded={true} />);
|
render(<MedDetailModal {...defaultProps} refillHistory={refillHistory} refillHistoryExpanded={true} />);
|
||||||
@@ -602,7 +604,7 @@ describe("MedDetailModal with refill history", () => {
|
|||||||
it("calls onRefillHistoryExpandedChange when toggle clicked", () => {
|
it("calls onRefillHistoryExpandedChange when toggle clicked", () => {
|
||||||
const onRefillHistoryExpandedChange = vi.fn();
|
const onRefillHistoryExpandedChange = vi.fn();
|
||||||
const refillHistory: RefillEntry[] = [
|
const refillHistory: RefillEntry[] = [
|
||||||
{ id: 1, medicationId: 1, timestamp: new Date().toISOString(), packsAdded: 1, looseAdded: 0 },
|
{ id: 1, refillDate: new Date().toISOString(), packsAdded: 1, loosePillsAdded: 0 },
|
||||||
];
|
];
|
||||||
|
|
||||||
render(
|
render(
|
||||||
@@ -642,9 +644,9 @@ describe("MedDetailModal intake schedule usage display", () => {
|
|||||||
};
|
};
|
||||||
render(<MedDetailModal {...defaultProps} selectedMed={med} />);
|
render(<MedDetailModal {...defaultProps} selectedMed={med} />);
|
||||||
|
|
||||||
const usageElements = document.querySelectorAll(".med-schedule-usage");
|
const rows = document.querySelectorAll(".med-schedule-row .med-schedule-usage");
|
||||||
// Each intake should show "1 pill" (not "2 pills")
|
// Each intake should show "1" in usage (not "2")
|
||||||
usageElements.forEach((el) => {
|
rows.forEach((el) => {
|
||||||
expect(el.textContent).toContain("1");
|
expect(el.textContent).toContain("1");
|
||||||
expect(el.textContent).not.toMatch(/^2\b/);
|
expect(el.textContent).not.toMatch(/^2\b/);
|
||||||
});
|
});
|
||||||
@@ -660,10 +662,10 @@ describe("MedDetailModal intake schedule usage display", () => {
|
|||||||
};
|
};
|
||||||
render(<MedDetailModal {...defaultProps} selectedMed={med} />);
|
render(<MedDetailModal {...defaultProps} selectedMed={med} />);
|
||||||
|
|
||||||
const usageElements = document.querySelectorAll(".med-schedule-usage");
|
const rows = document.querySelectorAll(".med-schedule-row .med-schedule-usage");
|
||||||
// Legacy: 1 pill * 2 people = "2 pills"
|
// Legacy: 1 pill * 2 people = "2 pills"
|
||||||
expect(usageElements.length).toBe(1);
|
expect(rows.length).toBe(1);
|
||||||
expect(usageElements[0].textContent).toContain("2");
|
expect(rows[0].textContent).toContain("2");
|
||||||
});
|
});
|
||||||
|
|
||||||
it("shows correct usage for single person with per-intake takenBy", () => {
|
it("shows correct usage for single person with per-intake takenBy", () => {
|
||||||
@@ -676,11 +678,11 @@ describe("MedDetailModal intake schedule usage display", () => {
|
|||||||
};
|
};
|
||||||
render(<MedDetailModal {...defaultProps} selectedMed={med} />);
|
render(<MedDetailModal {...defaultProps} selectedMed={med} />);
|
||||||
|
|
||||||
const usageElements = document.querySelectorAll(".med-schedule-usage");
|
const rows = document.querySelectorAll(".med-schedule-row .med-schedule-usage");
|
||||||
expect(usageElements.length).toBe(1);
|
expect(rows.length).toBe(1);
|
||||||
// Should show "2 pills (1000 mg)" - usage=2, not multiplied
|
// Should show "2 pills (1000 mg)" - usage=2, not multiplied
|
||||||
expect(usageElements[0].textContent).toContain("2");
|
expect(rows[0].textContent).toContain("2");
|
||||||
expect(usageElements[0].textContent).toContain("1000");
|
expect(rows[0].textContent).toContain("1000");
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
@@ -1,4 +1,5 @@
|
|||||||
import { fireEvent, render, screen } from "@testing-library/react";
|
import { fireEvent, render, screen } from "@testing-library/react";
|
||||||
|
import type { FormEvent } from "react";
|
||||||
import { beforeEach, describe, expect, it, vi } from "vitest";
|
import { beforeEach, describe, expect, it, vi } from "vitest";
|
||||||
import { MobileEditModal } from "../../components/MobileEditModal";
|
import { MobileEditModal } from "../../components/MobileEditModal";
|
||||||
import type { FormState } from "../../types";
|
import type { FormState } from "../../types";
|
||||||
@@ -7,15 +8,22 @@ const defaultForm: FormState = {
|
|||||||
name: "",
|
name: "",
|
||||||
genericName: "",
|
genericName: "",
|
||||||
takenBy: [],
|
takenBy: [],
|
||||||
|
medicationForm: "tablet",
|
||||||
|
pillForm: "tablet",
|
||||||
|
lifecycleCategory: "refill_when_empty",
|
||||||
packageType: "blister",
|
packageType: "blister",
|
||||||
packCount: "1",
|
packCount: "1",
|
||||||
blistersPerPack: "1",
|
blistersPerPack: "1",
|
||||||
pillsPerBlister: "1",
|
pillsPerBlister: "1",
|
||||||
|
packageAmountValue: "0",
|
||||||
|
packageAmountUnit: "ml",
|
||||||
looseTablets: "0",
|
looseTablets: "0",
|
||||||
totalPills: "",
|
totalPills: "",
|
||||||
pillWeightMg: "",
|
pillWeightMg: "",
|
||||||
doseUnit: "mg",
|
doseUnit: "mg",
|
||||||
medicationStartDate: "",
|
medicationStartDate: "",
|
||||||
|
medicationEndDate: "",
|
||||||
|
autoMarkObsoleteAfterEndDate: true,
|
||||||
expiryDate: "",
|
expiryDate: "",
|
||||||
notes: "",
|
notes: "",
|
||||||
intakeRemindersEnabled: false,
|
intakeRemindersEnabled: false,
|
||||||
@@ -78,6 +86,7 @@ const defaultProps = {
|
|||||||
meds: [],
|
meds: [],
|
||||||
onUploadMedImage: vi.fn(),
|
onUploadMedImage: vi.fn(),
|
||||||
onDeleteMedImage: vi.fn(),
|
onDeleteMedImage: vi.fn(),
|
||||||
|
imageUploadError: null,
|
||||||
onClose: vi.fn(),
|
onClose: vi.fn(),
|
||||||
onResetForm: vi.fn(),
|
onResetForm: vi.fn(),
|
||||||
onSaveMedication: vi.fn(),
|
onSaveMedication: vi.fn(),
|
||||||
@@ -233,6 +242,54 @@ describe("MobileEditModal", () => {
|
|||||||
const header = document.querySelector(".edit-modal-header");
|
const header = document.querySelector(".edit-modal-header");
|
||||||
expect(header).toBeInTheDocument();
|
expect(header).toBeInTheDocument();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it("uses plain numeric input for tube amount without stepper controls", () => {
|
||||||
|
render(
|
||||||
|
<MobileEditModal
|
||||||
|
{...defaultProps}
|
||||||
|
form={{
|
||||||
|
...defaultForm,
|
||||||
|
packageType: "tube",
|
||||||
|
medicationForm: "topical",
|
||||||
|
packageAmountValue: "150",
|
||||||
|
packageAmountUnit: "g",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
const amountInput = screen.getByLabelText("form.packageAmountPerTube") as HTMLInputElement;
|
||||||
|
expect(amountInput).toBeInTheDocument();
|
||||||
|
expect(amountInput.tagName).toBe("INPUT");
|
||||||
|
expect(amountInput).toHaveAttribute("inputmode", "decimal");
|
||||||
|
|
||||||
|
const unitSelect = screen.getByLabelText("form.packageAmountUnitG") as HTMLSelectElement;
|
||||||
|
expect(unitSelect).toBeDisabled();
|
||||||
|
expect(unitSelect.value).toBe("g");
|
||||||
|
});
|
||||||
|
|
||||||
|
it("uses plain numeric input for liquid container package amount", () => {
|
||||||
|
render(
|
||||||
|
<MobileEditModal
|
||||||
|
{...defaultProps}
|
||||||
|
form={{
|
||||||
|
...defaultForm,
|
||||||
|
packageType: "liquid_container",
|
||||||
|
medicationForm: "liquid",
|
||||||
|
packageAmountValue: "250",
|
||||||
|
packageAmountUnit: "ml",
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
|
||||||
|
const amountInput = screen.getByLabelText("form.packageAmountPerBottle") as HTMLInputElement;
|
||||||
|
expect(amountInput).toBeInTheDocument();
|
||||||
|
expect(amountInput.tagName).toBe("INPUT");
|
||||||
|
expect(amountInput).toHaveAttribute("inputmode", "decimal");
|
||||||
|
|
||||||
|
const unitSelect = screen.getByLabelText("form.packageAmountUnitMl") as HTMLSelectElement;
|
||||||
|
expect(unitSelect).toBeDisabled();
|
||||||
|
expect(unitSelect.value).toBe("ml");
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
describe("MobileEditModal with existing people", () => {
|
describe("MobileEditModal with existing people", () => {
|
||||||
@@ -383,7 +440,7 @@ describe("MobileEditModal form submission", () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
it("calls onSaveMedication when form submitted", () => {
|
it("calls onSaveMedication when form submitted", () => {
|
||||||
const onSaveMedication = vi.fn((e: Event) => e.preventDefault());
|
const onSaveMedication = vi.fn((e: FormEvent) => e.preventDefault());
|
||||||
const validForm = { ...defaultForm, name: "TestMed" };
|
const validForm = { ...defaultForm, name: "TestMed" };
|
||||||
|
|
||||||
render(<MobileEditModal {...defaultProps} form={validForm} onSaveMedication={onSaveMedication} />);
|
render(<MobileEditModal {...defaultProps} form={validForm} onSaveMedication={onSaveMedication} />);
|
||||||
|
|||||||
@@ -66,27 +66,7 @@ describe("ProfileModal", () => {
|
|||||||
expect(onClose).not.toHaveBeenCalled();
|
expect(onClose).not.toHaveBeenCalled();
|
||||||
});
|
});
|
||||||
|
|
||||||
it("calls onClose when Escape is pressed on overlay", () => {
|
// ESC key handling is tested at the App level — the global handler in
|
||||||
const onClose = vi.fn();
|
// App.tsx manages Escape for all modals, so per-component ESC tests are
|
||||||
render(<ProfileModal isOpen={true} onClose={onClose} />);
|
// not applicable here.
|
||||||
|
|
||||||
const overlay = document.querySelector(".modal-overlay");
|
|
||||||
if (overlay) {
|
|
||||||
fireEvent.keyDown(overlay, { key: "Escape" });
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(onClose).toHaveBeenCalledTimes(1);
|
|
||||||
});
|
|
||||||
|
|
||||||
it("does not close on non-escape keydown", () => {
|
|
||||||
const onClose = vi.fn();
|
|
||||||
render(<ProfileModal isOpen={true} onClose={onClose} />);
|
|
||||||
|
|
||||||
const overlay = document.querySelector(".modal-overlay");
|
|
||||||
if (overlay) {
|
|
||||||
fireEvent.keyDown(overlay, { key: "Enter" });
|
|
||||||
}
|
|
||||||
|
|
||||||
expect(onClose).not.toHaveBeenCalled();
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -15,6 +15,7 @@ const mockMedication: Medication = {
|
|||||||
id: 1,
|
id: 1,
|
||||||
name: "Test Med",
|
name: "Test Med",
|
||||||
genericName: "Generic Name",
|
genericName: "Generic Name",
|
||||||
|
packageType: "blister",
|
||||||
packCount: 1,
|
packCount: 1,
|
||||||
blistersPerPack: 1,
|
blistersPerPack: 1,
|
||||||
pillsPerBlister: 30,
|
pillsPerBlister: 30,
|
||||||
|
|||||||
@@ -39,18 +39,23 @@ vi.mock("../../utils/formatters", () => ({
|
|||||||
getSystemLocale: () => "en-US",
|
getSystemLocale: () => "en-US",
|
||||||
}));
|
}));
|
||||||
|
|
||||||
vi.mock("../../utils/schedule", () => ({
|
vi.mock("../../utils/schedule", async () => {
|
||||||
buildSchedulePreview: (...args: unknown[]) => mockBuildSchedulePreview(...args),
|
const actual = await vi.importActual<typeof import("../../utils/schedule")>("../../utils/schedule");
|
||||||
calculateCoverage: (...args: unknown[]) => mockCalculateCoverage(...args),
|
return {
|
||||||
computeMissedPastDoseIds: (...args: unknown[]) => mockComputeMissedPastDoseIds(...args),
|
...actual,
|
||||||
isDoseDismissed: vi.fn(() => false),
|
buildSchedulePreview: (...args: unknown[]) => mockBuildSchedulePreview(...args),
|
||||||
}));
|
calculateCoverage: (...args: unknown[]) => mockCalculateCoverage(...args),
|
||||||
|
computeMissedPastDoseIds: (...args: unknown[]) => mockComputeMissedPastDoseIds(...args),
|
||||||
|
isDoseDismissed: vi.fn(() => false),
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
const meds: Medication[] = [
|
const meds: Medication[] = [
|
||||||
{
|
{
|
||||||
id: 11,
|
id: 11,
|
||||||
name: "Aspirin",
|
name: "Aspirin",
|
||||||
takenBy: ["Max", "Anna"],
|
takenBy: ["Max", "Anna"],
|
||||||
|
packageType: "blister",
|
||||||
packCount: 1,
|
packCount: 1,
|
||||||
blistersPerPack: 1,
|
blistersPerPack: 1,
|
||||||
pillsPerBlister: 10,
|
pillsPerBlister: 10,
|
||||||
@@ -463,7 +468,7 @@ describe("useAppContext", () => {
|
|||||||
all: [
|
all: [
|
||||||
{
|
{
|
||||||
name: "Aspirin",
|
name: "Aspirin",
|
||||||
daysLeft: 2,
|
daysLeft: 8,
|
||||||
medsLeft: 5,
|
medsLeft: 5,
|
||||||
depletionTime: Date.now() + 100000,
|
depletionTime: Date.now() + 100000,
|
||||||
},
|
},
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user