mirror of
https://github.com/dankamongmen/notcurses.git
synced 2024-11-08 01:10:23 +00:00
242 lines
5.4 KiB
C++
242 lines
5.4 KiB
C++
#ifndef __NCPP_NOTCURSES_HH
|
|
#define __NCPP_NOTCURSES_HH
|
|
|
|
#include <cstdio>
|
|
#include <ctime>
|
|
#include <csignal>
|
|
#include <mutex>
|
|
|
|
#include <notcurses/notcurses.h>
|
|
|
|
#include "CellStyle.hh"
|
|
#include "NCKey.hh"
|
|
#include "NCLogLevel.hh"
|
|
#include "Palette256.hh"
|
|
#include "Plane.hh"
|
|
#include "Root.hh"
|
|
#include "_helpers.hh"
|
|
|
|
namespace ncpp
|
|
{
|
|
class NCPP_API_EXPORT NotCurses : public Root
|
|
{
|
|
public:
|
|
static notcurses_options default_notcurses_options;
|
|
|
|
public:
|
|
explicit NotCurses (FILE *fp = nullptr)
|
|
: NotCurses (default_notcurses_options, fp)
|
|
{}
|
|
|
|
explicit NotCurses (const notcurses_options &nc_opts, FILE *fp = nullptr);
|
|
|
|
// Must not move or copy a NotCurses instance because we have no way to guarantee validity of any other copy
|
|
// when even a single instance is destructed as that operation would close notcurses.
|
|
NotCurses (const NotCurses &other) = delete;
|
|
NotCurses (NotCurses &&other) = delete;
|
|
~NotCurses ();
|
|
|
|
notcurses* operator*() noexcept
|
|
{
|
|
return nc;
|
|
}
|
|
|
|
static NotCurses& get_instance ()
|
|
{
|
|
if (_instance == nullptr)
|
|
throw invalid_state_error ("NotCurses instance not found.");
|
|
if (_instance->nc == nullptr)
|
|
throw invalid_state_error (ncpp_invalid_state_message);
|
|
|
|
return *_instance;
|
|
}
|
|
|
|
static bool is_notcurses_stopped ()
|
|
{
|
|
return _instance == nullptr || _instance->nc == nullptr;
|
|
}
|
|
|
|
static const char* ncmetric (uintmax_t val, unsigned decimal, char *buf, int omitdec, unsigned mult, int uprefix) noexcept
|
|
{
|
|
return ::ncmetric (val, decimal, buf, omitdec, mult, uprefix);
|
|
}
|
|
|
|
static const char* qprefix (uintmax_t val, unsigned decimal, char *buf, int omitdec) noexcept
|
|
{
|
|
return ::qprefix (val, decimal, buf, omitdec);
|
|
}
|
|
|
|
static const char* bprefix (uintmax_t val, unsigned decimal, char *buf, int omitdec) noexcept
|
|
{
|
|
return ::bprefix (val, decimal, buf, omitdec);
|
|
}
|
|
|
|
static const char* version () noexcept
|
|
{
|
|
return notcurses_version ();
|
|
}
|
|
|
|
// This is potentially dangerous, but alas necessary. It can cause other calls here to fail in a bad way, but we
|
|
// need a way to report errors to std{out,err} in case of failure and that will work only if notcurses is
|
|
// stopped, so...
|
|
//
|
|
bool stop ()
|
|
{
|
|
if (nc == nullptr)
|
|
throw invalid_state_error (ncpp_invalid_state_message);
|
|
|
|
bool ret = !notcurses_stop (nc);
|
|
nc = nullptr;
|
|
|
|
return ret;
|
|
}
|
|
|
|
bool can_fade () const noexcept
|
|
{
|
|
return notcurses_canfade (nc);
|
|
}
|
|
|
|
bool can_open () const noexcept
|
|
{
|
|
return notcurses_canopen (nc);
|
|
}
|
|
|
|
bool can_change_color () const noexcept
|
|
{
|
|
return notcurses_canchangecolor (nc);
|
|
}
|
|
|
|
void get_stats (ncstats *stats) const noexcept
|
|
{
|
|
if (stats == nullptr)
|
|
return;
|
|
|
|
notcurses_stats (nc, stats);
|
|
}
|
|
|
|
void reset_stats (ncstats *stats) const
|
|
{
|
|
if (stats == nullptr)
|
|
throw invalid_argument ("'stats' must be a valid pointer");
|
|
|
|
notcurses_reset_stats (nc, stats);
|
|
}
|
|
|
|
bool use (const Palette256 *p) const
|
|
{
|
|
if (p == nullptr)
|
|
throw invalid_argument ("'p' must be a valid pointer");
|
|
|
|
return use (*p);
|
|
}
|
|
|
|
bool use (const Palette256 &p) const NOEXCEPT_MAYBE
|
|
{
|
|
return error_guard (palette256_use (nc, reinterpret_cast<const palette256*>(&p)), -1);
|
|
}
|
|
|
|
bool render () const NOEXCEPT_MAYBE
|
|
{
|
|
return error_guard (notcurses_render (nc), -1);
|
|
}
|
|
|
|
void get_term_dim (int *rows, int *cols) const noexcept
|
|
{
|
|
notcurses_term_dim_yx (nc, rows, cols);
|
|
}
|
|
|
|
void get_term_dim (int &rows, int &cols) const noexcept
|
|
{
|
|
get_term_dim (&rows, &cols);
|
|
}
|
|
|
|
bool refresh (int* rows, int* cols) const NOEXCEPT_MAYBE
|
|
{
|
|
return error_guard (notcurses_refresh (nc, rows, cols), -1);
|
|
}
|
|
|
|
bool refresh (int& rows, int& cols) const NOEXCEPT_MAYBE
|
|
{
|
|
return refresh (&rows, &cols);
|
|
}
|
|
|
|
int get_palette_size () const noexcept
|
|
{
|
|
return notcurses_palette_size (static_cast<const notcurses*> (nc));
|
|
}
|
|
|
|
bool mouse_enable () const NOEXCEPT_MAYBE
|
|
{
|
|
return error_guard (notcurses_mouse_enable (nc), -1);
|
|
}
|
|
|
|
bool mouse_disable () const NOEXCEPT_MAYBE
|
|
{
|
|
return error_guard (notcurses_mouse_disable (nc), -1);
|
|
}
|
|
|
|
CellStyle get_supported_styles () const noexcept
|
|
{
|
|
return static_cast<CellStyle>(notcurses_supported_styles (nc));
|
|
}
|
|
|
|
char32_t getc (const timespec *ts, sigset_t *sigmask = nullptr, ncinput *ni = nullptr) const noexcept
|
|
{
|
|
return notcurses_getc (nc, ts, sigmask, ni);
|
|
}
|
|
|
|
char32_t getc (bool blocking = false, ncinput *ni = nullptr) const noexcept
|
|
{
|
|
if (blocking)
|
|
return notcurses_getc_blocking (nc, ni);
|
|
|
|
return notcurses_getc_nblock (nc, ni);
|
|
}
|
|
|
|
char* get_at (int yoff, int xoff, uint32_t* attr, uint64_t* channels) const noexcept
|
|
{
|
|
return notcurses_at_yx (nc, yoff, xoff, attr, channels);
|
|
}
|
|
|
|
int get_inputready_fd () const noexcept
|
|
{
|
|
return notcurses_inputready_fd (nc);
|
|
}
|
|
|
|
void drop_planes () const noexcept
|
|
{
|
|
notcurses_drop_planes (nc);
|
|
}
|
|
|
|
Plane* get_stdplane () noexcept
|
|
{
|
|
return new Plane (notcurses_stdplane (nc), true);
|
|
}
|
|
|
|
Plane* get_stdplane (int *y, int *x)
|
|
{
|
|
if (y == nullptr)
|
|
throw invalid_argument ("'y' must be a valid pointer");
|
|
if (x == nullptr)
|
|
throw invalid_argument ("'x' must be a valid pointer");
|
|
|
|
return get_stdplane (*y, *x);
|
|
}
|
|
|
|
Plane* get_stdplane (int &y, int &x) noexcept
|
|
{
|
|
return new Plane (notcurses_stddim_yx (nc, &y, &x));
|
|
}
|
|
|
|
Plane* get_top () noexcept;
|
|
|
|
private:
|
|
notcurses *nc;
|
|
|
|
static NotCurses *_instance;
|
|
static std::mutex init_mutex;
|
|
};
|
|
}
|
|
|
|
#endif
|