Files
StellarX/CHANGELOG.en.md

24 KiB
Raw Blame History

Changelog

All notable changes to the StellarX project will be documented in this file.

The format is based on Keep a Changelog, and this project adheres to Semantic Versioning.

中文文档

[v2.3.1] - 2025-11-30

🙏 Acknowledgements

  • Special thanks to user @To-KongBai for providing stable reproduction steps and key phenomenon comparisons (container nested child control coordinate transformation issue), which helped us quickly identify and fix the control coordinate transformation problem in deeply nested containers. (Issues#6)
  • In the upcoming website (currently undergoing ICP filing), we plan to add a contributors' wall to acknowledge users. We welcome everyone to report bugs or share interfaces created with StellarX, and we will carefully read and include acknowledgements.
  • Sincere thanks to every user who reports bugs—your feedback makes StellarX more stable and robust.

⚙️ Changes

  • Dialog Background Snapshot Mechanism: Dialog no longer captures and destroys its own snapshot. The methods for capturing and restoring snapshots have been removed from Dialog, and the base class Canvas now handles all snapshot management. The draw method in Dialog no longer deals with snapshots.
  • Timing Adjustment for Window and Control Redrawing on Size Change: In the main event loop, when the window size changes, control sizes are processed first. Old snapshots are discarded, followed by the redrawing of the window with the new size, and finally, the controls are redrawn.

Fixes

  • Child Control Coordinate Transformation in Nested Containers: Canvas overrides the base class's setX/Y methods to synchronize the global coordinates of child controls when the parent container's global coordinates change. This prevents child controls in nested containers from incorrectly treating the container's relative coordinates as global coordinates.
  • Solid Background Window Title Not Applied: In the Window's draw() method, the window title is forcibly set to ensure that the title passed when creating the window is applied correctly, preventing the issue where the window title doesn't take effect.
  • Tab Control Background Residue when Changing Coordinates: In the overridden setX/Y methods of TabControl, all tabs and their child controls are forced to discard snapshots when the tab's coordinates change. This prevents background residue caused by incorrect snapshot restoration order after modifying coordinates.
  • Table Page Number Label and Pagination Button Misalignment when Changing Coordinates: In Table's setX/Y, the isNeedButtonAndPageNum state is reset to true, ensuring that the pagination button and page number label are recalculated and remain centered directly beneath the table when redrawn.

[v2.3.0] - 2025-11-18

Added

  • Introduced LayoutMode adaptive layout mode and Anchor anchor points. Controls can now call setLayoutMode to set layout mode and steAnchor to set anchor points, enabling controls to adapt to window changes during resizing. Dialog controls only recalculate their position during window resizing to maintain center alignment.
  • Added adaptiveLayout() API to Window, called by the main event loop to recalculate control positions and sizes based on anchor points during window resizing, enabling dual-anchored controls (left-right or top-bottom) to automatically stretch with window changes.

⚙️ Changed

  • Optimized Window Resizing Mechanism: Refactored WndProcThunk, runEventLoop, and pumpResizeIfNeeded to uniformly record window size changes and perform one-time repainting at the end of the event loop, preventing jitter and sequencing issues caused by repeated drawing during resizing.
  • Added Dialog Size Scheduling Interface: Introduced Window::scheduleResizeFromModal(), used in conjunction with pumpResizeIfNeeded(). During modal dialog display, the parent window can update client area size in real time and relayout child controls during unified finalization, while the dialog maintains its original size.
  • Redraw Sequence Optimization: Replaced InvalidateRect with ValidateRect during the finalization phase after window size changes, preventing the system from sending additional WM_PAINT messages that could cause reentrant drawing.
  • Other Improvements: Fixed delayed background snapshot updates for table pagination buttons and page number labels; improved dialog background capture logic.

Fixed

  • Modal Dialog Resizing Fix: Resolved the issue where window resizing during modal dialog display prevented underlying controls from updating their sizes and positions according to anchor points; simultaneously eliminated ghosting artifacts caused by repeated dialog redrawing.
  • Drawing Sequence Disorder Fix: Addressed sporadic drawing sequence disorders, control ghosting, and border flickering during window resizing, ensuring controls are drawn in the order they were added.
  • Stability Fixes: Corrected abnormal frames caused by sudden window size changes in certain scenarios; resolved delayed background snapshot updates for tables and dialogs under edge conditions.

[v2.2.2] - 2025 - 11- 08

⚙️ Changes

  • Modified the coordinate transfer method for the Canvas container. Child control coordinates are now passed as relative coordinates (with the origin at the top-left corner of the container, obtainable via the getX/getY interface), instead of the original global coordinates. Child control coordinates can now be set to negative values.
  • The example under examples\register-viewer has been updated to the latest version, aligning container child controls to use relative coordinates.

Fixes

  • Fixed jittering/bouncing and flickering when resizing the window (left/top edges):
    • WM_SIZING only clamps the minimum size; WM_GETMINMAXINFO sets the window-level minimum tracking size.
    • Freezes redrawing during dragging and handles resizing uniformly upon release; WM_SIZE only records the new size without interfering with drawing.
    • Disabled WM_ERASEBKGND background erasing and removed CS_HREDRAW/CS_VREDRAW to reduce flickering.
  • Fixed issues related to dialog boxes:
    • Resolved occasional residual functional buttons after closing a dialog.
    • Fixed issues where window resizing failed to redraw or displayed a corrupted background when a modal dialog was active.
  • Fixed delayed updates of background snapshots for the table control's pagination buttons and page number labels during window changes.

[v2.2.1] - 2025-11-04

==This release is a hotfix for v2.2.0==

Fixed

  • The TabControl class overrode the base class's setDirty method to ensure synchronized update status between the tab and its page list.
  • The Canvas container, special container TabControl, and dialog Dialog overrode the requestRepaint method. When control bubbling propagates upward, the parent pointer is passed. Repaint requests now only bubble up one level to the parent and no longer propagate to the root. Furthermore, the entire parent container is no longer repainted; instead, the parent container repaints only the dirtied child controls, avoiding flickering caused by frequent repaints of the entire container.
  • The saveBackground and restoreBackground methods were overridden in Dialog to ensure no border remnants remain after the dialog is closed.

⚙️ Changed

  • Completely disabled copy and move semantics for the Control class: Control(const Control&) = delete; Control& operator=(const Control&) = delete; Control(Control&&) = delete; Control& operator=(Control&&) = delete;

[v2.2.0] - 2025-11-02

Highlights: Officially introduces the TabControl, enhances control show/hide and layout responsiveness, and refines the text styling mechanism; fixes several UI details to improve stability.

Added

  • TabControl (tabbed container control): Added the TabControl class to implement a multi-page tabbed UI. The tab bar supports top/bottom/left/right positions via TabControl::setTabPlacement(...). Provides TabControl::add(std::pair<std::unique_ptr<Button>, std::unique_ptr<Canvas>>&&) to add a “tab button + page content” pair in one go and automatically manage switching. Each pages content area is hosted by a Canvas; clicking different tabs (Button, TOGGLE mode) switches the visible page (see API). TabControl can automatically adjust tab layout when the window size changes and uses background snapshots to avoid ghosting under transparent themes.
  • Control visibility: All controls now support runtime show/hide toggling. The base class adds Control::setIsVisible(bool) to control a controls own visibility; hidden controls no longer participate in drawing and events. Container controls (Canvas) override this to implement cascading hide: hiding a container automatically hides all its child controls, and showing it restores them. This makes it more convenient to toggle the visibility of a group of UI elements.
  • Window resize handling: Introduces the virtual function Control::onWindowResize(), called when the parent window size changes. Controls can implement this to respond to window resizing (e.g., reset background caches, adjust layout). Canvas implements this and recursively notifies child controls to call onWindowResize(), ensuring nested layouts update correctly. This improvement fixes possible misalignment or background issues that occurred after resizing.
  • Label text style structure: The Label control now uses a unified ControlText style structure to manage fonts and colors. By accessing the public member Label::textStyle, you can set font name, size, color, underline, etc., achieving full text style customization (replacing the previous setTextColor interface). This enables richer formatting when displaying text with Label.
  • Dialog de-duplication mechanism: Window adds an internal check to prevent popping up duplicate dialogs with the same content. When using MessageBox::showAsync for a non-modal popup, the framework checks if a dialog with the same title and message is already open; if so, it avoids creating another one. This prevents multiple identical prompts from appearing in quick succession.

⚙️ Changed

  • Text color interface adjustment: Label removes the outdated setTextColor method; use the public textStyle.color to set text color instead. To change a Labels text color, modify label.textStyle.color and redraw. This improves consistency in text property management but may be incompatible with older code and require replacement.
  • Tooltip styling and toggle: The Button Tooltip interface is adjusted to support more customization. Button::setTooltipStyle can now flexibly set tooltip text color, background color, and transparency; setTooltipTextsForToggle(onText, offText) is added so toggle buttons can display different tooltip texts in ON/OFF states. The original tooltip-text setting interface remains compatible, but the internal implementation is optimized, fixing the previous issue where toggle-button tooltip text didnt update.
  • Control coordinate system and layout: Controls now maintain both global coordinates and local coordinates. Control adds members (such as getLocalX()/getLocalY()) to get positions relative to the parent container, and a parent pointer to the parent container. This makes layout calculations in nested containers more convenient and accurate. In absolute-layout scenarios, a controls global coordinates are automatically converted and stored as local coordinates when added to a container. Note: in the new version, when moving controls or changing sizes, prefer using the controls own setters to keep internal coordinates in sync.
  • Window resizing defaults: Resizing is changed from experimental to enabled by default. The framework always enables resizable styles for the main window (WS_THICKFRAME | WS_MAXIMIZEBOX | ...), so theres no need to call a separate method to enable it. This simplifies usage and means created windows can be resized by users by default. For scenarios where resizing is not desired, pass specific mode flags at creation time to disable it.

Fixed

  • Toggle-button tooltip updates: Fixed an issue where tooltips for toggle-mode buttons did not update promptly when the state changed. With setTooltipTextsForToggle, the tooltip now correctly shows the corresponding text when switching between ON/OFF.
  • Background ghosting and coordinate sync for controls: Fixed defects where, in some cases, control backgrounds were not refreshed in time and position calculations deviated after window/container resizing. By using onWindowResize to uniformly discard and update background snapshots, background ghosting and misalignment during window resizing are avoided, improving overall stability.
  • Control class background-snapshot memory leak: The destructor now calls discardBackground to free and restore the background snapshot, preventing the memory leak caused by not releasing *saveBkImage in the previous version.
  • Duplicate dialog pop-ups: Fixed an issue where repeatedly calling a non-modal message box in quick succession could create multiple identical dialogs. The new de-duplication check ensures that only one non-modal dialog with the same content exists at a time, avoiding UI disruption.
  • Other: Optimized the control drawing/refresh strategy to reduce unnecessary repaints in certain scenarios and improve performance; corrected minor memory-management details to eliminate potential leaks. These enhancements further improve the frameworks performance and reliability.

[v2.1.0] - 2025-10-27

Focus: Resizable/maximized window (EasyX reinforced by Win32), first-phase layout manager (HBox/VBox), early Tabs control. We also fixed black borders, maximize “ghosts”, flicker, and the issue where controls only appeared after interaction. Control-level background snapshot/restore, single-line truncation, and tooltips are now standardized.

Added

  • Bilingual API Documentation (Chinese and English)

    • The documentation provides a detailed introduction of each class, including API descriptions, functionalities, and points to note, with a comprehensive explanation of each control.
  • Window resize/maximize reinforcement (EasyX + Win32)

    • Window::enableResize(bool enable, int minW, int minH); toggle at runtime and set min track size.
    • Subclassed WndProc for WM_GETMINMAXINFO / WM_SIZE / WM_EXITSIZEMOVE / WM_ERASEBKGND / WM_PAINT with WS_THICKFRAME | WS_MAXIMIZEBOX | WS_MINIMIZEBOX | WS_CLIPCHILDREN | WS_CLIPSIBLINGS.
    • Full-surface background paint in WM_PAINT (solid/image), removing black borders and maximize ghosts.
    • One-shot redraw on resize with pendingW/H + needResizeDirty coalescing.
  • Layout manager (phase 1)

    • On Canvas: LayoutKind::{Absolute, HBox, VBox, Grid(placeholder), Flow(placeholder), Stack(placeholder)} with LayoutParams (margins, fixedW/fixedH(-1=same), weight, Align{Start,Center,End,Stretch}).
    • Implemented HBox/VBox auto layout inside containers; containers stay absolutely positioned; nesting supported.
  • Tabs control (early)

    • Decoupled tab strip and page container; background snapshot for transparent themes.
  • Button single-line truncation (MBCS; CN/EN aware)

    • Pixel-width threshold with ... avoiding half-glyph artifacts.
  • Hover tooltip

    • Implemented via Label with delay & auto-hide; per-control background snapshot/restore; customizable text with sensible fallback.

🔧 Changed

  • Window rendering path
    • Reduced reliance on “offscreen frame blit”; in WM_PAINT use GDI full background + EasyX batch drawing (BeginBatchDraw/EndBatchDraw + FlushBatchDraw) to suppress flicker.
    • On resize, only rebuild the scaled background (zoomBackground); actual painting happens next frame.
  • Control base
    • Standardized captureBackground/restoreBackground; transparent/stacked visuals are stable.
    • Unified dirty: containers paint their own background when dirty; children still evaluate/draw as needed each frame.
  • Table
    • In transparent mode, pagination widgets now inherit table text/fill style.
    • Reworked pagination math + block centering (header included in available width).
    • Fixed background snapshot sizing (header inclusion) that caused failed restores.
  • Event loop
    • Coalesced WM_SIZE to the loop tail to avoid redraw storms and pointer-hover lag.

Fixed

  • Black borders / maximize ghost / flicker: blocked system background erase; full-surface paint in WM_PAINT; cleared clipping to prevent stale fragments.
  • Containers not drawn; controls only after interaction: first-frame & post-input full-tree dirty; children draw even if the container isnt dirty.
  • Table pagination overlap & transparent shadowing: corrected snapshot area; recomputed coordinates; instant restore + redraw after paging.

⚠️ Breaking

  • If external code accessed Window private members (e.g., dialogs), use getControls() / getdialogs().
  • Pagination math & header inclusion may shift hard-coded offsets in custom renderers.
  • Custom controls that dont restore SetWorkingImage(nullptr) before drawing should be reviewed.

📌 Upgrade notes

  1. Migrate manual cleardevice()+putimage paths to unified full-surface background in WM_PAINT.
  2. For transparent controls, captureBackground() before first draw; restoreBackground() when hiding/overdrawing.
  3. For layout, set container layout.kind = HBox/VBox and child LayoutParams (margin/fixed/weight/align).
  4. Keep a single truncation pass for buttons to avoid duplicate ....
  5. Prefer built-in table pagination.

[v2.0.1] - 2025-10-03

Added

  • New example: 32-bit register viewer tool implemented based on StellarX (supports bit inversion, left shift, right shift, hexadecimal/decimal signed/unsigned toggle, grouped binary display).
    • Example path: examples/register-viewer/
  • TextBox added setText API, allowing external setting of text box content
  • TextBox::setText API modified: immediately calls draw method to redraw after setting text
  • Button added setButtonClick API, allowing external functions to modify button click state and execute corresponding callback functions
  • ==All documents updated with corresponding English versions(.en)==

[v2.0.0] - 2025-09-21

Overview

v2.0.0 is a major release. This release introduces dialog and message box factory (Dialog / MessageBox), with several important fixes and improvements to event distribution, API semantics, and internal resource management.

Some APIs/behaviors have breaking changes that are not backward compatible.

Added

  • Dialog System:

    • Added Dialog class, inheriting from Canvas, for building modal and non-modal dialogs
    • Added MessageBox factory class, providing two convenient APIs: ShowModal (synchronous blocking) and ShowAsync (asynchronous callback)
    • Supports six standard message box types: OK, OKCancel, YesNo, YesNoCancel, RetryCancel, AbortRetryIgnore
    • Automatically handles dialog layout, background saving and restoration, event propagation, and lifecycle management
  • Enhanced Event System:

    • All controls' handleEvent methods now return bool type, indicating whether the event was consumed
    • Introduced event consumption mechanism, supporting finer-grained event propagation control
    • Window class event loop now prioritizes dialog event processing
  • Control State Management:

    • Control base class added dirty flag and setDirty() method, uniformly managing redraw state
    • All controls now implement IsVisible() and model() methods
  • API Enhancements:

    • Button class added setButtonFalseColor() method
    • TextBox class setMaxCharLen() now accepts size_t type parameter
    • Window class added dialog management methods and duplicate detection mechanism

Breaking Changes

  • API Signature Changes:

    • All controls' handleEvent(const ExMessage& msg) method changed from returning void to returning bool
    • Control base class added pure virtual functions IsVisible() const and model() const, all derived classes must implement them
  • Resource Management Changes:

    • Control base class style saving changed from stack objects to heap objects, managed using pointers
    • Enhanced resource release safety, all resources are properly released in destructors
  • Event Handling Logic:

    • Window's runEventLoop() method completely rewritten, now prioritizes dialog events
    • Introduced event consumption mechanism, events do not continue propagating after being consumed

Fixed

  • Memory Management:

    • Fixed memory leak issue in Button::setFillIma()
    • Fixed resource release issues in Control base class destructor
    • Fixed background image resource management issues in Dialog class
  • Layout and Rendering:

    • Fixed pagination calculation, column width, and row height boundary issues in Table component
    • Fixed layout disorder caused by pX coordinate accumulation error in Table
    • Fixed dirty determination issue in Canvas::draw() that prevented child controls from being drawn
    • Fixed asymmetric call issues between TextBox::draw() and restoreStyle()
  • Event Handling:

    • Fixed window event distribution logic to ensure dialog/top-level controls prioritize event processing
    • Fixed delayed state updates when mouse moves out of button area
    • Fixed race conditions in non-modal dialog event handling
  • Other Issues:

    • Fixed potential errors in text measurement and rendering
    • Fixed incomplete background restoration after dialog closure
    • Fixed z-order management issues in multi-dialog scenarios

Changed

  • Code Quality:

    • Refactored numerous internal APIs, enhancing exception safety and maintainability
    • Used smart pointers and modern C++ features to replace raw new/delete
    • Unified code style and naming conventions
  • Performance Optimization:

    • Optimized event processing flow, reducing unnecessary redraws
    • Improved dialog background saving and restoration mechanism
    • Reduced memory allocation and copy operations
  • Documentation and Examples:

    • Added detailed usage examples for all new features
    • Improved code comments and API documentation
    • Updated README and CHANGELOG to reflect latest changes

[1.1.0] - 2025-09-08

Added

  • Window Class API Enhancements:
    • Added complete getter method set, improving class encapsulation and usability
    • getHwnd() - Get window handle for integration with native Windows API
    • getWidth() - Get window width
    • getHeight() - Get window height
    • getHeadline() - Get window title
    • getBkcolor() - Get window background color
    • getBkImage() - Get window background image pointer
    • getControls() - Get reference to control management container, allowing iteration and manipulation of added controls

Improved

  • API Consistency: Provided symmetric setter and getter methods for all important attributes
  • Code Documentation: Further improved class comments, making them clearer and more professional

[1.0.0] - 2025-09-08

Release Summary

First stable release

Added

  • First stable version of StellarX framework
  • Complete control library: Button, Label, TextBox, Canvas, Table, and Window
  • CMake-based build system
  • Detailed documentation and example code
  • Explicit declaration: This framework is specifically designed for Windows platform

Released

  • First release of pre-compiled binary library files, facilitating quick integration without compiling from source
  • Provided release packages include:
    • StellarX-v1.0.0-x64-Windows-msvc-x64.zip
      • Build Environment: Visual Studio 2022 (MSVC v143)
      • Architecture: x64 (64-bit)
      • Runtime Library: /MD
      • Build Modes: Release | Debug
      • Contents: Includes all necessary header files (include/) and static library files (lib/StellarX-Debug.lib StellarX-Release.lib)

Core Features

  • Modular design following SOLID principles
  • Unified control interface (draw() and handleEvent())
  • Support for multiple control shapes and styles
  • Custom event handling callbacks
  • Lightweight design with no external dependencies

[0.1.0] - 2025-08-15

Added

  • Initial project structure and core architecture
  • Control base class and basic event handling system
  • Basic examples and documentation setup