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); }; }