// Copyright 2017 Google Inc. All Rights Reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
//     http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

#ifndef COBALT_RENDERER_RASTERIZER_EGL_DRAW_RECT_SHADOW_BLUR_H_
#define COBALT_RENDERER_RASTERIZER_EGL_DRAW_RECT_SHADOW_BLUR_H_

#include <vector>

#include "cobalt/math/rect_f.h"
#include "cobalt/render_tree/color_rgba.h"
#include "cobalt/renderer/rasterizer/egl/draw_object.h"
#include "egl/generated_shader_impl.h"

namespace cobalt {
namespace renderer {
namespace rasterizer {
namespace egl {

// Example CSS box shadow (outset):
//   +-------------------------------------+
//   | Box shadow "blur" region            |
//   |   +-----------------------------+   |
//   |   | Box shadow "spread" region  |   |
//   |   |   +---------------------+   |   |
//   |   |   | Box shadow rect     |   |   |
//   |   |   | (exclude geometry)  |   |   |
//   |   |   +---------------------+   |   |
//   |   |                             |   |
//   |   +-----------------------------+   |
//   | (include scissor)                   |
//   +-------------------------------------+
// NOTE: Despite the CSS naming, the actual blur effect starts inside the
// "spread" region.

// Handles drawing a box shadow with blur. This uses a gaussian kernel to fade
// the "blur" region.
class DrawRectShadowBlur : public DrawObject {
 public:
  // Draw a blurred box shadow.
  // The box shadow exists in the area between |base_rect| and |spread_rect|
  // extended (inset or outset accordingly) to cover the blur kernel.
  DrawRectShadowBlur(GraphicsState* graphics_state,
                     const BaseState& base_state,
                     const math::RectF& base_rect,
                     const OptionalRoundedCorners& base_corners,
                     const math::RectF& spread_rect,
                     const OptionalRoundedCorners& spread_corners,
                     const render_tree::ColorRGBA& color,
                     float blur_sigma, bool inset);

  void ExecuteUpdateVertexBuffer(GraphicsState* graphics_state,
      ShaderProgramManager* program_manager) OVERRIDE;
  void ExecuteRasterize(GraphicsState* graphics_state,
      ShaderProgramManager* program_manager) OVERRIDE;
  base::TypeId GetTypeId() const OVERRIDE;

 private:
  struct VertexAttributesSquare {
    VertexAttributesSquare(float x, float y, float offset_scale);
    float position[2];
    float offset[2];
  };

  struct VertexAttributesRound {
    VertexAttributesRound(float x, float y, const RCorner& init);
    float position[2];
    RCorner rcorner_scissor;
  };

  void SetupVertexShader(GraphicsState* graphics_state,
                         const ShaderVertexOffset& shader);
  void SetupVertexShader(GraphicsState* graphics_state,
                         const ShaderVertexOffsetRcorner& shader);
  void SetFragmentUniforms(GLint color_uniform, GLint scale_add_uniform);

  void SetGeometry(GraphicsState* graphics_state,
                   const math::RectF& base_rect,
                   const OptionalRoundedCorners& base_corners);
  void SetGeometry(GraphicsState* graphics_state,
                   const math::RectF& inner_rect,
                   const math::RectF& outer_rect);
  void SetGeometry(GraphicsState* graphics_state,
                   const RRectAttributes (&rrect)[8]);
  void SetGeometry(GraphicsState* graphics_state,
                   const RRectAttributes (&rrect_outer)[4],
                   const RRectAttributes (&rrect_inner)[8]);

  math::RectF spread_rect_;
  OptionalRoundedCorners spread_corners_;
  render_tree::ColorRGBA color_;
  float blur_sigma_;
  bool is_inset_;

  std::vector<VertexAttributesSquare> attributes_square_;
  std::vector<VertexAttributesRound> attributes_round_;
  std::vector<uint16_t> indices_;

  uint8_t* vertex_buffer_;
  uint16_t* index_buffer_;
};

}  // namespace egl
}  // namespace rasterizer
}  // namespace renderer
}  // namespace cobalt

#endif  // COBALT_RENDERER_RASTERIZER_EGL_DRAW_RECT_SHADOW_BLUR_H_
