candycore/candy-zone 问题修复 & 功能扩展

解决BUG、新增功能、兼容多环境部署,快速响应你的开发需求

邮箱:yvsm@zunyunkeji.com | QQ:316430983 | 微信:yvsm316

candycore/candy-zone

最新稳定版本:v0.2.0

Composer 安装命令:

composer require candycore/candy-zone

包简介

PHP port of lrstanley/bubblezone — mouse zone tracker for TUI apps.

README 文档

README

candy-zone

CandyZone

CI codecov Packagist Version License PHP

PHP port of lrstanley/bubblezone — mouse-zone tracker for TUI apps. Wrap rendered chunks with named markers, let CandyZone discover their bounding boxes, then ask zones whether a {@see \SugarCraft\Core\Msg\MouseMsg} fell inside them.

composer require sugarcraft/candy-zone
use SugarCraft\Zone\Manager;
use SugarCraft\Sprinkles\Style;

$z = Manager::newGlobal();

// Build a frame
$btnOk     = $z->mark('btn:ok',     Style::new()->padding(0, 2)->render('OK'));
$btnCancel = $z->mark('btn:cancel', Style::new()->padding(0, 2)->render('Cancel'));
$frame     = $btnOk . '   ' . $btnCancel;

// Scan once before printing — Manager records marker positions and strips them.
$displayable = $z->scan($frame);
echo $displayable;

// Later, when a MouseMsg arrives:
if ($z->get('btn:ok')?->inBounds($mouseMsg)) {
    // ...
}

Markers are APC escape sequences (ESC _ ... ESC \) — terminals ignore them, so they don't affect layout. {@see Manager::scan()} computes each zone's bounding box in 1-based terminal cells, accounting for ANSI styling and Unicode width.

Manager API

Beyond mark() / scan() / get():

  • setEnabled(bool) / isEnabled() — flip marker emission off in non-interactive contexts (CI logs, file dumps). When off, mark() returns content verbatim and scan() is identity.
  • Manager::newPrefix(?string) — namespace every id with a prefix so two CandyZone-aware components don't collide on 'item-0'. Auto- generates a monotonic prefix when called bare.
  • prefix() — read-only accessor for the prefix string.
  • get($id) / all() / clear(?$id) — single-zone lookup, every zone, and targeted-or-wipe-all clear.
  • close() — drop every zone + flip the manager into pass-through mode. Idempotent. PHP synchronous-scan has no worker to stop, so this is purely a state cleanup.

Package-level facade

SugarCraft\Zone\Zones mirrors bubblezone's package-level surface (bubblezone.DefaultManager + Mark / Scan / Clear / Get / Close / SetEnabled / Enabled / NewPrefix / AnyInBounds*) as static methods over a single shared Manager:

use SugarCraft\Zone\Zones;

$marked = Zones::mark('header', $header);
$cleaned = Zones::scan($marked);
if (Zones::get('header')?->inBounds($mouse)) { /* … */ }

Zones::setDefaultManager(?Manager) swaps in a custom manager — useful in tests (Zones::setDefaultManager(null) flushes state) or when you want every package-level call routed through a prefixed manager.

Tips

  • Each id should be unique within a Manager. Use Manager::newPrefix() per UI sub-tree so two child models don't shadow each other's ids.
  • Run scan() once on the full root frame, not per sub-tree — nested zone bounds depend on the outer layout.
  • lipgloss.Width() (CandySprinkles) and CandyZone interact cleanly: scan() strips markers before measurement.
  • Zone::isZero() distinguishes "never rendered" from "rendered but empty bounding box".
  • Organic shapes (ASCII art) report a rectangular bounding box — the marker pair only carries 4 corners' worth of information.
  • The PHP port has a synchronous scan() (no background worker), so close() is purely a state reset / disable rather than a thread join.

Test

cd candy-zone && composer install && vendor/bin/phpunit

统计信息

  • 总下载量: 0
  • 月度下载量: 0
  • 日度下载量: 0
  • 收藏数: 1
  • 点击次数: 6
  • 依赖项目数: 0
  • 推荐数: 0

GitHub 信息

  • Stars: 1
  • Watchers: 0
  • Forks: 0
  • 开发语言: PHP

其他信息

  • 授权协议: MIT
  • 更新时间: 2026-05-07

承接程序开发

PHP开发

VUE

Vue开发

前端开发

小程序开发

公众号开发

系统定制

数据库设计

云部署

网站建设

安全加固