fix: reasoning chip dropdown visible + monochrome SVG icon + /btw answer preserved (closes #933) (#934)
* fix: reasoning chip dropdown visible + SVG icon + /btw answer no longer wiped (closes #933) * fix(ui): resize handler symmetry + lock regressions for PR #934 fixes Two small additions on top of the core PR: 1. Resize handler now re-positions the reasoning dropdown when the window resizes while it's open, matching the existing model-dropdown branch. Without this, resizing while the dropdown is open leaves it aligned to the pre-resize chip position — fine in practice (most resizes close the dropdown via the global click handler) but inconsistent with the model-dropdown sibling. 2. Regression test file tests/test_reasoning_chip_btw_fixes.py with 10 tests locking all four fixes in place so they can't silently regress: - Dropdown sits OUTSIDE .composer-left (so overflow-y: hidden can't clip it) - Dropdown is grouped with the other composer-level dropdowns - Chip button contains stroke="currentColor" SVG (not a 🧠 emoji) - _applyReasoningChip() body doesn't include 🧠 - cmdReasoning calls _applyReasoningChip(eff) directly with the server-confirmed effort, not syncReasoningChip() (stale cache) - _streamDone flag declared, set in done handler, checked in onerror - _ensureBtwRow() called in done handler (creates bubble when no tokens arrive) - resize handler re-positions composerReasoningDropdown Full suite: 2056 passed, 0 failed. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com> --------- Co-authored-by: nesquena-hermes <nesquena-hermes@users.noreply.github.com> Co-authored-by: Nathan Esquenazi <nesquena@gmail.com> Co-authored-by: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
224
tests/test_reasoning_chip_btw_fixes.py
Normal file
224
tests/test_reasoning_chip_btw_fixes.py
Normal file
@@ -0,0 +1,224 @@
|
||||
"""Regression tests for PR #934 UI fixes.
|
||||
|
||||
Four invariants this file locks in place:
|
||||
|
||||
1. `#composerReasoningDropdown` lives OUTSIDE `.composer-left` (as a sibling of
|
||||
the other composer dropdowns), so it isn't clipped by that container's
|
||||
`overflow-y: hidden`. Regresses to invisible-dropdown if moved back.
|
||||
|
||||
2. The reasoning chip label uses an SVG icon (`stroke="currentColor"`) instead
|
||||
of the `🧠` emoji, matching every other composer chip.
|
||||
|
||||
3. `cmdReasoning()` calls `_applyReasoningChip(eff)` directly with the
|
||||
server-confirmed effort, not `syncReasoningChip()` which re-applies the
|
||||
stale cached value.
|
||||
|
||||
4. `attachBtwStream()` sets a `_streamDone` flag in `done`/`apperror` and
|
||||
gates `onerror`'s row removal on `!_streamDone` — otherwise the browser's
|
||||
post-`stream_end` error event wipes the just-rendered answer.
|
||||
"""
|
||||
from __future__ import annotations
|
||||
|
||||
import pathlib
|
||||
import re
|
||||
|
||||
|
||||
REPO = pathlib.Path(__file__).resolve().parent.parent
|
||||
INDEX = (REPO / "static" / "index.html").read_text(encoding="utf-8")
|
||||
UI_JS = (REPO / "static" / "ui.js").read_text(encoding="utf-8")
|
||||
COMMANDS_JS = (REPO / "static" / "commands.js").read_text(encoding="utf-8")
|
||||
MESSAGES_JS = (REPO / "static" / "messages.js").read_text(encoding="utf-8")
|
||||
|
||||
|
||||
# ── #1 dropdown escapes composer-left ─────────────────────────────────────────
|
||||
|
||||
|
||||
class TestReasoningDropdownEscapesComposerLeft:
|
||||
"""The dropdown must sit as a sibling of .composer-footer, not inside
|
||||
.composer-left which has overflow-y: hidden and clips absolute children."""
|
||||
|
||||
def test_dropdown_lives_outside_composer_left(self):
|
||||
# Find the <div class="composer-left">...</div> block and confirm the
|
||||
# reasoning dropdown is NOT inside it.
|
||||
m = re.search(
|
||||
r'<div class="composer-left"[^>]*>(?P<body>[\s\S]*?)<div class="composer-footer-right"',
|
||||
INDEX,
|
||||
)
|
||||
# Some templates use different closing structures; fall back to a
|
||||
# coarser search that at least locates composer-left.
|
||||
if m:
|
||||
inner = m.group("body")
|
||||
assert 'id="composerReasoningDropdown"' not in inner, (
|
||||
"composerReasoningDropdown is still nested inside .composer-left — "
|
||||
"this is the exact bug #933 flagged: overflow-y: hidden clips "
|
||||
"upward-opening absolute dropdowns. Move it alongside "
|
||||
"#composerModelDropdown / #composerWsDropdown / #profileDropdown."
|
||||
)
|
||||
# Either way, check that the dropdown sits next to the other composer
|
||||
# dropdowns (reliable structural marker).
|
||||
assert '<div class="profile-dropdown" id="profileDropdown"></div>' in INDEX
|
||||
assert 'id="composerReasoningDropdown"' in INDEX
|
||||
|
||||
def test_dropdown_is_sibling_of_other_composer_dropdowns(self):
|
||||
# The four composer-level dropdowns must appear contiguously — if one
|
||||
# of them is nested inside an overflow-hidden container, this would
|
||||
# typically split the group.
|
||||
positions = [
|
||||
("profileDropdown", INDEX.find('id="profileDropdown"')),
|
||||
("composerWsDropdown", INDEX.find('id="composerWsDropdown"')),
|
||||
("composerReasoningDropdown", INDEX.find('id="composerReasoningDropdown"')),
|
||||
("composerModelDropdown", INDEX.find('id="composerModelDropdown"')),
|
||||
]
|
||||
for name, pos in positions:
|
||||
assert pos > -1, f"{name} not found in index.html"
|
||||
# They should all be in the same area of the document — within ~1.5 KB
|
||||
window = [p for _, p in positions]
|
||||
assert max(window) - min(window) < 2000, (
|
||||
"composer dropdowns are no longer grouped — reasoning dropdown may "
|
||||
"have drifted back inside a nested container"
|
||||
)
|
||||
|
||||
|
||||
# ── #2 monochrome SVG replaces emoji ──────────────────────────────────────────
|
||||
|
||||
|
||||
class TestReasoningChipIcon:
|
||||
"""The chip must render a currentColor SVG, not a 🧠 emoji, for cross-platform
|
||||
rendering consistency with the other composer chips."""
|
||||
|
||||
def test_chip_button_contains_svg_with_currentColor(self):
|
||||
# Locate the chip button and confirm it contains a stroke="currentColor" SVG
|
||||
m = re.search(
|
||||
r'<button class="composer-reasoning-chip"[^>]*>([\s\S]*?)</button>',
|
||||
INDEX,
|
||||
)
|
||||
assert m, "composer-reasoning-chip button not found"
|
||||
btn_body = m.group(1)
|
||||
assert 'stroke="currentColor"' in btn_body, (
|
||||
"reasoning chip must use stroke='currentColor' SVG matching other chips"
|
||||
)
|
||||
assert '<svg' in btn_body, "reasoning chip must contain an <svg> icon"
|
||||
|
||||
def test_apply_reasoning_chip_label_has_no_emoji(self):
|
||||
# Locate _applyReasoningChip and confirm the label assignment doesn't
|
||||
# concatenate a 🧠 emoji.
|
||||
m = re.search(
|
||||
r"function\s+_applyReasoningChip\b[\s\S]*?^\}",
|
||||
UI_JS,
|
||||
re.MULTILINE,
|
||||
)
|
||||
assert m, "_applyReasoningChip not found in ui.js"
|
||||
fn = m.group(0)
|
||||
assert "🧠" not in fn, (
|
||||
"_applyReasoningChip should not concatenate a 🧠 emoji into the label — "
|
||||
"the chip already has a monochrome SVG icon next to the label"
|
||||
)
|
||||
|
||||
|
||||
# ── #3 /reasoning immediately updates chip ────────────────────────────────────
|
||||
|
||||
|
||||
class TestReasoningCommandUpdatesChip:
|
||||
"""cmdReasoning must apply the SERVER-CONFIRMED effort, not the cached value."""
|
||||
|
||||
def test_cmd_reasoning_calls_apply_not_sync(self):
|
||||
# Locate cmdReasoning and verify the success branch calls
|
||||
# _applyReasoningChip(eff) directly, not syncReasoningChip() which
|
||||
# would read stale _currentReasoningEffort.
|
||||
m = re.search(
|
||||
r"function\s+cmdReasoning\b[\s\S]*?(?=^function\s|\Z)",
|
||||
COMMANDS_JS,
|
||||
re.MULTILINE,
|
||||
)
|
||||
assert m, "cmdReasoning not found in commands.js"
|
||||
fn = m.group(0)
|
||||
assert "_applyReasoningChip(eff)" in fn, (
|
||||
"cmdReasoning must call _applyReasoningChip(eff) with the "
|
||||
"server-confirmed effort from the /api/reasoning POST response"
|
||||
)
|
||||
|
||||
|
||||
# ── #4 /btw answer not wiped by onerror after clean close ─────────────────────
|
||||
|
||||
|
||||
class TestBtwStreamDoneGuard:
|
||||
"""attachBtwStream must guard onerror with a _streamDone flag so the
|
||||
browser's post-stream_end error event doesn't wipe the just-rendered row."""
|
||||
|
||||
def get_attach_btw(self):
|
||||
m = re.search(
|
||||
r"function\s+attachBtwStream\b[\s\S]*?(?=^function\s|\Z)",
|
||||
MESSAGES_JS,
|
||||
re.MULTILINE,
|
||||
)
|
||||
assert m, "attachBtwStream not found in messages.js"
|
||||
return m.group(0)
|
||||
|
||||
def test_stream_done_flag_declared(self):
|
||||
fn = self.get_attach_btw()
|
||||
assert "_streamDone" in fn, (
|
||||
"attachBtwStream must declare a _streamDone flag to distinguish "
|
||||
"clean server-closed streams from real errors"
|
||||
)
|
||||
|
||||
def test_stream_done_set_in_done_handler(self):
|
||||
fn = self.get_attach_btw()
|
||||
# Inside the 'done' listener body, _streamDone must be set true.
|
||||
done_block_m = re.search(
|
||||
r"addEventListener\('done'[\s\S]*?(?=addEventListener\(')",
|
||||
fn,
|
||||
)
|
||||
assert done_block_m, "done handler not found in attachBtwStream"
|
||||
assert "_streamDone=true" in done_block_m.group(0) or \
|
||||
"_streamDone = true" in done_block_m.group(0), (
|
||||
"_streamDone must be set to true in the done handler so onerror "
|
||||
"knows the stream completed successfully"
|
||||
)
|
||||
|
||||
def test_onerror_gated_on_stream_done(self):
|
||||
fn = self.get_attach_btw()
|
||||
# onerror must NOT unconditionally call btwRow.remove()
|
||||
m = re.search(r"src\.onerror\s*=\s*\(?\)?\s*=>\s*\{[^}]*\}", fn)
|
||||
assert m, "src.onerror assignment not found"
|
||||
handler = m.group(0)
|
||||
assert "_streamDone" in handler, (
|
||||
"src.onerror must check !_streamDone before removing the btw row — "
|
||||
"otherwise the browser's post-stream_end error fire wipes the "
|
||||
"answer that was just rendered by the done handler"
|
||||
)
|
||||
|
||||
def test_ensure_btw_row_called_in_done(self):
|
||||
"""The done handler must create the row even if no token events arrived
|
||||
(e.g., agent returned a non-streaming single-shot answer)."""
|
||||
fn = self.get_attach_btw()
|
||||
done_block_m = re.search(
|
||||
r"addEventListener\('done'[\s\S]*?(?=addEventListener\(')",
|
||||
fn,
|
||||
)
|
||||
assert done_block_m
|
||||
assert "_ensureBtwRow()" in done_block_m.group(0), (
|
||||
"done handler must call _ensureBtwRow() so the answer bubble exists "
|
||||
"even if no token events arrived before done"
|
||||
)
|
||||
|
||||
|
||||
# ── #5 resize handler symmetry (non-blocking polish) ─────────────────────────
|
||||
|
||||
|
||||
class TestResizeHandlerSymmetry:
|
||||
"""When the window resizes while either the model OR reasoning dropdown is
|
||||
open, the dropdown must be re-positioned so it stays aligned under its chip."""
|
||||
|
||||
def test_resize_repositions_reasoning_dropdown(self):
|
||||
# The global resize handler must handle both composerModelDropdown AND
|
||||
# composerReasoningDropdown to keep them aligned when the window resizes.
|
||||
m = re.search(
|
||||
r"window\.addEventListener\(\s*['\"]resize['\"][\s\S]*?\}\s*\)\s*;",
|
||||
UI_JS,
|
||||
)
|
||||
assert m, "window resize handler not found in ui.js"
|
||||
handler = m.group(0)
|
||||
assert "composerReasoningDropdown" in handler, (
|
||||
"window resize handler must also re-position composerReasoningDropdown "
|
||||
"while it's open (symmetric with the existing model-dropdown branch)"
|
||||
)
|
||||
Reference in New Issue
Block a user