blob: ffa04b958804a5a86dc10bd0cf4c4e0521d47abf [file] [log] [blame]
// Copyright 2011 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 "cc/layer_tiling_data.h"
#include "base/logging.h"
using namespace std;
namespace cc {
scoped_ptr<LayerTilingData> LayerTilingData::create(const gfx::Size& tileSize, BorderTexelOption border)
{
return make_scoped_ptr(new LayerTilingData(tileSize, border));
}
LayerTilingData::LayerTilingData(const gfx::Size& tileSize, BorderTexelOption border)
: m_tilingData(tileSize, gfx::Size(), border == HasBorderTexels)
{
setTileSize(tileSize);
}
LayerTilingData::~LayerTilingData()
{
}
void LayerTilingData::setTileSize(const gfx::Size& size)
{
if (tileSize() == size)
return;
reset();
m_tilingData.SetMaxTextureSize(size);
}
gfx::Size LayerTilingData::tileSize() const
{
return m_tilingData.max_texture_size();
}
void LayerTilingData::setBorderTexelOption(BorderTexelOption borderTexelOption)
{
bool borderTexels = borderTexelOption == HasBorderTexels;
if (hasBorderTexels() == borderTexels)
return;
reset();
m_tilingData.SetHasBorderTexels(borderTexels);
}
const LayerTilingData& LayerTilingData::operator=(const LayerTilingData& tiler)
{
m_tilingData = tiler.m_tilingData;
return *this;
}
void LayerTilingData::addTile(scoped_ptr<Tile> tile, int i, int j)
{
DCHECK(!tileAt(i, j));
tile->moveTo(i, j);
m_tiles.add(make_pair(i, j), tile.Pass());
}
scoped_ptr<LayerTilingData::Tile> LayerTilingData::takeTile(int i, int j)
{
return m_tiles.take_and_erase(make_pair(i, j));
}
LayerTilingData::Tile* LayerTilingData::tileAt(int i, int j) const
{
return m_tiles.get(make_pair(i, j));
}
void LayerTilingData::reset()
{
m_tiles.clear();
}
void LayerTilingData::contentRectToTileIndices(const gfx::Rect& contentRect, int& left, int& top, int& right, int& bottom) const
{
// An empty rect doesn't result in an empty set of tiles, so don't pass an empty rect.
// FIXME: Possibly we should fill a vector of tiles instead,
// since the normal use of this function is to enumerate some tiles.
DCHECK(!contentRect.IsEmpty());
left = m_tilingData.TileXIndexFromSrcCoord(contentRect.x());
top = m_tilingData.TileYIndexFromSrcCoord(contentRect.y());
right = m_tilingData.TileXIndexFromSrcCoord(contentRect.right() - 1);
bottom = m_tilingData.TileYIndexFromSrcCoord(contentRect.bottom() - 1);
}
gfx::Rect LayerTilingData::tileRect(const Tile* tile) const
{
gfx::Rect tileRect = m_tilingData.TileBoundsWithBorder(tile->i(), tile->j());
tileRect.set_size(tileSize());
return tileRect;
}
Region LayerTilingData::opaqueRegionInContentRect(const gfx::Rect& contentRect) const
{
if (contentRect.IsEmpty())
return Region();
Region opaqueRegion;
int left, top, right, bottom;
contentRectToTileIndices(contentRect, left, top, right, bottom);
for (int j = top; j <= bottom; ++j) {
for (int i = left; i <= right; ++i) {
Tile* tile = tileAt(i, j);
if (!tile)
continue;
gfx::Rect tileOpaqueRect = gfx::IntersectRects(contentRect, tile->opaqueRect());
opaqueRegion.Union(tileOpaqueRect);
}
}
return opaqueRegion;
}
void LayerTilingData::setBounds(const gfx::Size& size)
{
m_tilingData.SetTotalSize(size);
if (size.IsEmpty()) {
m_tiles.clear();
return;
}
// Any tiles completely outside our new bounds are invalid and should be dropped.
int left, top, right, bottom;
contentRectToTileIndices(gfx::Rect(gfx::Point(), size), left, top, right, bottom);
std::vector<TileMapKey> invalidTileKeys;
for (TileMap::const_iterator it = m_tiles.begin(); it != m_tiles.end(); ++it) {
if (it->first.first > right || it->first.second > bottom)
invalidTileKeys.push_back(it->first);
}
for (size_t i = 0; i < invalidTileKeys.size(); ++i)
m_tiles.erase(invalidTileKeys[i]);
}
gfx::Size LayerTilingData::bounds() const
{
return m_tilingData.total_size();
}
} // namespace cc