feat: add flexible ratio input for line item quality ratings

This commit is contained in:
Sean McElwain 2026-05-01 17:30:24 -05:00
parent f2f2ac3310
commit 6ae16c1808
4 changed files with 249 additions and 14 deletions

View File

@ -40,6 +40,15 @@ def _to_decimal(value: str | None) -> Decimal | None:
return None
def _format_decimal_compact(value: Decimal | None) -> str:
if value is None:
return ""
text = format(value.normalize(), "f")
if "." in text:
text = text.rstrip("0").rstrip(".")
return text or "0"
def _line_item_extra(item: DocumentLineItem) -> dict:
return dict(item.raw_json or {})
@ -49,6 +58,28 @@ def _line_item_quality_rating(item: DocumentLineItem) -> str:
return "" if value is None else str(value)
def _line_item_quality_rating_value(item: DocumentLineItem) -> str:
extra = _line_item_extra(item)
value = extra.get("quality_rating_value")
if value is not None:
return str(value)
raw = _to_decimal(extra.get("quality_rating"))
return "" if raw is None else _format_decimal_compact(raw)
def _line_item_quality_rating_scale(item: DocumentLineItem) -> str:
extra = _line_item_extra(item)
scale = extra.get("quality_rating_scale")
if scale is not None and str(scale).strip():
return str(scale)
raw = _to_decimal(extra.get("quality_rating"))
if raw is None:
return "10"
return "100" if raw > 10 else "10"
def _line_item_quality_note(item: DocumentLineItem) -> str:
value = _line_item_extra(item).get("quality_note")
return "" if value is None else str(value)
@ -105,6 +136,8 @@ def _build_row(item: DocumentLineItem) -> dict | None:
"category": item.category or "",
"confidence": "",
"quality_rating": _line_item_quality_rating(item),
"quality_rating_value": _line_item_quality_rating_value(item),
"quality_rating_scale": _line_item_quality_rating_scale(item),
"quality_note": _line_item_quality_note(item),
"quality_status": _line_item_quality_status(item),
"is_reviewed": bool(_line_item_extra(item).get("reviewed_at")),
@ -260,7 +293,8 @@ def save_line_item_review(
rating_min: str = Form(""),
rating_max: str = Form(""),
return_to: str = Form("list"),
quality_rating: str = Form(""),
quality_rating_value: str = Form(""),
quality_rating_scale: str = Form(""),
quality_note: str = Form(""),
is_approved: str = Form(""),
is_excluded: str = Form(""),

View File

@ -6083,3 +6083,151 @@ table {
}
}
/* ===== end line item queue header relayout ===== */
/* ===== line item queue card polish ===== */
.line-queue-card-head {
display: grid;
grid-template-columns: minmax(0, 1fr) auto auto;
gap: 0.72rem 0.85rem;
align-items: start;
margin-bottom: 0.62rem;
}
.line-queue-title-block {
min-width: 0;
}
.line-queue-kicker {
font-size: 0.82rem;
line-height: 1.12;
color: #6b7280;
margin-bottom: 0.08rem;
}
.line-queue-title {
margin: 0 !important;
font-size: 0.96rem;
line-height: 1.02;
font-weight: 700;
color: #1f2937;
word-break: break-word;
}
.line-queue-flags {
display: flex;
flex-direction: column;
gap: 0.16rem;
align-items: flex-start;
padding-top: 0.02rem;
}
.line-queue-flag {
display: grid;
grid-template-columns: 0.95rem auto;
align-items: center;
column-gap: 0.38rem;
font-size: 0.78rem;
line-height: 1.02;
white-space: nowrap;
margin: 0;
}
.line-queue-flag input {
width: 0.92rem;
height: 0.92rem;
margin: 0;
}
.line-queue-badges {
display: flex;
flex-wrap: wrap;
justify-content: flex-end;
gap: 0.28rem;
}
.line-queue-badges .badge {
font-size: 0.74rem;
padding: 0.2rem 0.48rem;
line-height: 1.04;
white-space: nowrap;
}
.queue-rating-input-row {
display: grid;
grid-template-columns: minmax(0, 1fr) auto 5.2rem;
gap: 0.38rem;
align-items: center;
}
.queue-rating-slash {
font-size: 1rem;
color: #6b7280;
text-align: center;
}
.queue-rating-input-row input {
height: 2.25rem;
font-size: 0.82rem;
}
.queue-rating-denominator {
border-radius: 0.92rem;
text-align: center;
}
.queue-rating-scale-toggle {
min-width: 4.5rem;
border-radius: 0.92rem;
border: 1px solid #cfd6df;
background: #fff;
color: #111827;
padding: 0 0.85rem;
cursor: pointer;
}
.queue-rating-scale-toggle:active {
transform: translateY(1px);
}
@media (max-width: 900px) {
.line-queue-card-head {
grid-template-columns: minmax(0, 1fr) auto auto;
gap: 0.52rem 0.62rem;
}
.line-queue-kicker {
font-size: 0.78rem;
}
.line-queue-title {
font-size: 0.9rem;
}
.line-queue-flag {
font-size: 0.74rem;
grid-template-columns: 0.9rem auto;
column-gap: 0.32rem;
}
.line-queue-flag input {
width: 0.86rem;
height: 0.86rem;
}
.line-queue-badges .badge {
font-size: 0.7rem;
padding: 0.18rem 0.42rem;
}
.queue-rating-input-row input {
height: 2.15rem;
font-size: 0.8rem;
}
.queue-rating-input-row {
grid-template-columns: minmax(0, 1fr) auto 4.7rem;
}
}
/* ===== end line item queue card polish ===== */

View File

@ -4,7 +4,7 @@
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title>{% block title %}Document Processor{% endblock %}</title>
<link rel="stylesheet" href="/static/app.css?v=166">
<link rel="stylesheet" href="/static/app.css?v=171">
<link rel="stylesheet" href="/static/app-shell.css?v=158">
</head>
<body>

View File

@ -1,5 +1,12 @@
{% extends "base.html" %}
{% block title %}Line Items{% endblock %}
{% block title %}Line Items
{% endblock %}
{% block content %}
<div class="page-header">
@ -210,31 +217,31 @@
<form method="post" action="/line-items/{{ row.line_item_id }}/review">
<input type="hidden" name="return_to" value="queue">
<div class="queue-review-header">
<div class="queue-review-title-block">
<div class="queue-review-meta">{{ row.transaction_date }} · {{ row.merchant }}</div>
<h3 class="card-title queue-review-title">{{ row.description }}</h3>
<div class="line-queue-card-head">
<div class="line-queue-title-block">
<div class="line-queue-kicker">{{ row.transaction_date }} · {{ row.merchant }}</div>
<h3 class="card-title line-queue-title">{{ row.description }}</h3>
{% if row.raw_description and row.raw_description != row.description %}
<div class="queue-review-raw">{{ row.raw_description }}</div>
{% endif %}
</div>
<div class="queue-review-flags">
<label class="queue-flag-row">
<div class="line-queue-flags">
<label class="line-queue-flag">
<input type="checkbox" name="is_approved" value="1" {% if row.is_approved %}checked{% endif %}>
<span>Approved</span>
</label>
<label class="queue-flag-row">
<label class="line-queue-flag">
<input type="checkbox" name="is_excluded" value="1" {% if row.is_excluded %}checked{% endif %}>
<span>Excluded</span>
</label>
<label class="queue-flag-row">
<label class="line-queue-flag">
<input type="checkbox" name="is_na" value="1" {% if row.is_na %}checked{% endif %}>
<span>N/A</span>
</label>
</div>
<div class="badges queue-review-badges">
<div class="line-queue-badges">
{% if row.category %}
<span class="badge">{{ row.category }}</span>
{% endif %}
@ -249,8 +256,27 @@
<div class="queue-review-body">
<div class="form-field">
<label for="queue_quality_rating_{{ row.line_item_id }}">Quality rating</label>
<input id="queue_quality_rating_{{ row.line_item_id }}" class="queue-review-rating-input" type="text" name="quality_rating" value="{{ row.quality_rating }}" placeholder="e.g. 8.5 or 4/5">
<label for="queue_quality_rating_value_{{ row.line_item_id }}">Quality rating</label>
<div class="queue-rating-input-row">
<input
class="queue-rating-value queue-review-rating-input"
id="queue_quality_rating_value_{{ row.line_item_id }}"
type="text"
name="quality_rating_value"
value="{{ row.quality_rating_value }}"
placeholder="8.5"
oninput="queueRatingAutoDenominator(this, document.getElementById('queue_quality_rating_scale_{{ row.line_item_id }}'))"
>
<span class="queue-rating-slash">/</span>
<input
class="queue-rating-denominator"
id="queue_quality_rating_scale_{{ row.line_item_id }}"
type="text"
name="quality_rating_scale"
value="{{ row.quality_rating_scale or '10' }}"
placeholder="10"
>
</div>
</div>
<div class="form-field">
@ -273,4 +299,31 @@
</div>
</div>
<script>
function queueRatingAutoDenominator(valueEl, denomEl) {
if (!valueEl || !denomEl) return;
const raw = String(valueEl.value || "").trim();
if (!raw) return;
const value = Number(raw);
if (!Number.isFinite(value)) return;
const current = String(denomEl.value || "").trim();
if (current === "" || current === "10" || current === "100") {
if (value >= 0 && value <= 10) {
denomEl.value = "10";
} else if (value > 10 && value <= 100) {
denomEl.value = "100";
}
}
}
</script>
{% endblock %}