// vim: set et fenc=utf-8 :
#define WIN32_LEAN_AND_MEAN
#define NOMINMAX
#define UNICODE 1
#define _UNICODE 1
#define OEMRESOURCE
#include <windows.h>
#include <intrin.h>
#define BOOST_PARAMETER_MAX_ARITY 15
#include <boost/parameter.hpp>
#include <boost/optional.hpp>
#include <boost/variant.hpp>
#include <boost/exception/all.hpp>
#include <boost/utility.hpp>
#include <string>
#include <iostream>
#pragma comment(lib, "user32.lib")
#pragma comment(lib, "gdi32.lib")
#define SBOK_ASSERT(x) \
do { if (!(x)) { std::cerr << "Assert failed: " << #x << " at " << __FILE__ << ":" << __LINE__ << std::endl; __debugbreak(); } } while (false)
#if !defined(UNICODE) || !defined(_UNICODE) || _UNICODE != 1 || UNICODE != 1
# error Springbok requires UNICODE to be set.
#endif
namespace detail {
typedef boost::error_info<struct tag_error_msg, std::string> error_msg;
typedef boost::error_info<struct tag_error_code, int> error_code;
template <typename Exception>
Exception make_error(boost::optional<DWORD> code = boost::none) {
DWORD real_code = (code ? *code : GetLastError());
std::string message;
LPSTR buffer;
DWORD result = FormatMessageA(
FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM,
nullptr, real_code, 0, reinterpret_cast<LPSTR>(&buffer), 0, nullptr
);
if (result) {
message = buffer;
LocalFree(buffer);
} else {
message = "<FormatMessage failed>";
}
return Exception() << error_msg(message) << error_code(real_code);
}
}
struct base_error : virtual boost::exception, virtual std::exception {};
struct windowing_error : virtual base_error {};
#define SBOK_THROW_WIN(exc) BOOST_THROW_EXCEPTION(detail::make_error<exc>())
namespace window_class {
enum styles {
enable_double_clicks = CS_DBLCLKS,
redraw_on_width_change = CS_HREDRAW,
redraw_on_height_change = CS_VREDRAW,
no_close_button = CS_NOCLOSE,
use_shared_context = CS_CLASSDC,
use_unique_context = CS_OWNDC,
use_parent_context = CS_PARENTDC,
no_style = 0
};
namespace keywords {
BOOST_PARAMETER_NAME(class_name)
BOOST_PARAMETER_NAME(class_style)
BOOST_PARAMETER_NAME(icon)
BOOST_PARAMETER_NAME(cursor)
BOOST_PARAMETER_NAME(background)
BOOST_PARAMETER_NAME(menu_resource)
BOOST_PARAMETER_NAME(small_icon)
}
};
namespace detail {
LRESULT CALLBACK dispatch_wndproc(HWND, UINT, WPARAM, LPARAM);
typedef boost::variant<WORD, std::wstring> resource_id;
struct resource_id_visitor : boost::static_visitor<>, private boost::noncopyable {
mutable LPCWSTR& resource_id;
resource_id_visitor(LPCWSTR& resource_id) : resource_id(resource_id) {
}
void operator()(const WORD& int_id) const {
resource_id = MAKEINTRESOURCE(int_id);
}
void operator()(const std::wstring& str_id) const {
resource_id = str_id.c_str();
}
};
}
namespace detail {
struct window_impl;
struct window_class_impl {
window_class_impl(
const std::wstring& class_name, window_class::styles class_style, HICON icon, boost::optional<HCURSOR> cursor,
boost::optional<HBRUSH> background, const detail::resource_id& menu_resource, HICON small_icon
) : name(class_name), atom(), class_info()
{
SecureZeroMemory(&class_info, sizeof class_info);
HCURSOR use_cursor;
HBRUSH use_background;
if (cursor) {
use_cursor = *cursor;
} else {
HANDLE arrow_cursor = LoadImage(
class_info.hInstance, MAKEINTRESOURCE(OCR_NORMAL), IMAGE_CURSOR, 0, 0,
LR_DEFAULTSIZE | LR_SHARED
);
if (!arrow_cursor) {
SBOK_THROW_WIN(windowing_error);
}
use_cursor = static_cast<HCURSOR>(arrow_cursor);
}
if (background) {
use_background = *background;
} else {
use_background = static_cast<HBRUSH>(GetStockObject(BLACK_BRUSH));
}
class_info.cbSize = sizeof class_info;
class_info.style = static_cast<UINT>(class_style);
class_info.lpfnWndProc = detail::dispatch_wndproc;
class_info.cbClsExtra = 0;
class_info.cbWndExtra = sizeof (window_impl*);
class_info.hInstance = GetModuleHandle(nullptr);
class_info.hIcon = icon;
class_info.hCursor = use_cursor;
class_info.hbrBackground = use_background;
class_info.lpszClassName = class_name.c_str();
class_info.hIconSm = small_icon;
boost::apply_visitor(detail::resource_id_visitor(class_info.lpszMenuName), menu_resource);
atom = RegisterClassEx(&class_info);
if (!atom) {
SBOK_THROW_WIN(windowing_error);
}
}
std::wstring get_name() const {
return name;
}
private:
std::wstring name;
ATOM atom;
WNDCLASSEX class_info;
};
}
namespace window_position {
struct specific {
explicit specific(int x = CW_USEDEFAULT, int y = 0) : x(x), y(y) {
}
int x, y;
};
struct centered {};
};
namespace window {
enum extended_styles {
no_extended_style = 0,
accept_files = WS_EX_ACCEPTFILES,
is_app_window = WS_EX_APPWINDOW,
has_client_edge = WS_EX_CLIENTEDGE,
is_composited = WS_EX_COMPOSITED,
has_context_help = WS_EX_CONTEXTHELP,
is_control_parent = WS_EX_CONTROLPARENT,
has_modal_frame = WS_EX_DLGMODALFRAME,
is_layered = WS_EX_LAYERED,
is_mdi_child = WS_EX_MDICHILD,
no_inherited_layout = WS_EX_NOINHERITLAYOUT,
no_parent_notify = WS_EX_NOPARENTNOTIFY,
is_overlapped_ex = WS_EX_OVERLAPPEDWINDOW,
has_static_edge = WS_EX_STATICEDGE,
is_topmost = WS_EX_TOPMOST,
has_window_edge = WS_EX_WINDOWEDGE
};
enum styles {
default_style = WS_OVERLAPPED,
has_border = WS_BORDER,
has_title_bar = WS_CAPTION,
is_child = WS_CHILD,
is_disabled = WS_DISABLED,
has_dialog_frame = WS_DLGFRAME,
has_horiz_scrollbar = WS_HSCROLL,
create_minimised = WS_MINIMIZE,
create_maximised = WS_MAXIMIZE,
has_maximise_button = WS_MAXIMIZEBOX,
has_minimise_button = WS_MINIMIZEBOX,
is_overlapped = WS_OVERLAPPEDWINDOW,
has_sizing_border = WS_SIZEBOX,
has_system_menu = WS_SYSMENU,
create_visible = WS_VISIBLE,
has_vert_scrollbar = WS_VSCROLL
};
namespace keywords {
BOOST_PARAMETER_NAME(window_class)
BOOST_PARAMETER_NAME(extended_style)
BOOST_PARAMETER_NAME(name)
BOOST_PARAMETER_NAME(style)
BOOST_PARAMETER_NAME(position)
BOOST_PARAMETER_NAME(width)
BOOST_PARAMETER_NAME(height)
BOOST_PARAMETER_NAME(parent)
BOOST_PARAMETER_NAME(menu)
}
}
namespace detail {
typedef boost::variant<window_position::specific, window_position::centered> window_position_variant;
// called to get x, y for CreateWindowEx
struct position_create_visitor : boost::static_visitor<>, private boost::noncopyable {
mutable int &x, &y;
int width, height;
position_create_visitor(int& x, int& y, int width, int height) : x(x), y(y), width(width), height(height) {}
void operator()(const window_position::specific& pos) const {
x = pos.x;
y = pos.y;
}
void operator()(const window_position::centered&) const {
RECT rect;
GetClientRect(GetDesktopWindow(), &rect);
x = (rect.right / 2) - (width / 2);
y = (rect.bottom / 2) - (height / 2);
}
};
struct unhandled {};
typedef boost::variant<unhandled, LRESULT> process_result;
// wraps a window
struct window_impl {
window_impl(
const window_class_impl& window_class, window::extended_styles extended_style, const std::wstring& name,
window::styles style, const window_position_variant& position, int width, int height, HWND parent, HMENU menu
) : handle()
{
int x, y;
boost::apply_visitor(position_create_visitor(x, y, width, height), position);
std::wstring class_name = window_class.get_name();
handle = CreateWindowEx(
extended_style, class_name.c_str(), name.c_str(), style,
x, y, width, height, parent, menu, GetModuleHandle(nullptr),
nullptr
);
if (handle == INVALID_HANDLE_VALUE) {
SBOK_THROW_WIN(windowing_error);
}
SetWindowLongPtr(handle, GWLP_USERDATA, reinterpret_cast<LONG_PTR>(this));
UpdateWindow(handle);
}
process_result process_message(UINT, WPARAM, LPARAM) {
return unhandled();
}
HWND handle;
};
}
// factory functions
BOOST_PARAMETER_FUNCTION(
(detail::window_class_impl), create_window_class, window_class::keywords::tag,
(required
(class_name, *)
)
(optional
(class_style, *, window_class::no_style)
(icon, *, nullptr )
(cursor, *, boost::none )
(background, *, boost::none )
(menu_resource, *, L"" )
(small_icon, *, nullptr )
)
) {
return detail::window_class_impl(class_name, class_style, icon, cursor, background, menu_resource, small_icon);
}
BOOST_PARAMETER_FUNCTION(
(detail::window_impl), create_window, window::keywords::tag,
(required
(window_class, *)
)
(optional
(extended_style, *, window::is_overlapped_ex)
(name, *, L"")
(style, *, static_cast<window::styles>(window::default_style | window::create_visible))
(position, *, window_position::centered())
(width, *, 800)
(height, *, 600)
(parent, *, nullptr)
(menu, *, nullptr)
)
) {
return detail::window_impl(window_class, extended_style, name, style, position, width, height, parent, menu);
}
namespace detail {
struct dispatch_visitor : boost::static_visitor<LRESULT>, private boost::noncopyable {
const HWND& handle;
const UINT& message;
const WPARAM& w;
const LPARAM& l;
dispatch_visitor(const HWND& handle, const UINT& message, const WPARAM& w, const LPARAM& l)
: handle(handle), message(message), w(w), l(l) {}
LRESULT operator()(const LRESULT& result) const {
return result;
}
LRESULT operator()(const unhandled&) const {
return DefWindowProc(handle, message, w, l);
}
};
// delegates the message to process_msg member function inside window instance
LRESULT CALLBACK dispatch_wndproc(HWND handle, UINT message, WPARAM w, LPARAM l) {
LONG_PTR self = GetWindowLongPtr(handle, GWLP_USERDATA);
//SBOK_ASSERT(self != 0);
auto real_wndproc = &window_impl::process_message;
auto real_handle = reinterpret_cast<window_impl*>(self);
detail::process_result result = (real_handle->*real_wndproc)(message, w, l);
return boost::apply_visitor(dispatch_visitor(handle, message, w, l), result);
}
}
int main() {
using namespace window_class::keywords;
using namespace window::keywords;
try {
auto cls = create_window_class(_class_name = L"hello world", _class_style = window_class::no_style);
auto wnd = create_window(_window_class = cls, _name = L"ohai");
int result = 0;
MSG msg;
while ((result = GetMessage(&msg, NULL, 0, 0)) != 0) {
if (result == -1) {
SBOK_THROW_WIN(windowing_error);
} else {
TranslateMessage(&msg);
DispatchMessage(&msg);
}
}
} catch (const std::exception& e) {
std::cerr << boost::diagnostic_information(e) << std::endl;
}
return 0;
}
// vim: set et fenc=utf-8 :
#define WIN32_LEAN_AND_MEAN
#define NOMINMAX
#define UNICODE 1
#define _UNICODE 1
#define OEMRESOURCE
#include <windows.h>
#include <intrin.h>
#define BOOST_PARAMETER_MAX_ARITY 15
#include <boost/parameter.hpp>
#include <boost/optional.hpp>
#include <boost/variant.hpp>
#include <boost/exception/all.hpp>
#include <boost/utility.hpp>
#include <string>
#include <iostream>

#pragma comment(lib, "user32.lib")
#pragma comment(lib, "gdi32.lib")

#define SBOK_ASSERT(x) \
    do { if (!(x)) { std::cerr << "Assert failed: " << #x << " at " << __FILE__ << ":" << __LINE__ << std::endl; __debugbreak(); } } while (false)

#if !defined(UNICODE) || !defined(_UNICODE) || _UNICODE != 1 || UNICODE != 1
#   error Springbok requires UNICODE to be set.
#endif

namespace detail {
    typedef boost::error_info<struct tag_error_msg,  std::string> error_msg;
    typedef boost::error_info<struct tag_error_code, int>         error_code;
    
    template <typename Exception>
    Exception make_error(boost::optional<DWORD> code = boost::none) {
        DWORD real_code = (code ? *code : GetLastError());
        std::string message;
        
        LPSTR buffer;
        DWORD result = FormatMessageA(
            FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM,
            nullptr, real_code, 0, reinterpret_cast<LPSTR>(&buffer), 0, nullptr
        );
        
        if (result) {
            message = buffer;
            LocalFree(buffer);
        } else {
            message = "<FormatMessage failed>";
        }
        
        return Exception() << error_msg(message) << error_code(real_code);
    }
}

struct base_error : virtual boost::exception, virtual std::exception {};
struct windowing_error : virtual base_error {};

#define SBOK_THROW_WIN(exc) BOOST_THROW_EXCEPTION(detail::make_error<exc>())

namespace window_class {
    enum styles {
        enable_double_clicks    = CS_DBLCLKS,
        redraw_on_width_change  = CS_HREDRAW,
        redraw_on_height_change = CS_VREDRAW,
        no_close_button         = CS_NOCLOSE,
        use_shared_context      = CS_CLASSDC,
        use_unique_context      = CS_OWNDC,
        use_parent_context      = CS_PARENTDC,
        no_style                = 0
    };
    
    namespace keywords {
        BOOST_PARAMETER_NAME(class_name)
        BOOST_PARAMETER_NAME(class_style)
        BOOST_PARAMETER_NAME(icon)
        BOOST_PARAMETER_NAME(cursor)
        BOOST_PARAMETER_NAME(background)
        BOOST_PARAMETER_NAME(menu_resource)
        BOOST_PARAMETER_NAME(small_icon)
    }
};

namespace detail {
    LRESULT CALLBACK dispatch_wndproc(HWND, UINT, WPARAM, LPARAM);
    typedef boost::variant<WORD, std::wstring> resource_id;
    
    struct resource_id_visitor : boost::static_visitor<>, private boost::noncopyable {
        mutable LPCWSTR& resource_id;
        
        resource_id_visitor(LPCWSTR& resource_id) : resource_id(resource_id) {
        }
        
        void operator()(const WORD& int_id) const {
            resource_id = MAKEINTRESOURCE(int_id);
        }
        
        void operator()(const std::wstring& str_id) const {
            resource_id = str_id.c_str();
        }
    };
}

namespace detail {
    struct window_impl;
    
    struct window_class_impl {
        window_class_impl(
            const std::wstring& class_name, window_class::styles class_style, HICON icon, boost::optional<HCURSOR> cursor,
            boost::optional<HBRUSH> background, const detail::resource_id& menu_resource, HICON small_icon
        ) : name(class_name), atom(), class_info()
        {
            SecureZeroMemory(&class_info, sizeof class_info);
            
            HCURSOR use_cursor;
            HBRUSH  use_background;
            
            if (cursor) {
                use_cursor = *cursor;
            } else {
                HANDLE arrow_cursor = LoadImage(
                    class_info.hInstance, MAKEINTRESOURCE(OCR_NORMAL), IMAGE_CURSOR, 0, 0,
                    LR_DEFAULTSIZE | LR_SHARED
                );
                
                if (!arrow_cursor) {
                    SBOK_THROW_WIN(windowing_error);
                }
                
                use_cursor = static_cast<HCURSOR>(arrow_cursor);
            }
            
            if (background) {
                use_background = *background;
            } else {
                use_background = static_cast<HBRUSH>(GetStockObject(BLACK_BRUSH));
            }
            
            class_info.cbSize        = sizeof class_info;
            class_info.style         = static_cast<UINT>(class_style);
            class_info.lpfnWndProc   = detail::dispatch_wndproc;
            class_info.cbClsExtra    = 0;
            class_info.cbWndExtra    = sizeof (window_impl*);
            class_info.hInstance     = GetModuleHandle(nullptr);
            class_info.hIcon         = icon;
            class_info.hCursor       = use_cursor;
            class_info.hbrBackground = use_background;
            class_info.lpszClassName = class_name.c_str();
            class_info.hIconSm       = small_icon;
            
            boost::apply_visitor(detail::resource_id_visitor(class_info.lpszMenuName), menu_resource);
            
            atom = RegisterClassEx(&class_info);
            if (!atom) {
                SBOK_THROW_WIN(windowing_error);
            }
        }
        
        std::wstring get_name() const {
            return name;
        }
    private:
        std::wstring name;
        ATOM atom;
        WNDCLASSEX class_info;
    };
}

namespace window_position {
    struct specific {
        explicit specific(int x = CW_USEDEFAULT, int y = 0) : x(x), y(y) {
        }
        
        int x, y;
    };
    
    struct centered {};
};

namespace window {
    enum extended_styles {
        no_extended_style   = 0,
        accept_files        = WS_EX_ACCEPTFILES,
        is_app_window       = WS_EX_APPWINDOW,
        has_client_edge     = WS_EX_CLIENTEDGE,
        is_composited       = WS_EX_COMPOSITED,
        has_context_help    = WS_EX_CONTEXTHELP,
        is_control_parent   = WS_EX_CONTROLPARENT,
        has_modal_frame     = WS_EX_DLGMODALFRAME,
        is_layered          = WS_EX_LAYERED,
        is_mdi_child        = WS_EX_MDICHILD,
        no_inherited_layout = WS_EX_NOINHERITLAYOUT,
        no_parent_notify    = WS_EX_NOPARENTNOTIFY,
        is_overlapped_ex    = WS_EX_OVERLAPPEDWINDOW,
        has_static_edge     = WS_EX_STATICEDGE,
        is_topmost          = WS_EX_TOPMOST,
        has_window_edge     = WS_EX_WINDOWEDGE
    };
    
    enum styles {
        default_style       = WS_OVERLAPPED,
        has_border          = WS_BORDER,
        has_title_bar       = WS_CAPTION,
        is_child            = WS_CHILD,
        is_disabled         = WS_DISABLED,
        has_dialog_frame    = WS_DLGFRAME,
        has_horiz_scrollbar = WS_HSCROLL,
        create_minimised    = WS_MINIMIZE,
        create_maximised    = WS_MAXIMIZE,
        has_maximise_button = WS_MAXIMIZEBOX,
        has_minimise_button = WS_MINIMIZEBOX,
        is_overlapped       = WS_OVERLAPPEDWINDOW,
        has_sizing_border   = WS_SIZEBOX,
        has_system_menu     = WS_SYSMENU,
        create_visible      = WS_VISIBLE,
        has_vert_scrollbar  = WS_VSCROLL
    };
    
    namespace keywords {
        BOOST_PARAMETER_NAME(window_class)
        BOOST_PARAMETER_NAME(extended_style)
        BOOST_PARAMETER_NAME(name)
        BOOST_PARAMETER_NAME(style)
        BOOST_PARAMETER_NAME(position)
        BOOST_PARAMETER_NAME(width)
        BOOST_PARAMETER_NAME(height)
        BOOST_PARAMETER_NAME(parent)
        BOOST_PARAMETER_NAME(menu)
    }
}

namespace detail {
    typedef boost::variant<window_position::specific, window_position::centered> window_position_variant;
    
    // called to get x, y for CreateWindowEx
    struct position_create_visitor : boost::static_visitor<>, private boost::noncopyable {
        mutable int &x, &y;
        int width, height;
        position_create_visitor(int& x, int& y, int width, int height) : x(x), y(y), width(width), height(height) {}
        
        void operator()(const window_position::specific& pos) const {
            x = pos.x;
            y = pos.y;
        }
        
        void operator()(const window_position::centered&) const {
            RECT rect;
            GetClientRect(GetDesktopWindow(), &rect);
            
            x = (rect.right  / 2) - (width  / 2);
            y = (rect.bottom / 2) - (height / 2);
        }
    };
    
    struct unhandled {};
    
    typedef boost::variant<unhandled, LRESULT> process_result;
    
    // wraps a window
    struct window_impl {
        window_impl(
            const window_class_impl& window_class, window::extended_styles extended_style, const std::wstring& name,
            window::styles style, const window_position_variant& position, int width, int height, HWND parent, HMENU menu
        ) : handle()
        {
            int x, y;
            boost::apply_visitor(position_create_visitor(x, y, width, height), position);
            
            std::wstring class_name = window_class.get_name();
            handle = CreateWindowEx(
                extended_style, class_name.c_str(), name.c_str(), style,
                x, y, width, height, parent, menu, GetModuleHandle(nullptr),
                nullptr
            );
            
            if (handle == INVALID_HANDLE_VALUE) {
                SBOK_THROW_WIN(windowing_error);
            }
            
            SetWindowLongPtr(handle, GWLP_USERDATA, reinterpret_cast<LONG_PTR>(this));
            UpdateWindow(handle);
        }
        
        process_result process_message(UINT, WPARAM, LPARAM) {
            return unhandled();
        }
        
        HWND handle;
    };
}

// factory functions

BOOST_PARAMETER_FUNCTION(
    (detail::window_class_impl), create_window_class, window_class::keywords::tag,
    (required
        (class_name, *)
    )
    (optional
        (class_style,     *, window_class::no_style)
        (icon,            *, nullptr               )
        (cursor,          *, boost::none           )
        (background,      *, boost::none           )
        (menu_resource,   *, L""                   )
        (small_icon,      *, nullptr               )
    )
) {
    return detail::window_class_impl(class_name, class_style, icon, cursor, background, menu_resource, small_icon);
}

BOOST_PARAMETER_FUNCTION(
    (detail::window_impl), create_window, window::keywords::tag,
    (required
        (window_class, *)
    )
    (optional
        (extended_style, *, window::is_overlapped_ex)
        (name,           *, L"")
        (style,          *, static_cast<window::styles>(window::default_style | window::create_visible))
        (position,       *, window_position::centered())
        (width,          *, 800)
        (height,         *, 600)
        (parent,         *, nullptr)
        (menu,           *, nullptr)
    )
) {
    return detail::window_impl(window_class, extended_style, name, style, position, width, height, parent, menu);
}

namespace detail {
    struct dispatch_visitor : boost::static_visitor<LRESULT>, private boost::noncopyable {
        const HWND& handle;
        const UINT& message;
        const WPARAM& w;
        const LPARAM& l;
        
        dispatch_visitor(const HWND& handle, const UINT& message, const WPARAM& w, const LPARAM& l)
        : handle(handle), message(message), w(w), l(l) {}
        
        LRESULT operator()(const LRESULT& result) const {
            return result;
        }
        
        LRESULT operator()(const unhandled&) const {
            return DefWindowProc(handle, message, w, l);
        }
    };
    
    // delegates the message to process_msg member function inside window instance
    LRESULT CALLBACK dispatch_wndproc(HWND handle, UINT message, WPARAM w, LPARAM l) {
        LONG_PTR self = GetWindowLongPtr(handle, GWLP_USERDATA);
        //SBOK_ASSERT(self != 0);
        
        auto real_wndproc = &window_impl::process_message;
        auto real_handle  = reinterpret_cast<window_impl*>(self);
        
        detail::process_result result = (real_handle->*real_wndproc)(message, w, l);
        return boost::apply_visitor(dispatch_visitor(handle, message, w, l), result);
    }
}

int main() {
    using namespace window_class::keywords;
    using namespace window::keywords;
    
    try {
        auto cls = create_window_class(_class_name = L"hello world", _class_style = window_class::no_style);
        auto wnd = create_window(_window_class = cls, _name = L"ohai");
        
        int result = 0;
        MSG msg;
        while ((result = GetMessage(&msg, NULL, 0, 0)) != 0) {
            if (result == -1) {
                SBOK_THROW_WIN(windowing_error);
            } else {
                TranslateMessage(&msg);
                DispatchMessage(&msg);
            }
        }
    } catch (const std::exception& e) {
        std::cerr << boost::diagnostic_information(e) << std::endl;
    }
    
    return 0;
}
