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