blob: d539457e8af59f9689a22a22508e09312cb628fa [file] [log] [blame]
// Copyright 2016 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 "core/css/cssom/CSSSkew.h"
#include "bindings/core/v8/ExceptionState.h"
#include "core/css/CSSFunctionValue.h"
#include "core/css/CSSPrimitiveValue.h"
#include "core/css/cssom/CSSNumericValue.h"
#include "core/css/cssom/CSSStyleValue.h"
#include "core/css/cssom/CSSUnitValue.h"
#include "core/geometry/DOMMatrix.h"
namespace blink {
namespace {
bool IsValidSkewAngle(CSSNumericValue* value) {
return value &&
value->Type().MatchesBaseType(CSSNumericValueType::BaseType::kAngle);
}
} // namespace
CSSSkew* CSSSkew::Create(CSSNumericValue* ax,
CSSNumericValue* ay,
ExceptionState& exception_state) {
if (!IsValidSkewAngle(ax) || !IsValidSkewAngle(ay)) {
exception_state.ThrowTypeError("CSSSkew does not support non-angles");
return nullptr;
}
return new CSSSkew(ax, ay);
}
void CSSSkew::setAx(CSSNumericValue* value, ExceptionState& exception_state) {
if (!IsValidSkewAngle(value)) {
exception_state.ThrowTypeError("Must specify an angle unit");
return;
}
ax_ = value;
}
void CSSSkew::setAy(CSSNumericValue* value, ExceptionState& exception_state) {
if (!IsValidSkewAngle(value)) {
exception_state.ThrowTypeError("Must specify an angle unit");
return;
}
ay_ = value;
}
CSSSkew* CSSSkew::FromCSSValue(const CSSFunctionValue& value) {
DCHECK_GT(value.length(), 0U);
const CSSPrimitiveValue& x_value = ToCSSPrimitiveValue(value.Item(0));
switch (value.FunctionType()) {
case CSSValueSkew:
if (value.length() == 2U) {
const CSSPrimitiveValue& y_value = ToCSSPrimitiveValue(value.Item(1));
return CSSSkew::Create(CSSNumericValue::FromCSSValue(x_value),
CSSNumericValue::FromCSSValue(y_value));
}
// Else fall through; skew(ax) == skewX(ax).
case CSSValueSkewX:
DCHECK_EQ(value.length(), 1U);
return CSSSkew::Create(
CSSNumericValue::FromCSSValue(x_value),
CSSUnitValue::Create(0, CSSPrimitiveValue::UnitType::kDegrees));
case CSSValueSkewY:
DCHECK_EQ(value.length(), 1U);
return CSSSkew::Create(
CSSUnitValue::Create(0, CSSPrimitiveValue::UnitType::kDegrees),
CSSNumericValue::FromCSSValue(x_value));
default:
NOTREACHED();
return nullptr;
}
}
const DOMMatrix* CSSSkew::AsMatrix(ExceptionState&) const {
CSSUnitValue* ax = ax_->to(CSSPrimitiveValue::UnitType::kRadians);
CSSUnitValue* ay = ay_->to(CSSPrimitiveValue::UnitType::kRadians);
DCHECK(ax);
DCHECK(ay);
DOMMatrix* result = DOMMatrix::Create();
result->setM12(std::tan(ay->value()));
result->setM21(std::tan(ax->value()));
return result;
}
const CSSFunctionValue* CSSSkew::ToCSSValue() const {
// TDOO(meade): Handle calc angles here.
CSSUnitValue* ax = ToCSSUnitValue(ax_);
CSSUnitValue* ay = ToCSSUnitValue(ay_);
CSSFunctionValue* result = CSSFunctionValue::Create(CSSValueSkew);
result->Append(
*CSSPrimitiveValue::Create(ax->value(), ax->GetInternalUnit()));
result->Append(
*CSSPrimitiveValue::Create(ay->value(), ay->GetInternalUnit()));
return result;
}
CSSSkew::CSSSkew(CSSNumericValue* ax, CSSNumericValue* ay)
: CSSTransformComponent(true /* is2D */), ax_(ax), ay_(ay) {
DCHECK(ax);
DCHECK(ay);
}
} // namespace blink