| OLD | NEW |
| (Empty) |
| 1 /* | |
| 2 * Copyright 2014 Google Inc. | |
| 3 * | |
| 4 * | |
| 5 * Use of this source code is governed by a BSD-style license that can be | |
| 6 * found in the LICENSE file. | |
| 7 * | |
| 8 */ | |
| 9 | |
| 10 #include "Path.h" | |
| 11 #include "Global.h" | |
| 12 | |
| 13 Global* Path::gGlobal = NULL; | |
| 14 | |
| 15 void Path::ConstructPath(const v8::FunctionCallbackInfo<Value>& args) { | |
| 16 HandleScope handleScope(gGlobal->getIsolate()); | |
| 17 Path* path = new Path(); | |
| 18 args.This()->SetInternalField( | |
| 19 0, External::New(gGlobal->getIsolate(), path)); | |
| 20 } | |
| 21 | |
| 22 #define ADD_METHOD(name, fn) \ | |
| 23 constructor->InstanceTemplate()->Set( \ | |
| 24 String::NewFromUtf8( \ | |
| 25 global->getIsolate(), name, \ | |
| 26 String::kInternalizedString), \ | |
| 27 FunctionTemplate::New(global->getIsolate(), fn)) | |
| 28 | |
| 29 // Install the constructor in the global scope so Paths can be constructed | |
| 30 // in JS. | |
| 31 void Path::AddToGlobal(Global* global) { | |
| 32 gGlobal = global; | |
| 33 | |
| 34 // Create a stack-allocated handle scope. | |
| 35 HandleScope handleScope(gGlobal->getIsolate()); | |
| 36 | |
| 37 Handle<Context> context = gGlobal->getContext(); | |
| 38 | |
| 39 // Enter the scope so all operations take place in the scope. | |
| 40 Context::Scope contextScope(context); | |
| 41 | |
| 42 Local<FunctionTemplate> constructor = FunctionTemplate::New( | |
| 43 gGlobal->getIsolate(), Path::ConstructPath); | |
| 44 constructor->InstanceTemplate()->SetInternalFieldCount(1); | |
| 45 | |
| 46 ADD_METHOD("closePath", ClosePath); | |
| 47 ADD_METHOD("moveTo", MoveTo); | |
| 48 ADD_METHOD("lineTo", LineTo); | |
| 49 ADD_METHOD("quadraticCurveTo", QuadraticCurveTo); | |
| 50 ADD_METHOD("bezierCurveTo", BezierCurveTo); | |
| 51 ADD_METHOD("arc", Arc); | |
| 52 ADD_METHOD("rect", Rect); | |
| 53 ADD_METHOD("oval", Oval); | |
| 54 ADD_METHOD("conicTo", ConicTo); | |
| 55 | |
| 56 context->Global()->Set(String::NewFromUtf8( | |
| 57 gGlobal->getIsolate(), "Path"), constructor->GetFunction()); | |
| 58 } | |
| 59 | |
| 60 Path* Path::Unwrap(const v8::FunctionCallbackInfo<Value>& args) { | |
| 61 Handle<External> field = Handle<External>::Cast( | |
| 62 args.This()->GetInternalField(0)); | |
| 63 void* ptr = field->Value(); | |
| 64 return static_cast<Path*>(ptr); | |
| 65 } | |
| 66 | |
| 67 void Path::ClosePath(const v8::FunctionCallbackInfo<Value>& args) { | |
| 68 Path* path = Unwrap(args); | |
| 69 path->fSkPath.close(); | |
| 70 } | |
| 71 | |
| 72 void Path::MoveTo(const v8::FunctionCallbackInfo<Value>& args) { | |
| 73 if (args.Length() != 2) { | |
| 74 args.GetIsolate()->ThrowException( | |
| 75 v8::String::NewFromUtf8( | |
| 76 args.GetIsolate(), "Error: 2 arguments required.")); | |
| 77 return; | |
| 78 } | |
| 79 double x = args[0]->NumberValue(); | |
| 80 double y = args[1]->NumberValue(); | |
| 81 Path* path = Unwrap(args); | |
| 82 path->fSkPath.moveTo(SkDoubleToScalar(x), SkDoubleToScalar(y)); | |
| 83 } | |
| 84 | |
| 85 void Path::LineTo(const v8::FunctionCallbackInfo<Value>& args) { | |
| 86 if (args.Length() != 2) { | |
| 87 args.GetIsolate()->ThrowException( | |
| 88 v8::String::NewFromUtf8( | |
| 89 args.GetIsolate(), "Error: 2 arguments required.")); | |
| 90 return; | |
| 91 } | |
| 92 double x = args[0]->NumberValue(); | |
| 93 double y = args[1]->NumberValue(); | |
| 94 Path* path = Unwrap(args); | |
| 95 path->fSkPath.lineTo(SkDoubleToScalar(x), SkDoubleToScalar(y)); | |
| 96 } | |
| 97 | |
| 98 void Path::QuadraticCurveTo(const v8::FunctionCallbackInfo<Value>& args) { | |
| 99 if (args.Length() != 4) { | |
| 100 args.GetIsolate()->ThrowException( | |
| 101 v8::String::NewFromUtf8( | |
| 102 args.GetIsolate(), "Error: 4 arguments required.")); | |
| 103 return; | |
| 104 } | |
| 105 double cpx = args[0]->NumberValue(); | |
| 106 double cpy = args[1]->NumberValue(); | |
| 107 double x = args[2]->NumberValue(); | |
| 108 double y = args[3]->NumberValue(); | |
| 109 Path* path = Unwrap(args); | |
| 110 // TODO(jcgregorio) Doesn't handle the empty last path case correctly per | |
| 111 // the HTML 5 spec. | |
| 112 path->fSkPath.quadTo( | |
| 113 SkDoubleToScalar(cpx), SkDoubleToScalar(cpy), | |
| 114 SkDoubleToScalar(x), SkDoubleToScalar(y)); | |
| 115 } | |
| 116 | |
| 117 void Path::BezierCurveTo(const v8::FunctionCallbackInfo<Value>& args) { | |
| 118 if (args.Length() != 6) { | |
| 119 args.GetIsolate()->ThrowException( | |
| 120 v8::String::NewFromUtf8( | |
| 121 args.GetIsolate(), "Error: 6 arguments required.")); | |
| 122 return; | |
| 123 } | |
| 124 double cp1x = args[0]->NumberValue(); | |
| 125 double cp1y = args[1]->NumberValue(); | |
| 126 double cp2x = args[2]->NumberValue(); | |
| 127 double cp2y = args[3]->NumberValue(); | |
| 128 double x = args[4]->NumberValue(); | |
| 129 double y = args[5]->NumberValue(); | |
| 130 Path* path = Unwrap(args); | |
| 131 // TODO(jcgregorio) Doesn't handle the empty last path case correctly per | |
| 132 // the HTML 5 spec. | |
| 133 path->fSkPath.cubicTo( | |
| 134 SkDoubleToScalar(cp1x), SkDoubleToScalar(cp1y), | |
| 135 SkDoubleToScalar(cp2x), SkDoubleToScalar(cp2y), | |
| 136 SkDoubleToScalar(x), SkDoubleToScalar(y)); | |
| 137 } | |
| 138 | |
| 139 void Path::Arc(const v8::FunctionCallbackInfo<Value>& args) { | |
| 140 if (args.Length() != 5 && args.Length() != 6) { | |
| 141 args.GetIsolate()->ThrowException( | |
| 142 v8::String::NewFromUtf8( | |
| 143 args.GetIsolate(), "Error: 5 or 6 args required.")); | |
| 144 return; | |
| 145 } | |
| 146 double x = args[0]->NumberValue(); | |
| 147 double y = args[1]->NumberValue(); | |
| 148 double radius = args[2]->NumberValue(); | |
| 149 double startAngle = args[3]->NumberValue(); | |
| 150 double endAngle = args[4]->NumberValue(); | |
| 151 bool antiClockwise = false; | |
| 152 if (args.Length() == 6) { | |
| 153 antiClockwise = args[5]->BooleanValue(); | |
| 154 } | |
| 155 double sweepAngle; | |
| 156 if (!antiClockwise) { | |
| 157 sweepAngle = endAngle - startAngle; | |
| 158 } else { | |
| 159 sweepAngle = startAngle - endAngle; | |
| 160 startAngle = endAngle; | |
| 161 } | |
| 162 | |
| 163 Path* path = Unwrap(args); | |
| 164 SkRect rect = { | |
| 165 SkDoubleToScalar(x-radius), | |
| 166 SkDoubleToScalar(y-radius), | |
| 167 SkDoubleToScalar(x+radius), | |
| 168 SkDoubleToScalar(y+radius) | |
| 169 }; | |
| 170 | |
| 171 path->fSkPath.addArc(rect, SkRadiansToDegrees(startAngle), | |
| 172 SkRadiansToDegrees(sweepAngle)); | |
| 173 } | |
| 174 | |
| 175 void Path::Rect(const v8::FunctionCallbackInfo<Value>& args) { | |
| 176 if (args.Length() != 4) { | |
| 177 args.GetIsolate()->ThrowException( | |
| 178 v8::String::NewFromUtf8( | |
| 179 args.GetIsolate(), "Error: 4 arguments required.")); | |
| 180 return; | |
| 181 } | |
| 182 double x = args[0]->NumberValue(); | |
| 183 double y = args[1]->NumberValue(); | |
| 184 double w = args[2]->NumberValue(); | |
| 185 double h = args[3]->NumberValue(); | |
| 186 | |
| 187 SkRect rect = { | |
| 188 SkDoubleToScalar(x), | |
| 189 SkDoubleToScalar(y), | |
| 190 SkDoubleToScalar(x) + SkDoubleToScalar(w), | |
| 191 SkDoubleToScalar(y) + SkDoubleToScalar(h) | |
| 192 }; | |
| 193 Path* path = Unwrap(args); | |
| 194 path->fSkPath.addRect(rect); | |
| 195 } | |
| 196 | |
| 197 void Path::Oval(const v8::FunctionCallbackInfo<Value>& args) { | |
| 198 if (args.Length() != 4 && args.Length() != 5) { | |
| 199 args.GetIsolate()->ThrowException( | |
| 200 v8::String::NewFromUtf8( | |
| 201 args.GetIsolate(), "Error: 4 or 5 args required.")); | |
| 202 return; | |
| 203 } | |
| 204 double x = args[0]->NumberValue(); | |
| 205 double y = args[1]->NumberValue(); | |
| 206 double radiusX = args[2]->NumberValue(); | |
| 207 double radiusY = args[3]->NumberValue(); | |
| 208 SkPath::Direction dir = SkPath::kCW_Direction; | |
| 209 if (args.Length() == 5 && !args[4]->BooleanValue()) { | |
| 210 dir = SkPath::kCCW_Direction; | |
| 211 } | |
| 212 Path* path = Unwrap(args); | |
| 213 SkRect rect = { | |
| 214 SkDoubleToScalar(x-radiusX), | |
| 215 SkDoubleToScalar(y-radiusX), | |
| 216 SkDoubleToScalar(x+radiusY), | |
| 217 SkDoubleToScalar(y+radiusY) | |
| 218 }; | |
| 219 | |
| 220 path->fSkPath.addOval(rect, dir); | |
| 221 } | |
| 222 | |
| 223 void Path::ConicTo(const v8::FunctionCallbackInfo<Value>& args) { | |
| 224 if (args.Length() != 5) { | |
| 225 args.GetIsolate()->ThrowException( | |
| 226 v8::String::NewFromUtf8( | |
| 227 args.GetIsolate(), "Error: 5 args required.")); | |
| 228 return; | |
| 229 } | |
| 230 double x1 = args[0]->NumberValue(); | |
| 231 double y1 = args[1]->NumberValue(); | |
| 232 double x2 = args[2]->NumberValue(); | |
| 233 double y2 = args[3]->NumberValue(); | |
| 234 double w = args[4]->NumberValue(); | |
| 235 Path* path = Unwrap(args); | |
| 236 | |
| 237 path->fSkPath.conicTo( | |
| 238 SkDoubleToScalar(x1), | |
| 239 SkDoubleToScalar(y1), | |
| 240 SkDoubleToScalar(x2), | |
| 241 SkDoubleToScalar(y2), | |
| 242 SkDoubleToScalar(w) | |
| 243 ); | |
| 244 } | |
| OLD | NEW |