summaryrefslogtreecommitdiffstats
path: root/other-licenses/nsis/Contrib/WebBrowser/CustomFunctions.cpp
blob: 4ad004b841f04ab706c8c9f0e438da2fd307e21f (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
// This Source Code Form is subject to the terms of the Mozilla Public
// License, v. 2.0.If a copy of the MPL was not distributed with this
// file, You can obtain one at http://mozilla.org/MPL/2.0/.

#include "WebBrowser.h"
#include "exdll.h"

extern WebBrowser* gBrowser;
void Init(HWND hWndParent, int string_size, TCHAR* variables,
          stack_t** stacktop, extra_parameters* extra);

static void CustomFunctionWrapper(void* NSISFunctionAddr, VARIANT jsArg,
                                  VARIANT* retVal) {
  // Marshal the argument passed to the JavaScript function onto the NSIS stack.
  switch (jsArg.vt) {
    case VT_BSTR:
      pushstring(jsArg.bstrVal);
      break;
    case VT_I4: {
      TCHAR intArgStr[32] = _T("");
      _itot_s(jsArg.intVal, intArgStr, 10);
      pushstring(intArgStr);
      break;
    }
    case VT_BOOL:
      pushstring(jsArg.boolVal == VARIANT_TRUE ? _T("1") : _T("0"));
      break;
    default:
      // No other argument types supported.
      pushstring(_T(""));
      break;
  }

  // Call the NSIS function.
  int rv = g_executeCodeSegment((int)NSISFunctionAddr, nullptr);

  // Retrieve the return value from the NSIS stack.
  TCHAR* nsisRetval =
      (TCHAR*)HeapAlloc(GetProcessHeap(), 0, g_stringsize * sizeof(TCHAR));
  popstring(nsisRetval);

  // Pass the return value back to JavaScript, if it asked for one.
  if (retVal) {
    VariantInit(retVal);
    retVal->vt = VT_BSTR;
    retVal->bstrVal = SysAllocString(nsisRetval);
  }

  HeapFree(GetProcessHeap(), 0, nsisRetval);
}

PLUGINFUNCTION(RegisterCustomFunction) {
  if (!gBrowser) {
    Init(hWndParent, string_size, variables, stacktop, extra);
  }

  TCHAR* funcAddrStr =
      (TCHAR*)HeapAlloc(GetProcessHeap(), 0, g_stringsize * sizeof(TCHAR));
  popstring(funcAddrStr);

  TCHAR* funcName =
      (TCHAR*)HeapAlloc(GetProcessHeap(), 0, g_stringsize * sizeof(TCHAR));
  popstring(funcName);

  if (gBrowser && funcAddrStr && funcName) {
    // Apparently GetFunctionAddress returnes a 1-indexed offset, but
    // ExecuteCodeSegment expects a 0-indexed one. Or something.
    uintptr_t funcAddr = static_cast<uintptr_t>(_ttoi64(funcAddrStr)) - 1;
    gBrowser->AddCustomFunction(funcName, CustomFunctionWrapper,
                                (void*)funcAddr);
  }

  HeapFree(GetProcessHeap(), 0, funcName);
  HeapFree(GetProcessHeap(), 0, funcAddrStr);
}