blob: e78e2b8d90c9ea700a8be22e01eb890f06bf7a71 [file] [log] [blame]
/*
* Copyright (C) 2006, 2007, 2009 Apple Inc. All rights reserved.
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*/
#include "config.h"
#include "JSCanvasRenderingContext2D.h"
#include "CanvasGradient.h"
#include "CanvasPattern.h"
#include "CanvasRenderingContext2D.h"
#include "CanvasStyle.h"
#include "ExceptionCode.h"
#include "HTMLCanvasElement.h"
#include "HTMLImageElement.h"
#include "ImageData.h"
#include "JSCanvasGradient.h"
#include "JSCanvasPattern.h"
#include "JSHTMLCanvasElement.h"
#include "JSHTMLImageElement.h"
#include "JSImageData.h"
using namespace JSC;
namespace WebCore {
static JSValue toJS(ExecState* exec, JSDOMGlobalObject* globalObject, CanvasStyle* style)
{
if (style->canvasGradient())
return toJS(exec, globalObject, style->canvasGradient());
if (style->canvasPattern())
return toJS(exec, globalObject, style->canvasPattern());
return jsString(exec, style->color());
}
static PassRefPtr<CanvasStyle> toHTMLCanvasStyle(ExecState*, JSValue value)
{
if (!value.isObject())
return 0;
JSObject* object = asObject(value);
if (object->inherits(&JSCanvasGradient::s_info))
return CanvasStyle::createFromGradient(static_cast<JSCanvasGradient*>(object)->impl());
if (object->inherits(&JSCanvasPattern::s_info))
return CanvasStyle::createFromPattern(static_cast<JSCanvasPattern*>(object)->impl());
return 0;
}
JSValue JSCanvasRenderingContext2D::strokeStyle(ExecState* exec) const
{
CanvasRenderingContext2D* context = static_cast<CanvasRenderingContext2D*>(impl());
return toJS(exec, globalObject(), context->strokeStyle());
}
void JSCanvasRenderingContext2D::setStrokeStyle(ExecState* exec, JSValue value)
{
CanvasRenderingContext2D* context = static_cast<CanvasRenderingContext2D*>(impl());
if (value.isString()) {
context->setStrokeColor(ustringToString(asString(value)->value(exec)));
return;
}
context->setStrokeStyle(toHTMLCanvasStyle(exec, value));
}
JSValue JSCanvasRenderingContext2D::fillStyle(ExecState* exec) const
{
CanvasRenderingContext2D* context = static_cast<CanvasRenderingContext2D*>(impl());
return toJS(exec, globalObject(), context->fillStyle());
}
void JSCanvasRenderingContext2D::setFillStyle(ExecState* exec, JSValue value)
{
CanvasRenderingContext2D* context = static_cast<CanvasRenderingContext2D*>(impl());
if (value.isString()) {
context->setFillColor(ustringToString(asString(value)->value(exec)));
return;
}
context->setFillStyle(toHTMLCanvasStyle(exec, value));
}
JSValue JSCanvasRenderingContext2D::createPattern(ExecState* exec)
{
CanvasRenderingContext2D* context = static_cast<CanvasRenderingContext2D*>(impl());
JSValue value = exec->argument(0);
if (!value.isObject()) {
setDOMException(exec, TYPE_MISMATCH_ERR);
return jsUndefined();
}
JSObject* o = asObject(value);
if (o->inherits(&JSHTMLImageElement::s_info)) {
ExceptionCode ec;
JSValue pattern = toJS(exec, globalObject(),
context->createPattern(static_cast<HTMLImageElement*>(static_cast<JSHTMLElement*>(o)->impl()),
valueToStringWithNullCheck(exec, exec->argument(1)), ec).get());
setDOMException(exec, ec);
return pattern;
}
if (o->inherits(&JSHTMLCanvasElement::s_info)) {
ExceptionCode ec;
JSValue pattern = toJS(exec, globalObject(),
context->createPattern(static_cast<HTMLCanvasElement*>(static_cast<JSHTMLElement*>(o)->impl()),
valueToStringWithNullCheck(exec, exec->argument(1)), ec).get());
setDOMException(exec, ec);
return pattern;
}
setDOMException(exec, TYPE_MISMATCH_ERR);
return jsUndefined();
}
JSValue JSCanvasRenderingContext2D::createImageData(ExecState* exec)
{
// createImageData has two variants
// createImageData(ImageData)
// createImageData(width, height)
CanvasRenderingContext2D* context = static_cast<CanvasRenderingContext2D*>(impl());
RefPtr<ImageData> imageData = 0;
ExceptionCode ec = 0;
if (exec->argumentCount() == 1)
imageData = context->createImageData(toImageData(exec->argument(0)), ec);
else if (exec->argumentCount() == 2)
imageData = context->createImageData(exec->argument(0).toFloat(exec), exec->argument(1).toFloat(exec), ec);
setDOMException(exec, ec);
return toJS(exec, globalObject(), WTF::getPtr(imageData));
}
JSValue JSCanvasRenderingContext2D::webkitLineDash(ExecState* exec) const
{
CanvasRenderingContext2D* context = static_cast<CanvasRenderingContext2D*>(impl());
const DashArray* dash = context->webkitLineDash();
MarkedArgumentBuffer list;
DashArray::const_iterator end = dash->end();
for (DashArray::const_iterator it = dash->begin(); it != end; ++it)
list.append(JSValue(*it));
return constructArray(exec, globalObject(), list);
}
void JSCanvasRenderingContext2D::setWebkitLineDash(ExecState* exec, JSValue value)
{
if (!isJSArray(&exec->globalData(), value))
return;
DashArray dash;
JSArray* valueArray = asArray(value);
for (unsigned i = 0; i < valueArray->length(); ++i) {
float elem = valueArray->getIndex(i).toFloat(exec);
if (elem <= 0 || !isfinite(elem))
return;
dash.append(elem);
}
CanvasRenderingContext2D* context = static_cast<CanvasRenderingContext2D*>(impl());
context->setWebkitLineDash(dash);
}
} // namespace WebCore