mirror of
https://github.com/genodelabs/genode.git
synced 2024-12-31 11:07:02 +00:00
themed_decorator: new 'motion' policy attribute
This commit adds the optional 'motion=<number>' attribute to the decorator's <policy> nodes. The default value is 0. If a value higher than 0 is specified, window-geometry changes are applied as an animation where the <number> denotes the number of animation steps. Issue #3096
This commit is contained in:
parent
a3bbef5f21
commit
2d95c4dc1c
@ -35,19 +35,31 @@ class Decorator::Config
|
|||||||
|
|
||||||
Genode::Xml_node _config;
|
Genode::Xml_node _config;
|
||||||
|
|
||||||
|
template <typename T>
|
||||||
|
T _policy_attribute(Window_title const &title, char const *attr,
|
||||||
|
T default_value) const
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
Genode::Session_policy policy(title, _config);
|
||||||
|
return policy.attribute_value(attr, default_value);
|
||||||
|
|
||||||
|
} catch (Genode::Session_policy::No_policy_defined) { }
|
||||||
|
|
||||||
|
return default_value;
|
||||||
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
Config(Genode::Xml_node node) : _config(node) {}
|
Config(Genode::Xml_node node) : _config(node) {}
|
||||||
|
|
||||||
bool show_decoration(Window_title const &title) const
|
bool show_decoration(Window_title const &title) const
|
||||||
{
|
{
|
||||||
try {
|
return _policy_attribute(title, "decoration", true);
|
||||||
Genode::Session_policy policy(title, _config);
|
}
|
||||||
return policy.attribute_value("decoration", true);
|
|
||||||
|
|
||||||
} catch (Genode::Session_policy::No_policy_defined) { }
|
unsigned motion(Window_title const &title) const
|
||||||
|
{
|
||||||
return true;
|
return _policy_attribute(title, "motion", 0U);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -306,6 +306,7 @@ void Decorator::Main::_handle_nitpicker_sync()
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
_window_stack.update_nitpicker_views();
|
_window_stack.update_nitpicker_views();
|
||||||
|
_nitpicker.execute();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -189,7 +189,7 @@ Decorator::Rect Decorator::Theme::element_geometry(Element_type type) const
|
|||||||
|
|
||||||
void Decorator::Theme::draw_background(Decorator::Pixel_surface &pixel_surface,
|
void Decorator::Theme::draw_background(Decorator::Pixel_surface &pixel_surface,
|
||||||
Decorator::Alpha_surface &alpha_surface,
|
Decorator::Alpha_surface &alpha_surface,
|
||||||
unsigned alpha) const
|
Area const area, unsigned alpha) const
|
||||||
{
|
{
|
||||||
/*
|
/*
|
||||||
* Back out early there is neither a decor nor an aura. In this case, we
|
* Back out early there is neither a decor nor an aura. In this case, we
|
||||||
@ -208,19 +208,19 @@ void Decorator::Theme::draw_background(Decorator::Pixel_surface &pixel_surface,
|
|||||||
unsigned const left = aura_margins().left + decor_margins().left;
|
unsigned const left = aura_margins().left + decor_margins().left;
|
||||||
unsigned const right = aura_margins().right + decor_margins().right;
|
unsigned const right = aura_margins().right + decor_margins().right;
|
||||||
|
|
||||||
unsigned const middle = left + right < pixel_surface.size().w()
|
unsigned const middle = left + right < area.w()
|
||||||
? pixel_surface.size().w() - left - right
|
? area.w() - left - right
|
||||||
: 0;
|
: 0;
|
||||||
|
|
||||||
Rect const orig_clip = pixel_surface.clip();
|
Rect const orig_clip = pixel_surface.clip();
|
||||||
|
|
||||||
/* left */
|
/* left */
|
||||||
if (left) {
|
if (left) {
|
||||||
Rect curr_clip = Rect(Point(0, 0), Area(left, pixel_surface.size().h()));
|
Rect curr_clip = Rect(Point(0, 0), Area(left, area.h()));
|
||||||
pixel_surface.clip(curr_clip);
|
pixel_surface.clip(curr_clip);
|
||||||
alpha_surface.clip(curr_clip);
|
alpha_surface.clip(curr_clip);
|
||||||
|
|
||||||
Rect const rect(Point(0, 0), pixel_surface.size());
|
Rect const rect(Point(0, 0), area);
|
||||||
|
|
||||||
Icon_painter::paint(pixel_surface, rect, texture, alpha);
|
Icon_painter::paint(pixel_surface, rect, texture, alpha);
|
||||||
Icon_painter::paint(alpha_surface, rect, texture, alpha);
|
Icon_painter::paint(alpha_surface, rect, texture, alpha);
|
||||||
@ -228,11 +228,11 @@ void Decorator::Theme::draw_background(Decorator::Pixel_surface &pixel_surface,
|
|||||||
|
|
||||||
/* middle */
|
/* middle */
|
||||||
if (middle) {
|
if (middle) {
|
||||||
Rect curr_clip = Rect(Point(left, 0), Area(middle, pixel_surface.size().h()));
|
Rect curr_clip = Rect(Point(left, 0), Area(middle, area.h()));
|
||||||
pixel_surface.clip(curr_clip);
|
pixel_surface.clip(curr_clip);
|
||||||
alpha_surface.clip(curr_clip);
|
alpha_surface.clip(curr_clip);
|
||||||
|
|
||||||
Rect const rect(Point(0, 0), pixel_surface.size());
|
Rect const rect(Point(0, 0), area);
|
||||||
|
|
||||||
Icon_painter::paint(pixel_surface, rect, texture, alpha);
|
Icon_painter::paint(pixel_surface, rect, texture, alpha);
|
||||||
Icon_painter::paint(alpha_surface, rect, texture, alpha);
|
Icon_painter::paint(alpha_surface, rect, texture, alpha);
|
||||||
@ -240,15 +240,15 @@ void Decorator::Theme::draw_background(Decorator::Pixel_surface &pixel_surface,
|
|||||||
|
|
||||||
/* right */
|
/* right */
|
||||||
if (right) {
|
if (right) {
|
||||||
Rect curr_clip = Rect(Point(left + middle, 0), Area(right, pixel_surface.size().h()));
|
Rect curr_clip = Rect(Point(left + middle, 0), Area(right, area.h()));
|
||||||
pixel_surface.clip(curr_clip);
|
pixel_surface.clip(curr_clip);
|
||||||
alpha_surface.clip(curr_clip);
|
alpha_surface.clip(curr_clip);
|
||||||
|
|
||||||
Point at(0, 0);
|
Point at(0, 0);
|
||||||
Area size = pixel_surface.size();
|
Area size = area;
|
||||||
|
|
||||||
if (texture.size().w() > pixel_surface.size().w()) {
|
if (texture.size().w() > area.w()) {
|
||||||
at = Point((int)pixel_surface.size().w() - (int)texture.size().w(), 0);
|
at = Point((int)area.w() - (int)texture.size().w(), 0);
|
||||||
size = Area(texture.size().w(), size.h());
|
size = Area(texture.size().w(), size.h());
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -262,7 +262,7 @@ void Decorator::Theme::draw_background(Decorator::Pixel_surface &pixel_surface,
|
|||||||
|
|
||||||
void Decorator::Theme::draw_title(Decorator::Pixel_surface &pixel_surface,
|
void Decorator::Theme::draw_title(Decorator::Pixel_surface &pixel_surface,
|
||||||
Decorator::Alpha_surface &,
|
Decorator::Alpha_surface &,
|
||||||
char const *title) const
|
Area const area, char const *title) const
|
||||||
{
|
{
|
||||||
/* skip title drawing if the metadata lacks a title declaration */
|
/* skip title drawing if the metadata lacks a title declaration */
|
||||||
if (!title_geometry().area().valid())
|
if (!title_geometry().area().valid())
|
||||||
@ -272,8 +272,8 @@ void Decorator::Theme::draw_title(Decorator::Pixel_surface &pixel_surface,
|
|||||||
|
|
||||||
Area const label_area(font.string_width(title).decimal(),
|
Area const label_area(font.string_width(title).decimal(),
|
||||||
font.bounding_box().h());
|
font.bounding_box().h());
|
||||||
Rect const surface_rect(Point(0, 0), pixel_surface.size());
|
Rect const target_rect(Point(0, 0), area);
|
||||||
Rect const title_rect = absolute(title_geometry(), surface_rect);
|
Rect const title_rect = absolute(title_geometry(), target_rect);
|
||||||
Point const pos = title_rect.center(label_area) - Point(0, 1);
|
Point const pos = title_rect.center(label_area) - Point(0, 1);
|
||||||
|
|
||||||
Text_painter::paint(pixel_surface, Text_painter::Position(pos.x(), pos.y()),
|
Text_painter::paint(pixel_surface, Text_painter::Position(pos.x(), pos.y()),
|
||||||
@ -283,6 +283,7 @@ void Decorator::Theme::draw_title(Decorator::Pixel_surface &pixel_surface,
|
|||||||
|
|
||||||
void Decorator::Theme::draw_element(Decorator::Pixel_surface &pixel_surface,
|
void Decorator::Theme::draw_element(Decorator::Pixel_surface &pixel_surface,
|
||||||
Decorator::Alpha_surface &alpha_surface,
|
Decorator::Alpha_surface &alpha_surface,
|
||||||
|
Area area,
|
||||||
Element_type element_type,
|
Element_type element_type,
|
||||||
unsigned alpha) const
|
unsigned alpha) const
|
||||||
{
|
{
|
||||||
@ -292,9 +293,9 @@ void Decorator::Theme::draw_element(Decorator::Pixel_surface &pixel_surface,
|
|||||||
Genode::Texture<Pixel_rgb888> const &texture =
|
Genode::Texture<Pixel_rgb888> const &texture =
|
||||||
texture_by_element_type(_ram, _rm, _alloc, element_type);
|
texture_by_element_type(_ram, _rm, _alloc, element_type);
|
||||||
|
|
||||||
Rect const surface_rect(Point(0, 0), pixel_surface.size());
|
Rect const target_rect(Point(0, 0), area);
|
||||||
Rect const element_rect = element_geometry(element_type);
|
Rect const element_rect = element_geometry(element_type);
|
||||||
Point const pos = absolute(element_rect.p1(), surface_rect);
|
Point const pos = absolute(element_rect.p1(), target_rect);
|
||||||
Rect const rect(pos, element_rect.area());
|
Rect const rect(pos, element_rect.area());
|
||||||
|
|
||||||
Icon_painter::paint(pixel_surface, rect, texture, alpha);
|
Icon_painter::paint(pixel_surface, rect, texture, alpha);
|
||||||
|
@ -65,11 +65,11 @@ class Decorator::Theme
|
|||||||
|
|
||||||
Margins decor_margins() const;
|
Margins decor_margins() const;
|
||||||
|
|
||||||
void draw_background(Pixel_surface &, Alpha_surface &, unsigned alpha) const;
|
void draw_background(Pixel_surface &, Alpha_surface &, Area, unsigned alpha) const;
|
||||||
|
|
||||||
void draw_title(Pixel_surface &, Alpha_surface &, char const *title) const;
|
void draw_title(Pixel_surface &, Alpha_surface &, Area, char const *title) const;
|
||||||
|
|
||||||
void draw_element(Pixel_surface &, Alpha_surface &, Element_type,
|
void draw_element(Pixel_surface &, Alpha_surface &, Area, Element_type,
|
||||||
unsigned alpha) const;
|
unsigned alpha) const;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -21,12 +21,9 @@
|
|||||||
#include <base/attached_dataspace.h>
|
#include <base/attached_dataspace.h>
|
||||||
#include <util/reconstructible.h>
|
#include <util/reconstructible.h>
|
||||||
|
|
||||||
/* demo includes */
|
|
||||||
#include <util/lazy_value.h>
|
|
||||||
|
|
||||||
/* gems includes */
|
/* gems includes */
|
||||||
#include <gems/animator.h>
|
|
||||||
#include <gems/nitpicker_buffer.h>
|
#include <gems/nitpicker_buffer.h>
|
||||||
|
#include <gems/animated_geometry.h>
|
||||||
|
|
||||||
/* local includes */
|
/* local includes */
|
||||||
#include "theme.h"
|
#include "theme.h"
|
||||||
@ -241,12 +238,22 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
|
|
||||||
bool _show_decoration = _config.show_decoration(_title);
|
bool _show_decoration = _config.show_decoration(_title);
|
||||||
|
|
||||||
|
unsigned _motion = _config.motion(_title);
|
||||||
|
|
||||||
|
Genode::Animated_rect _animated_rect { _animator };
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Geometry most recently propagated to nitpicker
|
||||||
|
*/
|
||||||
|
Rect _nitpicker_view_rect { };
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Nitpicker session that contains the upper and lower window
|
* Nitpicker session that contains the upper and lower window
|
||||||
* decorations.
|
* decorations.
|
||||||
*/
|
*/
|
||||||
Nitpicker::Connection _nitpicker_top_bottom { _env };
|
Nitpicker::Connection _nitpicker_top_bottom { _env };
|
||||||
Genode::Constructible<Nitpicker_buffer> _buffer_top_bottom { };
|
Genode::Constructible<Nitpicker_buffer> _buffer_top_bottom { };
|
||||||
|
Area _size_top_bottom { };
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Nitpicker session that contains the left and right window
|
* Nitpicker session that contains the left and right window
|
||||||
@ -254,6 +261,21 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
*/
|
*/
|
||||||
Nitpicker::Connection _nitpicker_left_right { _env };
|
Nitpicker::Connection _nitpicker_left_right { _env };
|
||||||
Genode::Constructible<Nitpicker_buffer> _buffer_left_right { };
|
Genode::Constructible<Nitpicker_buffer> _buffer_left_right { };
|
||||||
|
Area _size_left_right { };
|
||||||
|
|
||||||
|
Area _visible_top_bottom_area(Area const inner_size) const
|
||||||
|
{
|
||||||
|
Area const outer_size = _outer_from_inner_size(inner_size);
|
||||||
|
|
||||||
|
return Area(outer_size.w(), _theme.background_size().h());
|
||||||
|
}
|
||||||
|
|
||||||
|
Area _visible_left_right_area(Area const inner_size) const
|
||||||
|
{
|
||||||
|
Area const outer_size = _outer_from_inner_size(inner_size);
|
||||||
|
|
||||||
|
return Area(outer_size.w() - inner_size.w(), outer_size.h());
|
||||||
|
}
|
||||||
|
|
||||||
Nitpicker_view _bottom_view { _nitpicker, _nitpicker_top_bottom },
|
Nitpicker_view _bottom_view { _nitpicker, _nitpicker_top_bottom },
|
||||||
_right_view { _nitpicker, _nitpicker_left_right },
|
_right_view { _nitpicker, _nitpicker_left_right },
|
||||||
@ -262,16 +284,48 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
|
|
||||||
Nitpicker_view _content_view { _nitpicker, (unsigned)id() };
|
Nitpicker_view _content_view { _nitpicker, (unsigned)id() };
|
||||||
|
|
||||||
|
void _repaint_decorations(Nitpicker_buffer &buffer, Area area)
|
||||||
|
{
|
||||||
|
buffer.reset_surface();
|
||||||
|
|
||||||
|
buffer.apply_to_surface([&] (Pixel_surface &pixel,
|
||||||
|
Alpha_surface &alpha) {
|
||||||
|
|
||||||
|
_theme.draw_background(pixel, alpha, area, (int)_alpha);
|
||||||
|
|
||||||
|
_theme.draw_title(pixel, alpha, area, _title.string());
|
||||||
|
|
||||||
|
_for_each_element([&] (Element const &element) {
|
||||||
|
_theme.draw_element(pixel, alpha, area, element.type, element.alpha); });
|
||||||
|
|
||||||
|
Color const tint_color = _color();
|
||||||
|
if (tint_color != Color(0, 0, 0))
|
||||||
|
Tint_painter::paint(pixel, Rect(Point(0, 0), area),
|
||||||
|
tint_color);
|
||||||
|
});
|
||||||
|
|
||||||
|
buffer.flush_surface();
|
||||||
|
|
||||||
|
buffer.nitpicker.framebuffer()->refresh(0, 0, buffer.size().w(), buffer.size().h());
|
||||||
|
}
|
||||||
|
|
||||||
|
void _repaint_decorations()
|
||||||
|
{
|
||||||
|
Area const inner_size = _curr_inner_geometry().area();
|
||||||
|
|
||||||
|
_repaint_decorations(*_buffer_top_bottom, _visible_top_bottom_area(inner_size));
|
||||||
|
_repaint_decorations(*_buffer_left_right, _visible_left_right_area(inner_size));
|
||||||
|
}
|
||||||
|
|
||||||
void _reallocate_nitpicker_buffers()
|
void _reallocate_nitpicker_buffers()
|
||||||
{
|
{
|
||||||
Area const theme_size = _theme.background_size();
|
|
||||||
bool const use_alpha = true;
|
bool const use_alpha = true;
|
||||||
|
|
||||||
Area const inner_size = geometry().area();
|
Area const size_top_bottom = _visible_top_bottom_area(geometry().area());
|
||||||
Area const outer_size = outer_geometry().area();
|
|
||||||
|
|
||||||
Area const size_top_bottom(outer_size.w(),
|
if (size_top_bottom.w() > _size_top_bottom.w()
|
||||||
theme_size.h());
|
|| size_top_bottom.h() > _size_top_bottom.h()
|
||||||
|
|| !_buffer_top_bottom.constructed()) {
|
||||||
|
|
||||||
_nitpicker_top_bottom.buffer(Framebuffer::Mode(size_top_bottom.w(),
|
_nitpicker_top_bottom.buffer(Framebuffer::Mode(size_top_bottom.w(),
|
||||||
size_top_bottom.h(),
|
size_top_bottom.h(),
|
||||||
@ -281,8 +335,14 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
_buffer_top_bottom.construct(_nitpicker_top_bottom, size_top_bottom,
|
_buffer_top_bottom.construct(_nitpicker_top_bottom, size_top_bottom,
|
||||||
_env.ram(), _env.rm());
|
_env.ram(), _env.rm());
|
||||||
|
|
||||||
Area const size_left_right(outer_size.w() - inner_size.w(),
|
_size_top_bottom = size_top_bottom;
|
||||||
outer_size.h());
|
}
|
||||||
|
|
||||||
|
Area const size_left_right = _visible_left_right_area(geometry().area());
|
||||||
|
|
||||||
|
if (size_left_right.w() > _size_left_right.w()
|
||||||
|
|| size_left_right.h() > _size_left_right.h()
|
||||||
|
|| !_buffer_left_right.constructed()) {
|
||||||
|
|
||||||
_nitpicker_left_right.buffer(Framebuffer::Mode(size_left_right.w(),
|
_nitpicker_left_right.buffer(Framebuffer::Mode(size_left_right.w(),
|
||||||
size_left_right.h(),
|
size_left_right.h(),
|
||||||
@ -291,31 +351,9 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
|
|
||||||
_buffer_left_right.construct(_nitpicker_left_right, size_left_right,
|
_buffer_left_right.construct(_nitpicker_left_right, size_left_right,
|
||||||
_env.ram(), _env.rm());
|
_env.ram(), _env.rm());
|
||||||
|
|
||||||
|
_size_left_right = size_left_right;
|
||||||
}
|
}
|
||||||
|
|
||||||
void _repaint_decorations(Nitpicker_buffer &buffer)
|
|
||||||
{
|
|
||||||
buffer.reset_surface();
|
|
||||||
|
|
||||||
buffer.apply_to_surface([&] (Pixel_surface &pixel,
|
|
||||||
Alpha_surface &alpha) {
|
|
||||||
|
|
||||||
_theme.draw_background(pixel, alpha, (int)_alpha);
|
|
||||||
|
|
||||||
_theme.draw_title(pixel, alpha, _title.string());
|
|
||||||
|
|
||||||
_for_each_element([&] (Element const &element) {
|
|
||||||
_theme.draw_element(pixel, alpha, element.type, element.alpha); });
|
|
||||||
|
|
||||||
Color const tint_color = _color();
|
|
||||||
if (tint_color != Color(0, 0, 0))
|
|
||||||
Tint_painter::paint(pixel, Rect(Point(0, 0), pixel.size()),
|
|
||||||
tint_color);
|
|
||||||
});
|
|
||||||
|
|
||||||
buffer.flush_surface();
|
|
||||||
|
|
||||||
buffer.nitpicker.framebuffer()->refresh(0, 0, buffer.size().w(), buffer.size().h());
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void _assign_color(Color color)
|
void _assign_color(Color color)
|
||||||
@ -375,15 +413,28 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
return _show_decoration ? _bottom_view.handle() : _content_view.handle();
|
return _show_decoration ? _bottom_view.handle() : _content_view.handle();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Return current inner geometry
|
||||||
|
*
|
||||||
|
* While the window is in motion, the returned rectangle corresponds to
|
||||||
|
* the intermediate window position and size whereas the 'geometry()'
|
||||||
|
* method returns the final geometry.
|
||||||
|
*/
|
||||||
|
Rect _curr_inner_geometry() const
|
||||||
|
{
|
||||||
|
return (_motion && _animated_rect.initialized())
|
||||||
|
? _animated_rect.rect() : geometry();
|
||||||
|
}
|
||||||
|
|
||||||
Rect _decor_geometry() const
|
Rect _decor_geometry() const
|
||||||
{
|
{
|
||||||
Theme::Margins const decor = _theme.decor_margins();
|
Theme::Margins const decor = _theme.decor_margins();
|
||||||
|
|
||||||
return Rect(geometry().p1() - Point(decor.left, decor.top),
|
return Rect(_curr_inner_geometry().p1() - Point(decor.left, decor.top),
|
||||||
geometry().p2() + Point(decor.right, decor.bottom));
|
_curr_inner_geometry().p2() + Point(decor.right, decor.bottom));
|
||||||
}
|
}
|
||||||
|
|
||||||
Rect outer_geometry() const override
|
Rect _outer_from_inner_geometry(Rect inner) const
|
||||||
{
|
{
|
||||||
Theme::Margins const aura = _theme.aura_margins();
|
Theme::Margins const aura = _theme.aura_margins();
|
||||||
Theme::Margins const decor = _theme.decor_margins();
|
Theme::Margins const decor = _theme.decor_margins();
|
||||||
@ -393,34 +444,47 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
unsigned const top = aura.top + decor.top;
|
unsigned const top = aura.top + decor.top;
|
||||||
unsigned const bottom = aura.bottom + decor.bottom;
|
unsigned const bottom = aura.bottom + decor.bottom;
|
||||||
|
|
||||||
return Rect(geometry().p1() - Point(left, top),
|
return Rect(inner.p1() - Point(left, top),
|
||||||
geometry().p2() + Point(right, bottom));
|
inner.p2() + Point(right, bottom));
|
||||||
}
|
}
|
||||||
|
|
||||||
void border_rects(Rect *top, Rect *left, Rect *right, Rect *bottom) const
|
Area _outer_from_inner_size(Area inner) const
|
||||||
{
|
{
|
||||||
outer_geometry().cut(geometry(), top, left, right, bottom);
|
return _outer_from_inner_geometry(Rect(Point(0, 0), inner)).area();
|
||||||
|
}
|
||||||
|
|
||||||
|
Rect outer_geometry() const override
|
||||||
|
{
|
||||||
|
return _outer_from_inner_geometry(geometry());
|
||||||
}
|
}
|
||||||
|
|
||||||
void update_nitpicker_views() override
|
void update_nitpicker_views() override
|
||||||
{
|
{
|
||||||
if (!_nitpicker_views_up_to_date) {
|
bool const nitpicker_view_rect_up_to_date =
|
||||||
|
_nitpicker_view_rect.p1() == geometry().p1() &&
|
||||||
|
_nitpicker_view_rect.p2() == geometry().p2();
|
||||||
|
|
||||||
|
if (!_nitpicker_views_up_to_date || !nitpicker_view_rect_up_to_date) {
|
||||||
|
|
||||||
Area const theme_size = _theme.background_size();
|
Area const theme_size = _theme.background_size();
|
||||||
|
Rect const inner = _curr_inner_geometry();
|
||||||
|
Rect const outer = _outer_from_inner_geometry(inner);
|
||||||
|
|
||||||
/* update view positions */
|
/* update view positions */
|
||||||
Rect top, left, right, bottom;
|
Rect top, left, right, bottom;
|
||||||
border_rects(&top, &left, &right, &bottom);
|
outer.cut(inner, &top, &left, &right, &bottom);
|
||||||
|
|
||||||
_content_view.place(geometry(), Point(0, 0));
|
_content_view.place(inner, Point(0, 0));
|
||||||
_top_view .place(top, Point(0, 0));
|
_top_view .place(top, Point(0, 0));
|
||||||
_left_view .place(left, Point(0, -top.h()));
|
_left_view .place(left, Point(0, -top.h()));
|
||||||
_right_view .place(right, Point(-right.w(), -top.h()));
|
_right_view .place(right, Point(-right.w(), -top.h()));
|
||||||
_bottom_view .place(bottom, Point(0, -theme_size.h() + bottom.h()));
|
_bottom_view .place(bottom, Point(0, -theme_size.h() + bottom.h()));
|
||||||
|
|
||||||
|
_nitpicker.execute();
|
||||||
|
|
||||||
|
_nitpicker_view_rect = inner;
|
||||||
_nitpicker_views_up_to_date = true;
|
_nitpicker_views_up_to_date = true;
|
||||||
}
|
}
|
||||||
_nitpicker.execute();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void draw(Canvas_base &, Rect, Draw_behind_fn const &) const override { }
|
void draw(Canvas_base &, Rect, Draw_behind_fn const &) const override { }
|
||||||
@ -431,6 +495,7 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
animate();
|
animate();
|
||||||
|
|
||||||
_show_decoration = _config.show_decoration(_title);
|
_show_decoration = _config.show_decoration(_title);
|
||||||
|
_motion = _config.motion(_title);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool update(Xml_node window_node) override
|
bool update(Xml_node window_node) override
|
||||||
@ -439,16 +504,39 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
|
|
||||||
bool trigger_animation = false;
|
bool trigger_animation = false;
|
||||||
|
|
||||||
Rect const old_geometry = geometry();
|
Window_title const title =
|
||||||
|
window_node.attribute_value("title", Window_title("<untitled>"));
|
||||||
|
|
||||||
geometry(rect_attribute(window_node));
|
if (_title != title) {
|
||||||
|
_title = title;
|
||||||
|
trigger_animation = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
_show_decoration = _config.show_decoration(_title);
|
||||||
|
_motion = _config.motion(_title);
|
||||||
|
|
||||||
|
Rect const old_geometry = geometry();
|
||||||
|
Rect const new_geometry = rect_attribute(window_node);
|
||||||
|
|
||||||
|
geometry(new_geometry);
|
||||||
|
|
||||||
|
bool const geometry_changed = (old_geometry.p1() != new_geometry.p1())
|
||||||
|
|| (old_geometry.p2() != new_geometry.p2());
|
||||||
|
|
||||||
|
bool const size_changed = (new_geometry.w() != old_geometry.w()
|
||||||
|
|| new_geometry.h() != old_geometry.h());
|
||||||
|
|
||||||
|
bool const motion_triggered =
|
||||||
|
_motion && (geometry_changed || !_animated_rect.initialized());
|
||||||
|
|
||||||
|
if (motion_triggered)
|
||||||
|
_animated_rect.move_to(new_geometry,
|
||||||
|
Genode::Animated_rect::Steps{_motion});
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Detect position changes
|
* Detect position changes
|
||||||
*/
|
*/
|
||||||
if (geometry().p1() != old_geometry.p1()
|
if (geometry_changed || motion_triggered) {
|
||||||
|| geometry().p2() != old_geometry.p2()) {
|
|
||||||
|
|
||||||
_nitpicker_views_up_to_date = false;
|
_nitpicker_views_up_to_date = false;
|
||||||
updated = true;
|
updated = true;
|
||||||
}
|
}
|
||||||
@ -456,8 +544,7 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
/*
|
/*
|
||||||
* Detect size changes
|
* Detect size changes
|
||||||
*/
|
*/
|
||||||
if (geometry().w() != old_geometry.w()
|
if (size_changed || motion_triggered) {
|
||||||
|| geometry().h() != old_geometry.h()) {
|
|
||||||
|
|
||||||
_reallocate_nitpicker_buffers();
|
_reallocate_nitpicker_buffers();
|
||||||
|
|
||||||
@ -473,16 +560,6 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
trigger_animation = true;
|
trigger_animation = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
Window_title const title =
|
|
||||||
window_node.attribute_value("title", Window_title("<untitled>"));
|
|
||||||
|
|
||||||
if (_title != title) {
|
|
||||||
_title = title;
|
|
||||||
trigger_animation = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
_show_decoration = _config.show_decoration(_title);
|
|
||||||
|
|
||||||
/* update color on title change as the title is used as policy selector */
|
/* update color on title change as the title is used as policy selector */
|
||||||
Color const base_color = _config.base_color(_title);
|
Color const base_color = _config.base_color(_title);
|
||||||
if (_base_color != base_color) {
|
if (_base_color != base_color) {
|
||||||
@ -529,8 +606,8 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
{
|
{
|
||||||
return (_alpha.dst() != (int)_alpha)
|
return (_alpha.dst() != (int)_alpha)
|
||||||
|| _r != _r.dst() || _g != _g.dst() || _b != _b.dst()
|
|| _r != _r.dst() || _g != _g.dst() || _b != _b.dst()
|
||||||
|| _closer.animated() || _maximizer.animated();
|
|| _closer.animated() || _maximizer.animated()
|
||||||
|
|| _animated_rect.animated();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -542,13 +619,13 @@ class Decorator::Window : public Window_base, public Animator::Item
|
|||||||
_r.animate();
|
_r.animate();
|
||||||
_g.animate();
|
_g.animate();
|
||||||
_b.animate();
|
_b.animate();
|
||||||
|
_animated_rect.animate();
|
||||||
|
|
||||||
_for_each_element([&] (Element &element) { element.animate(); });
|
_for_each_element([&] (Element &element) { element.animate(); });
|
||||||
|
|
||||||
Animator::Item::animated(animated());
|
_repaint_decorations();
|
||||||
|
|
||||||
_repaint_decorations(*_buffer_top_bottom);
|
Animator::Item::animated(animated());
|
||||||
_repaint_decorations(*_buffer_left_right);
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user