Screen is a terminal renderer written in pure PHP. It powers Solo for Laravel and can be used to build rich text-based user interfaces in any PHP application.
Note
Screen is a library intended to be integrated into PHP applications. It is not a standalone terminal application.
A terminal renderer processes text and ANSI escape sequences to create a virtual representation of terminal output. Unlike a full terminal emulator, Screen focuses specifically on correctly interpreting and rendering text content with formatting rather than handling input, interactive sessions, or process management.
Terminal renderers interpret escape sequences to:
- Track cursor position
- Apply text colors and styles (bold, underline, etc.)
- Manage screen content
- Handle special character sets
- Generate a final rendered output
Screen implements this functionality in pure PHP, allowing developers to build terminal user interfaces without relying on external dependencies or native code.
Screen was originally created to solve a specific problem in Solo for Laravel.
Solo provides a TUI (Text User Interface) that runs multiple processes simultaneously in separate panels, similar to tmux. However, when these processes output ANSI escape codes for cursor movement and screen manipulation, they could potentially "break out" of their visual containers and interfere with other parts of the interface.
To solve this problem, Screen creates a virtual terminal buffer where:
- All ANSI operations (cursor movements, color changes, screen clears) are safely interpreted within an isolated environment
- The final rendered state is captured after all operations are processed
- Only the final visual output is displayed to the user's terminal
This approach provides complete control over how terminal output is rendered, ensuring that complex ANSI operations stay contained within their designated areas. While initially built for Solo, Screen has evolved into a standalone library that can be used in any PHP application requiring terminal rendering.
- Pure PHP Implementation: Only one dependency (Grapheme, another Solo library)
- Comprehensive ANSI Support: Handles cursor positioning, text styling, and screen manipulation
- Unicode/Multibyte Support: Properly handles UTF-8 characters including emojis and wide characters
- Buffer Management: Maintains separate buffers for text content and styling
- Character Width Handling: Correctly calculates display width for CJK and other double-width characters
- Scrolling: Support for vertical scrolling with proper content management
Install via Composer:
composer require soloterm/screen
- PHP 8.1 or higher
- mbstring extension
Here's a simple example of using Screen:
use SoloTerm\Screen\Screen;
// Create a screen with dimensions (columns, rows)
$screen = new Screen(80, 24);
// Write text and ANSI escape sequences
$screen->write("Hello, \e[1;32mWorld!\e[0m");
// Move cursor and add more text
$screen->write("\e[5;10HPositioned text");
// Get the rendered content
echo $screen->output();
Screen operates with several key components:
The main class that coordinates all functionality. It takes care of cursor positioning, content writing, and rendering the final output.
$screen = new Screen(80, 24); // width, height
$screen->write("Text and ANSI codes");
Screen uses multiple buffer types to track content and styling:
- PrintableBuffer: Stores visible characters and handles width calculations
- AnsiBuffer: Tracks styling information (colors, bold, underline, etc.)
Screen correctly interprets ANSI escape sequences for:
- Cursor movement (up, down, left, right, absolute positioning)
- Text styling (colors, bold, italic, underline)
- Screen clearing and line manipulation
- Scrolling
// Move cursor to position (row 5, column 10)
$screen->write("\e[5;10H");
// Move cursor up 3 lines
$screen->write("\e[3A");
// Save and restore cursor position
$screen->write("\e7"); // Save
$screen->write("More text");
$screen->write("\e8"); // Restore
// Bold red text
$screen->write("\e[1;31mImportant message\e[0m");
// Background colors
$screen->write("\e[44mBlue background\e[0m");
// 256-color support
$screen->write("\e[38;5;208mOrange text\e[0m");
// RGB colors
$screen->write("\e[38;2;255;100;0mCustom color\e[0m");
// Clear screen
$screen->write("\e[2J");
// Clear from cursor to end of line
$screen->write("\e[0K");
// Insert lines
$screen->write("\e[2L");
// Scroll up
$screen->write("\e[2S");
You can respond to terminal queries by setting a callback:
$screen->respondToQueriesVia(function($response) {
// Process response (like cursor position)
echo $response;
});
Note
This is still a work in progress. We need some more tests / use cases here.
use SoloTerm\Screen\Screen;
$screen = new Screen(80, 24);
// Draw a border
$screen->write("┌" . str_repeat("─", 78) . "┐\n");
for ($i = 0; $i < 22; $i++) {
$screen->write("│" . str_repeat(" ", 78) . "│\n");
}
$screen->write("└" . str_repeat("─", 78) . "┘");
// Add a title
$screen->write("\e[1;30H\e[1;36mMy Application\e[0m");
// Add some content
$screen->write("\e[5;5HWelcome to the application!");
$screen->write("\e[7;5HPress 'q' to quit.");
// Render
echo $screen->output();
Screen properly handles Unicode characters including emoji and CJK characters that take up multiple columns:
$screen->write("Regular text: Hello");
$screen->write("\nWide characters: 你好世界");
$screen->write("\nEmoji: 🚀 👨👩👧👦 🌍");
Screen includes a comprehensive testing suite that features a unique visual comparison system:
composer test
Screen employs an innovative screenshot-based testing approach (see ComparesVisually
trait) that validates the visual
output:
- The test renders content in a real terminal (iTerm)
- It captures a screenshot of the terminal output
- It runs the same content through the Screen renderer
- It captures a screenshot of the rendered output
- It compares the screenshots pixel-by-pixel to ensure accuracy
This testing strategy ensures that Screen's rendering accurately matches real terminal behavior, especially for complex scenarios involving:
- Multi-byte characters
- Complex ANSI formatting
- Cursor movements
- Scrolling behavior
- Line wrapping
For environments without screenshot capabilities, tests can fall back to fixture-based comparison, making the test suite versatile for CI/CD pipelines.
To enable screenshots for all tests, use the following command:
ENABLE_SCREENSHOT_TESTING=1 composer test
To enable screenshots for only the tests that don't already have fixtures, use the following command:
ENABLE_SCREENSHOT_TESTING=2 composer test
Contributions are welcome! Please feel free to submit a pull request.
The MIT License (MIT).
This is free! If you want to support me:
- Sponsor my open source work: aaronfrancis.com/backstage
- Check out my courses:
- Help spread the word about things I make
Solo Screen was developed by Aaron Francis. If you like it, please let me know!
- Twitter: https://twitter.com/aarondfrancis
- Website: https://aaronfrancis.com
- YouTube: https://youtube.com/@aarondfrancis
- GitHub: https://github.com/aarondfrancis