diff --git a/meson.build b/meson.build
index 3efced1..0d63a62 100644
--- a/meson.build
+++ b/meson.build
@@ -16,6 +16,7 @@ project(
gtkmm = dependency('gtkmm-3.0')
wayland_client = dependency('wayland-client')
wayland_protos = dependency('wayland-protocols')
+gtkls = dependency('gtk-layer-shell-0')
add_project_link_arguments(['-rdynamic'], language:'cpp')
add_project_arguments(['-Wno-unused-parameter'], language: 'cpp')
diff --git a/proto/meson.build b/proto/meson.build
index a05e26e..eb07910 100644
--- a/proto/meson.build
+++ b/proto/meson.build
@@ -15,8 +15,8 @@ wayland_scanner_client = generator(
)
client_protocols = [
- 'wayfire-shell.xml',
- 'virtual-keyboard-unstable-v1.xml'
+ ['wayfire-shell-unstable-v2.xml'],
+ ['virtual-keyboard-unstable-v1.xml']
]
wl_protos_src = []
diff --git a/proto/wayfire-shell-unstable-v2.xml b/proto/wayfire-shell-unstable-v2.xml
new file mode 100644
index 0000000..7c7b087
--- /dev/null
+++ b/proto/wayfire-shell-unstable-v2.xml
@@ -0,0 +1,120 @@
+
+
+
+ This protocol provides additional events and requests for special DE
+ clients like panels, docks, etc.
+
+ It is meant as an addition for protocols like wlr-layer-shell.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Represents a single output.
+ Each output is managed independently from the others.
+
+
+
+
+ Emitted when a window gets fullscreened on the given output. In this
+ mode, windows in the TOP layer are not visible.
+
+ There will be no two consecutive enter_fullscreen calls, i.e. if
+ fullscreen mode is entered it will be exited before going into this mode
+ again.
+
+
+
+
+
+ Emitted when the output is no longer in fullscreen mode. Each
+ leave_fullscreen has a corresponding enter_fullscreen before it.
+
+
+
+
+
+ Request the compositor to not render the output, so the output usually
+ is cleared to black color. To enable output rendering again, call
+ inhibit_output_done.
+
+
+
+
+
+ Stop inhibiting the output. This must be called as many times as
+ inhibit_output was called to actually uninhibit rendering.
+
+ The inhibit/inhibit_done requests can be called multiple times, even
+ from different apps, so don't assume that a call to inhibit_done would
+ always mean actually starting the rendering process.
+
+
+
+
+
+
+
+
+
+
+
+
+ A hotspot on the output is an edge or a corner region of the
+ output where the mouse or touch point has been residing for a given
+ amount of time.
+
+ The hotspot can be used for example for autohiding panels, where the
+ panel is shown when the input hovers on the edge of the output for a
+ specific amount of time.
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ Means that the mouse and/or touch finger was inside the indicated
+ hotspot for the given amount of time.
+
+ Emitted at most once for each entry of the input inside the hotspot.
+
+
+
+
+
+ This event indicates that the mouse or touch point has left the hotspot
+ area.
+
+ Emitted only once after each enter.
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/proto/wayfire-shell.xml b/proto/wayfire-shell.xml
deleted file mode 100644
index 8abb7e6..0000000
--- a/proto/wayfire-shell.xml
+++ /dev/null
@@ -1,172 +0,0 @@
-
-
-
- IMPORTANT: most of wayfire-shell is going to be deprecated. Try to
- use layer-shell instead.
-
- The purpose of this protocol is to enable the creation of different
- desktop-interface windows like panels, backgrounds, docks,
- lockscreens, etc. It also aims to allow the creation of full-blown
- DEs using Wayfire.
-
- Note that in contrast to some other efforts to create a similar
- protocol, such as wlr-layer-shell, this isn't a new "shell" for
- giving a role to wl_surfaces. This protocol can be used with any
- type of toplevel surface (xdg_toplevel, xdg_toplevel_v6, etc.)
- to give them to the corresponding WM role.
-
-
-
-
-
-
-
-
-
-
- Assign the given role to the given surface and add it to the
- given output. A client can specify a null output, in which case
- the compositor will assign the surface to the focused output,
- if any such output.
-
- The role cannot be changed later, and neither can the surface be
- moved to a different output, except by the compositor.
-
-
-
-
-
-
-
-
-
-
-
- Represents a single output.
- Each output is managed independently from the others.
-
-
-
- Panels are always rendered on top, even above fullscreen windows.
- If autohide is 1, the event indicates that the panels should hide
- itself, by for example unmapping or sliding outside of the output.
- If autohide is 0, this means that the reason for the last request
- with autohide == 1 is no longer valid, i.e the panels can show
- themselves.
-
- The output_hide_panels can be called multiple times with
- autohide = 1, and the panel should show itself only when
- it has received a matching number of events with autohide = 0
-
-
-
-
-
-
-
- Request the compositor to not render the output, so
- the output usually is cleared to black color.
- To enable output rendering again, call inhibit_output_done
-
-
-
-
-
- Stop inhibiting the output. This must be called as many times
- as inhibit_output was called to actually uninhibit rendering.
-
- The inhibit/inhibit_done requests can be called multiple times,
- even from different apps, so don't assume that a call to
- inhibit_done would always mean actually starting the rendering process.
-
-
-
-
-
-
- Represents a surface with a specific WM role.
- It belongs to the output which it was created for.
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- Sets the position on the screen where the compositor should
- position the view. Can be reset by specifying anchor 0. If not
- set, the compositor will assume manual positioning via the
- configure request.
-
- If one anchor edge is provided, the wm surface is "stuck" to
- that edge.
- If two anchor edges are provided, the wm surface is considered
- anchored to the corner of the screen between them.
-
- Any other anchor edge configuration is considered invalid.
-
-
-
-
-
-
- Set the offset from the anchored edges to the wm surface. This
- is an alternative to the configure request. Using both will
- result in undefined results.
-
- Margin has effect only for edges the wm surface is anchored to.
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
-
- Sets how the wm surface will interact with keyboard focus.
- Setting no_focus means that the surface will never receive
- keyboard focus, click_to_focus means normal focus semantics (i.e
- what you expect from "normal" windows), and exclusive focus means
- that no other window can get keyboard focus.
-
-
-
-
-
-
- Request the compositor to reserve the given amount of pixels
- for the wm surface(like STRUTS in X11). This has effect only
- if the surface is anchored to a single edge. Margin doesn't
- affect exclusive zone in any way.
-
-
-
-
-
diff --git a/src/main.cpp b/src/main.cpp
index 0adef44..c62a1b7 100644
--- a/src/main.cpp
+++ b/src/main.cpp
@@ -3,6 +3,8 @@
#include
#include
+#include "util/clara.hpp"
+
#define ABC_TOGGLE 0x12345678
#define NUM_TOGGLE 0x87654321
@@ -14,11 +16,10 @@ namespace wf
{
namespace osk
{
- int spacing = 8;
- int default_x = 100;
- int default_y = 100;
+ int spacing = OSK_SPACING;
int default_width = 800;
int default_height = 400;
+ std::string anchor;
KeyButton::KeyButton(Key key, int width, int height)
{
@@ -111,18 +112,14 @@ namespace wf
void Keyboard::set_layout(KeyboardLayout *new_layout)
{
- if (this->current_layout)
- this->window->remove();
-
this->current_layout = new_layout;
- this->window->add(new_layout->box);
- this->window->show_all();
+ window->set_widget(new_layout->box);
}
Keyboard::Keyboard()
{
window = std::make_unique
- (default_x, default_y, default_width, default_height);
+ (default_width, default_height, anchor);
vk = std::make_unique ();
init_layouts();
@@ -175,33 +172,25 @@ namespace wf
int main(int argc, char **argv)
{
- struct option opts[] = {
- { "geometry", required_argument, NULL, 'g' },
- { 0, 0, NULL, 0 }
- };
+ bool show_help = false;
- int c, i;
- while((c = getopt_long(argc, argv, "g:", opts, &i)) != -1)
- {
- using namespace wf::osk;
- switch(c)
- {
- case 'g':
- if (sscanf(optarg, "%d,%d %dx%d", &default_x, &default_y,
- &default_width, &default_height) != 4)
- {
- std::cerr << "Invalid geometry: " << optarg << std::endl;
- std::exit(-1);
- } else
- {
- std::cout << "Geometry " << default_x << "," << default_y << " "
- << default_width << "x" << default_height;
- }
+ auto cli = clara::detail::Help(show_help) |
+ clara::detail::Opt(wf::osk::default_width, "int")["-w"]["--width"]
+ ("keyboard width") |
+ clara::detail::Opt(wf::osk::default_height, "int")["-h"]["--height"]
+ ("keyboard height") |
+ clara::detail::Opt(wf::osk::anchor, "top|left|bottom|right")["-a"]
+ ["--anchor"]("where the keyboard should anchor in the screen");
- break;
- default:
- std::cerr << "Unrecognized argument " << char(c) << std::endl;
- }
+ auto res = cli.parse(clara::detail::Args(argc, argv));
+ if (!res) {
+ std::cerr << "Error: " << res.errorMessage() << std::endl;
+ return 1;
+ }
+
+ if (show_help) {
+ std::cout << cli << std::endl;
+ return 0;
}
auto app = Gtk::Application::create();
diff --git a/src/meson.build b/src/meson.build
index be9209f..ebdbd3f 100644
--- a/src/meson.build
+++ b/src/meson.build
@@ -1,3 +1,3 @@
executable('wf-osk', ['main.cpp', 'wayland-window.cpp', 'virtual-keyboard.cpp', 'shared/os-compatibility.c'],
- dependencies: [gtkmm, wf_protos],
+ dependencies: [gtkmm, wf_protos, gtkls],
install: true)
diff --git a/src/util/clara.hpp b/src/util/clara.hpp
new file mode 100644
index 0000000..7e32288
--- /dev/null
+++ b/src/util/clara.hpp
@@ -0,0 +1,1264 @@
+// Copyright 2017 Two Blue Cubes Ltd. All rights reserved.
+//
+// Distributed under the Boost Software License, Version 1.0. (See accompanying
+// file LICENSE_1_0.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
+//
+// See https://github.com/philsquared/Clara for more details
+
+// Clara v1.1.5
+
+#ifndef CLARA_HPP_INCLUDED
+#define CLARA_HPP_INCLUDED
+
+#ifndef CLARA_CONFIG_CONSOLE_WIDTH
+#define CLARA_CONFIG_CONSOLE_WIDTH 80
+#endif
+
+#ifndef CLARA_TEXTFLOW_CONFIG_CONSOLE_WIDTH
+#define CLARA_TEXTFLOW_CONFIG_CONSOLE_WIDTH CLARA_CONFIG_CONSOLE_WIDTH
+#endif
+
+#ifndef CLARA_CONFIG_OPTIONAL_TYPE
+#ifdef __has_include
+#if __has_include() && __cplusplus >= 201703L
+#include
+#define CLARA_CONFIG_OPTIONAL_TYPE std::optional
+#endif
+#endif
+#endif
+
+
+// ----------- #included from clara_textflow.hpp -----------
+
+// TextFlowCpp
+//
+// A single-header library for wrapping and laying out basic text, by Phil Nash
+//
+// Distributed under the Boost Software License, Version 1.0. (See accompanying
+// file LICENSE.txt or copy at http://www.boost.org/LICENSE_1_0.txt)
+//
+// This project is hosted at https://github.com/philsquared/textflowcpp
+
+#ifndef CLARA_TEXTFLOW_HPP_INCLUDED
+#define CLARA_TEXTFLOW_HPP_INCLUDED
+
+#include
+#include
+#include
+#include
+
+#ifndef CLARA_TEXTFLOW_CONFIG_CONSOLE_WIDTH
+#define CLARA_TEXTFLOW_CONFIG_CONSOLE_WIDTH 80
+#endif
+
+
+namespace clara { namespace TextFlow {
+
+ inline auto isWhitespace( char c ) -> bool {
+ static std::string chars = " \t\n\r";
+ return chars.find( c ) != std::string::npos;
+ }
+ inline auto isBreakableBefore( char c ) -> bool {
+ static std::string chars = "[({<|";
+ return chars.find( c ) != std::string::npos;
+ }
+ inline auto isBreakableAfter( char c ) -> bool {
+ static std::string chars = "])}>.,:;*+-=&/\\";
+ return chars.find( c ) != std::string::npos;
+ }
+
+ class Columns;
+
+ class Column {
+ std::vector m_strings;
+ size_t m_width = CLARA_TEXTFLOW_CONFIG_CONSOLE_WIDTH;
+ size_t m_indent = 0;
+ size_t m_initialIndent = std::string::npos;
+
+ public:
+ class iterator {
+ friend Column;
+
+ Column const& m_column;
+ size_t m_stringIndex = 0;
+ size_t m_pos = 0;
+
+ size_t m_len = 0;
+ size_t m_end = 0;
+ bool m_suffix = false;
+
+ iterator( Column const& column, size_t stringIndex )
+ : m_column( column ),
+ m_stringIndex( stringIndex )
+ {}
+
+ auto line() const -> std::string const& { return m_column.m_strings[m_stringIndex]; }
+
+ auto isBoundary( size_t at ) const -> bool {
+ assert( at > 0 );
+ assert( at <= line().size() );
+
+ return at == line().size() ||
+ ( isWhitespace( line()[at] ) && !isWhitespace( line()[at-1] ) ) ||
+ isBreakableBefore( line()[at] ) ||
+ isBreakableAfter( line()[at-1] );
+ }
+
+ void calcLength() {
+ assert( m_stringIndex < m_column.m_strings.size() );
+
+ m_suffix = false;
+ auto width = m_column.m_width-indent();
+ m_end = m_pos;
+ while( m_end < line().size() && line()[m_end] != '\n' )
+ ++m_end;
+
+ if( m_end < m_pos + width ) {
+ m_len = m_end - m_pos;
+ }
+ else {
+ size_t len = width;
+ while (len > 0 && !isBoundary(m_pos + len))
+ --len;
+ while (len > 0 && isWhitespace( line()[m_pos + len - 1] ))
+ --len;
+
+ if (len > 0) {
+ m_len = len;
+ } else {
+ m_suffix = true;
+ m_len = width - 1;
+ }
+ }
+ }
+
+ auto indent() const -> size_t {
+ auto initial = m_pos == 0 && m_stringIndex == 0 ? m_column.m_initialIndent : std::string::npos;
+ return initial == std::string::npos ? m_column.m_indent : initial;
+ }
+
+ auto addIndentAndSuffix(std::string const &plain) const -> std::string {
+ return std::string( indent(), ' ' ) + (m_suffix ? plain + "-" : plain);
+ }
+
+ public:
+ using difference_type = std::ptrdiff_t;
+ using value_type = std::string;
+ using pointer = value_type*;
+ using reference = value_type&;
+ using iterator_category = std::forward_iterator_tag;
+
+ explicit iterator( Column const& column ) : m_column( column ) {
+ assert( m_column.m_width > m_column.m_indent );
+ assert( m_column.m_initialIndent == std::string::npos || m_column.m_width > m_column.m_initialIndent );
+ calcLength();
+ if( m_len == 0 )
+ m_stringIndex++; // Empty string
+ }
+
+ auto operator *() const -> std::string {
+ assert( m_stringIndex < m_column.m_strings.size() );
+ assert( m_pos <= m_end );
+ return addIndentAndSuffix(line().substr(m_pos, m_len));
+ }
+
+ auto operator ++() -> iterator& {
+ m_pos += m_len;
+ if( m_pos < line().size() && line()[m_pos] == '\n' )
+ m_pos += 1;
+ else
+ while( m_pos < line().size() && isWhitespace( line()[m_pos] ) )
+ ++m_pos;
+
+ if( m_pos == line().size() ) {
+ m_pos = 0;
+ ++m_stringIndex;
+ }
+ if( m_stringIndex < m_column.m_strings.size() )
+ calcLength();
+ return *this;
+ }
+ auto operator ++(int) -> iterator {
+ iterator prev( *this );
+ operator++();
+ return prev;
+ }
+
+ auto operator ==( iterator const& other ) const -> bool {
+ return
+ m_pos == other.m_pos &&
+ m_stringIndex == other.m_stringIndex &&
+ &m_column == &other.m_column;
+ }
+ auto operator !=( iterator const& other ) const -> bool {
+ return !operator==( other );
+ }
+ };
+ using const_iterator = iterator;
+
+ explicit Column( std::string const& text ) { m_strings.push_back( text ); }
+
+ auto width( size_t newWidth ) -> Column& {
+ assert( newWidth > 0 );
+ m_width = newWidth;
+ return *this;
+ }
+ auto indent( size_t newIndent ) -> Column& {
+ m_indent = newIndent;
+ return *this;
+ }
+ auto initialIndent( size_t newIndent ) -> Column& {
+ m_initialIndent = newIndent;
+ return *this;
+ }
+
+ auto width() const -> size_t { return m_width; }
+ auto begin() const -> iterator { return iterator( *this ); }
+ auto end() const -> iterator { return { *this, m_strings.size() }; }
+
+ inline friend std::ostream& operator << ( std::ostream& os, Column const& col ) {
+ bool first = true;
+ for( auto line : col ) {
+ if( first )
+ first = false;
+ else
+ os << "\n";
+ os << line;
+ }
+ return os;
+ }
+
+ auto operator + ( Column const& other ) -> Columns;
+
+ auto toString() const -> std::string {
+ std::ostringstream oss;
+ oss << *this;
+ return oss.str();
+ }
+ };
+
+ class Spacer : public Column {
+
+ public:
+ explicit Spacer( size_t spaceWidth ) : Column( "" ) {
+ width( spaceWidth );
+ }
+ };
+
+ class Columns {
+ std::vector m_columns;
+
+ public:
+
+ class iterator {
+ friend Columns;
+ struct EndTag {};
+
+ std::vector const& m_columns;
+ std::vector m_iterators;
+ size_t m_activeIterators;
+
+ iterator( Columns const& columns, EndTag )
+ : m_columns( columns.m_columns ),
+ m_activeIterators( 0 )
+ {
+ m_iterators.reserve( m_columns.size() );
+
+ for( auto const& col : m_columns )
+ m_iterators.push_back( col.end() );
+ }
+
+ public:
+ using difference_type = std::ptrdiff_t;
+ using value_type = std::string;
+ using pointer = value_type*;
+ using reference = value_type&;
+ using iterator_category = std::forward_iterator_tag;
+
+ explicit iterator( Columns const& columns )
+ : m_columns( columns.m_columns ),
+ m_activeIterators( m_columns.size() )
+ {
+ m_iterators.reserve( m_columns.size() );
+
+ for( auto const& col : m_columns )
+ m_iterators.push_back( col.begin() );
+ }
+
+ auto operator ==( iterator const& other ) const -> bool {
+ return m_iterators == other.m_iterators;
+ }
+ auto operator !=( iterator const& other ) const -> bool {
+ return m_iterators != other.m_iterators;
+ }
+ auto operator *() const -> std::string {
+ std::string row, padding;
+
+ for( size_t i = 0; i < m_columns.size(); ++i ) {
+ auto width = m_columns[i].width();
+ if( m_iterators[i] != m_columns[i].end() ) {
+ std::string col = *m_iterators[i];
+ row += padding + col;
+ if( col.size() < width )
+ padding = std::string( width - col.size(), ' ' );
+ else
+ padding = "";
+ }
+ else {
+ padding += std::string( width, ' ' );
+ }
+ }
+ return row;
+ }
+ auto operator ++() -> iterator& {
+ for( size_t i = 0; i < m_columns.size(); ++i ) {
+ if (m_iterators[i] != m_columns[i].end())
+ ++m_iterators[i];
+ }
+ return *this;
+ }
+ auto operator ++(int) -> iterator {
+ iterator prev( *this );
+ operator++();
+ return prev;
+ }
+ };
+ using const_iterator = iterator;
+
+ auto begin() const -> iterator { return iterator( *this ); }
+ auto end() const -> iterator { return { *this, iterator::EndTag() }; }
+
+ auto operator += ( Column const& col ) -> Columns& {
+ m_columns.push_back( col );
+ return *this;
+ }
+ auto operator + ( Column const& col ) -> Columns {
+ Columns combined = *this;
+ combined += col;
+ return combined;
+ }
+
+ inline friend std::ostream& operator << ( std::ostream& os, Columns const& cols ) {
+
+ bool first = true;
+ for( auto line : cols ) {
+ if( first )
+ first = false;
+ else
+ os << "\n";
+ os << line;
+ }
+ return os;
+ }
+
+ auto toString() const -> std::string {
+ std::ostringstream oss;
+ oss << *this;
+ return oss.str();
+ }
+ };
+
+ inline auto Column::operator + ( Column const& other ) -> Columns {
+ Columns cols;
+ cols += *this;
+ cols += other;
+ return cols;
+ }
+}}
+
+#endif // CLARA_TEXTFLOW_HPP_INCLUDED
+
+// ----------- end of #include from clara_textflow.hpp -----------
+// ........... back in clara.hpp
+
+
+#include
+#include
+#include
+
+#if !defined(CLARA_PLATFORM_WINDOWS) && ( defined(WIN32) || defined(__WIN32__) || defined(_WIN32) || defined(_MSC_VER) )
+#define CLARA_PLATFORM_WINDOWS
+#endif
+
+namespace clara {
+namespace detail {
+
+ // Traits for extracting arg and return type of lambdas (for single argument lambdas)
+ template
+ struct UnaryLambdaTraits : UnaryLambdaTraits {};
+
+ template
+ struct UnaryLambdaTraits {
+ static const bool isValid = false;
+ };
+
+ template
+ struct UnaryLambdaTraits {
+ static const bool isValid = true;
+ using ArgType = typename std::remove_const::type>::type;
+ using ReturnType = ReturnT;
+ };
+
+ class TokenStream;
+
+ // Transport for raw args (copied from main args, or supplied via init list for testing)
+ class Args {
+ friend TokenStream;
+ std::string m_exeName;
+ std::vector m_args;
+
+ public:
+ Args( int argc, char const* const* argv )
+ : m_exeName(argv[0]),
+ m_args(argv + 1, argv + argc) {}
+
+ Args( std::initializer_list args )
+ : m_exeName( *args.begin() ),
+ m_args( args.begin()+1, args.end() )
+ {}
+
+ auto exeName() const -> std::string {
+ return m_exeName;
+ }
+ };
+
+ // Wraps a token coming from a token stream. These may not directly correspond to strings as a single string
+ // may encode an option + its argument if the : or = form is used
+ enum class TokenType {
+ Option, Argument
+ };
+ struct Token {
+ TokenType type;
+ std::string token;
+ };
+
+ inline auto isOptPrefix( char c ) -> bool {
+ return c == '-'
+#ifdef CLARA_PLATFORM_WINDOWS
+ || c == '/'
+#endif
+ ;
+ }
+
+ // Abstracts iterators into args as a stream of tokens, with option arguments uniformly handled
+ class TokenStream {
+ using Iterator = std::vector::const_iterator;
+ Iterator it;
+ Iterator itEnd;
+ std::vector m_tokenBuffer;
+
+ void loadBuffer() {
+ m_tokenBuffer.resize( 0 );
+
+ // Skip any empty strings
+ while( it != itEnd && it->empty() )
+ ++it;
+
+ if( it != itEnd ) {
+ auto const &next = *it;
+ if( isOptPrefix( next[0] ) ) {
+ auto delimiterPos = next.find_first_of( " :=" );
+ if( delimiterPos != std::string::npos ) {
+ m_tokenBuffer.push_back( { TokenType::Option, next.substr( 0, delimiterPos ) } );
+ m_tokenBuffer.push_back( { TokenType::Argument, next.substr( delimiterPos + 1 ) } );
+ } else {
+ if( next[1] != '-' && next.size() > 2 ) {
+ std::string opt = "- ";
+ for( size_t i = 1; i < next.size(); ++i ) {
+ opt[1] = next[i];
+ m_tokenBuffer.push_back( { TokenType::Option, opt } );
+ }
+ } else {
+ m_tokenBuffer.push_back( { TokenType::Option, next } );
+ }
+ }
+ } else {
+ m_tokenBuffer.push_back( { TokenType::Argument, next } );
+ }
+ }
+ }
+
+ public:
+ explicit TokenStream( Args const &args ) : TokenStream( args.m_args.begin(), args.m_args.end() ) {}
+
+ TokenStream( Iterator it, Iterator itEnd ) : it( it ), itEnd( itEnd ) {
+ loadBuffer();
+ }
+
+ explicit operator bool() const {
+ return !m_tokenBuffer.empty() || it != itEnd;
+ }
+
+ auto count() const -> size_t { return m_tokenBuffer.size() + (itEnd - it); }
+
+ auto operator*() const -> Token {
+ assert( !m_tokenBuffer.empty() );
+ return m_tokenBuffer.front();
+ }
+
+ auto operator->() const -> Token const * {
+ assert( !m_tokenBuffer.empty() );
+ return &m_tokenBuffer.front();
+ }
+
+ auto operator++() -> TokenStream & {
+ if( m_tokenBuffer.size() >= 2 ) {
+ m_tokenBuffer.erase( m_tokenBuffer.begin() );
+ } else {
+ if( it != itEnd )
+ ++it;
+ loadBuffer();
+ }
+ return *this;
+ }
+ };
+
+
+ class ResultBase {
+ public:
+ enum Type {
+ Ok, LogicError, RuntimeError
+ };
+
+ protected:
+ ResultBase( Type type ) : m_type( type ) {}
+ virtual ~ResultBase() = default;
+
+ virtual void enforceOk() const = 0;
+
+ Type m_type;
+ };
+
+ template
+ class ResultValueBase : public ResultBase {
+ public:
+ auto value() const -> T const & {
+ enforceOk();
+ return m_value;
+ }
+
+ protected:
+ ResultValueBase( Type type ) : ResultBase( type ) {}
+
+ ResultValueBase( ResultValueBase const &other ) : ResultBase( other ) {
+ if( m_type == ResultBase::Ok )
+ new( &m_value ) T( other.m_value );
+ }
+
+ ResultValueBase( Type, T const &value ) : ResultBase( Ok ) {
+ new( &m_value ) T( value );
+ }
+
+ auto operator=( ResultValueBase const &other ) -> ResultValueBase & {
+ if( m_type == ResultBase::Ok )
+ m_value.~T();
+ ResultBase::operator=(other);
+ if( m_type == ResultBase::Ok )
+ new( &m_value ) T( other.m_value );
+ return *this;
+ }
+
+ ~ResultValueBase() override {
+ if( m_type == Ok )
+ m_value.~T();
+ }
+
+ union {
+ T m_value;
+ };
+ };
+
+ template<>
+ class ResultValueBase : public ResultBase {
+ protected:
+ using ResultBase::ResultBase;
+ };
+
+ template
+ class BasicResult : public ResultValueBase {
+ public:
+ template
+ explicit BasicResult( BasicResult const &other )
+ : ResultValueBase( other.type() ),
+ m_errorMessage( other.errorMessage() )
+ {
+ assert( type() != ResultBase::Ok );
+ }
+
+ template
+ static auto ok( U const &value ) -> BasicResult { return { ResultBase::Ok, value }; }
+ static auto ok() -> BasicResult { return { ResultBase::Ok }; }
+ static auto logicError( std::string const &message ) -> BasicResult { return { ResultBase::LogicError, message }; }
+ static auto runtimeError( std::string const &message ) -> BasicResult { return { ResultBase::RuntimeError, message }; }
+
+ explicit operator bool() const { return m_type == ResultBase::Ok; }
+ auto type() const -> ResultBase::Type { return m_type; }
+ auto errorMessage() const -> std::string { return m_errorMessage; }
+
+ protected:
+ void enforceOk() const override {
+
+ // Errors shouldn't reach this point, but if they do
+ // the actual error message will be in m_errorMessage
+ assert( m_type != ResultBase::LogicError );
+ assert( m_type != ResultBase::RuntimeError );
+ if( m_type != ResultBase::Ok )
+ std::abort();
+ }
+
+ std::string m_errorMessage; // Only populated if resultType is an error
+
+ BasicResult( ResultBase::Type type, std::string const &message )
+ : ResultValueBase(type),
+ m_errorMessage(message)
+ {
+ assert( m_type != ResultBase::Ok );
+ }
+
+ using ResultValueBase::ResultValueBase;
+ using ResultBase::m_type;
+ };
+
+ enum class ParseResultType {
+ Matched, NoMatch, ShortCircuitAll, ShortCircuitSame
+ };
+
+ class ParseState {
+ public:
+
+ ParseState( ParseResultType type, TokenStream const &remainingTokens )
+ : m_type(type),
+ m_remainingTokens( remainingTokens )
+ {}
+
+ auto type() const -> ParseResultType { return m_type; }
+ auto remainingTokens() const -> TokenStream { return m_remainingTokens; }
+
+ private:
+ ParseResultType m_type;
+ TokenStream m_remainingTokens;
+ };
+
+ using Result = BasicResult;
+ using ParserResult = BasicResult;
+ using InternalParseResult = BasicResult;
+
+ struct HelpColumns {
+ std::string left;
+ std::string right;
+ };
+
+ template
+ inline auto convertInto( std::string const &source, T& target ) -> ParserResult {
+ std::stringstream ss;
+ ss << source;
+ ss >> target;
+ if( ss.fail() )
+ return ParserResult::runtimeError( "Unable to convert '" + source + "' to destination type" );
+ else
+ return ParserResult::ok( ParseResultType::Matched );
+ }
+ inline auto convertInto( std::string const &source, std::string& target ) -> ParserResult {
+ target = source;
+ return ParserResult::ok( ParseResultType::Matched );
+ }
+ inline auto convertInto( std::string const &source, bool &target ) -> ParserResult {
+ std::string srcLC = source;
+ std::transform( srcLC.begin(), srcLC.end(), srcLC.begin(), []( char c ) { return static_cast( ::tolower(c) ); } );
+ if (srcLC == "y" || srcLC == "1" || srcLC == "true" || srcLC == "yes" || srcLC == "on")
+ target = true;
+ else if (srcLC == "n" || srcLC == "0" || srcLC == "false" || srcLC == "no" || srcLC == "off")
+ target = false;
+ else
+ return ParserResult::runtimeError( "Expected a boolean value but did not recognise: '" + source + "'" );
+ return ParserResult::ok( ParseResultType::Matched );
+ }
+#ifdef CLARA_CONFIG_OPTIONAL_TYPE
+ template
+ inline auto convertInto( std::string const &source, CLARA_CONFIG_OPTIONAL_TYPE& target ) -> ParserResult {
+ T temp;
+ auto result = convertInto( source, temp );
+ if( result )
+ target = std::move(temp);
+ return result;
+ }
+#endif // CLARA_CONFIG_OPTIONAL_TYPE
+
+ struct NonCopyable {
+ NonCopyable() = default;
+ NonCopyable( NonCopyable const & ) = delete;
+ NonCopyable( NonCopyable && ) = delete;
+ NonCopyable &operator=( NonCopyable const & ) = delete;
+ NonCopyable &operator=( NonCopyable && ) = delete;
+ };
+
+ struct BoundRef : NonCopyable {
+ virtual ~BoundRef() = default;
+ virtual auto isContainer() const -> bool { return false; }
+ virtual auto isFlag() const -> bool { return false; }
+ };
+ struct BoundValueRefBase : BoundRef {
+ virtual auto setValue( std::string const &arg ) -> ParserResult = 0;
+ };
+ struct BoundFlagRefBase : BoundRef {
+ virtual auto setFlag( bool flag ) -> ParserResult = 0;
+ virtual auto isFlag() const -> bool { return true; }
+ };
+
+ template
+ struct BoundValueRef : BoundValueRefBase {
+ T &m_ref;
+
+ explicit BoundValueRef( T &ref ) : m_ref( ref ) {}
+
+ auto setValue( std::string const &arg ) -> ParserResult override {
+ return convertInto( arg, m_ref );
+ }
+ };
+
+ template
+ struct BoundValueRef> : BoundValueRefBase {
+ std::vector &m_ref;
+
+ explicit BoundValueRef( std::vector &ref ) : m_ref( ref ) {}
+
+ auto isContainer() const -> bool override { return true; }
+
+ auto setValue( std::string const &arg ) -> ParserResult override {
+ T temp;
+ auto result = convertInto( arg, temp );
+ if( result )
+ m_ref.push_back( temp );
+ return result;
+ }
+ };
+
+ struct BoundFlagRef : BoundFlagRefBase {
+ bool &m_ref;
+
+ explicit BoundFlagRef( bool &ref ) : m_ref( ref ) {}
+
+ auto setFlag( bool flag ) -> ParserResult override {
+ m_ref = flag;
+ return ParserResult::ok( ParseResultType::Matched );
+ }
+ };
+
+ template
+ struct LambdaInvoker {
+ static_assert( std::is_same::value, "Lambda must return void or clara::ParserResult" );
+
+ template
+ static auto invoke( L const &lambda, ArgType const &arg ) -> ParserResult {
+ return lambda( arg );
+ }
+ };
+
+ template<>
+ struct LambdaInvoker {
+ template
+ static auto invoke( L const &lambda, ArgType const &arg ) -> ParserResult {
+ lambda( arg );
+ return ParserResult::ok( ParseResultType::Matched );
+ }
+ };
+
+ template
+ inline auto invokeLambda( L const &lambda, std::string const &arg ) -> ParserResult {
+ ArgType temp{};
+ auto result = convertInto( arg, temp );
+ return !result
+ ? result
+ : LambdaInvoker::ReturnType>::invoke( lambda, temp );
+ }
+
+
+ template
+ struct BoundLambda : BoundValueRefBase {
+ L m_lambda;
+
+ static_assert( UnaryLambdaTraits::isValid, "Supplied lambda must take exactly one argument" );
+ explicit BoundLambda( L const &lambda ) : m_lambda( lambda ) {}
+
+ auto setValue( std::string const &arg ) -> ParserResult override {
+ return invokeLambda::ArgType>( m_lambda, arg );
+ }
+ };
+
+ template
+ struct BoundFlagLambda : BoundFlagRefBase {
+ L m_lambda;
+
+ static_assert( UnaryLambdaTraits::isValid, "Supplied lambda must take exactly one argument" );
+ static_assert( std::is_same::ArgType, bool>::value, "flags must be boolean" );
+
+ explicit BoundFlagLambda( L const &lambda ) : m_lambda( lambda ) {}
+
+ auto setFlag( bool flag ) -> ParserResult override {
+ return LambdaInvoker::ReturnType>::invoke( m_lambda, flag );
+ }
+ };
+
+ enum class Optionality { Optional, Required };
+
+ struct Parser;
+
+ class ParserBase {
+ public:
+ virtual ~ParserBase() = default;
+ virtual auto validate() const -> Result { return Result::ok(); }
+ virtual auto parse( std::string const& exeName, TokenStream const &tokens) const -> InternalParseResult = 0;
+ virtual auto cardinality() const -> size_t { return 1; }
+
+ auto parse( Args const &args ) const -> InternalParseResult {
+ return parse( args.exeName(), TokenStream( args ) );
+ }
+ };
+
+ template
+ class ComposableParserImpl : public ParserBase {
+ public:
+ template
+ auto operator|( T const &other ) const -> Parser;
+
+ template
+ auto operator+( T const &other ) const -> Parser;
+ };
+
+ // Common code and state for Args and Opts
+ template
+ class ParserRefImpl : public ComposableParserImpl {
+ protected:
+ Optionality m_optionality = Optionality::Optional;
+ std::shared_ptr m_ref;
+ std::string m_hint;
+ std::string m_description;
+
+ explicit ParserRefImpl( std::shared_ptr const &ref ) : m_ref( ref ) {}
+
+ public:
+ template
+ ParserRefImpl( T &ref, std::string const &hint )
+ : m_ref( std::make_shared>( ref ) ),
+ m_hint( hint )
+ {}
+
+ template
+ ParserRefImpl( LambdaT const &ref, std::string const &hint )
+ : m_ref( std::make_shared>( ref ) ),
+ m_hint(hint)
+ {}
+
+ auto operator()( std::string const &description ) -> DerivedT & {
+ m_description = description;
+ return static_cast( *this );
+ }
+
+ auto optional() -> DerivedT & {
+ m_optionality = Optionality::Optional;
+ return static_cast( *this );
+ };
+
+ auto required() -> DerivedT & {
+ m_optionality = Optionality::Required;
+ return static_cast( *this );
+ };
+
+ auto isOptional() const -> bool {
+ return m_optionality == Optionality::Optional;
+ }
+
+ auto cardinality() const -> size_t override {
+ if( m_ref->isContainer() )
+ return 0;
+ else
+ return 1;
+ }
+
+ auto hint() const -> std::string { return m_hint; }
+ };
+
+ class ExeName : public ComposableParserImpl {
+ std::shared_ptr m_name;
+ std::shared_ptr m_ref;
+
+ template
+ static auto makeRef(LambdaT const &lambda) -> std::shared_ptr {
+ return std::make_shared>( lambda) ;
+ }
+
+ public:
+ ExeName() : m_name( std::make_shared( "" ) ) {}
+
+ explicit ExeName( std::string &ref ) : ExeName() {
+ m_ref = std::make_shared>( ref );
+ }
+
+ template
+ explicit ExeName( LambdaT const& lambda ) : ExeName() {
+ m_ref = std::make_shared>( lambda );
+ }
+
+ // The exe name is not parsed out of the normal tokens, but is handled specially
+ auto parse( std::string const&, TokenStream const &tokens ) const -> InternalParseResult override {
+ return InternalParseResult::ok( ParseState( ParseResultType::NoMatch, tokens ) );
+ }
+
+ auto name() const -> std::string { return *m_name; }
+ auto set( std::string const& newName ) -> ParserResult {
+
+ auto lastSlash = newName.find_last_of( "\\/" );
+ auto filename = ( lastSlash == std::string::npos )
+ ? newName
+ : newName.substr( lastSlash+1 );
+
+ *m_name = filename;
+ if( m_ref )
+ return m_ref->setValue( filename );
+ else
+ return ParserResult::ok( ParseResultType::Matched );
+ }
+ };
+
+ class Arg : public ParserRefImpl {
+ public:
+ using ParserRefImpl::ParserRefImpl;
+
+ auto parse( std::string const &, TokenStream const &tokens ) const -> InternalParseResult override {
+ auto validationResult = validate();
+ if( !validationResult )
+ return InternalParseResult( validationResult );
+
+ auto remainingTokens = tokens;
+ auto const &token = *remainingTokens;
+ if( token.type != TokenType::Argument )
+ return InternalParseResult::ok( ParseState( ParseResultType::NoMatch, remainingTokens ) );
+
+ assert( !m_ref->isFlag() );
+ auto valueRef = static_cast( m_ref.get() );
+
+ auto result = valueRef->setValue( remainingTokens->token );
+ if( !result )
+ return InternalParseResult( result );
+ else
+ return InternalParseResult::ok( ParseState( ParseResultType::Matched, ++remainingTokens ) );
+ }
+ };
+
+ inline auto normaliseOpt( std::string const &optName ) -> std::string {
+#ifdef CLARA_PLATFORM_WINDOWS
+ if( optName[0] == '/' )
+ return "-" + optName.substr( 1 );
+ else
+#endif
+ return optName;
+ }
+
+ class Opt : public ParserRefImpl {
+ protected:
+ std::vector m_optNames;
+
+ public:
+ template
+ explicit Opt( LambdaT const &ref ) : ParserRefImpl( std::make_shared>( ref ) ) {}
+
+ explicit Opt( bool &ref ) : ParserRefImpl( std::make_shared( ref ) ) {}
+
+ template
+ Opt( LambdaT const &ref, std::string const &hint ) : ParserRefImpl( ref, hint ) {}
+
+ template
+ Opt( T &ref, std::string const &hint ) : ParserRefImpl( ref, hint ) {}
+
+ auto operator[]( std::string const &optName ) -> Opt & {
+ m_optNames.push_back( optName );
+ return *this;
+ }
+
+ auto getHelpColumns() const -> std::vector {
+ std::ostringstream oss;
+ bool first = true;
+ for( auto const &opt : m_optNames ) {
+ if (first)
+ first = false;
+ else
+ oss << ", ";
+ oss << opt;
+ }
+ if( !m_hint.empty() )
+ oss << " <" << m_hint << ">";
+ return { { oss.str(), m_description } };
+ }
+
+ auto isMatch( std::string const &optToken ) const -> bool {
+ auto normalisedToken = normaliseOpt( optToken );
+ for( auto const &name : m_optNames ) {
+ if( normaliseOpt( name ) == normalisedToken )
+ return true;
+ }
+ return false;
+ }
+
+ using ParserBase::parse;
+
+ auto parse( std::string const&, TokenStream const &tokens ) const -> InternalParseResult override {
+ auto validationResult = validate();
+ if( !validationResult )
+ return InternalParseResult( validationResult );
+
+ auto remainingTokens = tokens;
+ if( remainingTokens && remainingTokens->type == TokenType::Option ) {
+ auto const &token = *remainingTokens;
+ if( isMatch(token.token ) ) {
+ if( m_ref->isFlag() ) {
+ auto flagRef = static_cast( m_ref.get() );
+ auto result = flagRef->setFlag( true );
+ if( !result )
+ return InternalParseResult( result );
+ if( result.value() == ParseResultType::ShortCircuitAll )
+ return InternalParseResult::ok( ParseState( result.value(), remainingTokens ) );
+ } else {
+ auto valueRef = static_cast( m_ref.get() );
+ ++remainingTokens;
+ if( !remainingTokens )
+ return InternalParseResult::runtimeError( "Expected argument following " + token.token );
+ auto const &argToken = *remainingTokens;
+ if( argToken.type != TokenType::Argument )
+ return InternalParseResult::runtimeError( "Expected argument following " + token.token );
+ auto result = valueRef->setValue( argToken.token );
+ if( !result )
+ return InternalParseResult( result );
+ if( result.value() == ParseResultType::ShortCircuitAll )
+ return InternalParseResult::ok( ParseState( result.value(), remainingTokens ) );
+ }
+ return InternalParseResult::ok( ParseState( ParseResultType::Matched, ++remainingTokens ) );
+ }
+ }
+ return InternalParseResult::ok( ParseState( ParseResultType::NoMatch, remainingTokens ) );
+ }
+
+ auto validate() const -> Result override {
+ if( m_optNames.empty() )
+ return Result::logicError( "No options supplied to Opt" );
+ for( auto const &name : m_optNames ) {
+ if( name.empty() )
+ return Result::logicError( "Option name cannot be empty" );
+#ifdef CLARA_PLATFORM_WINDOWS
+ if( name[0] != '-' && name[0] != '/' )
+ return Result::logicError( "Option name must begin with '-' or '/'" );
+#else
+ if( name[0] != '-' )
+ return Result::logicError( "Option name must begin with '-'" );
+#endif
+ }
+ return ParserRefImpl::validate();
+ }
+ };
+
+ struct Help : Opt {
+ Help( bool &showHelpFlag )
+ : Opt([&]( bool flag ) {
+ showHelpFlag = flag;
+ return ParserResult::ok( ParseResultType::ShortCircuitAll );
+ })
+ {
+ static_cast( *this )
+ ("display usage information")
+ ["-?"]["-h"]["--help"]
+ .optional();
+ }
+ };
+
+
+ struct Parser : ParserBase {
+
+ mutable ExeName m_exeName;
+ std::vector m_options;
+ std::vector m_args;
+
+ auto operator|=( ExeName const &exeName ) -> Parser & {
+ m_exeName = exeName;
+ return *this;
+ }
+
+ auto operator|=( Arg const &arg ) -> Parser & {
+ m_args.push_back(arg);
+ return *this;
+ }
+
+ auto operator|=( Opt const &opt ) -> Parser & {
+ m_options.push_back(opt);
+ return *this;
+ }
+
+ auto operator|=( Parser const &other ) -> Parser & {
+ m_options.insert(m_options.end(), other.m_options.begin(), other.m_options.end());
+ m_args.insert(m_args.end(), other.m_args.begin(), other.m_args.end());
+ return *this;
+ }
+
+ template
+ auto operator|( T const &other ) const -> Parser {
+ return Parser( *this ) |= other;
+ }
+
+ // Forward deprecated interface with '+' instead of '|'
+ template
+ auto operator+=( T const &other ) -> Parser & { return operator|=( other ); }
+ template
+ auto operator+( T const &other ) const -> Parser { return operator|( other ); }
+
+ auto getHelpColumns() const -> std::vector {
+ std::vector cols;
+ for (auto const &o : m_options) {
+ auto childCols = o.getHelpColumns();
+ cols.insert( cols.end(), childCols.begin(), childCols.end() );
+ }
+ return cols;
+ }
+
+ void writeToStream( std::ostream &os ) const {
+ if (!m_exeName.name().empty()) {
+ os << "usage:\n" << " " << m_exeName.name() << " ";
+ bool required = true, first = true;
+ for( auto const &arg : m_args ) {
+ if (first)
+ first = false;
+ else
+ os << " ";
+ if( arg.isOptional() && required ) {
+ os << "[";
+ required = false;
+ }
+ os << "<" << arg.hint() << ">";
+ if( arg.cardinality() == 0 )
+ os << " ... ";
+ }
+ if( !required )
+ os << "]";
+ if( !m_options.empty() )
+ os << " options";
+ os << "\n\nwhere options are:" << std::endl;
+ }
+
+ auto rows = getHelpColumns();
+ size_t consoleWidth = CLARA_CONFIG_CONSOLE_WIDTH;
+ size_t optWidth = 0;
+ for( auto const &cols : rows )
+ optWidth = (std::max)(optWidth, cols.left.size() + 2);
+
+ optWidth = (std::min)(optWidth, consoleWidth/2);
+
+ for( auto const &cols : rows ) {
+ auto row =
+ TextFlow::Column( cols.left ).width( optWidth ).indent( 2 ) +
+ TextFlow::Spacer(4) +
+ TextFlow::Column( cols.right ).width( consoleWidth - 7 - optWidth );
+ os << row << std::endl;
+ }
+ }
+
+ friend auto operator<<( std::ostream &os, Parser const &parser ) -> std::ostream& {
+ parser.writeToStream( os );
+ return os;
+ }
+
+ auto validate() const -> Result override {
+ for( auto const &opt : m_options ) {
+ auto result = opt.validate();
+ if( !result )
+ return result;
+ }
+ for( auto const &arg : m_args ) {
+ auto result = arg.validate();
+ if( !result )
+ return result;
+ }
+ return Result::ok();
+ }
+
+ using ParserBase::parse;
+
+ auto parse( std::string const& exeName, TokenStream const &tokens ) const -> InternalParseResult override {
+
+ struct ParserInfo {
+ ParserBase const* parser = nullptr;
+ size_t count = 0;
+ };
+ const size_t totalParsers = m_options.size() + m_args.size();
+ assert( totalParsers < 512 );
+ // ParserInfo parseInfos[totalParsers]; // <-- this is what we really want to do
+ ParserInfo parseInfos[512];
+
+ {
+ size_t i = 0;
+ for (auto const &opt : m_options) parseInfos[i++].parser = &opt;
+ for (auto const &arg : m_args) parseInfos[i++].parser = &arg;
+ }
+
+ m_exeName.set( exeName );
+
+ auto result = InternalParseResult::ok( ParseState( ParseResultType::NoMatch, tokens ) );
+ while( result.value().remainingTokens() ) {
+ bool tokenParsed = false;
+
+ for( size_t i = 0; i < totalParsers; ++i ) {
+ auto& parseInfo = parseInfos[i];
+ if( parseInfo.parser->cardinality() == 0 || parseInfo.count < parseInfo.parser->cardinality() ) {
+ result = parseInfo.parser->parse(exeName, result.value().remainingTokens());
+ if (!result)
+ return result;
+ if (result.value().type() != ParseResultType::NoMatch) {
+ tokenParsed = true;
+ ++parseInfo.count;
+ break;
+ }
+ }
+ }
+
+ if( result.value().type() == ParseResultType::ShortCircuitAll )
+ return result;
+ if( !tokenParsed )
+ return InternalParseResult::runtimeError( "Unrecognised token: " + result.value().remainingTokens()->token );
+ }
+ // !TBD Check missing required options
+ return result;
+ }
+ };
+
+ template
+ template
+ auto ComposableParserImpl::operator|( T const &other ) const -> Parser {
+ return Parser() | static_cast( *this ) | other;
+ }
+} // namespace detail
+
+
+// A Combined parser
+using detail::Parser;
+
+// A parser for options
+using detail::Opt;
+
+// A parser for arguments
+using detail::Arg;
+
+// Wrapper for argc, argv from main()
+using detail::Args;
+
+// Specifies the name of the executable
+using detail::ExeName;
+
+// Convenience wrapper for option parser that specifies the help option
+using detail::Help;
+
+// enum of result types from a parse
+using detail::ParseResultType;
+
+// Result type for parser operation
+using detail::ParserResult;
+
+
+} // namespace clara
+
+#endif // CLARA_HPP_INCLUDED
diff --git a/src/virtual-keyboard.cpp b/src/virtual-keyboard.cpp
index 4850f2e..d84fef5 100644
--- a/src/virtual-keyboard.cpp
+++ b/src/virtual-keyboard.cpp
@@ -7,14 +7,18 @@
#include
#include
+#include
+#include
+#include
+
namespace wf
{
VirtualKeyboardDevice::VirtualKeyboardDevice()
{
auto& display = WaylandDisplay::get();
-
+ auto seat = Gdk::Display::get_default()->get_default_seat();
vk = zwp_virtual_keyboard_manager_v1_create_virtual_keyboard(
- display.vk_manager, display.seat);
+ display.vk_manager, gdk_wayland_seat_get_wl_seat(seat->gobj()));
this->send_keymap();
}
diff --git a/src/wayland-window.cpp b/src/wayland-window.cpp
index 7671b68..a6e04d1 100644
--- a/src/wayland-window.cpp
+++ b/src/wayland-window.cpp
@@ -1,28 +1,31 @@
#include "wayland-window.hpp"
#include
+#include
+#include
+#include
#include
#include
+#include
+#include
+
+#include
+#include
+
+static constexpr int HEADERBAR_SIZE = 60;
namespace wf
{
// listeners
- static void registry_add_object(void *data, struct wl_registry *registry, uint32_t name,
- const char *interface, uint32_t version)
+ static void registry_add_object(void *data, struct wl_registry *registry,
+ uint32_t name, const char *interface, uint32_t version)
{
auto display = static_cast (data);
- if (strcmp(interface, wl_seat_interface.name) == 0 && !display->seat)
+ if (strcmp(interface, zwf_shell_manager_v2_interface.name) == 0)
{
- display->seat = (wl_seat*) wl_registry_bind(registry, name,
- &wl_seat_interface, std::min(version, 1u));
- }
-
- if (strcmp(interface, zwf_shell_manager_v1_interface.name) == 0)
- {
- display->wf_manager =
- (zwf_shell_manager_v1*) wl_registry_bind(registry, name,
- &zwf_shell_manager_v1_interface,
- std::min(version, 1u));
+ display->zwf_manager =
+ (zwf_shell_manager_v2*) wl_registry_bind(registry, name,
+ &zwf_shell_manager_v2_interface, std::min(version, 1u));
}
if (strcmp(interface, zwp_virtual_keyboard_manager_v1_interface.name) == 0)
@@ -58,12 +61,13 @@ namespace wf
wl_registry *registry = wl_display_get_registry(display);
wl_registry_add_listener(registry, ®istry_listener, this);
+ wl_display_dispatch(display);
wl_display_roundtrip(display);
- if (!vk_manager || !seat || !wf_manager)
+ if (!vk_manager)
{
std::cerr << "Compositor doesn't support the virtual-keyboard-v1 "
- << "and/or the wayfire-shell protocols, exiting" << std::endl;
+ << "protocol, exiting" << std::endl;
std::exit(-1);
}
}
@@ -74,30 +78,108 @@ namespace wf
return instance;
}
- WaylandWindow::WaylandWindow(int x, int y, int width, int height)
- : Gtk::Window()
+ int32_t WaylandWindow::check_anchor(std::string anchor)
{
- auto display = WaylandDisplay::get();
+ std::transform(anchor.begin(), anchor.end(), anchor.begin(), ::tolower);
+
+ int32_t parsed_anchor = -1;
+ if (anchor.compare("top") == 0)
+ {
+ parsed_anchor = GTK_LAYER_SHELL_EDGE_TOP;
+ } else if (anchor.compare("bottom") == 0)
+ {
+ parsed_anchor = GTK_LAYER_SHELL_EDGE_BOTTOM;
+ } else if (anchor.compare("left") == 0)
+ {
+ parsed_anchor = GTK_LAYER_SHELL_EDGE_LEFT;
+ } else if (anchor.compare("right") == 0)
+ {
+ parsed_anchor = GTK_LAYER_SHELL_EDGE_RIGHT;
+ }
+
+ return parsed_anchor;
+ }
+
+ void WaylandWindow::init(int width, int height, std::string anchor)
+ {
+ gtk_layer_init_for_window(this->gobj());
+ gtk_layer_set_layer(this->gobj(), GTK_LAYER_SHELL_LAYER_OVERLAY);
+ gtk_layer_set_namespace(this->gobj(), "keyboard");
+ auto layer_anchor = check_anchor(anchor);
+ if (layer_anchor > -1)
+ {
+ gtk_layer_set_anchor(this->gobj(),
+ (GtkLayerShellEdge)layer_anchor, true);
+ }
- /* Trick: first show the window, get frame size, then subtract it again */
this->set_size_request(width, height);
- this->set_default_size(width, height);
- this->set_type_hint(Gdk::WINDOW_TYPE_HINT_DOCK);
this->show_all();
-
auto gdk_window = this->get_window()->gobj();
auto surface = gdk_wayland_window_get_wl_surface(gdk_window);
- if (!surface)
+ if (surface && WaylandDisplay::get().zwf_manager)
{
- std::cerr << "Error: created window was not a wayland surface" << std::endl;
- std::exit(-1);
+ this->wf_surface = zwf_shell_manager_v2_get_wf_surface(
+ WaylandDisplay::get().zwf_manager, surface);
}
+ }
- wm_surface = zwf_shell_manager_v1_get_wm_surface(display.wf_manager,
- surface, ZWF_WM_SURFACE_V1_ROLE_DESKTOP_WIDGET, NULL);
- zwf_wm_surface_v1_set_keyboard_mode(wm_surface,
- ZWF_WM_SURFACE_V1_KEYBOARD_FOCUS_MODE_NO_FOCUS);
- zwf_wm_surface_v1_configure(wm_surface, x, y);
+ WaylandWindow::WaylandWindow(int width, int height, std::string anchor)
+ : Gtk::Window()
+ {
+ // setup close button
+ close_button.get_style_context()->add_class("image-button");
+ close_button.set_image_from_icon_name("window-close-symbolic",
+ Gtk::ICON_SIZE_LARGE_TOOLBAR);
+ close_button.signal_clicked().connect_notify([=] () {
+ this->get_application()->quit();
+ });
+
+ // setup move gesture
+ headerbar_drag = Gtk::GestureDrag::create(drag_box);
+ headerbar_drag->signal_drag_begin().connect_notify([=] (double, double) {
+ if (this->wf_surface)
+ {
+ zwf_surface_v2_interactive_move(this->wf_surface);
+ /* Taken from GDK's Wayland impl of begin_move_drag() */
+ Gdk::Display::get_default()->get_default_seat()->ungrab();
+ headerbar_drag->reset();
+ }
+ });
+ Gtk::HeaderBar bar;
+ headerbar_box.override_background_color(bar.get_style_context()->get_background_color());
+
+
+ // setup headerbar layout
+ headerbar_box.set_size_request(-1, HEADERBAR_SIZE);
+
+ close_button.set_size_request(HEADERBAR_SIZE * 0.8, HEADERBAR_SIZE * 0.8);
+ close_button.set_margin_bottom(OSK_SPACING);
+ close_button.set_margin_top(OSK_SPACING);
+ close_button.set_margin_left(OSK_SPACING);
+ close_button.set_margin_right(OSK_SPACING);
+
+ headerbar_box.pack_end(close_button, false, false);
+ headerbar_box.pack_start(drag_box, true, true);
+ layout_box.pack_start(headerbar_box);
+ layout_box.set_spacing(OSK_SPACING);
+ this->add(layout_box);
+
+ // setup gtk layer shell
+ init(width, height, anchor);
+ }
+
+ void WaylandWindow::set_widget(Gtk::Widget& w)
+ {
+ if (current_widget)
+ this->layout_box.remove(*current_widget);
+
+ this->layout_box.pack_end(w);
+ current_widget = &w;
+
+ w.set_margin_bottom(OSK_SPACING);
+ w.set_margin_left(OSK_SPACING);
+ w.set_margin_right(OSK_SPACING);
+ this->show_all();
}
}
diff --git a/src/wayland-window.hpp b/src/wayland-window.hpp
index 4fea46f..fa64509 100644
--- a/src/wayland-window.hpp
+++ b/src/wayland-window.hpp
@@ -1,9 +1,16 @@
#pragma once
+#include
+#include
#include
-#include
+#include
+#include
+#include
+#include
#include
+#define OSK_SPACING 8
+
namespace wf
{
class WaylandDisplay
@@ -13,15 +20,26 @@ namespace wf
public:
static WaylandDisplay& get();
- wl_seat *seat = nullptr;
- zwf_shell_manager_v1 *wf_manager = nullptr;
+ zwf_shell_manager_v2 *zwf_manager = nullptr;
zwp_virtual_keyboard_manager_v1 *vk_manager = nullptr;
};
class WaylandWindow : public Gtk::Window
{
- zwf_wm_surface_v1 *wm_surface;
- public:
- WaylandWindow(int x, int y, int width, int height);
+ zwf_surface_v2 *wf_surface = nullptr;
+
+ Gtk::Widget* current_widget = nullptr;
+ Glib::RefPtr headerbar_drag;
+ Gtk::EventBox drag_box;
+ Gtk::Button close_button;
+ Gtk::HBox headerbar_box;
+ Gtk::VBox layout_box;
+
+ int32_t check_anchor(std::string anchor);
+ void init(int width, int height, std::string anchor);
+
+ public:
+ WaylandWindow(int width, int height, std::string anchor);
+ void set_widget(Gtk::Widget& w);
};
}