Claude-skill-registry approval

Imported skill approval from langchain

install
source · Clone the upstream repo
git clone https://github.com/majiayu000/claude-skill-registry
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/majiayu000/claude-skill-registry "$T" && mkdir -p ~/.claude/skills && cp -r "$T/skills/data/approval" ~/.claude/skills/majiayu000-claude-skill-registry-approval && rm -rf "$T"
manifest: skills/data/approval/SKILL.md
source content

"""Approval widget for HITL - using standard Textual patterns."""

from future import annotations

import asyncio from typing import Any, ClassVar

from textual import events from textual.app import ComposeResult from textual.binding import Binding, BindingType from textual.containers import Container, Vertical, VerticalScroll from textual.message import Message from textual.widgets import Static

from deepagents_cli.widgets.tool_renderers import get_renderer

class ApprovalMenu(Container): """Approval menu using standard Textual patterns.

Key design decisions (following mistral-vibe reference):
- Container base class with compose()
- BINDINGS for key handling (not on_key)
- can_focus_children = False to prevent focus theft
- Simple Static widgets for options
- Standard message posting
- Tool-specific widgets via renderer pattern
"""

can_focus = True
can_focus_children = False

# CSS is in app.tcss - no DEFAULT_CSS needed

BINDINGS: ClassVar[list[BindingType]] = [
    Binding("up", "move_up", "Up", show=False),
    Binding("k", "move_up", "Up", show=False),
    Binding("down", "move_down", "Down", show=False),
    Binding("j", "move_down", "Down", show=False),
    Binding("enter", "select", "Select", show=False),
    Binding("1", "select_approve", "Approve", show=False),
    Binding("y", "select_approve", "Approve", show=False),
    Binding("2", "select_reject", "Reject", show=False),
    Binding("n", "select_reject", "Reject", show=False),
    Binding("3", "select_auto", "Auto-approve", show=False),
    Binding("a", "select_auto", "Auto-approve", show=False),
]

class Decided(Message):
    """Message sent when user makes a decision."""

    def __init__(self, decision: dict[str, str]) -> None:
        super().__init__()
        self.decision = decision

def __init__(
    self,
    action_request: dict[str, Any],
    assistant_id: str | None = None,
    id: str | None = None,  # noqa: A002
    **kwargs: Any,
) -> None:
    super().__init__(id=id or "approval-menu", classes="approval-menu", **kwargs)
    self._action_request = action_request
    self._assistant_id = assistant_id
    self._tool_name = action_request.get("name", "unknown")
    self._tool_args = action_request.get("args", {})
    self._description = action_request.get("description", "")
    self._selected = 0
    self._future: asyncio.Future[dict[str, str]] | None = None
    self._option_widgets: list[Static] = []
    self._tool_info_container: Vertical | None = None

def set_future(self, future: asyncio.Future[dict[str, str]]) -> None:
    """Set the future to resolve when user decides."""
    self._future = future

def compose(self) -> ComposeResult:
    """Compose the widget with Static children.

    Layout prioritizes options visibility - they appear at the top so users
    always see them even in small terminals.
    """
    # Title
    yield Static(
        f">>> {self._tool_name} Requires Approval <<<",
        classes="approval-title",
    )

    # Options container FIRST - always visible at top
    with Container(classes="approval-options-container"):
        # Options - create 3 Static widgets
        for i in range(3):
            widget = Static("", classes="approval-option")
            self._option_widgets.append(widget)
            yield widget

    # Help text right after options
    yield Static(
        "↑/↓ navigate • Enter select • y/n/a quick keys",
        classes="approval-help",
    )

    # Separator between options and tool details
    yield Static("─" * 40, classes="approval-separator")

    # Tool info in scrollable container BELOW options
    with VerticalScroll(classes="tool-info-scroll"):
        self._tool_info_container = Vertical(classes="tool-info-container")
        yield self._tool_info_container

async def on_mount(self) -> None:
    """Focus self on mount and update tool info."""
    await self._update_tool_info()
    self._update_options()
    self.focus()

async def _update_tool_info(self) -> None:
    """Mount the tool-specific approval widget."""
    if not self._tool_info_container:
        return

    # Get the appropriate renderer for this tool
    renderer = get_renderer(self._tool_name)
    widget_class, data = renderer.get_approval_widget(self._tool_args)

    # Clear existing content and mount new widget
    await self._tool_info_container.remove_children()
    approval_widget = widget_class(data)
    await self._tool_info_container.mount(approval_widget)

def _update_options(self) -> None:
    """Update option widgets based on selection."""
    options = [
        "1. Approve (y)",
        "2. Reject (n)",
        "3. Auto-approve all this session (a)",
    ]

    for i, (text, widget) in enumerate(zip(options, self._option_widgets, strict=True)):
        cursor = "› " if i == self._selected else "  "
        widget.update(f"{cursor}{text}")

        # Update classes
        widget.remove_class("approval-option-selected")
        if i == self._selected:
            widget.add_class("approval-option-selected")

def action_move_up(self) -> None:
    """Move selection up."""
    self._selected = (self._selected - 1) % 3
    self._update_options()

def action_move_down(self) -> None:
    """Move selection down."""
    self._selected = (self._selected + 1) % 3
    self._update_options()

def action_select(self) -> None:
    """Select current option."""
    self._handle_selection(self._selected)

def action_select_approve(self) -> None:
    """Select approve option."""
    self._selected = 0
    self._update_options()
    self._handle_selection(0)

def action_select_reject(self) -> None:
    """Select reject option."""
    self._selected = 1
    self._update_options()
    self._handle_selection(1)

def action_select_auto(self) -> None:
    """Select auto-approve option."""
    self._selected = 2
    self._update_options()
    self._handle_selection(2)

def _handle_selection(self, option: int) -> None:
    """Handle the selected option."""
    decision_map = {
        0: "approve",
        1: "reject",
        2: "auto_approve_all",
    }
    decision = {"type": decision_map[option]}

    # Resolve the future
    if self._future and not self._future.done():
        self._future.set_result(decision)

    # Post message
    self.post_message(self.Decided(decision))

def on_blur(self, event: events.Blur) -> None:
    """Re-focus on blur to keep focus trapped."""
    self.call_after_refresh(self.focus)