/* * The contents of this file are subject to the Mozilla Public * License Version 1.1 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS * IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or * implied. See the License for the specific language governing * rights and limitations under the License. * * The Original Code is the Netscape security libraries. * * The Initial Developer of the Original Code is Netscape * Communications Corporation. Portions created by Netscape are * Copyright (C) 1994-2000 Netscape Communications Corporation. All * Rights Reserved. * * Contributor(s): * * Adam Lock * * Alternatively, the contents of this file may be used under the * terms of the GNU General Public License Version 2 or later (the * "GPL"), in which case the provisions of the GPL are applicable * instead of those above. If you wish to allow use of your * version of this file only under the terms of the GPL and not to * allow others to use your version of this file under the MPL, * indicate your decision by deleting the provisions above and * replace them with the notice and other provisions required by * the GPL. If you do not delete the provisions above, a recipient * may use your version of this file under either the MPL or the * GPL. */ #include "stdafx.h" #include "Pluginhostctrl.h" #include "nsPluginHostCtrl.h" #include "nsURLDataCallback.h" #include "npn.h" #define NS_4XPLUGIN_CALLBACK(_type, _name) _type (__stdcall * _name) typedef NS_4XPLUGIN_CALLBACK(NPError, NP_GETENTRYPOINTS) (NPPluginFuncs* pCallbacks); typedef NS_4XPLUGIN_CALLBACK(NPError, NP_PLUGININIT) (const NPNetscapeFuncs* pCallbacks); typedef NS_4XPLUGIN_CALLBACK(NPError, NP_PLUGINSHUTDOWN) (void); const kArraySizeIncrement = 10; nsSimpleArray nsPluginHostCtrl::m_LoadedPlugins; ///////////////////////////////////////////////////////////////////////////// // nsPluginHostCtrl nsPluginHostCtrl::nsPluginHostCtrl() { m_bWindowOnly = TRUE; m_bPluginIsAlive = FALSE; m_bCreatePluginFromStreamData = FALSE; m_pLoadedPlugin = NULL; m_nArgs = 0; m_nArgsMax = 0; m_pszArgNames = NULL; m_pszArgValues = NULL; InitPluginCallbacks(); memset(&m_NPPFuncs, 0, sizeof(m_NPPFuncs)); } nsPluginHostCtrl::~nsPluginHostCtrl() { } LRESULT nsPluginHostCtrl::OnCreate(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/) { SetWindowLong(GWL_STYLE, GetWindowLong(GWL_STYLE) | WS_CLIPCHILDREN); // Load a list of plugins CreatePluginList( PLUGINS_FROM_IE | PLUGINS_FROM_NS4X | PLUGINS_FROM_NS6X | PLUGINS_FROM_MOZILLA); HRESULT hr = E_FAIL; if (m_bstrContentType.Length() == 0 && m_bstrSource.Length() != 0) { USES_CONVERSION; // Do a late instantiation of the plugin based on the content type of // the stream data. m_bCreatePluginFromStreamData = TRUE; hr = OpenURLStream(OLE2T(m_bstrSource), NULL, NULL, 0); } else { // Create a plugin based upon the specified content type property USES_CONVERSION; hr = LoadPluginByContentType(OLE2T(m_bstrContentType)); if (SUCCEEDED(hr)) { hr = CreatePluginInstance(); if (m_bstrSource.Length()) { OpenURLStream(OLE2T(m_bstrSource), NULL, NULL, 0); } } } return SUCCEEDED(hr) ? 0 : -1; } LRESULT nsPluginHostCtrl::OnDestroy(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/) { DestroyPluginInstance(); UnloadPlugin(); CleanupPluginList(); return 0; } LRESULT nsPluginHostCtrl::OnSize(UINT /*uMsg*/, WPARAM /*wParam*/, LPARAM /*lParam*/, BOOL& /*bHandled*/) { SizeToFitPluginInstance(); return 0; } LRESULT nsPluginHostCtrl::OnPaint(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled) { PAINTSTRUCT ps; HDC hdc; RECT rc; hdc = BeginPaint(&ps); GetClientRect(&rc); FillRect(hdc, &rc, (HBRUSH) GetStockObject(GRAY_BRUSH)); EndPaint(&ps); return 0; } NPNetscapeFuncs nsPluginHostCtrl::g_NPNFuncs; HRESULT nsPluginHostCtrl::InitPluginCallbacks() { static BOOL gCallbacksSet = FALSE; if (gCallbacksSet) { return S_OK; } gCallbacksSet = TRUE; memset(&g_NPNFuncs, 0, sizeof(g_NPNFuncs)); g_NPNFuncs.size = sizeof(g_NPNFuncs); g_NPNFuncs.version = (NP_VERSION_MAJOR << 8) + NP_VERSION_MINOR; g_NPNFuncs.geturl = NewNPN_GetURLProc(NPN_GetURL); g_NPNFuncs.posturl = NewNPN_PostURLProc(NPN_PostURL); g_NPNFuncs.requestread = NewNPN_RequestReadProc(NPN_RequestRead); g_NPNFuncs.newstream = NewNPN_NewStreamProc(NPN_NewStream); g_NPNFuncs.write = NewNPN_WriteProc(NPN_Write); g_NPNFuncs.destroystream = NewNPN_DestroyStreamProc(NPN_DestroyStream); g_NPNFuncs.status = NewNPN_StatusProc(NPN_Status); g_NPNFuncs.uagent = NewNPN_UserAgentProc(NPN_UserAgent); g_NPNFuncs.memalloc = NewNPN_MemAllocProc(NPN_MemAlloc); g_NPNFuncs.memfree = NewNPN_MemFreeProc(NPN_MemFree); g_NPNFuncs.memflush = NewNPN_MemFlushProc(NPN_MemFlush); g_NPNFuncs.reloadplugins = NewNPN_ReloadPluginsProc(NPN_ReloadPlugins); g_NPNFuncs.getJavaEnv = NewNPN_GetJavaEnvProc(NPN_GetJavaEnv); g_NPNFuncs.getJavaPeer = NewNPN_GetJavaPeerProc(NPN_GetJavaPeer); g_NPNFuncs.geturlnotify = NewNPN_GetURLNotifyProc(NPN_GetURLNotify); g_NPNFuncs.posturlnotify = NewNPN_PostURLNotifyProc(NPN_PostURLNotify); g_NPNFuncs.getvalue = NewNPN_GetValueProc(NPN_GetValue); g_NPNFuncs.setvalue = NewNPN_SetValueProc(NPN_SetValue); g_NPNFuncs.invalidaterect = NewNPN_InvalidateRectProc(NPN_InvalidateRect); g_NPNFuncs.invalidateregion = NewNPN_InvalidateRegionProc(NPN_InvalidateRegion); g_NPNFuncs.forceredraw = NewNPN_ForceRedrawProc(NPN_ForceRedraw); return S_OK; } HRESULT nsPluginHostCtrl::GetWebBrowserApp(IWebBrowserApp **pBrowser) { ATLASSERT(pBrowser); if (!pBrowser) { return E_INVALIDARG; } // Get the web browser through the site the control is attached to. // Note: The control could be running in some other container than IE // so code shouldn't expect this function to work all the time. CComPtr cpWebBrowser; CComQIPtr cpServiceProvider = m_spClientSite; HRESULT hr; if (cpServiceProvider) { hr = cpServiceProvider->QueryService(IID_IWebBrowserApp, &cpWebBrowser); } if (!cpWebBrowser) { return E_FAIL; } *pBrowser = cpWebBrowser; (*pBrowser)->AddRef(); return S_OK; } HRESULT nsPluginHostCtrl::GetBaseURL(TCHAR **ppszBaseURL) { ATLASSERT(ppszBaseURL); *ppszBaseURL = NULL; CComPtr cpWebBrowser; GetWebBrowserApp(&cpWebBrowser); if (!cpWebBrowser) { return E_FAIL; } USES_CONVERSION; CComBSTR bstrURL; cpWebBrowser->get_LocationURL(&bstrURL); DWORD cbBaseURL = (bstrURL.Length() + 1) * sizeof(WCHAR); DWORD cbBaseURLUsed = 0; WCHAR *pszBaseURL = (WCHAR *) malloc(cbBaseURL); ATLASSERT(pszBaseURL); CoInternetParseUrl( bstrURL.m_str, PARSE_ROOTDOCUMENT, 0, pszBaseURL, cbBaseURL, &cbBaseURLUsed, 0); *ppszBaseURL = _tcsdup(W2T(pszBaseURL)); free(pszBaseURL); return S_OK; } HRESULT nsPluginHostCtrl::LoadPluginByContentType(const TCHAR *pszContentType) { TCHAR * pszPluginPath = NULL; // Set the content type USES_CONVERSION; put_PluginContentType(T2OLE(pszContentType)); // Search for a plugin that can handle this content HRESULT hr = FindPluginPathByContentType(pszContentType, &pszPluginPath); if (FAILED(hr)) { // Try the default 'catch-all' plugin hr = FindPluginPathByContentType(_T("*"), &pszPluginPath); if (FAILED(hr)) { return hr; } } hr = LoadPlugin(pszPluginPath); free(pszPluginPath); return hr; } HRESULT nsPluginHostCtrl::CreatePluginList(unsigned long ulFlags) { // This function trawls through the plugin directory and builds a list // of plugins and what MIME types each plugin handles. CleanupPluginList(); // Try and obtain a path to the plugins in Netscape 6.x or Mozilla if (ulFlags & PLUGINS_FROM_NS6X || ulFlags & PLUGINS_FROM_MOZILLA) { // TODO search for Mozilla/NS 6.x plugins dir } // Try and obtain a path to the plugins in Netscape 4.x if (ulFlags & PLUGINS_FROM_NS4X) { TCHAR szPluginsDir[_MAX_PATH]; memset(szPluginsDir, 0, sizeof(szPluginsDir)); CRegKey keyNS; const TCHAR *kNav4xKey = _T("Software\\Netscape\\Netscape Navigator"); if (keyNS.Open(HKEY_LOCAL_MACHINE, kNav4xKey, KEY_READ) == ERROR_SUCCESS) { TCHAR szVersion[10]; DWORD nVersion = sizeof(szVersion) / sizeof(szVersion[0]); keyNS.QueryValue(szVersion, _T("CurrentVersion"), &nVersion); CRegKey keyVersion; if (keyVersion.Open(keyNS, szVersion, KEY_READ) == ERROR_SUCCESS) { CRegKey keyMain; if (keyMain.Open(keyVersion, _T("Main"), KEY_READ) == ERROR_SUCCESS) { DWORD nPluginsDir = sizeof(szPluginsDir) / sizeof(szPluginsDir[0]); keyMain.QueryValue(szPluginsDir, _T("Plugins Directory"), &nPluginsDir); keyMain.Close(); } keyVersion.Close(); } keyNS.Close(); } if (szPluginsDir[0]) { CreatePluginListFrom(szPluginsDir); } } // Try and obtain a path to the plugins in Internet Explorer if (ulFlags & PLUGINS_FROM_IE) { TCHAR szPluginsDir[_MAX_PATH]; memset(szPluginsDir, 0, sizeof(szPluginsDir)); CRegKey keyIE; const TCHAR *kIEKey = _T("Software\\Microsoft\\Windows\\CurrentVersion\\App Paths\\IEXPLORE.EXE"); if (keyIE.Open(HKEY_LOCAL_MACHINE, kIEKey, KEY_READ) == ERROR_SUCCESS) { DWORD nPluginsDir = sizeof(szPluginsDir) / sizeof(szPluginsDir[0]); keyIE.QueryValue(szPluginsDir, _T("Path"), &nPluginsDir); TCHAR *szSemiColon = _tcschr(szPluginsDir, _TCHAR(';')); if (szSemiColon) { *szSemiColon = _TCHAR('\0'); } ULONG nLen = _tcslen(szPluginsDir); if (nLen > 0 && szPluginsDir[nLen - 1] == _TCHAR('\\')) { szPluginsDir[nLen - 1] = _TCHAR('\0'); } _tcscat(szPluginsDir, _T("\\Plugins")); keyIE.Close(); } if (szPluginsDir[0]) { CreatePluginListFrom(szPluginsDir); } } return S_OK; } HRESULT nsPluginHostCtrl::CreatePluginListFrom(const TCHAR *szPluginsDir) { HANDLE hFind; WIN32_FIND_DATA finddata; // Change to the plugin directory TCHAR szCurrentDir[MAX_PATH + 1]; GetCurrentDirectory(sizeof(szCurrentDir) / sizeof(szCurrentDir[0]), szCurrentDir); SetCurrentDirectory(szPluginsDir); // Search for files matching the "np*dll" pattern hFind = FindFirstFile(_T("np*dll"), &finddata); if (hFind != INVALID_HANDLE_VALUE) { do { PluginInfo *pInfo = new PluginInfo; if (!pInfo) { CleanupPluginList(); SetCurrentDirectory(szCurrentDir); return E_OUTOFMEMORY; } if (SUCCEEDED(GetPluginInfo(finddata.cFileName, pInfo))) { pInfo->szPluginName = _tcsdup(finddata.cFileName); pInfo->szPluginPath = _tcsdup(szPluginsDir); m_Plugins.AppendElement(pInfo); } else { ATLTRACE(_T("Error: Cannot plugin info for \"%s\".\n"), finddata.cFileName); delete pInfo; } } while (FindNextFile(hFind, &finddata)); FindClose(hFind); } SetCurrentDirectory(szCurrentDir); return S_OK; } HRESULT nsPluginHostCtrl::CleanupPluginList() { // Free the memory used by the plugin info list for (unsigned long i = 0; i < m_Plugins.Count(); i++) { PluginInfo *pI = m_Plugins[i]; if (pI->szMIMEType) free(pI->szMIMEType); if (pI->szPluginName) free(pI->szPluginName); if (pI->szPluginPath) free(pI->szPluginPath); free(pI); } return S_OK; } HRESULT nsPluginHostCtrl::GetPluginInfo(const TCHAR *pszPluginPath, PluginInfo *pInfo) { // Get the version info from the plugin USES_CONVERSION; DWORD nVersionInfoSize; DWORD nZero = 0; void *pVersionInfo = NULL; nVersionInfoSize = GetFileVersionInfoSize((TCHAR *)pszPluginPath, &nZero); if (nVersionInfoSize) { pVersionInfo = malloc(nVersionInfoSize); } if (!pVersionInfo) { return E_OUTOFMEMORY; } GetFileVersionInfo((TCHAR *)pszPluginPath, NULL, nVersionInfoSize, pVersionInfo); // Extract the MIMEType info TCHAR *szValue = NULL; UINT nValueLength = 0; if (!VerQueryValue(pVersionInfo, _T("\\StringFileInfo\\040904E4\\MIMEType"), (void **) &szValue, &nValueLength)) { return E_FAIL; } if (pInfo) { pInfo->szMIMEType = _tcsdup(szValue); } free(pVersionInfo); return S_OK; } HRESULT nsPluginHostCtrl::FindPluginPathByContentType(const TCHAR *pszContentType, TCHAR **ppszPluginPath) { *ppszPluginPath = NULL; // Search the list of plugins for one that will handle the content type TCHAR szPluginPath[_MAX_PATH + 1]; unsigned long nContentType = _tcslen(pszContentType); for (unsigned long i = 0; i < m_Plugins.Count(); i++) { PluginInfo *pI = m_Plugins[i]; if (pI->szMIMEType) { TCHAR *pszMIMEType = pI->szMIMEType; do { if (_tcsncmp(pszContentType, pszMIMEType, nContentType) == 0) { // Found a match _tmakepath(szPluginPath, NULL, pI->szPluginPath, pI->szPluginName, NULL); *ppszPluginPath = _tcsdup(szPluginPath); return S_OK; } // Check the other types the plugin handles pszMIMEType = _tcschr(pszMIMEType, TCHAR('|')); if (pszMIMEType) { pszMIMEType++; } } while (pszMIMEType && *pszMIMEType); } } return E_FAIL; } HRESULT nsPluginHostCtrl::LoadPlugin(const TCHAR *szPluginPath) { ATLASSERT(m_pLoadedPlugin == NULL); if (m_pLoadedPlugin) { return E_UNEXPECTED; } // TODO critical section // Test if the plugin has already been loaded for (unsigned long i = 0; i < m_LoadedPlugins.Count(); i++) { if (_tcscmp(m_LoadedPlugins[i]->szFullPath, szPluginPath) == 0) { m_pLoadedPlugin = m_LoadedPlugins[i]; memcpy(&m_NPPFuncs, &m_pLoadedPlugin->NPPFuncs, sizeof(m_NPPFuncs)); m_pLoadedPlugin->nRefCount++; return S_OK; } } // Plugin library is being loaded for the first time so initialise it // and store an entry in the loaded plugins array. HINSTANCE hInstance = LoadLibrary(szPluginPath); if (!hInstance) { return E_FAIL; } m_pLoadedPlugin = new LoadedPluginInfo; if (!m_pLoadedPlugin) { ATLASSERT(m_pLoadedPlugin); return E_OUTOFMEMORY; } // Get the plugin function entry points NP_GETENTRYPOINTS pfnGetEntryPoints = (NP_GETENTRYPOINTS) GetProcAddress(hInstance, "NP_GetEntryPoints"); if (pfnGetEntryPoints) { pfnGetEntryPoints(&m_NPPFuncs); } // Tell the plugin to initialize itself NP_PLUGININIT pfnInitialize = (NP_PLUGININIT) GetProcAddress(hInstance, "NP_Initialize"); if (!pfnInitialize) { pfnInitialize = (NP_PLUGININIT) GetProcAddress(hInstance, "NP_PluginInit"); } if (pfnInitialize) { pfnInitialize(&g_NPNFuncs); } // Create a new entry for the plugin m_pLoadedPlugin->szFullPath = _tcsdup(szPluginPath); m_pLoadedPlugin->nRefCount = 1; m_pLoadedPlugin->hInstance = hInstance; memcpy(&m_pLoadedPlugin->NPPFuncs, &m_NPPFuncs, sizeof(m_NPPFuncs)); // Add it to the array m_LoadedPlugins.AppendElement(m_pLoadedPlugin); return S_OK; } HRESULT nsPluginHostCtrl::UnloadPlugin() { if (!m_pLoadedPlugin) { return E_FAIL; } // TODO critical section ATLASSERT(m_pLoadedPlugin->nRefCount > 0); if (m_pLoadedPlugin->nRefCount == 1) { NP_PLUGINSHUTDOWN pfnShutdown = (NP_PLUGINSHUTDOWN) GetProcAddress( m_pLoadedPlugin->hInstance, "NP_Shutdown"); if (pfnShutdown) { pfnShutdown(); } FreeLibrary(m_pLoadedPlugin->hInstance); // Delete the entry from the array m_LoadedPlugins.RemoveElement(m_pLoadedPlugin); free(m_pLoadedPlugin->szFullPath); delete m_pLoadedPlugin; } else { m_pLoadedPlugin->nRefCount--; } m_pLoadedPlugin = NULL; return S_OK; } HRESULT nsPluginHostCtrl::AddPluginParam(const char *szName, const char *szValue) { ATLASSERT(szName); ATLASSERT(szValue); if (!szName || !szValue) { return E_INVALIDARG; } // Skip params that already there for (unsigned long i = 0; i < m_nArgs; i++) { if (stricmp(szName, m_pszArgNames[i]) == 0) { return S_OK; } } // Add the value if (!m_pszArgNames) { ATLASSERT(!m_pszArgValues); m_nArgsMax = kArraySizeIncrement; m_pszArgNames = (char **) malloc(sizeof(char *) * m_nArgsMax); m_pszArgValues = (char **) malloc(sizeof(char *) * m_nArgsMax); } else if (m_nArgs == m_nArgsMax) { m_nArgsMax += kArraySizeIncrement; m_pszArgNames = (char **) realloc(m_pszArgNames, sizeof(char *) * m_nArgsMax); m_pszArgValues = (char **) realloc(m_pszArgValues, sizeof(char *) * m_nArgsMax); } if (!m_pszArgNames || !m_pszArgValues) { return E_OUTOFMEMORY; } m_pszArgNames[m_nArgs] = strdup(szName); m_pszArgValues[m_nArgs] = strdup(szValue); m_nArgs++; return S_OK; } HRESULT nsPluginHostCtrl::CreatePluginInstance() { m_NPP.pdata = NULL; m_NPP.ndata = this; USES_CONVERSION; char *szContentType = strdup(OLE2A(m_bstrContentType.m_str)); // Create a child window to house the plugin RECT rc; GetClientRect(&rc); m_wndPlugin.Create(m_hWnd, rc, NULL, WS_CHILD | WS_VISIBLE); m_NPWindow.window = (void *) m_wndPlugin.m_hWnd; m_NPWindow.type = NPWindowTypeWindow; if (m_NPPFuncs.newp) { // Create the arguments to be fed into the plugin if (m_bstrSource.m_str) { AddPluginParam("SRC", OLE2A(m_bstrSource.m_str)); } if (m_bstrContentType.m_str) { AddPluginParam("TYPE", OLE2A(m_bstrContentType.m_str)); } if (m_bstrPluginsPage.m_str) { AddPluginParam("PLUGINSPAGE", OLE2A(m_bstrPluginsPage.m_str)); } char szTmp[50]; sprintf(szTmp, "%d", (int) (rc.right - rc.left)); AddPluginParam("WIDTH", szTmp); sprintf(szTmp, "%d", (int) (rc.bottom - rc.top)); AddPluginParam("HEIGHT", szTmp); NPSavedData *pSaved = NULL; // Create the plugin instance NPError npres = m_NPPFuncs.newp(szContentType, &m_NPP, NP_EMBED, (short) m_nArgs, m_pszArgNames, m_pszArgValues, pSaved); if (npres != NPERR_NO_ERROR) { return E_FAIL; } } m_bPluginIsAlive = TRUE; SizeToFitPluginInstance(); return S_OK; } HRESULT nsPluginHostCtrl::DestroyPluginInstance() { if (!m_bPluginIsAlive) { return S_OK; } // Destroy the plugin if (m_NPPFuncs.destroy) { NPSavedData *pSavedData = NULL; NPError npres = m_NPPFuncs.destroy(&m_NPP, &pSavedData); // TODO could store saved data instead of just deleting it. if (pSavedData && pSavedData->buf) { NPN_MemFree(pSavedData->buf); } } // Destroy the arguments if (m_pszArgNames) { for (unsigned long i = 0; i < m_nArgs; i++) { free(m_pszArgNames[i]); } free(m_pszArgNames); m_pszArgNames = NULL; } if (m_pszArgValues) { for (unsigned long i = 0; i < m_nArgs; i++) { free(m_pszArgValues[i]); } free(m_pszArgValues); m_pszArgValues = NULL; } m_wndPlugin.DestroyWindow(); m_bPluginIsAlive = FALSE; return S_OK; } HRESULT nsPluginHostCtrl::SizeToFitPluginInstance() { if (!m_bPluginIsAlive) { return S_OK; } // Resize the plugin to fit the window RECT rc; GetClientRect(&rc); m_wndPlugin.SetWindowPos(HWND_TOP, rc.left, rc.top, rc.right - rc.left, rc.bottom - rc.top, SWP_NOZORDER); m_NPWindow.x = 0; m_NPWindow.y = 0; m_NPWindow.width = rc.right - rc.left; m_NPWindow.height = rc.bottom - rc.top; m_NPWindow.clipRect.left = 0; m_NPWindow.clipRect.top = 0; m_NPWindow.clipRect.right = m_NPWindow.width; m_NPWindow.clipRect.bottom = m_NPWindow.height; if (m_NPPFuncs.setwindow) { NPError npres = m_NPPFuncs.setwindow(&m_NPP, &m_NPWindow); } return S_OK; } HRESULT nsPluginHostCtrl::OpenURLStream(const TCHAR *szURL, void *pNotifyData, const void *pPostData, unsigned long nPostDataLength) { nsURLDataCallback::OpenURL(this, szURL, pNotifyData, pPostData, nPostDataLength); return S_OK; } /////////////////////////////////////////////////////////////////////////////// // IMozPluginHostCtrl STDMETHODIMP nsPluginHostCtrl::Load(LPPROPERTYBAG pPropBag, LPERRORLOG pErrorLog) { CComQIPtr cpPropBag2 = pPropBag; if (cpPropBag2) { // Read *all* the properties via IPropertyBag2 and store them somewhere // so they can be fed into the plugin instance at creation.. ULONG nProperties; cpPropBag2->CountProperties(&nProperties); if (nProperties > 0) { PROPBAG2 *pProperties = (PROPBAG2 *) malloc(sizeof(PROPBAG2) * nProperties); ULONG nPropertiesGotten = 0; cpPropBag2->GetPropertyInfo(0, nProperties, pProperties, &nPropertiesGotten); for (ULONG i = 0; i < nPropertiesGotten; i++) { if (pProperties[i].vt == VT_BSTR) { USES_CONVERSION; CComVariant v; HRESULT hrRead; cpPropBag2->Read(1, &pProperties[i], NULL, &v, &hrRead); AddPluginParam(OLE2A(pProperties[i].pstrName), OLE2A(v.bstrVal)); } if (pProperties[i].pstrName) { CoTaskMemFree(pProperties[i].pstrName); } } free(pProperties); } } return IPersistPropertyBagImpl::Load(pPropBag, pErrorLog); } /////////////////////////////////////////////////////////////////////////////// // IMozPluginHostCtrl STDMETHODIMP nsPluginHostCtrl::get_PluginContentType(BSTR *pVal) { if (!pVal) { return E_INVALIDARG; } *pVal = m_bstrContentType.Copy(); return S_OK; } STDMETHODIMP nsPluginHostCtrl::put_PluginContentType(BSTR newVal) { // Security. Copying the source BSTR this way ensures that embedded NULL // characters do not end up in the destination BSTR. SysAllocString will // create a copy truncated at the first NULL char. m_bstrContentType.Empty(); m_bstrContentType.Attach(SysAllocString(newVal)); return S_OK; } STDMETHODIMP nsPluginHostCtrl::get_PluginSource(BSTR *pVal) { if (!pVal) { return E_INVALIDARG; } *pVal = m_bstrSource.Copy(); return S_OK; } STDMETHODIMP nsPluginHostCtrl::put_PluginSource(BSTR newVal) { // Security. Copying the source BSTR this way ensures that embedded NULL // characters do not end up in the destination BSTR. SysAllocString will // create a copy truncated at the first NULL char. m_bstrSource.Empty(); m_bstrSource.Attach(SysAllocString(newVal)); return S_OK; } STDMETHODIMP nsPluginHostCtrl::get_PluginsPage(BSTR *pVal) { if (!pVal) { return E_INVALIDARG; } *pVal = m_bstrPluginsPage.Copy(); return S_OK; } STDMETHODIMP nsPluginHostCtrl::put_PluginsPage(BSTR newVal) { // Security. Copying the source BSTR this way ensures that embedded NULL // characters do not end up in the destination BSTR. SysAllocString will // create a copy truncated at the first NULL char. m_bstrPluginsPage.Empty(); m_bstrPluginsPage.Attach(SysAllocString(newVal)); return S_OK; }