summaryrefslogtreecommitdiffstats
path: root/widget/gtk/WindowSurfaceXRender.cpp
blob: 9f040d9ce3626bfa89230ac8a19d0744130e45fa (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 2 -*-
 *
 * This Source Code Form is subject to the terms of the Mozilla Public
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */

#include "WindowSurfaceXRender.h"

#include "mozilla/gfx/2D.h"
#include "mozilla/gfx/Types.h"
#include "gfxPlatform.h"

namespace mozilla {
namespace widget {

WindowSurfaceXRender::WindowSurfaceXRender(Display* aDisplay, Window aWindow,
                                           Visual* aVisual, unsigned int aDepth)
    : WindowSurfaceX11(aDisplay, aWindow, aVisual, aDepth),
      mXlibSurface(nullptr),
      mGC(X11None) {}

WindowSurfaceXRender::~WindowSurfaceXRender() {
  if (mGC != X11None) {
    XFreeGC(mDisplay, mGC);
  }
}

already_AddRefed<gfx::DrawTarget> WindowSurfaceXRender::Lock(
    const LayoutDeviceIntRegion& aRegion) {
  gfx::IntRect bounds = aRegion.GetBounds().ToUnknownRect();
  gfx::IntSize size(bounds.XMost(), bounds.YMost());
  if (!mXlibSurface || mXlibSurface->CairoStatus() ||
      !(size <= mXlibSurface->GetSize())) {
    mXlibSurface = gfxXlibSurface::Create(DefaultScreenOfDisplay(mDisplay),
                                          mVisual, size, mWindow);
  }
  if (!mXlibSurface || mXlibSurface->CairoStatus()) {
    return nullptr;
  }

  return gfxPlatform::CreateDrawTargetForSurface(mXlibSurface, size);
}

void WindowSurfaceXRender::Commit(const LayoutDeviceIntRegion& aInvalidRegion) {
  AutoTArray<XRectangle, 32> xrects;
  xrects.SetCapacity(aInvalidRegion.GetNumRects());

  for (auto iter = aInvalidRegion.RectIter(); !iter.Done(); iter.Next()) {
    const LayoutDeviceIntRect& r = iter.Get();
    XRectangle xrect = {(short)r.x, (short)r.y, (unsigned short)r.width,
                        (unsigned short)r.height};
    xrects.AppendElement(xrect);
  }

  if (!mGC) {
    mGC = XCreateGC(mDisplay, mWindow, 0, nullptr);
    if (!mGC) {
      NS_WARNING("Couldn't create X11 graphics context for window!");
      return;
    }
  }

  XSetClipRectangles(mDisplay, mGC, 0, 0, xrects.Elements(), xrects.Length(),
                     YXBanded);

  MOZ_ASSERT(mXlibSurface && mXlibSurface->CairoStatus() == 0,
             "Attempted to commit invalid surface!");
  gfx::IntRect bounds = aInvalidRegion.GetBounds().ToUnknownRect();
  gfx::IntSize size(bounds.XMost(), bounds.YMost());
  XCopyArea(mDisplay, mXlibSurface->XDrawable(), mWindow, mGC, bounds.x,
            bounds.y, size.width, size.height, bounds.x, bounds.y);
}

}  // namespace widget
}  // namespace mozilla