ObjFW  Artifact [abfd187834]

Artifact abfd1878348bad0d894a4d2f6422d60592cac608ba9b6bdc7e2cf4f92230b067:

  • File src/OFStdIOStream.h — part of check-in [0586655c9d] at 2025-03-27 20:17:12 on branch trunk — Add -[OFStdIOStream isCursorVisible] (user: js size: 7868) [more...]

/*
 * Copyright (c) 2008-2025 Jonathan Schleifer <js@nil.im>
 *
 * All rights reserved.
 *
 * This program is free software: you can redistribute it and/or modify it
 * under the terms of the GNU Lesser General Public License version 3.0 only,
 * as published by the Free Software Foundation.
 *
 * This program is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License
 * version 3.0 for more details.
 *
 * You should have received a copy of the GNU Lesser General Public License
 * version 3.0 along with this program. If not, see
 * <https://www.gnu.org/licenses/>.
 */

#import "OFStream.h"
#import "OFKernelEventObserver.h"

#ifdef OF_AMIGAOS
# include <dos/dos.h>
#endif

OF_ASSUME_NONNULL_BEGIN

/** @file */

@class OFColor;

/**
 * @class OFStdIOStream OFStdIOStream.h ObjFW/ObjFW.h
 *
 * @brief A class for providing standard input, output and error as OFStream.
 *
 * The global variables @ref OFStdIn, @ref OFStdOut and @ref OFStdErr are
 * instances of this class and need no initialization.
 */
@interface OFStdIOStream: OFStream
#if !defined(OF_WINDOWS) && !defined(OF_AMIGAOS) && !defined(OF_WII_U)
    <OFReadyForReadingObserving, OFReadyForWritingObserving>
#endif
{
#if defined(OF_AMIGAOS)
	BPTR _handle;
	bool _closable;
#elif !defined(OF_WII_U)
	int _fd;
#endif
	bool _atEndOfStream;
	int _colors;
	OFColor *_Nullable _foregroundColor, *_Nullable _backgroundColor;
	bool _bold, _italic, _underlined, _blinking;
	uintptr_t _cursorVisible;  /* Change type on ABI bump */
	OF_RESERVE_IVARS(OFStdIOStream, 3)
}

/**
 * @brief Whether there is an underlying terminal.
 */
@property (readonly, nonatomic) bool hasTerminal;

/**
 * @brief The number of columns, or -1 if there is no underlying terminal or
 *	  the number of columns could not be queried.
 */
@property (readonly, nonatomic) int columns;

/**
 * @brief The number of rows, or -1 if there is no underlying terminal or the
 *	  number of rows could not be queried.
 */
@property (readonly, nonatomic) int rows;

/**
 * @brief The number of colors supported by the underyling terminal or -1 if
 *	  there is no underlying terminal.
 */
@property (readonly, nonatomic) int colors;

/**
 * @brief The current foreground color on the underlying terminal.
 *
 * Setting this does nothing if there is no underlying terminal or colors are
 * unsupported.
 *
 * If the specified color is @ref OFColor#black, @ref OFColor#silver,
 * @ref OFColor#gray, @ref OFColor#white, @ref OFColor#maroon, @ref OFColor#red,
 * @ref OFColor#purple, @ref OFColor#fuchsia, @ref OFColor#green,
 * @ref OFColor#lime, @ref OFColor#olive, @ref OFColor#yellow,
 * @ref OFColor#navy, @ref OFColor#blue, @ref OFColor#teal or @ref OFColor#aqua,
 * one of the 16 terminal colors will be used which doesn't necessarily match
 * the RGB value of the color. If you want an exact color, create a new
 * @ref OFColor with the RGB value you want. In that case, it will be
 * represented exactly on a truecolor terminal or by the closest color on a 256
 * color terminal.
 */
@property OF_NULLABLE_PROPERTY (retain, nonatomic) OFColor *foregroundColor;

/**
 * @brief The current background color on the underlying terminal.
 *
 * Setting this does nothing if there is no underlying terminal or colors are
 * unsupported.
 *
 * If the specified color is @ref OFColor#black, @ref OFColor#silver,
 * @ref OFColor#gray, @ref OFColor#white, @ref OFColor#maroon, @ref OFColor#red,
 * @ref OFColor#purple, @ref OFColor#fuchsia, @ref OFColor#green,
 * @ref OFColor#lime, @ref OFColor#olive, @ref OFColor#yellow,
 * @ref OFColor#navy, @ref OFColor#blue, @ref OFColor#teal or @ref OFColor#aqua,
 * one of the 16 terminal colors will be used which doesn't necessarily match
 * the RGB value of the color. If you want an exact color, create a new
 * @ref OFColor with the RGB value you want. In that case, it will be
 * represented exactly on a truecolor terminal or by the closest color on a 256
 * color terminal.
 */
@property OF_NULLABLE_PROPERTY (retain, nonatomic) OFColor *backgroundColor;

/**
 * @brief Whether bold is on on the underlying terminal.
 *
 * Setting this does nothing if there is no underlying terminal or bold is
 * unsupported.
 */
@property (nonatomic, getter=isBold) bool bold;

/**
 * @brief Whether italic is on on the underlying terminal.
 *
 * Setting this does nothing if there is no underlying terminal or italic is
 * unsupported.
 */
@property (nonatomic, getter=isItalic) bool italic;

/**
 * @brief Whether underlined is on on the underlying terminal.
 *
 * Setting this does nothing if there is no underlying terminal or underlined is
 * unsupported.
 */
@property (nonatomic, getter=isUnderlined) bool underlined;

/**
 * @brief Whether blinking is on on the underlying terminal.
 *
 * Setting this does nothing if there is no underlying terminal or blinking is
 * unsupported.
 */
@property (nonatomic, getter=isBlinking) bool blinking;

/**
 * @brief Whether the cursor of the underlying terminal is visible.
 *
 * Setting this does nothing if there is no underlying terminal or hiding /
 * showing the cursor is unsupported.
 */
@property (nonatomic, getter=isCursorVisible) bool cursorVisible;

#if defined(OF_WII) || defined(OF_NINTENDO_DS) || defined(OF_NINTENDO_3DS) || \
    defined(DOXYGEN)
/**
 * @brief Sets up a console for @ref OFStdOut / @ref OFStdErr output on systems
 *	  that don't have a console by default.
 *
 * @note This method is only available on Wii, Nintendo DS and Nintendo 3DS.
 */
+ (void)setUpConsole;
#endif

- (instancetype)init OF_UNAVAILABLE;

/**
 * @brief Resets all attributes (color, bold, etc.). Does nothing if there is
 *	  no underlying terminal.
 */
- (void)reset;

/**
 * @brief Clears the entire underlying terminal. Does nothing if there is no
 *	  underlying terminal.
 */
- (void)clear;

/**
 * @brief Erases the entire current line on the underlying terminal. Does
 *	  nothing if there is no underlying terminal.
 */
- (void)eraseLine;

/**
 * @brief Moves the cursor to the specified column in the current row. Does
 *	  nothing if there is no underlying terminal.
 *
 * @param column The column in the current row to move the cursor to
 */
- (void)setCursorColumn: (unsigned int)column;

/**
 * @brief Moves the cursor to the specified absolute position. Does nothing if
 *	  there is no underlying terminal.
 *
 * @param position The position to move the cursor to
 */
- (void)setCursorPosition: (OFPoint)position;

/**
 * @brief Moves the cursor to the specified relative position. Does nothing if
 *	  there is no underlying terminal.
 *
 * @param position The position to move the cursor to
 */
- (void)setRelativeCursorPosition: (OFPoint)position;
@end

#ifdef __cplusplus
extern "C" {
#endif
/** @file */

/**
 * @brief The standard input as an OFStream.
 */
extern OFStdIOStream *_Nullable OFStdIn;

/**
 * @brief The standard output as an OFStream.
 */
extern OFStdIOStream *_Nullable OFStdOut;

/**
 * @brief The standard error as an OFStream.
 */
extern OFStdIOStream *_Nullable OFStdErr;

/**
 * @brief Logs the specified printf-style format to @ref OFStdErr.
 *
 * This prefixes the output with the date, timestamp, process name and PID.
 *
 * @param format The format for the line to log. See @ref OFStream#writeFormat:.
 */
extern void OFLog(OFConstantString *format, ...);

/**
 * @brief Logs the specified printf-style format to @ref OFStdErr.
 *
 * This prefixes the output with the date, timestamp, process name and PID.
 *
 * @param format The format for the line to log. See @ref OFStream#writeFormat:.
 * @param arguments The arguments for the format
 */
extern void OFLogV(OFConstantString *format, va_list arguments);
#ifdef __cplusplus
}
#endif

OF_ASSUME_NONNULL_END