summaryrefslogtreecommitdiff
path: root/Libraries/LibGUI
diff options
context:
space:
mode:
authorAndreas Kling <kling@serenityos.org>2020-12-31 11:56:19 +0100
committerAndreas Kling <kling@serenityos.org>2020-12-31 12:10:01 +0100
commitddaa13577ec264dc08b6ef6446e5ab7cc01b3251 (patch)
tree617f11ee99d6d7a0c354e9b84599b0e63e1fd648 /Libraries/LibGUI
parentacb87cd164f3b7fc242eafef21a7773c48cab861 (diff)
downloadserenity-ddaa13577ec264dc08b6ef6446e5ab7cc01b3251.zip
LibGUI: Add a new GUI::OpacitySlider widget :^)
This widgets offers a more visually intuitive way to adjust the opacity of something.
Diffstat (limited to 'Libraries/LibGUI')
-rw-r--r--Libraries/LibGUI/CMakeLists.txt1
-rw-r--r--Libraries/LibGUI/Forward.h1
-rw-r--r--Libraries/LibGUI/OpacitySlider.cpp163
-rw-r--r--Libraries/LibGUI/OpacitySlider.h56
4 files changed, 221 insertions, 0 deletions
diff --git a/Libraries/LibGUI/CMakeLists.txt b/Libraries/LibGUI/CMakeLists.txt
index 2d2da72abc..e936af38db 100644
--- a/Libraries/LibGUI/CMakeLists.txt
+++ b/Libraries/LibGUI/CMakeLists.txt
@@ -63,6 +63,7 @@ set(SOURCES
ModelSelection.cpp
MultiView.cpp
Notification.cpp
+ OpacitySlider.cpp
Painter.cpp
ProcessChooser.cpp
ProgressBar.cpp
diff --git a/Libraries/LibGUI/Forward.h b/Libraries/LibGUI/Forward.h
index 0e771b7b38..2631a7e658 100644
--- a/Libraries/LibGUI/Forward.h
+++ b/Libraries/LibGUI/Forward.h
@@ -64,6 +64,7 @@ class ModelIndex;
class MouseEvent;
class MultiPaintEvent;
class MultiView;
+class OpacitySlider;
class PaintEvent;
class Painter;
class ResizeCorner;
diff --git a/Libraries/LibGUI/OpacitySlider.cpp b/Libraries/LibGUI/OpacitySlider.cpp
new file mode 100644
index 0000000000..3bfb7e0938
--- /dev/null
+++ b/Libraries/LibGUI/OpacitySlider.cpp
@@ -0,0 +1,163 @@
+/*
+ * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright notice, this
+ * list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright notice,
+ * this list of conditions and the following disclaimer in the documentation
+ * and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+ * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+ * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
+ * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#include <LibGUI/OpacitySlider.h>
+#include <LibGUI/Painter.h>
+#include <LibGfx/Palette.h>
+#include <LibGfx/StylePainter.h>
+
+namespace GUI {
+
+OpacitySlider::OpacitySlider(Gfx::Orientation orientation)
+ : AbstractSlider(orientation)
+{
+ // FIXME: Implement vertical mode.
+ ASSERT(orientation == Gfx::Orientation::Horizontal);
+
+ set_min(0);
+ set_max(100);
+ set_value(100);
+ set_fixed_height(20);
+}
+
+OpacitySlider::~OpacitySlider()
+{
+}
+
+Gfx::IntRect OpacitySlider::frame_inner_rect() const
+{
+ return rect().shrunken(4, 4);
+}
+
+void OpacitySlider::paint_event(PaintEvent& event)
+{
+ GUI::Painter painter(*this);
+ painter.add_clip_rect(event.rect());
+
+ auto inner_rect = frame_inner_rect();
+
+ // Grid pattern
+ Gfx::StylePainter::paint_transparency_grid(painter, inner_rect, palette());
+
+ // Alpha gradient
+ for (int x = inner_rect.left(); x <= inner_rect.right(); ++x) {
+ float relative_offset = (float)x / (float)width();
+ float alpha = relative_offset * 255.0f;
+ Color color { 0, 0, 0, (u8)alpha };
+ painter.fill_rect({ x, inner_rect.y(), 1, inner_rect.height() }, color);
+ }
+
+ constexpr int notch_size = 3;
+ int notch_y_top = inner_rect.top() + notch_size;
+ int notch_y_bottom = inner_rect.bottom() - notch_size;
+ int notch_x = inner_rect.left() + ((float)value() / (float)max() * (float)inner_rect.width());
+
+ // Top notch
+ painter.set_pixel(notch_x, notch_y_top, palette().threed_shadow2());
+ for (int i = notch_size; i >= 0; --i) {
+ painter.set_pixel(notch_x - (i + 1), notch_y_top - i - 1, palette().threed_highlight());
+ for (int j = 0; j < i * 2; ++j) {
+ painter.set_pixel(notch_x - (i + 1) + j + 1, notch_y_top - i - 1, palette().button());
+ }
+ painter.set_pixel(notch_x + (i + 0), notch_y_top - i - 1, palette().threed_shadow1());
+ painter.set_pixel(notch_x + (i + 1), notch_y_top - i - 1, palette().threed_shadow2());
+ }
+
+ // Bottom notch
+ painter.set_pixel(notch_x, notch_y_bottom, palette().threed_shadow2());
+ for (int i = 0; i < notch_size; ++i) {
+ painter.set_pixel(notch_x - (i + 1), notch_y_bottom + i + 1, palette().threed_highlight());
+ for (int j = 0; j < i * 2; ++j) {
+ painter.set_pixel(notch_x - (i + 1) + j + 1, notch_y_bottom + i + 1, palette().button());
+ }
+ painter.set_pixel(notch_x + (i + 0), notch_y_bottom + i + 1, palette().threed_shadow1());
+ painter.set_pixel(notch_x + (i + 1), notch_y_bottom + i + 1, palette().threed_shadow2());
+ }
+
+ // Hairline
+ // NOTE: If we're in the whiter part of the gradient, the notch is painted as shadow between the notches.
+ // If we're in the darker part, the notch is painted as highlight.
+ // We adjust the hairline's x position so it lines up with the shadow/highlight of the notches.
+ u8 h = ((float)value() / (float)max()) * 255.0f;
+ if (h < 128)
+ painter.draw_line({ notch_x, notch_y_top }, { notch_x, notch_y_bottom }, Color(h, h, h, h));
+ else
+ painter.draw_line({ notch_x - 1, notch_y_top }, { notch_x - 1, notch_y_bottom }, Color(h, h, h, h));
+
+ // Text label
+ auto percent_text = String::formatted("{}%", (int)((float)value() / (float)max() * 100.0f));
+ painter.draw_text(inner_rect.translated(1, 1), percent_text, Gfx::TextAlignment::Center, Color::Black);
+ painter.draw_text(inner_rect, percent_text, Gfx::TextAlignment::Center, Color::White);
+
+ // Frame
+ Gfx::StylePainter::paint_frame(painter, rect(), palette(), Gfx::FrameShape::Container, Gfx::FrameShadow::Sunken, 2);
+}
+
+int OpacitySlider::value_at(const Gfx::IntPoint& position) const
+{
+ auto inner_rect = frame_inner_rect();
+ if (position.x() < inner_rect.left())
+ return min();
+ if (position.x() > inner_rect.right())
+ return max();
+ float relative_offset = (float)(position.x() - inner_rect.x()) / (float)inner_rect.width();
+ return relative_offset * (float)max();
+}
+
+void OpacitySlider::mousedown_event(MouseEvent& event)
+{
+ if (event.button() == MouseButton::Left) {
+ m_dragging = true;
+ set_value(value_at(event.position()));
+ return;
+ }
+ AbstractSlider::mousedown_event(event);
+}
+
+void OpacitySlider::mousemove_event(MouseEvent& event)
+{
+ if (m_dragging) {
+ set_value(value_at(event.position()));
+ return;
+ }
+ AbstractSlider::mousemove_event(event);
+}
+
+void OpacitySlider::mouseup_event(MouseEvent& event)
+{
+ if (event.button() == MouseButton::Left) {
+ m_dragging = false;
+ return;
+ }
+ AbstractSlider::mouseup_event(event);
+}
+
+void OpacitySlider::mousewheel_event(MouseEvent& event)
+{
+ set_value(value() - event.wheel_delta());
+}
+
+}
diff --git a/Libraries/LibGUI/OpacitySlider.h b/Libraries/LibGUI/OpacitySlider.h
new file mode 100644
index 0000000000..7f61018f0a
--- /dev/null
+++ b/Libraries/LibGUI/OpacitySlider.h
@@ -0,0 +1,56 @@
+/*
+ * Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
+ * All rights reserved.
+ *
+ * Redistribution and use in source and binary forms, with or without
+ * modification, are permitted provided that the following conditions are met:
+ *
+ * 1. Redistributions of source code must retain the above copyright notice, this
+ * list of conditions and the following disclaimer.
+ *
+ * 2. Redistributions in binary form must reproduce the above copyright notice,
+ * this list of conditions and the following disclaimer in the documentation
+ * and/or other materials provided with the distribution.
+ *
+ * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
+ * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
+ * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
+ * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
+ * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
+ * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
+ * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
+ * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
+ * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
+ * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
+ */
+
+#pragma once
+
+#include <LibGUI/AbstractSlider.h>
+
+namespace GUI {
+
+class OpacitySlider : public AbstractSlider {
+ C_OBJECT(OpacitySlider);
+
+public:
+ virtual ~OpacitySlider() override;
+
+protected:
+ virtual void paint_event(PaintEvent&) override;
+ virtual void mousedown_event(MouseEvent&) override;
+ virtual void mousemove_event(MouseEvent&) override;
+ virtual void mouseup_event(MouseEvent&) override;
+ virtual void mousewheel_event(MouseEvent&) override;
+
+private:
+ explicit OpacitySlider(Gfx::Orientation = Gfx::Orientation::Horizontal);
+
+ Gfx::IntRect frame_inner_rect() const;
+
+ int value_at(const Gfx::IntPoint&) const;
+
+ bool m_dragging { false };
+};
+
+}