blob: 6eed39ea37ad9b3b7d5ade3f432e7bfe8b99268a [file] [log] [blame]
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -04001/*
2 * Copyright 2008, The Android Open Source Project
3 *
4 * Redistribution and use in source and binary forms, with or without
5 * modification, are permitted provided that the following conditions
6 * are met:
7 * * Redistributions of source code must retain the above copyright
8 * notice, this list of conditions and the following disclaimer.
9 * * Redistributions in binary form must reproduce the above copyright
10 * notice, this list of conditions and the following disclaimer in the
11 * documentation and/or other materials provided with the distribution.
12 *
13 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ``AS IS'' AND ANY
14 * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR
17 * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21 * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23 * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
24 */
25
26#include "BackgroundPlugin.h"
27#include "android_npapi.h"
28
29#include <stdio.h>
30#include <sys/time.h>
31#include <time.h>
32#include <math.h>
33#include <string.h>
34
35extern NPNetscapeFuncs* browser;
36extern ANPBitmapInterfaceV0 gBitmapI;
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -040037extern ANPCanvasInterfaceV0 gCanvasI;
Derek Sollenbergerd049ec12009-08-07 11:26:44 -040038extern ANPLogInterfaceV0 gLogI;
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -040039extern ANPPaintInterfaceV0 gPaintI;
Derek Sollenbergerd049ec12009-08-07 11:26:44 -040040extern ANPSurfaceInterfaceV0 gSurfaceI;
Derek Sollenbergerb4a23912009-11-09 15:38:58 -050041extern ANPSystemInterfaceV0 gSystemI;
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -040042extern ANPTypefaceInterfaceV0 gTypefaceI;
Derek Sollenberger51cce582009-12-01 08:26:20 -050043extern ANPWindowInterfaceV0 gWindowI;
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -040044
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -040045#define ARRAY_COUNT(array) (sizeof(array) / sizeof(array[0]))
46
Derek Sollenberger0c763b62009-08-11 10:05:46 -040047static uint32_t getMSecs() {
48 struct timeval tv;
49 gettimeofday(&tv, NULL);
50 return (uint32_t) (tv.tv_sec * 1000 + tv.tv_usec / 1000 ); // microseconds to milliseconds
51}
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -040052
53///////////////////////////////////////////////////////////////////////////////
54
Derek Sollenberger08581f12009-09-08 18:36:29 -040055BackgroundPlugin::BackgroundPlugin(NPP inst) : SurfaceSubPlugin(inst) {
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -040056
Derek Sollenbergerd049ec12009-08-07 11:26:44 -040057 // initialize the drawing surface
Derek Sollenberger08581f12009-09-08 18:36:29 -040058 m_surface = NULL;
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -040059
Derek Sollenbergerd049ec12009-08-07 11:26:44 -040060 //initialize bitmap transparency variables
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -040061 mFinishedStageOne = false;
62 mFinishedStageTwo = false;
63 mFinishedStageThree = false;
64
65 // test basic plugin functionality
66 test_logging(); // android logging
67 test_timers(); // plugin timers
68 test_bitmaps(); // android bitmaps
Derek Sollenbergerc356b9f2009-07-30 16:57:07 -040069 test_domAccess();
70 test_javascript();
Derek Sollenbergerb4a23912009-11-09 15:38:58 -050071 test_loadJavaClass();
Grace Kloba2c0f89a2009-12-09 00:13:05 -080072
73 //register for touch events
74 ANPEventFlags flags = kTouch_ANPEventFlag;
75 NPError err = browser->setvalue(inst, kAcceptEvents_ANPSetValue, &flags);
76 if (err != NPERR_NO_ERROR) {
77 gLogI.log(kError_ANPLogType, "Error selecting input events.");
78 }
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -040079}
80
Derek Sollenbergerb8947ee2009-10-05 14:40:18 -040081BackgroundPlugin::~BackgroundPlugin() {
Derek Sollenbergerd53b56d2010-01-11 12:31:49 -050082 setContext(NULL);
83 destroySurface();
Derek Sollenbergerb8947ee2009-10-05 14:40:18 -040084}
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -040085
Derek Sollenbergerd53b56d2010-01-11 12:31:49 -050086jobject BackgroundPlugin::getSurface() {
87
88 if (m_surface) {
89 return m_surface;
90 }
91
92 // load the appropriate java class and instantiate it
93 JNIEnv* env = NULL;
94 if (gVM->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK) {
95 gLogI.log(kError_ANPLogType, " ---- getSurface: failed to get env");
96 return NULL;
97 }
98
99 const char* className = "com.android.sampleplugin.BackgroundSurface";
100 jclass backgroundClass = gSystemI.loadJavaClass(inst(), className);
101
102 if(!backgroundClass) {
103 gLogI.log(kError_ANPLogType, " ---- getSurface: failed to load class");
104 return NULL;
105 }
106
107 jmethodID constructor = env->GetMethodID(backgroundClass, "<init>", "(Landroid/content/Context;)V");
108 jobject backgroundSurface = env->NewObject(backgroundClass, constructor, m_context);
109
110 if(!backgroundSurface) {
111 gLogI.log(kError_ANPLogType, " ---- getSurface: failed to construct object");
112 return NULL;
113 }
114
115 m_surface = env->NewGlobalRef(backgroundSurface);
116 return m_surface;
Derek Sollenberger21f39912009-07-09 09:19:39 -0400117}
118
Derek Sollenbergerd53b56d2010-01-11 12:31:49 -0500119void BackgroundPlugin::destroySurface() {
Derek Sollenberger08581f12009-09-08 18:36:29 -0400120 JNIEnv* env = NULL;
Derek Sollenbergerb8947ee2009-10-05 14:40:18 -0400121 if (m_surface && gVM->GetEnv((void**) &env, JNI_VERSION_1_4) == JNI_OK) {
Derek Sollenberger08581f12009-09-08 18:36:29 -0400122 env->DeleteGlobalRef(m_surface);
123 m_surface = NULL;
124 }
125}
126
Derek Sollenbergerd049ec12009-08-07 11:26:44 -0400127void BackgroundPlugin::drawPlugin(int surfaceWidth, int surfaceHeight) {
128
129 // get the plugin's dimensions according to the DOM
130 PluginObject *obj = (PluginObject*) inst()->pdata;
131 const int W = obj->window->width;
132 const int H = obj->window->height;
133
134 // compute the current zoom level
135 const float zoomFactorW = static_cast<float>(surfaceWidth) / W;
136 const float zoomFactorH = static_cast<float>(surfaceHeight) / H;
137
138 // check to make sure the zoom level is uniform
139 if (zoomFactorW + .01 < zoomFactorH && zoomFactorW - .01 > zoomFactorH)
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500140 gLogI.log(kError_ANPLogType, " ------ %p zoom is out of sync (%f,%f)",
Derek Sollenbergerd049ec12009-08-07 11:26:44 -0400141 inst(), zoomFactorW, zoomFactorH);
142
143 // scale the variables based on the zoom level
144 const int fontSize = (int)(zoomFactorW * 16);
145 const int leftMargin = (int)(zoomFactorW * 10);
146
147 // lock the surface
148 ANPBitmap bitmap;
Derek Sollenberger08581f12009-09-08 18:36:29 -0400149 JNIEnv* env = NULL;
Derek Sollenbergerb8947ee2009-10-05 14:40:18 -0400150 if (!m_surface || gVM->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK ||
Derek Sollenberger08581f12009-09-08 18:36:29 -0400151 !gSurfaceI.lock(env, m_surface, &bitmap, NULL)) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500152 gLogI.log(kError_ANPLogType, " ------ %p unable to lock the plugin", inst());
Derek Sollenbergerd049ec12009-08-07 11:26:44 -0400153 return;
154 }
155
156 // create a canvas
Derek Sollenbergerc0f26572009-07-16 11:38:02 -0400157 ANPCanvas* canvas = gCanvasI.newCanvas(&bitmap);
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400158 gCanvasI.drawColor(canvas, 0xFFFFFFFF);
159
Derek Sollenbergerd049ec12009-08-07 11:26:44 -0400160 ANPPaint* paint = gPaintI.newPaint();
161 gPaintI.setFlags(paint, gPaintI.getFlags(paint) | kAntiAlias_ANPPaintFlag);
162 gPaintI.setColor(paint, 0xFFFF0000);
163 gPaintI.setTextSize(paint, fontSize);
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400164
Derek Sollenbergerd049ec12009-08-07 11:26:44 -0400165 ANPTypeface* tf = gTypefaceI.createFromName("serif", kItalic_ANPTypefaceStyle);
166 gPaintI.setTypeface(paint, tf);
167 gTypefaceI.unref(tf);
168
169 ANPFontMetrics fm;
170 gPaintI.getFontMetrics(paint, &fm);
171
172 gPaintI.setColor(paint, 0xFF0000FF);
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400173 const char c[] = "This is a background plugin.";
Derek Sollenbergerd049ec12009-08-07 11:26:44 -0400174 gCanvasI.drawText(canvas, c, sizeof(c)-1, leftMargin, -fm.fTop, paint);
175
176 // clean up variables and unlock the surface
177 gPaintI.deletePaint(paint);
178 gCanvasI.deleteCanvas(canvas);
Derek Sollenberger08581f12009-09-08 18:36:29 -0400179 gSurfaceI.unlock(env, m_surface);
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400180}
181
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400182int16 BackgroundPlugin::handleEvent(const ANPEvent* evt) {
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400183 switch (evt->eventType) {
184 case kDraw_ANPEventType:
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500185 gLogI.log(kError_ANPLogType, " ------ %p the plugin did not request draw events", inst());
Derek Sollenbergerd049ec12009-08-07 11:26:44 -0400186 break;
Derek Sollenbergerd049ec12009-08-07 11:26:44 -0400187 case kLifecycle_ANPEventType:
Derek Sollenberger5b674142010-02-23 08:52:16 -0500188 switch (evt->data.lifecycle.action) {
189 case kOnLoad_ANPLifecycleAction:
190 gLogI.log(kDebug_ANPLogType, " ------ %p onLoad", inst());
191 return 1;
192 case kOnScreen_ANPLifecycleAction:
193 gLogI.log(kDebug_ANPLogType, " ------ %p onScreen", inst());
194 return 1;
195 case kOffScreen_ANPLifecycleAction:
196 gLogI.log(kDebug_ANPLogType, " ------ %p offScreen", inst());
197 return 1;
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400198 }
Derek Sollenberger5b674142010-02-23 08:52:16 -0500199 break; // end kLifecycle_ANPEventType
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400200 case kTouch_ANPEventType:
Grace Kloba2c0f89a2009-12-09 00:13:05 -0800201 if (kDown_ANPTouchAction == evt->data.touch.action)
202 return kHandleLongPress_ANPTouchResult | kHandleDoubleTap_ANPTouchResult;
Derek Sollenbergerd53b56d2010-01-11 12:31:49 -0500203 else if (kLongPress_ANPTouchAction == evt->data.touch.action) {
Grace Kloba2c0f89a2009-12-09 00:13:05 -0800204 browser->geturl(inst(), "javascript:alert('Detected long press event.')", 0);
Derek Sollenbergerd53b56d2010-01-11 12:31:49 -0500205 gWindowI.requestFullScreen(inst());
206 }
Grace Kloba2c0f89a2009-12-09 00:13:05 -0800207 else if (kDoubleTap_ANPTouchAction == evt->data.touch.action)
208 browser->geturl(inst(), "javascript:alert('Detected double tap event.')", 0);
Derek Sollenbergerd049ec12009-08-07 11:26:44 -0400209 break;
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400210 case kKey_ANPEventType:
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500211 gLogI.log(kError_ANPLogType, " ------ %p the plugin did not request key events", inst());
Derek Sollenbergerd049ec12009-08-07 11:26:44 -0400212 break;
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400213 default:
214 break;
215 }
216 return 0; // unknown or unhandled event
217}
218
219///////////////////////////////////////////////////////////////////////////////
220// LOGGING TESTS
221///////////////////////////////////////////////////////////////////////////////
222
223
224void BackgroundPlugin::test_logging() {
225 NPP instance = this->inst();
226
227 //LOG_ERROR(instance, " ------ %p Testing Log Error", instance);
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500228 gLogI.log(kError_ANPLogType, " ------ %p Testing Log Error", instance);
229 gLogI.log(kWarning_ANPLogType, " ------ %p Testing Log Warning", instance);
230 gLogI.log(kDebug_ANPLogType, " ------ %p Testing Log Debug", instance);
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400231}
232
233///////////////////////////////////////////////////////////////////////////////
234// TIMER TESTS
235///////////////////////////////////////////////////////////////////////////////
236
237#define TIMER_INTERVAL 50
238static void timer_oneshot(NPP instance, uint32 timerID);
239static void timer_repeat(NPP instance, uint32 timerID);
240static void timer_neverfires(NPP instance, uint32 timerID);
241static void timer_latency(NPP instance, uint32 timerID);
242
243void BackgroundPlugin::test_timers() {
244 NPP instance = this->inst();
245
246 //Setup the testing counters
247 mTimerRepeatCount = 5;
248 mTimerLatencyCount = 5;
249
250 // test for bogus timerID
251 browser->unscheduletimer(instance, 999999);
252 // test one-shot
253 browser->scheduletimer(instance, 100, false, timer_oneshot);
254 // test repeat
255 browser->scheduletimer(instance, 50, true, timer_repeat);
256 // test timer latency
257 browser->scheduletimer(instance, TIMER_INTERVAL, true, timer_latency);
258 mStartTime = mPrevTime = getMSecs();
259 // test unschedule immediately
260 uint32 id = browser->scheduletimer(instance, 100, false, timer_neverfires);
261 browser->unscheduletimer(instance, id);
262 // test double unschedule (should be no-op)
263 browser->unscheduletimer(instance, id);
264
265}
266
267static void timer_oneshot(NPP instance, uint32 timerID) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500268 gLogI.log(kDebug_ANPLogType, "-------- oneshot timer\n");
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400269}
270
271static void timer_repeat(NPP instance, uint32 timerID) {
272 BackgroundPlugin *obj = ((BackgroundPlugin*) ((PluginObject*) instance->pdata)->activePlugin);
273
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500274 gLogI.log(kDebug_ANPLogType, "-------- repeat timer %d\n",
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400275 obj->mTimerRepeatCount);
276 if (--obj->mTimerRepeatCount == 0) {
277 browser->unscheduletimer(instance, timerID);
278 }
279}
280
281static void timer_neverfires(NPP instance, uint32 timerID) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500282 gLogI.log(kError_ANPLogType, "-------- timer_neverfires!!!\n");
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400283}
284
285static void timer_latency(NPP instance, uint32 timerID) {
286 BackgroundPlugin *obj = ((BackgroundPlugin*) ((PluginObject*) instance->pdata)->activePlugin);
287
288 obj->mTimerLatencyCurrentCount += 1;
289
290 uint32_t now = getMSecs();
291 uint32_t interval = now - obj->mPrevTime;
292 uint32_t dur = now - obj->mStartTime;
293 uint32_t expectedDur = obj->mTimerLatencyCurrentCount * TIMER_INTERVAL;
294 int32_t drift = dur - expectedDur;
295 int32_t avgDrift = drift / obj->mTimerLatencyCurrentCount;
296
297 obj->mPrevTime = now;
298
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500299 gLogI.log(kDebug_ANPLogType,
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400300 "-------- latency test: [%3d] interval %d expected %d, total %d expected %d, drift %d avg %d\n",
301 obj->mTimerLatencyCurrentCount, interval, TIMER_INTERVAL, dur,
302 expectedDur, drift, avgDrift);
303
304 if (--obj->mTimerLatencyCount == 0) {
305 browser->unscheduletimer(instance, timerID);
306 }
307}
308
309///////////////////////////////////////////////////////////////////////////////
310// BITMAP TESTS
311///////////////////////////////////////////////////////////////////////////////
312
313static void test_formats(NPP instance);
314
315void BackgroundPlugin::test_bitmaps() {
316 test_formats(this->inst());
317}
318
319static void test_formats(NPP instance) {
320
321 // TODO pull names from enum in npapi instead of hardcoding them
322 static const struct {
323 ANPBitmapFormat fFormat;
324 const char* fName;
325 } gRecs[] = {
326 { kUnknown_ANPBitmapFormat, "unknown" },
327 { kRGBA_8888_ANPBitmapFormat, "8888" },
328 { kRGB_565_ANPBitmapFormat, "565" },
329 };
330
331 ANPPixelPacking packing;
332 for (size_t i = 0; i < ARRAY_COUNT(gRecs); i++) {
333 if (gBitmapI.getPixelPacking(gRecs[i].fFormat, &packing)) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500334 gLogI.log(kDebug_ANPLogType,
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400335 "pixel format [%d] %s has packing ARGB [%d %d] [%d %d] [%d %d] [%d %d]\n",
336 gRecs[i].fFormat, gRecs[i].fName,
337 packing.AShift, packing.ABits,
338 packing.RShift, packing.RBits,
339 packing.GShift, packing.GBits,
340 packing.BShift, packing.BBits);
341 } else {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500342 gLogI.log(kDebug_ANPLogType,
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400343 "pixel format [%d] %s has no packing\n",
344 gRecs[i].fFormat, gRecs[i].fName);
345 }
346 }
347}
348
349void BackgroundPlugin::test_bitmap_transparency(const ANPEvent* evt) {
350 NPP instance = this->inst();
351
352 // check default & set transparent
353 if (!mFinishedStageOne) {
354
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500355 gLogI.log(kDebug_ANPLogType, "BEGIN: testing bitmap transparency");
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400356
357 //check to make sure it is not transparent
358 if (evt->data.draw.data.bitmap.format == kRGBA_8888_ANPBitmapFormat) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500359 gLogI.log(kError_ANPLogType, "bitmap default format is transparent");
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400360 }
361
362 //make it transparent (any non-null value will set it to true)
363 bool value = true;
364 NPError err = browser->setvalue(instance, NPPVpluginTransparentBool, &value);
365 if (err != NPERR_NO_ERROR) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500366 gLogI.log(kError_ANPLogType, "Error setting transparency.");
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400367 }
368
369 mFinishedStageOne = true;
370 browser->invalidaterect(instance, NULL);
371 }
372 // check transparent & set opaque
373 else if (!mFinishedStageTwo) {
374
375 //check to make sure it is transparent
376 if (evt->data.draw.data.bitmap.format != kRGBA_8888_ANPBitmapFormat) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500377 gLogI.log(kError_ANPLogType, "bitmap did not change to transparent format");
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400378 }
379
380 //make it opaque
381 NPError err = browser->setvalue(instance, NPPVpluginTransparentBool, NULL);
382 if (err != NPERR_NO_ERROR) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500383 gLogI.log(kError_ANPLogType, "Error setting transparency.");
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400384 }
385
386 mFinishedStageTwo = true;
387 }
388 // check opaque
389 else if (!mFinishedStageThree) {
390
391 //check to make sure it is not transparent
392 if (evt->data.draw.data.bitmap.format == kRGBA_8888_ANPBitmapFormat) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500393 gLogI.log(kError_ANPLogType, "bitmap default format is transparent");
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400394 }
395
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500396 gLogI.log(kDebug_ANPLogType, "END: testing bitmap transparency");
Derek Sollenbergerd7ebf272009-06-18 11:19:41 -0400397
398 mFinishedStageThree = true;
399 }
400}
Derek Sollenbergerc356b9f2009-07-30 16:57:07 -0400401
402///////////////////////////////////////////////////////////////////////////////
403// DOM TESTS
404///////////////////////////////////////////////////////////////////////////////
405
406void BackgroundPlugin::test_domAccess() {
407 NPP instance = this->inst();
408
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500409 gLogI.log(kDebug_ANPLogType, " ------ %p Testing DOM Access", instance);
Derek Sollenbergerc356b9f2009-07-30 16:57:07 -0400410
411 // Get the plugin's DOM object
412 NPObject* windowObject = NULL;
413 browser->getvalue(instance, NPNVWindowNPObject, &windowObject);
414
415 if (!windowObject)
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500416 gLogI.log(kError_ANPLogType, " ------ %p Unable to retrieve DOM Window", instance);
Derek Sollenbergerc356b9f2009-07-30 16:57:07 -0400417
418 // Retrieve a property from the plugin's DOM object
419 NPIdentifier topIdentifier = browser->getstringidentifier("top");
420 NPVariant topObjectVariant;
421 browser->getproperty(instance, windowObject, topIdentifier, &topObjectVariant);
422
423 if (topObjectVariant.type != NPVariantType_Object)
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500424 gLogI.log(kError_ANPLogType, " ------ %p Invalid Variant type for DOM Property: %d,%d", instance, topObjectVariant.type, NPVariantType_Object);
Derek Sollenbergerc356b9f2009-07-30 16:57:07 -0400425}
426
427
428///////////////////////////////////////////////////////////////////////////////
429// JAVASCRIPT TESTS
430///////////////////////////////////////////////////////////////////////////////
431
432
433void BackgroundPlugin::test_javascript() {
434 NPP instance = this->inst();
435
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500436 gLogI.log(kDebug_ANPLogType, " ------ %p Testing JavaScript Access", instance);
Derek Sollenbergerc356b9f2009-07-30 16:57:07 -0400437
438 // Get the plugin's DOM object
439 NPObject* windowObject = NULL;
440 browser->getvalue(instance, NPNVWindowNPObject, &windowObject);
441
442 if (!windowObject)
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500443 gLogI.log(kError_ANPLogType, " ------ %p Unable to retrieve DOM Window", instance);
Derek Sollenbergerc356b9f2009-07-30 16:57:07 -0400444
445 // create a string (JS code) that is stored in memory allocated by the browser
446 const char* jsString = "1200 + 34";
447 void* stringMem = browser->memalloc(strlen(jsString));
448 memcpy(stringMem, jsString, strlen(jsString));
449
450 // execute the javascript in the plugin's DOM object
451 NPString script = { (char*)stringMem, strlen(jsString) };
452 NPVariant scriptVariant;
453 if (!browser->evaluate(instance, windowObject, &script, &scriptVariant))
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500454 gLogI.log(kError_ANPLogType, " ------ %p Unable to eval the JS.", instance);
Derek Sollenbergerc356b9f2009-07-30 16:57:07 -0400455
456 if (scriptVariant.type == NPVariantType_Int32) {
457 if (scriptVariant.value.intValue != 1234)
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500458 gLogI.log(kError_ANPLogType, " ------ %p Invalid Value for JS Return: %d,1234", instance, scriptVariant.value.intValue);
Derek Sollenbergerc356b9f2009-07-30 16:57:07 -0400459 } else {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500460 gLogI.log(kError_ANPLogType, " ------ %p Invalid Variant type for JS Return: %d,%d", instance, scriptVariant.type, NPVariantType_Int32);
Derek Sollenbergerc356b9f2009-07-30 16:57:07 -0400461 }
462
463 // free the memory allocated within the browser
464 browser->memfree(stringMem);
465}
Derek Sollenbergerb4a23912009-11-09 15:38:58 -0500466
467///////////////////////////////////////////////////////////////////////////////
468// Load Java Classes Tests
469///////////////////////////////////////////////////////////////////////////////
470
471void BackgroundPlugin::test_loadJavaClass() {
472
473 JNIEnv* env = NULL;
474 if (gVM->GetEnv((void**) &env, JNI_VERSION_1_4) != JNI_OK) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500475 gLogI.log(kError_ANPLogType, " ---- LoadJavaTest: failed to get env");
Derek Sollenbergerb4a23912009-11-09 15:38:58 -0500476 return;
477 }
478
479 const char* className = "com.android.sampleplugin.BackgroundTest";
480 jclass backgroundClass = gSystemI.loadJavaClass(inst(), className);
481
482 if(!backgroundClass) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500483 gLogI.log(kError_ANPLogType, " ---- LoadJavaTest: failed to load class");
Derek Sollenbergerb4a23912009-11-09 15:38:58 -0500484 return;
485 }
486
487 jmethodID constructor = env->GetMethodID(backgroundClass, "<init>", "()V");
488 jmethodID addMethod = env->GetMethodID(backgroundClass, "addInt", "(II)I");
489 jobject backgroundObject = env->NewObject(backgroundClass, constructor);
490
491 if(!backgroundObject) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500492 gLogI.log(kError_ANPLogType, " ---- LoadJavaTest: failed to construct object");
Derek Sollenbergerb4a23912009-11-09 15:38:58 -0500493 return;
494 }
495
496 jint result = env->CallIntMethod(backgroundObject, addMethod, 2, 2);
497
498 if (result != 4) {
Derek Sollenbergere62ce172009-11-30 11:52:06 -0500499 gLogI.log(kError_ANPLogType, " ---- LoadJavaTest: invalid result (%d != 4)", result);
Derek Sollenbergerb4a23912009-11-09 15:38:58 -0500500 }
501}