Skip to main content
Version: v4

Designing the Plugin API

The first - and arguably most important - step when building a Capacitor plugin is to design the API. The API is the contract we will adhere to when writing each platform’s specific implementation.

We can define the plugin API using TypeScript; it will serve as our contract when implementing and provides the niceties that come with TypeScript, such as code completion and type checking.

Wait, do you even need a plugin for that?

Believe it or not, modern web browsers can do many things that we think of as “native functionality,” such as checking battery status, speech recognition, and, yes, screen orientation. It’s not uncommon when building Web Native applications to see functionality that once required plugins to access are now available as Web APIs.

Before building a plugin for a particular feature, we recommend checking out sites such as What Web Can Do Today to see if the functionality you are looking for is already available as a Web API.

If screen orientation already has a Web API, why would we go out of our way to build one? Taking a look at the Screen Orientation Web API we can see that iOS does not implement the API (as of this writing), which means we will need to provide the implementation ourselves. As it relates to Android, we could just use the Screen Orientation Web API when our app runs on the Android platform - but we will implement screen orientation functionality natively for educational purposes.

Defining the ScreenOrientation API

We might not be able to use the Screen Orientation Web API outright, but we can model our plugin’s API against it:

Method NameInput ParametersReturn Value
orientationPromise<{ type: OrientationType }>
lock{ orientation: OrientationLockType }Promise<void>
unlockPromise<void>
addListener(orientation: { type: OrientationType }) Promise<PluginListenerHandle> & PluginListenerHandle
removeAllListenersPromise<void>

There is an added advantage here; we can use the OrientationType and OrientationLockType types available through TypeScript’s existing DOM typings.

Let's set up a directory to hold our plugin API. Create a new subfolder src/plugins/screen-orientation and add the following files within:

  • definitions.ts
  • index.ts

Populate definitions.ts with the following code:

import type { PluginListenerHandle } from '@capacitor/core';

export interface ScreenOrientationPlugin {
/**
* Returns the screen's current orientation.
*/
orientation(): Promise<{ type: OrientationType }>;

/**
* Locks the screen orientation.
*/
lock(opts: { orientation: OrientationLockType }): Promise<void>;

/**
* Unlocks the screen's orientation.
*/
unlock(): Promise<void>;

/**
* Listens for screen orientation changes.
*/
addListener(
eventName: 'screenOrientationChange',
listenerFunc: (orientation: { type: OrientationType }) => void,
): Promise<PluginListenerHandle> & PluginListenerHandle;

/**
* Removes all listeners
*/
removeAllListeners(): Promise<void>;
}

Registering the ScreenOrientation plugin

In order to use the plugin in the Capacitor application, we need to register it using the registerPlugin() module exported from @capacitor/core.

Populate index.ts with the following code:

import { registerPlugin } from '@capacitor/core';

import type { ScreenOrientationPlugin } from './definitions';

const ScreenOrientation = registerPlugin<ScreenOrientationPlugin>(
'ScreenOrientation',
);

export * from './definitions';
export { ScreenOrientation };

The code above creates an object linked to our plugin's implementation code.

Designing the API is complete; let’s build a user interface that will call it. In doing so, we will make testing easier as we implement each platform integration. Our next step: using the plugin API.