Custom UI

INK Editor allows plugins to create their own custom sidebar and its content. Using this, plugins can easily create an interface for their settings, personalization, or maybe just their descriptions.

INK Editor use simple JSON based architecture to create custom sidebar and it’s child components. Which allows the power to create complex UI with simplicity.

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
import INKAPI from './inkapi.js'

INKAPI.ready(()=>{
  const UI = INKAPI.ui;

  const sidebar = await UI.create({
    type: "Sidebar",
    sidebarTitle: "My Personal Sidebar",
    icon: "./sample.svg",
  });
})

In the above code we are making use of UI.create API to create a custom sidebar with title ‘My Personal Sidebar’ and custom icon. Yes! it is that easy.

As you can see in the example that we are passing JavaScript object with some data to the UI.create API. Let’s break it down. Following are the most common and standard fields for all ui components.

Name

Type

Description

type

String

This is a mandatory field and explains the type of component you are trying to create. Currently INK Editor supports only few element types including Sidebar, Input, Button and Toggle.

id

String

This field assigns a unique id to the created component. If no id is provided, an automatically generated id will be assigned. It is recommended to assign id yourself.

styles

Object

This field can be used for implementing custom CSS styles to the component.

Above are the standard fields and can be used in all element. Other than above fields, each element have their own unique fields.

Now, if we look at above code example it will only create an empty sidebar which isn’t much useful. To create child components in the custom sidebar, we use children field which accepts an array of objects.

 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
import INKAPI from './inkapi.js'

INKAPI.ready(()=>{
  const UI = INKAPI.ui;

  const sidebar = await UI.create({
    type: "Sidebar",
    sidebarTitle: "My Personal Sidebar",
    icon: "./sample.svg",
    children: [
      {
        type: "Input",
        id: "input",
        styles: {
          margin: "10px auto",
          width: "95%",
        },
      },
      {
        type: "Button",
        id: "imgBtn",
        image: "./dice.jpg",
        styles: {
          width: "220px",
          height: "220px",
        },
      },
    ],
  });
})

In the above example we are create a custom sidebar with two children components (an input and a button).

Shadow UI Objects

Every UI component created by plugin (even if its child component), there is an object associated with it in global context. Using this object we can trigger methods related to that UI Component. For eg. an Input component have getValue and setValue methods. We call these objects Shadow UI Objects. We can access these shadow objects using a global variable called shadowUIObjects, which is an object itself with UI component Id as field name and their respective shadow object as value. Let’s take the above code example again and set the input component’s initial value to “Hello, World!”.

 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
import INKAPI from './inkapi.js'

INKAPI.ready(()=>{
  const UI = INKAPI.ui;

  // keeping all component Ids here for better reusability and code quality
  const componentIDs = {
    inputId: "inputId",
    imgBtnId: "imgBtnId",
  };

  const sidebar = await UI.create({
    type: "Sidebar",
    sidebarTitle: "My Personal Sidebar",
    icon: "./sample.svg",
    children: [
      {
        type: "Input",
        id: componentIDs.inputId,
        styles: {
          margin: "10px auto",
          width: "95%",
        },
      },
      {
        type: "Button",
        id: componentIDs.imgBtnId,
        image: "./dice.jpg",
        styles: {
          width: "220px",
          height: "220px",
        },
      },
    ],
  });

  // extracting shadow object for input component
  const inputShadowObj = shadowUIObjects[componentIDs.inputId];

  // setting input initial value to "Hello, World!"
  inputShadowObj?.setValue("Hello, World!");

})

Notice how I have grouped all component Ids in a single object. That is a recommended way to deal with UI components and will help once your plugin scales.

Note

To learn more about all the methods you can call from shadow object. Check the UI components detail page.