Creating a New Electron Browser Module

Welcome to the Electron API guide! If you are unfamiliar with creating a new Electron API module within the browser directory, this guide serves as a checklist for some of the necessary steps that you will need to implement.

This is not a comprehensive end-all guide to creating an Electron Browser API, rather an outline documenting some of the more unintuitive steps.

Add your files to Electron’s project configuration

Electron uses GN as a meta build system to generate files for its compiler, Ninja. This means that in order to tell Electron to compile your code, we have to add your API’s code and header file names into filenames.gni.

You will need to append your API file names alphabetically into the appropriate files like so:

```cpp title=’filenames.gni’ lib_sources = [ “path/to/api/api_name.cc”, “path/to/api/api_name.h”, ]

lib_sources_mac = [ “path/to/api/api_name_mac.h”, “path/to/api/api_name_mac.mm”, ]

lib_sources_win = [ “path/to/api/api_name_win.cc”, “path/to/api/api_name_win.h”, ]

lib_sources_linux = [ “path/to/api/api_name_linux.cc”, “path/to/api/api_name_linux.h”, ]

  1. Note that the Windows, macOS and Linux array additions are optional and should only be added if your API has specific platform implementations.
  2. ## Create API documentation
  3. Type definitions are generated by Electron using [`@electron/docs-parser`](https://github.com/electron/docs-parser) and [`@electron/typescript-definitions`](https://github.com/electron/typescript-definitions). This step is necessary to ensure consistency across Electron's API documentation. This means that for your API type definition to appear in the `electron.d.ts` file, we must create a `.md` file. Examples can be found in [this folder](https://github.com/electron/electron/tree/main/docs/api).
  4. ## Set up `ObjectTemplateBuilder` and `Wrappable`
  5. Electron constructs its modules using [`object_template_builder`](https://www.electronjs.org/blog/from-native-to-js#mateobjecttemplatebuilder).
  6. [`wrappable`](https://chromium.googlesource.com/chromium/src/+/refs/heads/main/gin/wrappable.h) is a base class for C++ objects that have corresponding v8 wrapper objects.
  7. Here is a basic example of code that you may need to add, in order to incorporate `object_template_builder` and `wrappable` into your API. For further reference, you can find more implementations [here](https://github.com/electron/electron/tree/main/shell/browser/api).
  8. In your `api_name.h` file:
  9. ```cpp title='api_name.h'
  10. #ifndef ELECTRON_SHELL_BROWSER_API_ELECTRON_API_{API_NAME}_H_
  11. #define ELECTRON_SHELL_BROWSER_API_ELECTRON_API_{API_NAME}_H_
  12. #include "gin/handle.h"
  13. #include "gin/wrappable.h"
  14. namespace electron {
  15. namespace api {
  16. class ApiName : public gin::Wrappable<ApiName> {
  17. public:
  18. static gin::Handle<ApiName> Create(v8::Isolate* isolate);
  19. // gin::Wrappable
  20. static gin::WrapperInfo kWrapperInfo;
  21. gin::ObjectTemplateBuilder GetObjectTemplateBuilder(
  22. v8::Isolate* isolate) override;
  23. const char* GetTypeName() override;
  24. } // namespace api
  25. } // namespace electron

In your api_name.cc file:

```cpp title=’api_name.cc’

include “shell/browser/api/electron_api_safe_storage.h”

include “shell/browser/browser.h”

include “shell/common/gin_converters/base_converter.h”

include “shell/common/gin_converters/callback_converter.h”

include “shell/common/gin_helper/dictionary.h”

include “shell/common/gin_helper/object_template_builder.h”

include “shell/common/node_includes.h”

include “shell/common/platform_util.h”

namespace electron {

namespace api {

gin::WrapperInfo ApiName::kWrapperInfo = {gin::kEmbedderNativeGin};

gin::ObjectTemplateBuilder ApiName::GetObjectTemplateBuilder( v8::Isolate* isolate) { return gin::ObjectTemplateBuilder(isolate) .SetMethod(“methodName”, &ApiName::methodName); }

const char* ApiName::GetTypeName() { return “ApiName”; }

// static gin::Handle ApiName::Create(v8::Isolate* isolate) { return gin::CreateHandle(isolate, new ApiName()); }

} // namespace api

} // namespace electron

namespace {

void Initialize(v8::Local exports, v8::Local unused, v8::Local context, void priv) { v8::Isolate isolate = context->GetIsolate(); gin_helper::Dictionary dict(isolate, exports); dict.Set(“apiName”, electron::api::ApiName::Create(isolate)); }

} // namespace

  1. ## Link your Electron API with Node
  2. In the [`typings/internal-ambient.d.ts`](https://github.com/electron/electron/blob/main/typings/internal-ambient.d.ts) file, we need to append a new property onto the `Process` interface like so:
  3. ```ts title='typings/internal-ambient.d.ts'
  4. interface Process {
  5. _linkedBinding(name: 'electron_browser_{api_name}', Electron.ApiName);
  6. }

At the very bottom of your api_name.cc file:

```cpp title=’apiname.cc’ NODE_LINKED_MODULE_CONTEXT_AWARE(electron_browser{api_name},Initialize)

  1. In your [`shell/common/node_bindings.cc`](https://github.com/electron/electron/blob/main/shell/common/node_bindings.cc) file, add your node binding name to Electron's built-in modules.
  2. ```cpp title='shell/common/node_bindings.cc'
  3. #define ELECTRON_BUILTIN_MODULES(V) \
  4. V(electron_browser_{api_name})

Note: More technical details on how Node links with Electron can be found on our blog.

Expose your API to TypeScript

Export your API as a module

We will need to create a new TypeScript file in the path that follows:

"lib/browser/api/{electron_browser_{api_name}}.ts"

An example of the contents of this file can be found here.

Expose your module to TypeScript

Add your module to the module list found at "lib/browser/api/module-list.ts" like so:

typescript title='lib/browser/api/module-list.ts' export const browserModuleList: ElectronInternal.ModuleEntry[] = [ { name: 'apiName', loader: () => require('./api-name') }, ];