diff --git a/Base/res/cursors/arrow.png b/Base/res/cursors/arrow.png new file mode 100644 index 00000000000..427b92027a9 Binary files /dev/null and b/Base/res/cursors/arrow.png differ diff --git a/Base/res/cursors/disallowed.png b/Base/res/cursors/disallowed.png new file mode 100644 index 00000000000..5d7d5f26cd4 Binary files /dev/null and b/Base/res/cursors/disallowed.png differ diff --git a/Base/res/cursors/i-beam.png b/Base/res/cursors/i-beam.png new file mode 100644 index 00000000000..febaa9d0a89 Binary files /dev/null and b/Base/res/cursors/i-beam.png differ diff --git a/Base/res/cursors/resize-diagonal-bltr.png b/Base/res/cursors/resize-diagonal-bltr.png new file mode 100644 index 00000000000..9a6fc76681f Binary files /dev/null and b/Base/res/cursors/resize-diagonal-bltr.png differ diff --git a/Base/res/cursors/resize-diagonal-tlbr.png b/Base/res/cursors/resize-diagonal-tlbr.png new file mode 100644 index 00000000000..2c51df5bfb7 Binary files /dev/null and b/Base/res/cursors/resize-diagonal-tlbr.png differ diff --git a/Base/res/cursors/resize-horizontal.png b/Base/res/cursors/resize-horizontal.png new file mode 100644 index 00000000000..e6b714f736c Binary files /dev/null and b/Base/res/cursors/resize-horizontal.png differ diff --git a/Base/res/cursors/resize-vertical.png b/Base/res/cursors/resize-vertical.png new file mode 100644 index 00000000000..a0b493ce8ce Binary files /dev/null and b/Base/res/cursors/resize-vertical.png differ diff --git a/Servers/WindowServer/Makefile b/Servers/WindowServer/Makefile index 0b8dcafbf28..3590168ae4a 100644 --- a/Servers/WindowServer/Makefile +++ b/Servers/WindowServer/Makefile @@ -21,12 +21,13 @@ WINDOWSERVER_OBJS = \ WSClientConnection.o \ WSWindowSwitcher.o \ WSClipboard.o \ + WSCursor.o \ main.o APP = WindowServer OBJS = $(SHAREDGRAPHICS_OBJS) $(WINDOWSERVER_OBJS) -STANDARD_FLAGS = -std=c++17 +STANDARD_FLAGS = -std=c++17 -Wno-sized-deallocation WARNING_FLAGS = -Wextra -Wall -Wundef -Wcast-qual -Wwrite-strings -Wimplicit-fallthrough FLAVOR_FLAGS = -fno-exceptions -fno-rtti OPTIMIZATION_FLAGS = -Os diff --git a/Servers/WindowServer/WSCursor.cpp b/Servers/WindowServer/WSCursor.cpp new file mode 100644 index 00000000000..12261a5f540 --- /dev/null +++ b/Servers/WindowServer/WSCursor.cpp @@ -0,0 +1,21 @@ +#include + +WSCursor::WSCursor(Retained&& bitmap, const Point& hotspot) + : m_bitmap(move(bitmap)) + , m_hotspot(hotspot) +{ +} + +WSCursor::~WSCursor() +{ +} + +Retained WSCursor::create(Retained&& bitmap) +{ + return adopt(*new WSCursor(move(bitmap), bitmap->rect().center())); +} + +Retained WSCursor::create(Retained&& bitmap, const Point& hotspot) +{ + return adopt(*new WSCursor(move(bitmap), hotspot)); +} diff --git a/Servers/WindowServer/WSCursor.h b/Servers/WindowServer/WSCursor.h new file mode 100644 index 00000000000..7125ad2a782 --- /dev/null +++ b/Servers/WindowServer/WSCursor.h @@ -0,0 +1,22 @@ +#pragma once + +#include + +class WSCursor : public Retainable { +public: + static Retained create(Retained&&, const Point& hotspot); + static Retained create(Retained&&); + ~WSCursor(); + + Point hotspot() const { return m_hotspot; } + const GraphicsBitmap& bitmap() const { return *m_bitmap; } + + Rect rect() const { return m_bitmap->rect(); } + Size size() const { return m_bitmap->size(); } + +private: + WSCursor(Retained&&, const Point&); + + RetainPtr m_bitmap; + Point m_hotspot; +}; diff --git a/Servers/WindowServer/WSWindowManager.cpp b/Servers/WindowServer/WSWindowManager.cpp index 2de9c04b981..23a8503115e 100644 --- a/Servers/WindowServer/WSWindowManager.cpp +++ b/Servers/WindowServer/WSWindowManager.cpp @@ -16,6 +16,7 @@ #include #include #include +#include "WSCursor.h" #ifdef KERNEL #include @@ -199,8 +200,13 @@ WSWindowManager::WSWindowManager() m_highlight_window_border_color2 = Color::from_rgb(0xfabbbb); m_highlight_window_title_color = Color::White; - m_cursor_bitmap_inner = CharacterBitmap::create_from_ascii(cursor_bitmap_inner_ascii, 12, 17); - m_cursor_bitmap_outer = CharacterBitmap::create_from_ascii(cursor_bitmap_outer_ascii, 12, 17); + m_arrow_cursor = WSCursor::create(*GraphicsBitmap::load_from_file("/res/cursors/arrow.png"), { 2, 2 }); + m_resize_horizontally_cursor = WSCursor::create(*GraphicsBitmap::load_from_file("/res/cursors/resize-horizontal.png")); + m_resize_vertically_cursor = WSCursor::create(*GraphicsBitmap::load_from_file("/res/cursors/resize-vertical.png")); + m_resize_diagonally_tlbr_cursor = WSCursor::create(*GraphicsBitmap::load_from_file("/res/cursors/resize-diagonal-tlbr.png")); + m_resize_diagonally_bltr_cursor = WSCursor::create(*GraphicsBitmap::load_from_file("/res/cursors/resize-diagonal-bltr.png")); + m_i_beam_cursor = WSCursor::create(*GraphicsBitmap::load_from_file("/res/cursors/i-beam.png")); + m_disallowed_cursor = WSCursor::create(*GraphicsBitmap::load_from_file("/res/cursors/disallowed.png")); m_wallpaper_path = "/res/wallpapers/retro.rgb"; m_wallpaper = GraphicsBitmap::load_from_file(GraphicsBitmap::Format::RGBA32, m_wallpaper_path, { 1024, 768 }); @@ -880,9 +886,8 @@ void WSWindowManager::process_mouse_event(WSMouseEvent& event, WSWindow*& event_ void WSWindowManager::compose() { auto dirty_rects = move(m_dirty_rects); - auto cursor_location = m_screen.cursor_location(); dirty_rects.add(m_last_cursor_rect); - dirty_rects.add({ cursor_location.x(), cursor_location.y(), (int)m_cursor_bitmap_inner->width(), (int)m_cursor_bitmap_inner->height() }); + dirty_rects.add(current_cursor_rect()); #ifdef DEBUG_COUNTERS dbgprintf("[WM] compose #%u (%u rects)\n", ++m_compose_count, dirty_rects.rects().size()); #endif @@ -988,11 +993,14 @@ void WSWindowManager::compose() flush(r); } +Rect WSWindowManager::current_cursor_rect() const +{ + return { m_screen.cursor_location().translated(-active_cursor().hotspot()), active_cursor().size() }; +} + void WSWindowManager::invalidate_cursor() { - auto cursor_location = m_screen.cursor_location(); - Rect cursor_rect { cursor_location.x(), cursor_location.y(), (int)m_cursor_bitmap_inner->width(), (int)m_cursor_bitmap_inner->height() }; - invalidate(cursor_rect); + invalidate(current_cursor_rect()); } Rect WSWindowManager::menubar_rect() const @@ -1073,14 +1081,12 @@ void WSWindowManager::draw_window_switcher() void WSWindowManager::draw_cursor() { - auto cursor_location = m_screen.cursor_location(); - Rect cursor_rect { cursor_location.x(), cursor_location.y(), (int)m_cursor_bitmap_inner->width(), (int)m_cursor_bitmap_inner->height() }; + Rect cursor_rect = current_cursor_rect(); Color inner_color = Color::White; Color outer_color = Color::Black; if (m_screen.mouse_button_state() & (unsigned)MouseButton::Left) swap(inner_color, outer_color); - m_back_painter->draw_bitmap(cursor_location, *m_cursor_bitmap_inner, inner_color); - m_back_painter->draw_bitmap(cursor_location, *m_cursor_bitmap_outer, outer_color); + m_back_painter->blit(cursor_rect.location(), active_cursor().bitmap(), active_cursor().rect()); m_last_cursor_rect = cursor_rect; } diff --git a/Servers/WindowServer/WSWindowManager.h b/Servers/WindowServer/WSWindowManager.h index 62e5a2eb508..751ad94f442 100644 --- a/Servers/WindowServer/WSWindowManager.h +++ b/Servers/WindowServer/WSWindowManager.h @@ -13,6 +13,7 @@ #include #include #include +#include #include class WSAPIClientRequest; @@ -23,7 +24,6 @@ class WSClientWantsToPaintMessage; class WSWindow; class WSClientConnection; class WSWindowSwitcher; -class CharacterBitmap; class GraphicsBitmap; enum class ResizeDirection { None, Left, UpLeft, Up, UpRight, Right, DownRight, Down, DownLeft }; @@ -84,6 +84,9 @@ public: bool set_wallpaper(const String& path); String wallpaper_path() const { return m_wallpaper_path; } + const WSCursor& active_cursor() const { return *m_arrow_cursor; } + Rect current_cursor_rect() const; + private: void process_mouse_event(WSMouseEvent&, WSWindow*& event_window); bool process_ongoing_window_resize(WSMouseEvent&, WSWindow*& event_window); @@ -154,8 +157,13 @@ private: bool m_pending_compose_event { false }; - RetainPtr m_cursor_bitmap_inner; - RetainPtr m_cursor_bitmap_outer; + RetainPtr m_arrow_cursor; + RetainPtr m_resize_horizontally_cursor; + RetainPtr m_resize_vertically_cursor; + RetainPtr m_resize_diagonally_tlbr_cursor; + RetainPtr m_resize_diagonally_bltr_cursor; + RetainPtr m_i_beam_cursor; + RetainPtr m_disallowed_cursor; OwnPtr m_back_painter; OwnPtr m_front_painter; diff --git a/SharedGraphics/Point.h b/SharedGraphics/Point.h index 5a00512b7d3..87680ce22f1 100644 --- a/SharedGraphics/Point.h +++ b/SharedGraphics/Point.h @@ -28,6 +28,13 @@ public: move_by(delta.x(), delta.y()); } + Point translated(const Point& delta) const + { + Point point = *this; + point.move_by(delta); + return point; + } + Point translated(int dx, int dy) const { Point point = *this;