home *** CD-ROM | disk | FTP | other *** search
/ Netscape Plug-Ins Developer's Kit / Netscape_Plug-Ins_Developers_Kit.iso / source / Chap04 / npshell.cpp < prev   
Encoding:
C/C++ Source or Header  |  1996-09-10  |  20.5 KB  |  588 lines

  1. //::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
  2. //
  3. // npshell.cpp
  4. //
  5. // This file defines a "shell" plugin that plugin developers can use
  6. // as the basis for a real plugin.  This shell just provides empty
  7. // implementations of all functions that the plugin can implement
  8. // that will be called by Netscape (the NPP_xxx methods defined in 
  9. // npapi.h). 
  10. //
  11. //::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::::
  12.  
  13. #include "stdafx.h"
  14. #include "npChap04.h"
  15.  
  16. #include "windows.h"
  17.  
  18. #ifndef WIN32
  19.     #include "string.h"
  20. #endif
  21.  
  22. #ifndef _NPAPI_H_
  23. #include "npapi.h"
  24. #endif
  25.  
  26. #define BACKGROUND_COLOR RGB(192, 192, 192)
  27.  
  28. // Global MFC CWinAppobject
  29. #ifdef _PLUG_IN
  30. CNpChap04App theApp;
  31. #endif
  32.  
  33. //
  34. // Instance state information about the plugin.
  35. //
  36. // *Developers*: Use this struct to hold per-instance
  37. //               information that you'll need in the
  38. //               various functions in this file.
  39. //
  40.  
  41. typedef struct _PluginInstance PluginInstance;
  42. typedef struct _PluginInstance
  43. {
  44.     NPWindow*       fWindow;
  45.     HWND            hWnd;
  46.     uint16          fMode;
  47.  
  48. // recommendation from Win32 Functions Reference for CallWndProc
  49. // Needed to get NS code to compile
  50. #ifdef STRICT
  51.     WNDPROC            lpfnOldWndProc;
  52. #else
  53.     FARPROC         lpfnOldWndProc;
  54. #endif
  55.     NPSavedData*    pSavedInstanceData;
  56.     PluginInstance* pNext;
  57. } PluginInstance;
  58.  
  59. LONG NP_LOADDS WINAPI 
  60. SubClassFunc(HWND hWnd,WORD Message,WORD wParam, LONG lParam);
  61.  
  62. HINSTANCE DLLInstance;
  63.  
  64. #ifdef WIN32
  65. BOOL WINAPI
  66. DllMain(HINSTANCE hDLL, DWORD dwReason, LPVOID lpReserved)
  67. {
  68.     DLLInstance = hDLL;
  69.     return 1;
  70. }
  71. #else
  72. int CALLBACK
  73. LibMain(HINSTANCE hinst, WORD wDataSeg, WORD cbHeap, LPSTR lpszCmdLine)
  74. {
  75.     DLLInstance = hinst;
  76.     return 1;
  77. }
  78. #endif
  79.  
  80. // A plugin instance typically will subclass the plugin's client window, so
  81. // it can get Windows messages, (such as paint, palettechanged, keybd, etc).
  82. // To do work associated with a specific plugin instance the WndProc which
  83. // receives the Windows messages, (named "SubClassFunc" herein), needs access
  84. // to the "This" (PluginInstance*) ptr.
  85.  
  86. // When Navigator registers the plugin client's window class, (the class for
  87. // the window passed in NPP_SetWindow()), Navigator does not reserve any
  88. // "extra" windows bytes.  If it had, the plugin could simply have stored its
  89. // "This" (PluginInstance*) ptr in the extra bytes.  But Nav did not, and the
  90. // plugin cannot, so a different technique must be used.  The technique used
  91. // is to keep a linked list of PluginInstance structures, and walk the list
  92. // to find which one is associated with the window handle.  Inefficient,
  93. // grungy, complicates the code, etc.  C'est la vie ...
  94.  
  95. PluginInstance* g_pHeadInstanceList = NULL; 
  96.  
  97. // Associate the hWnd with pInstance by setting the hWnd member of the
  98. // PluginInstance struct.  Also, add the PluginInstance struct to the list
  99. // if necessary
  100. static void AssociateInstance(HWND hWnd, PluginInstance* pInstance)
  101. {
  102.     pInstance->hWnd = hWnd;
  103.  
  104.     // add this PluginInstance to the list if it's not already
  105.     if(g_pHeadInstanceList != NULL) { // anything in the list?
  106.         if(g_pHeadInstanceList != pInstance) { // its not first in the list
  107.             PluginInstance* pInst = g_pHeadInstanceList;
  108.             while(pInst->pNext != NULL) {
  109.                 if(pInst->pNext == pInstance)
  110.                     return; // found it, done
  111.                 pInst = pInst->pNext;
  112.             }
  113.             // didn't find it, add it
  114.             pInst->pNext = pInstance;
  115.         }
  116.     }
  117.     else // list is empty, just add it
  118.         g_pHeadInstanceList = pInstance;
  119. }
  120.  
  121. // Find the PluginInstance associated with this hWnd and return it
  122. static PluginInstance* GetInstance(HWND hWnd)
  123. {
  124.     for(PluginInstance* pInstance = g_pHeadInstanceList;
  125.         pInstance != NULL;
  126.         pInstance = pInstance->pNext) {
  127.             if(pInstance->hWnd == hWnd)
  128.                 return pInstance; // found it, done
  129.     }
  130.     return NULL;
  131. }
  132.  
  133. //----------------------------------------------------------------------------
  134. // NPP_Initialize:
  135. //----------------------------------------------------------------------------
  136. NPError NPP_Initialize(void)
  137. {
  138.     // do your one time initialization here, such as dynamically loading
  139.     // dependant DLLs
  140.     return NPERR_NO_ERROR;
  141. }
  142.  
  143. //----------------------------------------------------------------------------
  144. // NPP_Shutdown:
  145. //----------------------------------------------------------------------------
  146. void NPP_Shutdown(void)
  147. {
  148.     // do your one time uninitialization here, such as unloading dynamically
  149.     // loaded DLLs
  150. }
  151.  
  152. //----------------------------------------------------------------------------
  153. // NPP_New:
  154. //----------------------------------------------------------------------------
  155. NPError NP_LOADDS
  156. NPP_New(NPMIMEType pluginType,
  157.                 NPP instance,
  158.                 uint16 mode,
  159.                 int16 argc,
  160.                 char* argn[],
  161.                 char* argv[],
  162.                 NPSavedData* saved)
  163. {
  164.     if (instance == NULL)
  165.         return NPERR_INVALID_INSTANCE_ERROR;
  166.         
  167.     instance->pdata = NPN_MemAlloc(sizeof(PluginInstance));
  168.     PluginInstance* This = (PluginInstance*) instance->pdata;
  169.  
  170.     if (This == NULL)
  171.         return NPERR_OUT_OF_MEMORY_ERROR;
  172.     //
  173.     // *Developers*: Initialize fields of your plugin
  174.     // instance data here.  If the NPSavedData is non-
  175.     // NULL, you can use that data (returned by you from
  176.     // NPP_Destroy to set up the new plugin instance.
  177.         
  178.     This->fWindow = NULL;
  179.     // mode is NP_EMBED, NP_FULL, or NP_BACKGROUND (see npapi.h)
  180.     This->fMode = mode;
  181.     This->hWnd = NULL;
  182.     This->pSavedInstanceData = saved;
  183.     This->pNext = NULL;
  184.     
  185.     return NPERR_NO_ERROR;
  186. }
  187.  
  188. //-----------------------------------------------------------------------------
  189. // NPP_Destroy:
  190. //-----------------------------------------------------------------------------
  191. NPError NP_LOADDS
  192. NPP_Destroy(NPP instance, NPSavedData** save)
  193. {
  194.     if (instance == NULL)
  195.         return NPERR_INVALID_INSTANCE_ERROR;
  196.  
  197.     PluginInstance* This = (PluginInstance*) instance->pdata;
  198.  
  199.     // *Developers*: If desired, call NP_MemAlloc to create a
  200.     // NPSavedDate structure containing any state information
  201.     // that you want restored if this plugin instance is later
  202.     // recreated.
  203.  
  204.     if (This != NULL)
  205.     {
  206. #ifdef _PLUG_IN
  207.         theApp.EatPlugin();
  208. #endif
  209.         // Remove the subclass for the client window
  210.         if(This->hWnd) {
  211.             SetWindowLong(This->hWnd,
  212.                         GWL_WNDPROC,
  213.                         (DWORD)This->lpfnOldWndProc);
  214.         }
  215.  
  216.         // Remove this PluginInstance from the list
  217.         if(g_pHeadInstanceList != NULL) { // anything in the list?
  218.             if(g_pHeadInstanceList == This) {
  219.             // handle the head of the list here to simplify the for loop below
  220.                 g_pHeadInstanceList = This->pNext;
  221.             }
  222.             else {
  223.                 for(PluginInstance* pInstance = g_pHeadInstanceList;
  224.                     pInstance != NULL;
  225.                     pInstance = pInstance->pNext) {
  226.                         if(pInstance->pNext == This) {
  227.                             pInstance->pNext = This->pNext;
  228.                             break; // done walking list
  229.                         }
  230.                 }
  231.             }
  232.         }
  233.  
  234.         // make some saved instance data if necessary
  235.         if(This->pSavedInstanceData == NULL) {
  236.             // make a struct header for the data
  237.             This->pSavedInstanceData =
  238.                 (NPSavedData*)NPN_MemAlloc(sizeof NPSavedData);
  239.  
  240.             // fill in the struct
  241.             if(This->pSavedInstanceData != NULL) {
  242.                 This->pSavedInstanceData->len = 0;
  243.                 This->pSavedInstanceData->buf = NULL;
  244.  
  245.                 // replace the def below and references to it with your data
  246.                 #define SIDATA "aSavedInstanceDataBlock"
  247.  
  248.                 // the data
  249.                 This->pSavedInstanceData->buf = NPN_MemAlloc(sizeof SIDATA);
  250.  
  251.                 if(This->pSavedInstanceData->buf != NULL) {
  252.                     strcpy((char*)This->pSavedInstanceData->buf, SIDATA);
  253.                     This->pSavedInstanceData->len = sizeof SIDATA;
  254.                 }
  255.             }
  256.         }
  257.  
  258.         // save some instance data
  259.         *save = This->pSavedInstanceData;
  260.  
  261.         NPN_MemFree(instance->pdata);
  262.         instance->pdata = NULL;
  263.     }
  264.  
  265.     return NPERR_NO_ERROR;
  266. }
  267.  
  268. //----------------------------------------------------------------------------
  269. // NPP_SetWindow:
  270. //----------------------------------------------------------------------------
  271. NPError NP_LOADDS
  272. NPP_SetWindow(NPP instance, NPWindow* window)
  273. {
  274.     if (instance == NULL)
  275.         return NPERR_INVALID_INSTANCE_ERROR;
  276.     PluginInstance* This = (PluginInstance*) instance->pdata;
  277.     //
  278.     // *Developers*: Before setting fWindow to point to the
  279.     // new window, you may wish to compare the new window
  280.     // info to the previous window (if any) to note window
  281.     // size changes, etc.
  282.     //
  283.     if((window->window != NULL) && (This->hWnd == NULL))
  284.     {
  285.         This->fWindow = window;
  286. #ifdef _PLUG_IN
  287.     theApp.PreparePlugin((HWND)window->window);
  288. #endif
  289.         This->hWnd    = (HWND)(DWORD)This->fWindow->window;
  290.  
  291.         // subclass the window
  292.         This->lpfnOldWndProc =
  293.              (FARPROC)SetWindowLong(This->hWnd,
  294.             (WNDPROC)SetWindowLong(This->hWnd,
  295.                                    GWL_WNDPROC,
  296.                                    (DWORD)SubClassFunc);
  297.         AssociateInstance(This->hWnd, This);
  298.     }
  299.     else {
  300.         // if window handle changed
  301.         if(This->hWnd != (HWND)(DWORD)window->window) {
  302.             // remember the new window
  303.             This->fWindow = window;
  304.  
  305.             // Remove the subclass for the old client window
  306.             SetWindowLong(This->hWnd,
  307.                         GWL_WNDPROC,
  308.                         (DWORD)This->lpfnOldWndProc);
  309.  
  310.             // remember the new window handle
  311.             This->hWnd = (HWND)(DWORD)This->fWindow->window;
  312.  
  313.             if(This->hWnd != NULL) {
  314.                 // subclass the new one
  315.                 This->lpfnOldWndProc =
  316.                      (FARPROC)SetWindowLong(This->hWnd,
  317.                     (WNDPROC)SetWindowLong(This->hWnd,
  318.                                            GWL_WNDPROC,
  319.                                            (DWORD)SubClassFunc);
  320.                 AssociateInstance(This->hWnd, This);
  321.             }
  322.         }
  323.     }
  324.     return NPERR_NO_ERROR;
  325. }
  326.  
  327. //----------------------------------------------------------------------------
  328. // NPP_NewStream:
  329. //----------------------------------------------------------------------------
  330. NPError NP_LOADDS
  331. NPP_NewStream(NPP instance,
  332.               NPMIMEType type,
  333.               NPStream *stream, 
  334.               NPBool seekable,
  335.               uint16 *stype)
  336. {
  337.     if (instance == NULL)
  338.         return NPERR_INVALID_INSTANCE_ERROR;
  339.     PluginInstance* This = (PluginInstance*) instance->pdata;
  340.     
  341.     // if your plugin must operate file based, you may wish to do this:
  342.     //    *stype = NP_ASFILE;
  343.     // remember, though, that use of NP_ASFILE is strongly discouraged;
  344.     // your plugin should attempt to work with data as it comes in on
  345.     // the stream if at all possible
  346.  
  347.     return NPERR_NO_ERROR;
  348. }
  349.  
  350. //
  351. // *Developers*: 
  352. // These next 2 functions are directly relevant in a plug-in which handles the
  353. // data in a streaming manner.  If you want zero bytes because no buffer space
  354. // is YET available, return 0.  As long as the stream has not been written
  355. // to the plugin, Navigator will continue trying to send bytes.  If the plugin
  356. // doesn't want them, just return some large number from NPP_WriteReady(), and
  357. // ignore them in NPP_Write().  For a NP_ASFILE stream, they are still called
  358. // but can safely be ignored using this strategy.
  359. //
  360.  
  361. int32 STREAMBUFSIZE = 0X0FFFFFFF;   // If we are reading from a file in
  362.                                     // NP_ASFILE mode, we can take any size
  363.                                     // stream in our write call (since we
  364.                                     // ignore it)
  365.  
  366. //----------------------------------------------------------------------------
  367. // NPP_WriteReady:
  368. //----------------------------------------------------------------------------
  369. int32 NP_LOADDS
  370. NPP_WriteReady(NPP instance, NPStream *stream)
  371. {
  372.     if (instance != NULL)
  373.         PluginInstance* This = (PluginInstance*) instance->pdata;
  374.  
  375.     return STREAMBUFSIZE;   // Number of bytes ready to accept in NPP_Write()
  376. }
  377.  
  378. //----------------------------------------------------------------------------
  379. // NPP_Write:
  380. //----------------------------------------------------------------------------
  381. int32 NP_LOADDS
  382. NPP_Write(NPP instance, NPStream *stream,
  383.           int32 offset, int32 len, void *buffer)
  384. {
  385.     if (instance != NULL)
  386.         PluginInstance* This = (PluginInstance*) instance->pdata;
  387.  
  388.     return len;     // The number of bytes accepted.  Return a
  389.                     // negative number here if, e.g., there was an error
  390.                     // during plugin operation and you want to abort the
  391.                     // stream
  392. }
  393.  
  394. //----------------------------------------------------------------------------
  395. // NPP_DestroyStream:
  396. //----------------------------------------------------------------------------
  397. NPError NP_LOADDS
  398. NPP_DestroyStream(NPP instance, NPStream *stream, NPError reason)
  399. {
  400.     if (instance == NULL)
  401.         return NPERR_INVALID_INSTANCE_ERROR;
  402.     PluginInstance* This = (PluginInstance*) instance->pdata;
  403.  
  404.     return NPERR_NO_ERROR;
  405. }
  406.  
  407. //----------------------------------------------------------------------------
  408. // NPP_StreamAsFile:
  409. //----------------------------------------------------------------------------
  410. void NP_LOADDS
  411. NPP_StreamAsFile(NPP instance, NPStream *stream, const char* fname)
  412. {
  413.     if (instance == NULL)
  414.         return;
  415.  
  416.     PluginInstance* This = (PluginInstance*) instance->pdata;
  417. }
  418.  
  419. //----------------------------------------------------------------------------
  420. // NPP_Print:
  421. //----------------------------------------------------------------------------
  422. void NP_LOADDS
  423. NPP_Print(NPP instance, NPPrint* printInfo)
  424. {
  425.     if(printInfo == NULL)   // trap invalid parm
  426.         return;
  427.     if (instance != NULL)
  428.     {
  429.         PluginInstance* This = (PluginInstance*) instance->pdata;
  430.     
  431.         if (printInfo->mode == NP_FULL)
  432.         {
  433.             //
  434.             // *Developers*: If your plugin would like to take over
  435.             // printing completely when it is in full-screen mode,
  436.             // set printInfo->pluginPrinted to TRUE and print your
  437.             // plugin as you see fit.  If your plugin wants Netscape
  438.             // to handle printing in this case, set printInfo->pluginPrinted
  439.             // to FALSE (the default) and do nothing.  If you do want
  440.             // to handle printing yourself, printOne is true if the
  441.             // print button (as opposed to the print menu) was clicked.
  442.             // On the Macintosh, platformPrint is a THPrint; on Windows,
  443.             // platformPrint is a structure (defined in npapi.h) containing
  444.             // the printer name, port, etc.
  445.             //
  446.             void* platformPrint = printInfo->print.fullPrint.platformPrint;
  447.             NPBool printOne = printInfo->print.fullPrint.printOne;
  448.             
  449.             printInfo->print.fullPrint.pluginPrinted = FALSE; // Do the default
  450.         
  451.         }
  452.         else    // If not fullscreen, we must be embedded
  453.         {
  454.             //
  455.             // *Developers*: If your plugin is embedded, or is full-screen
  456.             // but you returned false in pluginPrinted above, NPP_Print
  457.             // will be called with mode == NP_EMBED.  The NPWindow
  458.             // in the printInfo gives the location and dimensions of
  459.             // the embedded plugin on the printed page.  On the Macintosh,
  460.             // platformPrint is the printer port; on Windows, platformPrint
  461.             // is the handle to the printing device context.
  462.             //
  463.             NPWindow* printWindow = &(printInfo->print.embedPrint.window);
  464.             void* platformPrint = printInfo->print.embedPrint.platformPrint;
  465.  
  466.             HPEN hPen, hPenOld;
  467. #ifdef WIN32
  468.             /* Initialize the pen's "brush" */
  469.             LOGBRUSH lb;
  470.             lb.lbStyle = BS_SOLID;
  471.             lb.lbColor = RGB(128, 128, 128);
  472.             lb.lbHatch = 0;
  473.             hPen = ExtCreatePen(PS_COSMETIC | PS_SOLID, 1, &lb, 0, NULL);
  474. #else
  475.             COLORREF cref = RGB(128, 128, 128);
  476.             hPen = CreatePen(PS_SOLID, 32, cref);
  477. #endif
  478.             HDC hDC = (HDC)(DWORD)platformPrint;
  479.             hPenOld = (HPEN)SelectObject(hDC, hPen);
  480.             BOOL result = Rectangle(hDC,
  481.                                     (int)(printWindow->x),
  482.                                     (int)(printWindow->y),
  483.                                     (int)(printWindow->x + printWindow->width),
  484.                                     (int)(printWindow->y + printWindow->height));
  485.             SelectObject(hDC, hPenOld);
  486.             DeleteObject(hPen);
  487.         }
  488.     }
  489. }
  490.  
  491. //----------------------------------------------------------------------------
  492. // NPP_HandleEvent:
  493. // Mac-only.
  494. //----------------------------------------------------------------------------
  495. int16 NPP_HandleEvent(NPP instance, void* event)
  496. {
  497.     NPBool eventHandled = FALSE;
  498.     if (instance == NULL)
  499.         return eventHandled;
  500.         
  501.     PluginInstance* This = (PluginInstance*) instance->pdata;
  502.     //
  503.     // *Developers*: The "event" passed in is a Macintosh
  504.     // EventRecord*.  The event.what field can be any of the
  505.     // normal Mac event types, or one of the following additional
  506.     // types defined in npapi.h: getFocusEvent, loseFocusEvent,
  507.     // adjustCursorEvent.  The focus events inform your plugin
  508.     // that it will become, or is no longer, the recepient of
  509.     // key events.  If your plugin doesn't want to receive key
  510.     // events, return false when passed at getFocusEvent.  The
  511.     // adjustCursorEvent is passed repeatedly when the mouse is
  512.     // over your plugin; if your plugin doesn't want to set the
  513.     // cursor, return false.  Handle the standard Mac events as
  514.     // normal.  The return value for all standard events is currently
  515.     // ignored except for the key event: for key events, only return
  516.     // true if your plugin has handled that particular key event. 
  517.     //
  518.     return eventHandled;
  519. }
  520. //
  521. // Here is a sample subclass function.
  522. //
  523. LONG NP_LOADDS WINAPI 
  524. SubClassFunc(  HWND hWnd,
  525.                WORD Message,
  526.                WORD wParam,
  527.                LONG lParam)
  528. {
  529.     PluginInstance *This = GetInstance(hWnd);
  530.     switch(Message) {
  531.     case WM_PALETTECHANGED:
  532.         InvalidateRect(hWnd, NULL, TRUE);
  533.         UpdateWindow(hWnd);    
  534.         return 0;
  535.         break;
  536.  
  537.     case WM_PAINT:
  538.         {
  539.         HBRUSH hBrush, hBrushOld;
  540.         PAINTSTRUCT paint;
  541.         HDC hDC = BeginPaint(hWnd, &paint);
  542.  
  543.         hBrush = CreateSolidBrush(BACKGROUND_COLOR);
  544.         hBrushOld = (HPEN)SelectObject(hDC, hBrush);
  545.     hBrush = (HBRUSH)SelectObject(hDC, hBrush);
  546.  
  547.         BOOL result = FillRect(hDC, &paint.rcPaint, hBrush);
  548.         SelectObject(hDC, hBrushOld);
  549.         DeleteObject(hBrush);
  550.         EndPaint(hWnd, &paint);
  551.         }
  552.         return 0;
  553.         break;
  554.     default:
  555.         break;
  556.     }
  557.     return CallWindowProc(This->lpfnOldWndProc,
  558.                           hWnd,
  559.                           Message,
  560.                           wParam,
  561.                           lParam);
  562. }
  563.  
  564. // NPP_GetJavaClass
  565. //
  566. //    Return the Java class representing this plugin
  567. //
  568. jref 
  569. NPP_GetJavaClass(void)
  570. {
  571.     //\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//.
  572.     // get the Java environment. You need this information pretty much for
  573.     // any jri (Java Runtime Interface) call. 
  574.     JRIEnv* env = NPN_GetJavaEnv();
  575.     //\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//\\//.
  576.     // init any classes that define native methods.
  577.     // The following functions are generated by javah running on the java
  578.     // class(es) representing this plugin. javah generates the files
  579.     // <java class name>.h and <java class name>.c (same for any additional
  580.     // class you may want to use)
  581.     // Return the main java class representing this plugin (derives from
  582.     // Plugin class on the java side)
  583. //    return init_AviPlayer(env);
  584.     // if no java is used
  585.      return NULL;
  586. }
  587.  
  588.