/* * Copyright 2008, The Android Open Source Project * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS ``AS IS'' AND ANY * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE COMPUTER, INC. OR * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. */ #include <stdlib.h> #include <string.h> #include <stdio.h> #include "main.h" #include "PluginObject.h" #include "AnimationPlugin.h" #include "AudioPlugin.h" #include "BackgroundPlugin.h" #include "FormPlugin.h" #include "NavigationPlugin.h" #include "PaintPlugin.h" #include "VideoPlugin.h" NPNetscapeFuncs* browser; JavaVM* gVM; #define EXPORT __attribute__((visibility("default"))) NPError NPP_New(NPMIMEType pluginType, NPP instance, uint16_t mode, int16_t argc, char* argn[], char* argv[], NPSavedData* saved); NPError NPP_Destroy(NPP instance, NPSavedData** save); NPError NPP_SetWindow(NPP instance, NPWindow* window); NPError NPP_NewStream(NPP instance, NPMIMEType type, NPStream* stream, NPBool seekable, uint16_t* stype); NPError NPP_DestroyStream(NPP instance, NPStream* stream, NPReason reason); int32_t NPP_WriteReady(NPP instance, NPStream* stream); int32_t NPP_Write(NPP instance, NPStream* stream, int32_t offset, int32_t len, void* buffer); void NPP_StreamAsFile(NPP instance, NPStream* stream, const char* fname); void NPP_Print(NPP instance, NPPrint* platformPrint); int16_t NPP_HandleEvent(NPP instance, void* event); void NPP_URLNotify(NPP instance, const char* URL, NPReason reason, void* notifyData); NPError NPP_GetValue(NPP instance, NPPVariable variable, void *value); NPError NPP_SetValue(NPP instance, NPNVariable variable, void *value); extern "C" { EXPORT NPError NP_Initialize(NPNetscapeFuncs* browserFuncs, NPPluginFuncs* pluginFuncs, void *java_env); EXPORT NPError NP_GetValue(NPP instance, NPPVariable variable, void *value); EXPORT const char* NP_GetMIMEDescription(void); EXPORT void NP_Shutdown(void); }; ANPAudioTrackInterfaceV0 gSoundI; ANPBitmapInterfaceV0 gBitmapI; ANPCanvasInterfaceV0 gCanvasI; ANPEventInterfaceV0 gEventI; ANPLogInterfaceV0 gLogI; ANPPaintInterfaceV0 gPaintI; ANPPathInterfaceV0 gPathI; ANPSurfaceInterfaceV0 gSurfaceI; ANPSystemInterfaceV0 gSystemI; ANPTypefaceInterfaceV0 gTypefaceI; ANPWindowInterfaceV1 gWindowI; ANPNativeWindowInterfaceV0 gNativeWindowI; #define ARRAY_COUNT(array) (sizeof(array) / sizeof(array[0])) #define DEBUG_PLUGIN_EVENTS 0 NPError NP_Initialize(NPNetscapeFuncs* browserFuncs, NPPluginFuncs* pluginFuncs, void *java_env) { // Make sure we have a function table equal or larger than we are built against. if (browserFuncs->size < sizeof(NPNetscapeFuncs)) { return NPERR_GENERIC_ERROR; } // Copy the function table (structure) browser = (NPNetscapeFuncs*) malloc(sizeof(NPNetscapeFuncs)); memcpy(browser, browserFuncs, sizeof(NPNetscapeFuncs)); // Build the plugin function table pluginFuncs->version = 11; pluginFuncs->size = sizeof(pluginFuncs); pluginFuncs->newp = NPP_New; pluginFuncs->destroy = NPP_Destroy; pluginFuncs->setwindow = NPP_SetWindow; pluginFuncs->newstream = NPP_NewStream; pluginFuncs->destroystream = NPP_DestroyStream; pluginFuncs->asfile = NPP_StreamAsFile; pluginFuncs->writeready = NPP_WriteReady; pluginFuncs->write = (NPP_WriteProcPtr)NPP_Write; pluginFuncs->print = NPP_Print; pluginFuncs->event = NPP_HandleEvent; pluginFuncs->urlnotify = NPP_URLNotify; pluginFuncs->getvalue = NPP_GetValue; pluginFuncs->setvalue = NPP_SetValue; static const struct { NPNVariable v; uint32_t size; ANPInterface* i; } gPairs[] = { { kAudioTrackInterfaceV0_ANPGetValue, sizeof(gSoundI), &gSoundI }, { kBitmapInterfaceV0_ANPGetValue, sizeof(gBitmapI), &gBitmapI }, { kCanvasInterfaceV0_ANPGetValue, sizeof(gCanvasI), &gCanvasI }, { kEventInterfaceV0_ANPGetValue, sizeof(gEventI), &gEventI }, { kLogInterfaceV0_ANPGetValue, sizeof(gLogI), &gLogI }, { kPaintInterfaceV0_ANPGetValue, sizeof(gPaintI), &gPaintI }, { kPathInterfaceV0_ANPGetValue, sizeof(gPathI), &gPathI }, { kSurfaceInterfaceV0_ANPGetValue, sizeof(gSurfaceI), &gSurfaceI }, { kSystemInterfaceV0_ANPGetValue, sizeof(gSystemI), &gSystemI }, { kTypefaceInterfaceV0_ANPGetValue, sizeof(gTypefaceI), &gTypefaceI }, { kWindowInterfaceV1_ANPGetValue, sizeof(gWindowI), &gWindowI }, { kNativeWindowInterfaceV0_ANPGetValue, sizeof(gNativeWindowI), &gNativeWindowI }, }; for (size_t i = 0; i < ARRAY_COUNT(gPairs); i++) { gPairs[i].i->inSize = gPairs[i].size; NPError err = browser->getvalue(NULL, gPairs[i].v, gPairs[i].i); if (err) { return err; } } // store the JavaVM for the plugin JNIEnv* env = (JNIEnv*)java_env; env->GetJavaVM(&gVM); return NPERR_NO_ERROR; } void NP_Shutdown(void) { } const char *NP_GetMIMEDescription(void) { return "application/x-testbrowserplugin:tst:Test plugin mimetype is application/x-testbrowserplugin"; } NPError NPP_New(NPMIMEType pluginType, NPP instance, uint16_t mode, int16_t argc, char* argn[], char* argv[], NPSavedData* saved) { /* BEGIN: STANDARD PLUGIN FRAMEWORK */ PluginObject *obj = NULL; // Scripting functions appeared in NPAPI version 14 if (browser->version >= 14) { instance->pdata = browser->createobject (instance, getPluginClass()); obj = static_cast<PluginObject*>(instance->pdata); obj->pluginType = 0; } /* END: STANDARD PLUGIN FRAMEWORK */ // select the drawing model based on user input ANPDrawingModel model = kBitmap_ANPDrawingModel; for (int i = 0; i < argc; i++) { if (!strcmp(argn[i], "DrawingModel")) { if (!strcmp(argv[i], "Bitmap")) { model = kBitmap_ANPDrawingModel; } else if (!strcmp(argv[i], "Surface")) { model = kSurface_ANPDrawingModel; } else if (!strcmp(argv[i], "OpenGL")) { model = kOpenGL_ANPDrawingModel; } gLogI.log(kDebug_ANPLogType, "------ %p DrawingModel is %d", instance, model); break; } } // notify the plugin API of the drawing model we wish to use. This must be // done prior to creating certain subPlugin objects (e.g. surfaceViews) NPError err = browser->setvalue(instance, kRequestDrawingModel_ANPSetValue, reinterpret_cast<void*>(model)); if (err) { gLogI.log(kError_ANPLogType, "request model %d err %d", model, err); return err; } const char* path = gSystemI.getApplicationDataDirectory(); if (path) { gLogI.log(kDebug_ANPLogType, "Application data dir is %s", path); } else { gLogI.log(kError_ANPLogType, "Can't find Application data dir"); } // select the pluginType for (int i = 0; i < argc; i++) { if (!strcmp(argn[i], "PluginType")) { if (!strcmp(argv[i], "Animation")) { obj->pluginType = kAnimation_PluginType; obj->activePlugin = new BallAnimation(instance); } else if (!strcmp(argv[i], "Audio")) { obj->pluginType = kAudio_PluginType; obj->activePlugin = new AudioPlugin(instance); } else if (!strcmp(argv[i], "Background")) { obj->pluginType = kBackground_PluginType; obj->activePlugin = new BackgroundPlugin(instance); } else if (!strcmp(argv[i], "Form")) { obj->pluginType = kForm_PluginType; obj->activePlugin = new FormPlugin(instance); } else if (!strcmp(argv[i], "Navigation")) { obj->pluginType = kNavigation_PluginType; obj->activePlugin = new NavigationPlugin(instance); } else if (!strcmp(argv[i], "Paint")) { obj->pluginType = kPaint_PluginType; obj->activePlugin = new PaintPlugin(instance); } else if (!strcmp(argv[i], "Video")) { obj->pluginType = kVideo_PluginType; obj->activePlugin = new VideoPlugin(instance); } else { gLogI.log(kError_ANPLogType, "PluginType %s unknown!", argv[i]); } break; } } // if no pluginType is specified then default to Animation if (!obj->pluginType) { gLogI.log(kError_ANPLogType, "------ %p No PluginType attribute was found", instance); obj->pluginType = kAnimation_PluginType; obj->activePlugin = new BallAnimation(instance); } gLogI.log(kDebug_ANPLogType, "------ %p PluginType is %d", instance, obj->pluginType); // check to ensure the pluginType supports the model if (!obj->activePlugin->supportsDrawingModel(model)) { gLogI.log(kError_ANPLogType, "------ %p Unsupported DrawingModel (%d)", instance, model); return NPERR_GENERIC_ERROR; } // if the plugin uses the surface drawing model then set the java context if (model == kSurface_ANPDrawingModel || model == kOpenGL_ANPDrawingModel) { SurfaceSubPlugin* surfacePlugin = static_cast<SurfaceSubPlugin*>(obj->activePlugin); jobject context; NPError err = browser->getvalue(instance, kJavaContext_ANPGetValue, static_cast<void*>(&context)); if (err) { gLogI.log(kError_ANPLogType, "request context err: %d", err); return err; } surfacePlugin->setContext(context); } return NPERR_NO_ERROR; } NPError NPP_Destroy(NPP instance, NPSavedData** save) { PluginObject *obj = (PluginObject*) instance->pdata; if (obj) { delete obj->activePlugin; browser->releaseobject(&obj->header); } return NPERR_NO_ERROR; } NPError NPP_SetWindow(NPP instance, NPWindow* window) { PluginObject *obj = (PluginObject*) instance->pdata; // Do nothing if browser didn't support NPN_CreateObject which would have created the PluginObject. if (obj != NULL) { obj->window = window; } browser->invalidaterect(instance, NULL); return NPERR_NO_ERROR; } NPError NPP_NewStream(NPP instance, NPMIMEType type, NPStream* stream, NPBool seekable, uint16_t* stype) { *stype = NP_ASFILEONLY; return NPERR_NO_ERROR; } NPError NPP_DestroyStream(NPP instance, NPStream* stream, NPReason reason) { return NPERR_NO_ERROR; } int32_t NPP_WriteReady(NPP instance, NPStream* stream) { return 0; } int32_t NPP_Write(NPP instance, NPStream* stream, int32_t offset, int32_t len, void* buffer) { return 0; } void NPP_StreamAsFile(NPP instance, NPStream* stream, const char* fname) { } void NPP_Print(NPP instance, NPPrint* platformPrint) { } int16_t NPP_HandleEvent(NPP instance, void* event) { PluginObject *obj = reinterpret_cast<PluginObject*>(instance->pdata); const ANPEvent* evt = reinterpret_cast<const ANPEvent*>(event); #if DEBUG_PLUGIN_EVENTS switch (evt->eventType) { case kDraw_ANPEventType: if (evt->data.draw.model == kBitmap_ANPDrawingModel) { static ANPBitmapFormat currentFormat = -1; if (evt->data.draw.data.bitmap.format != currentFormat) { currentFormat = evt->data.draw.data.bitmap.format; gLogI.log(kDebug_ANPLogType, "---- %p Draw (bitmap)" " clip=%d,%d,%d,%d format=%d", instance, evt->data.draw.clip.left, evt->data.draw.clip.top, evt->data.draw.clip.right, evt->data.draw.clip.bottom, evt->data.draw.data.bitmap.format); } } break; case kKey_ANPEventType: gLogI.log(kDebug_ANPLogType, "---- %p Key action=%d" " code=%d vcode=%d unichar=%d repeat=%d mods=%x", instance, evt->data.key.action, evt->data.key.nativeCode, evt->data.key.virtualCode, evt->data.key.unichar, evt->data.key.repeatCount, evt->data.key.modifiers); break; case kLifecycle_ANPEventType: gLogI.log(kDebug_ANPLogType, "---- %p Lifecycle action=%d", instance, evt->data.lifecycle.action); break; case kTouch_ANPEventType: gLogI.log(kDebug_ANPLogType, "---- %p Touch action=%d [%d %d]", instance, evt->data.touch.action, evt->data.touch.x, evt->data.touch.y); break; case kMouse_ANPEventType: gLogI.log(kDebug_ANPLogType, "---- %p Mouse action=%d [%d %d]", instance, evt->data.mouse.action, evt->data.mouse.x, evt->data.mouse.y); break; case kVisibleRect_ANPEventType: gLogI.log(kDebug_ANPLogType, "---- %p VisibleRect [%d %d %d %d]", instance, evt->data.visibleRect.rect.left, evt->data.visibleRect.rect.top, evt->data.visibleRect.rect.right, evt->data.visibleRect.rect.bottom); break; default: gLogI.log(kError_ANPLogType, "---- %p Unknown Event [%d]", instance, evt->eventType); break; } #endif if(!obj->activePlugin) { gLogI.log(kError_ANPLogType, "the active plugin is null."); return 0; // unknown or unhandled event } else { return obj->activePlugin->handleEvent(evt); } } void NPP_URLNotify(NPP instance, const char* url, NPReason reason, void* notifyData) { } EXPORT NPError NP_GetValue(NPP instance, NPPVariable variable, void *value) { if (variable == NPPVpluginNameString) { const char **str = (const char **)value; *str = "Test Plugin"; return NPERR_NO_ERROR; } if (variable == NPPVpluginDescriptionString) { const char **str = (const char **)value; *str = "Description of Test Plugin"; return NPERR_NO_ERROR; } return NPERR_GENERIC_ERROR; } NPError NPP_GetValue(NPP instance, NPPVariable variable, void* value) { if (variable == NPPVpluginScriptableNPObject) { void **v = (void **)value; PluginObject *obj = (PluginObject*) instance->pdata; if (obj) browser->retainobject(&obj->header); *v = &(obj->header); return NPERR_NO_ERROR; } if (variable == kJavaSurface_ANPGetValue) { //get the surface sub-plugin PluginObject* obj = static_cast<PluginObject*>(instance->pdata); if (obj && obj->activePlugin) { if(obj->activePlugin->supportsDrawingModel(kSurface_ANPDrawingModel) || obj->activePlugin->supportsDrawingModel(kOpenGL_ANPDrawingModel)) { SurfaceSubPlugin* plugin = static_cast<SurfaceSubPlugin*>(obj->activePlugin); jobject* surface = static_cast<jobject*>(value); *surface = plugin->getSurface(); return NPERR_NO_ERROR; } else { gLogI.log(kError_ANPLogType, "-- %p Tried to retrieve surface for non-surface plugin", instance); } } } return NPERR_GENERIC_ERROR; } NPError NPP_SetValue(NPP instance, NPNVariable variable, void *value) { return NPERR_GENERIC_ERROR; }