| // Copyright 2015 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "components/exo/surface.h" |
| |
| #include <utility> |
| |
| #include "base/callback_helpers.h" |
| #include "base/logging.h" |
| #include "base/macros.h" |
| #include "base/trace_event/trace_event.h" |
| #include "base/trace_event/trace_event_argument.h" |
| #include "cc/resources/single_release_callback.h" |
| #include "components/exo/buffer.h" |
| #include "components/exo/surface_delegate.h" |
| #include "components/exo/surface_observer.h" |
| #include "ui/aura/window_delegate.h" |
| #include "ui/aura/window_property.h" |
| #include "ui/base/cursor/cursor.h" |
| #include "ui/base/hit_test.h" |
| #include "ui/compositor/layer.h" |
| #include "ui/gfx/buffer_format_util.h" |
| #include "ui/gfx/gpu_memory_buffer.h" |
| #include "ui/gfx/transform_util.h" |
| |
| DECLARE_WINDOW_PROPERTY_TYPE(exo::Surface*); |
| |
| namespace exo { |
| namespace { |
| |
| // A property key containing the surface that is associated with |
| // window. If unset, no surface is associated with window. |
| DEFINE_WINDOW_PROPERTY_KEY(Surface*, kSurfaceKey, nullptr); |
| |
| // Helper function that returns an iterator to the first entry in |list| |
| // with |key|. |
| template <typename T, typename U> |
| typename T::iterator FindListEntry(T& list, U key) { |
| return std::find_if(list.begin(), list.end(), |
| [key](const typename T::value_type& entry) { |
| return entry.first == key; |
| }); |
| } |
| |
| // Helper function that returns true if |list| contains an entry with |key|. |
| template <typename T, typename U> |
| bool ListContainsEntry(T& list, U key) { |
| return FindListEntry(list, key) != list.end(); |
| } |
| |
| // A window delegate which does nothing. Used to create a window that |
| // is an event target, but do nothing. |
| class EmptyWindowDelegate : public aura::WindowDelegate { |
| public: |
| EmptyWindowDelegate() {} |
| ~EmptyWindowDelegate() override {} |
| |
| // Overridden from aura::WindowDelegate: |
| gfx::Size GetMinimumSize() const override { return gfx::Size(); } |
| gfx::Size GetMaximumSize() const override { return gfx::Size(); } |
| void OnBoundsChanged(const gfx::Rect& old_bounds, |
| const gfx::Rect& new_bounds) override {} |
| gfx::NativeCursor GetCursor(const gfx::Point& point) override { |
| return gfx::kNullCursor; |
| } |
| int GetNonClientComponent(const gfx::Point& point) const override { |
| return HTNOWHERE; |
| } |
| bool ShouldDescendIntoChildForEventHandling( |
| aura::Window* child, |
| const gfx::Point& location) override { |
| return true; |
| } |
| bool CanFocus() override { return true; } |
| void OnCaptureLost() override {} |
| void OnPaint(const ui::PaintContext& context) override {} |
| void OnDeviceScaleFactorChanged(float device_scale_factor) override {} |
| void OnWindowDestroying(aura::Window* window) override {} |
| void OnWindowDestroyed(aura::Window* window) override { delete this; } |
| void OnWindowTargetVisibilityChanged(bool visible) override {} |
| bool HasHitTestMask() const override { return false; } |
| void GetHitTestMask(gfx::Path* mask) const override {} |
| |
| private: |
| DISALLOW_COPY_AND_ASSIGN(EmptyWindowDelegate); |
| }; |
| |
| } // namespace |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // Surface, public: |
| |
| Surface::Surface() |
| : aura::Window(new EmptyWindowDelegate), |
| has_pending_contents_(false), |
| pending_buffer_scale_(1.0f), |
| needs_commit_surface_hierarchy_(false), |
| update_contents_after_successful_compositing_(false), |
| compositor_(nullptr), |
| delegate_(nullptr) { |
| SetType(ui::wm::WINDOW_TYPE_CONTROL); |
| SetName("ExoSurface"); |
| Init(ui::LAYER_SOLID_COLOR); |
| SetProperty(kSurfaceKey, this); |
| set_owned_by_parent(false); |
| } |
| |
| Surface::~Surface() { |
| FOR_EACH_OBSERVER(SurfaceObserver, observers_, OnSurfaceDestroying(this)); |
| |
| layer()->SetShowSolidColorContent(); |
| |
| if (compositor_) |
| compositor_->RemoveObserver(this); |
| |
| // Call pending frame callbacks with a null frame time to indicate that they |
| // have been cancelled. |
| frame_callbacks_.splice(frame_callbacks_.end(), pending_frame_callbacks_); |
| active_frame_callbacks_.splice(active_frame_callbacks_.end(), |
| frame_callbacks_); |
| for (const auto& frame_callback : active_frame_callbacks_) |
| frame_callback.Run(base::TimeTicks()); |
| } |
| |
| // static |
| Surface* Surface::AsSurface(aura::Window* window) { |
| return window->GetProperty(kSurfaceKey); |
| } |
| |
| void Surface::Attach(Buffer* buffer) { |
| TRACE_EVENT1("exo", "Surface::Attach", "buffer", buffer->AsTracedValue()); |
| |
| has_pending_contents_ = true; |
| pending_buffer_ = buffer ? buffer->AsWeakPtr() : base::WeakPtr<Buffer>(); |
| } |
| |
| void Surface::Damage(const gfx::Rect& damage) { |
| TRACE_EVENT1("exo", "Surface::Damage", "damage", damage.ToString()); |
| |
| pending_damage_.Union(damage); |
| } |
| |
| void Surface::RequestFrameCallback(const FrameCallback& callback) { |
| TRACE_EVENT0("exo", "Surface::RequestFrameCallback"); |
| |
| pending_frame_callbacks_.push_back(callback); |
| } |
| |
| void Surface::SetOpaqueRegion(const SkRegion& region) { |
| TRACE_EVENT1("exo", "Surface::SetOpaqueRegion", "region", |
| gfx::SkIRectToRect(region.getBounds()).ToString()); |
| |
| pending_opaque_region_ = region; |
| } |
| |
| void Surface::SetBufferScale(float scale) { |
| TRACE_EVENT1("exo", "Surface::SetBufferScale", "scale", scale); |
| |
| pending_buffer_scale_ = scale; |
| } |
| |
| void Surface::AddSubSurface(Surface* sub_surface) { |
| TRACE_EVENT1("exo", "Surface::AddSubSurface", "sub_surface", |
| sub_surface->AsTracedValue()); |
| |
| DCHECK(!sub_surface->parent()); |
| DCHECK(!sub_surface->IsVisible()); |
| DCHECK(sub_surface->bounds().origin() == gfx::Point()); |
| AddChild(sub_surface); |
| |
| DCHECK(!ListContainsEntry(pending_sub_surfaces_, sub_surface)); |
| pending_sub_surfaces_.push_back(std::make_pair(sub_surface, gfx::Point())); |
| } |
| |
| void Surface::RemoveSubSurface(Surface* sub_surface) { |
| TRACE_EVENT1("exo", "Surface::AddSubSurface", "sub_surface", |
| sub_surface->AsTracedValue()); |
| |
| RemoveChild(sub_surface); |
| |
| DCHECK(ListContainsEntry(pending_sub_surfaces_, sub_surface)); |
| pending_sub_surfaces_.erase( |
| FindListEntry(pending_sub_surfaces_, sub_surface)); |
| } |
| |
| void Surface::SetSubSurfacePosition(Surface* sub_surface, |
| const gfx::Point& position) { |
| TRACE_EVENT2("exo", "Surface::SetSubSurfacePosition", "sub_surface", |
| sub_surface->AsTracedValue(), "position", position.ToString()); |
| |
| auto it = FindListEntry(pending_sub_surfaces_, sub_surface); |
| DCHECK(it != pending_sub_surfaces_.end()); |
| it->second = position; |
| } |
| |
| void Surface::PlaceSubSurfaceAbove(Surface* sub_surface, Surface* reference) { |
| TRACE_EVENT2("exo", "Surface::PlaceSubSurfaceAbove", "sub_surface", |
| sub_surface->AsTracedValue(), "reference", |
| reference->AsTracedValue()); |
| |
| if (sub_surface == reference) { |
| DLOG(WARNING) << "Client tried to place sub-surface above itself"; |
| return; |
| } |
| |
| auto position_it = pending_sub_surfaces_.begin(); |
| if (reference != this) { |
| position_it = FindListEntry(pending_sub_surfaces_, reference); |
| if (position_it == pending_sub_surfaces_.end()) { |
| DLOG(WARNING) << "Client tried to place sub-surface above a reference " |
| "surface that is neither a parent nor a sibling"; |
| return; |
| } |
| |
| // Advance iterator to have |position_it| point to the sibling surface |
| // above |reference|. |
| ++position_it; |
| } |
| |
| DCHECK(ListContainsEntry(pending_sub_surfaces_, sub_surface)); |
| pending_sub_surfaces_.splice( |
| position_it, pending_sub_surfaces_, |
| FindListEntry(pending_sub_surfaces_, sub_surface)); |
| } |
| |
| void Surface::PlaceSubSurfaceBelow(Surface* sub_surface, Surface* sibling) { |
| TRACE_EVENT2("exo", "Surface::PlaceSubSurfaceBelow", "sub_surface", |
| sub_surface->AsTracedValue(), "sibling", |
| sibling->AsTracedValue()); |
| |
| if (sub_surface == sibling) { |
| DLOG(WARNING) << "Client tried to place sub-surface below itself"; |
| return; |
| } |
| |
| auto sibling_it = FindListEntry(pending_sub_surfaces_, sibling); |
| if (sibling_it == pending_sub_surfaces_.end()) { |
| DLOG(WARNING) << "Client tried to place sub-surface below a surface that " |
| "is not a sibling"; |
| return; |
| } |
| |
| DCHECK(ListContainsEntry(pending_sub_surfaces_, sub_surface)); |
| pending_sub_surfaces_.splice( |
| sibling_it, pending_sub_surfaces_, |
| FindListEntry(pending_sub_surfaces_, sub_surface)); |
| } |
| |
| void Surface::Commit() { |
| TRACE_EVENT0("exo", "Surface::Commit"); |
| |
| needs_commit_surface_hierarchy_ = true; |
| |
| if (delegate_) |
| delegate_->OnSurfaceCommit(); |
| else |
| CommitSurfaceHierarchy(); |
| } |
| |
| void Surface::CommitSurfaceHierarchy() { |
| DCHECK(needs_commit_surface_hierarchy_); |
| needs_commit_surface_hierarchy_ = false; |
| |
| // We update contents if Attach() has been called since last commit. |
| if (has_pending_contents_) { |
| has_pending_contents_ = false; |
| |
| current_buffer_ = pending_buffer_; |
| pending_buffer_.reset(); |
| |
| cc::TextureMailbox texture_mailbox; |
| scoped_ptr<cc::SingleReleaseCallback> texture_mailbox_release_callback; |
| if (current_buffer_) { |
| texture_mailbox_release_callback = |
| current_buffer_->ProduceTextureMailbox(&texture_mailbox); |
| } |
| |
| if (texture_mailbox_release_callback) { |
| // Update layer with the new contents. |
| gfx::Size contents_size(gfx::ScaleToFlooredSize( |
| texture_mailbox.size_in_pixels(), 1.0f / pending_buffer_scale_)); |
| layer()->SetTextureMailbox(texture_mailbox, |
| std::move(texture_mailbox_release_callback), |
| contents_size); |
| layer()->SetTextureFlipped(false); |
| layer()->SetBounds(gfx::Rect(layer()->bounds().origin(), contents_size)); |
| layer()->SetFillsBoundsOpaquely(pending_opaque_region_.contains( |
| gfx::RectToSkIRect(gfx::Rect(contents_size)))); |
| } else { |
| // Show solid color content if no buffer is attached or we failed |
| // to produce a texture mailbox for the currently attached buffer. |
| layer()->SetShowSolidColorContent(); |
| layer()->SetColor(SK_ColorBLACK); |
| } |
| |
| // Schedule redraw of the damage region. |
| layer()->SchedulePaint(pending_damage_); |
| pending_damage_ = gfx::Rect(); |
| } |
| |
| ui::Compositor* compositor = layer()->GetCompositor(); |
| if (compositor && !pending_frame_callbacks_.empty()) { |
| // Start observing the compositor for frame callbacks. |
| if (!compositor_) { |
| compositor->AddObserver(this); |
| compositor_ = compositor; |
| } |
| |
| // Move pending frame callbacks to the end of |frame_callbacks_|. |
| frame_callbacks_.splice(frame_callbacks_.end(), pending_frame_callbacks_); |
| } |
| |
| // Synchronize window hierarchy. This will position and update the stacking |
| // order of all sub-surfaces after committing all pending state of sub-surface |
| // descendants. |
| aura::Window* stacking_target = nullptr; |
| for (auto& sub_surface_entry : pending_sub_surfaces_) { |
| Surface* sub_surface = sub_surface_entry.first; |
| |
| // Synchronsouly commit all pending state of the sub-surface and its |
| // decendents. |
| if (sub_surface->needs_commit_surface_hierarchy()) |
| sub_surface->CommitSurfaceHierarchy(); |
| |
| // Enable/disable sub-surface based on if it has contents. |
| if (sub_surface->has_contents()) |
| sub_surface->Show(); |
| else |
| sub_surface->Hide(); |
| |
| // Move sub-surface to its new position in the stack. |
| if (stacking_target) |
| StackChildAbove(sub_surface, stacking_target); |
| |
| // Stack next sub-surface above this sub-surface. |
| stacking_target = sub_surface; |
| |
| // Update sub-surface position relative to surface origin. |
| sub_surface->SetBounds( |
| gfx::Rect(sub_surface_entry.second, sub_surface->layer()->size())); |
| } |
| } |
| |
| gfx::Size Surface::GetPreferredSize() const { |
| return pending_buffer_ ? gfx::ScaleToFlooredSize(pending_buffer_->GetSize(), |
| 1.0f / pending_buffer_scale_) |
| : layer()->size(); |
| } |
| |
| bool Surface::IsSynchronized() const { |
| return delegate_ ? delegate_->IsSurfaceSynchronized() : false; |
| } |
| |
| void Surface::SetSurfaceDelegate(SurfaceDelegate* delegate) { |
| DCHECK(!delegate_ || !delegate); |
| delegate_ = delegate; |
| } |
| |
| bool Surface::HasSurfaceDelegate() const { |
| return !!delegate_; |
| } |
| |
| void Surface::AddSurfaceObserver(SurfaceObserver* observer) { |
| observers_.AddObserver(observer); |
| } |
| |
| void Surface::RemoveSurfaceObserver(SurfaceObserver* observer) { |
| observers_.RemoveObserver(observer); |
| } |
| |
| bool Surface::HasSurfaceObserver(const SurfaceObserver* observer) const { |
| return observers_.HasObserver(observer); |
| } |
| |
| scoped_refptr<base::trace_event::TracedValue> Surface::AsTracedValue() const { |
| scoped_refptr<base::trace_event::TracedValue> value = |
| new base::trace_event::TracedValue; |
| value->SetString("name", layer()->name()); |
| return value; |
| } |
| |
| //////////////////////////////////////////////////////////////////////////////// |
| // ui::CompositorObserver overrides: |
| |
| void Surface::OnCompositingDidCommit(ui::Compositor* compositor) { |
| // Move frame callbacks to the end of |active_frame_callbacks_|. |
| active_frame_callbacks_.splice(active_frame_callbacks_.end(), |
| frame_callbacks_); |
| } |
| |
| void Surface::OnCompositingStarted(ui::Compositor* compositor, |
| base::TimeTicks start_time) { |
| // Run all frame callbacks associated with the compositor's active tree. |
| while (!active_frame_callbacks_.empty()) { |
| active_frame_callbacks_.front().Run(start_time); |
| active_frame_callbacks_.pop_front(); |
| } |
| } |
| |
| void Surface::OnCompositingEnded(ui::Compositor* compositor) { |
| // Nothing to do in here unless this has been set. |
| if (!update_contents_after_successful_compositing_) |
| return; |
| |
| update_contents_after_successful_compositing_ = false; |
| |
| // Early out if no contents is currently assigned to the surface. |
| if (!current_buffer_) |
| return; |
| |
| // Update contents by producing a new texture mailbox for the current buffer. |
| cc::TextureMailbox texture_mailbox; |
| scoped_ptr<cc::SingleReleaseCallback> texture_mailbox_release_callback = |
| current_buffer_->ProduceTextureMailbox(&texture_mailbox); |
| if (texture_mailbox_release_callback) { |
| layer()->SetTextureMailbox(texture_mailbox, |
| std::move(texture_mailbox_release_callback), |
| texture_mailbox.size_in_pixels()); |
| layer()->SetTextureFlipped(false); |
| layer()->SchedulePaint(gfx::Rect(texture_mailbox.size_in_pixels())); |
| } |
| } |
| |
| void Surface::OnCompositingAborted(ui::Compositor* compositor) { |
| // The contents of this surface might be lost if compositing aborted because |
| // of a lost graphics context. We recover from this by updating the contents |
| // of the surface next time the compositor successfully ends compositing. |
| update_contents_after_successful_compositing_ = true; |
| } |
| |
| void Surface::OnCompositingShuttingDown(ui::Compositor* compositor) { |
| compositor->RemoveObserver(this); |
| compositor_ = nullptr; |
| } |
| |
| } // namespace exo |