Skip to content

How to make a Widget

TimGuiteDiamond edited this page Nov 1, 2019 · 13 revisions

This page describes how to create widgets for use in CS-Web-Proto, taking advantage of the work we have done to provide a base interface and connection to PVs.

NOTE: We separate components from Widgets in this document. A component is a React component which can be rendered directly from its props. A Widget is what will actually be used by CS-Web-Proto as it has a specified interface and is part of a toolchain which makes the app work well.

We have used a mixture of TypeScript (TS) and PropTypes to perform checking on the props which are provided to components and Widgets. This page will talk through our practices but there is a section at the end for if you would rather use plain JavaScript (JS).

TypeScript provides compile time checking and is useful for ensuring that the internals of your component work as expected. PropTypes provide runtime checking and is useful for ensuring that Widget definitions in screen files are correct.

Functional Components

We recommend using functional components. This a style for writing React components in which functions return JSX Elements. It encourages stateless components which can be easily tested and have clear links between input and output. This also makes components easier to test and compose.

Find out more about functional components here and here.

Creating your widget

Think about all the things which your component needs to render properly. Try to minimise this as much as possible. For a label, you might only need text.

We have provided some types for you to reduce boilerplate code:

  • Component is for components which do not use PVs
  • PVComponent is for components which will use PV information to render correctly.

Both of these types can be imported from widget.tsx

This allows you to think about the additional props which are specific to your component. These additional props should be specified with prop types:

import PropTypes from "prop-types";
const MyExtraProps = {
    // Specifies optional prop 'MyProp' to be either a string or number
    MyProp: PropType.oneOfType([PropType.string, PropType.number])
};

NOTE: Additional props should have a flat and simple shape, specifying props to be numbers, string, boolean etc. as opposed to a complex series of objects and arrays

The next step is to actually write your component. At this step, combine your additional props with the relevant base props. For a basic component (not using PVs):

import React from "react";
import {Component} from "../Widget/widget";

const MyComponent = (
    // InferProps turns your PropType into a TS type for compile time use
    props: PropTypes.InferProps<typeof MyExtraProps> & Component
): JSX.Element =>
  // Style is available because it is specified in component
  <div style={props.style}>
    {props.MyProp}
  </div>
);

NOTE: TS files which use JSX should use the *.tsx extension

CSS Classes

If you want to add classes to your component, use CSS modules. This provides you with the capability to swap out CSS classes dynamically depending on the input.

You should also add a global class to your component with a simple name. This allows configuration of the styles from a global stylesheet and themes.

import classes from "./mycomponent.module.css"

const MyComponent = (
    props: PropTypes.InferProps<typeof MyExtraProps> & Component
): JSX.Element =>
  <div className={`Component ${classes.component}` style={props.style}>
    {props.MyProp}
  </div>
);

Widgifying

To turn your component into a Widget, import Widget and WidgetPropType from the Widget component files:

import { Widget, WidgetPropType } from "../Widget/widget";
  • Widget is a special component which provides the component you have written with information about macros, PVs, alarms, etc. when necessary.

  • WidgetPropType is a specified PropType which ensures that the component was correctly specified at runtime.

Combine your extra props with WidgetPropType to provide an exact specification for your Widget at runtime. This can be done with the spread operator because they are both just normal JS objects:

const MyWidgetProps = {
  ...MyExtraProps,
  ...WidgetPropType
}

Now combine your component with the Widget by including it with the baseWidget prop and inferring your new PropType:

const MyWidget = (
  props: PropTypes.InferProps<typeof MyWidgetProps>
): JSX.Element => <Widget baseWidget={MyComponent} {...props} />;

Finally, add your PropType to the Widget:

MyWidget.propTypes = MyWidgetProps;

Congratulations! You now have a Widget which you can export and include in all your CS-Web-Proto screens!

Include the extra information which is handled by Widget when you instantiate your widget:

<MyWidget 
  MyProp={"This is my widget!"}
  containerStyling={{
    position: "relative",
    height: "",
    width: "",
    margin: "",
    padding: ""
  }}
/>

You might notice that when doing this, TS is very fussy and even requires you specify props which are marked as optional. This is due to some disagreement between TS and PropTypes in how to handle this and is a small development cost to pay for the benefits it provides, given that almost all Widgets rendered will be specified in a screen file rather than directly in JSX.

My component needs PVs!

Great! The process is very similar but there are a few things to be aware of.

PV components may use the following props to get information about the PV they are connected to:

  • pvName - get a string with the name of the PV
  • initializingPvName - string ofthe PV name before any macro substitution took place
  • connected - boolean on whether or not the PV has been connected to
  • value - VType with latest PV data
  • readonly - bool on if the PV is readonly to the client

You may use any number of these props when defining your component. If you don't need any of them you might be better off with a standard Widget (see above).

Import PVComponent, PVWidget and PVWidgetPropType in place of the normal widget imports then follow the instructions above as normal. If you are using an editor with auto-completion and TS integration you should be able to see all of the props available when you type props. and this should include your extra component props, the base component props and the PV component props listed just above.

A brief example of a PV Widget is given below:

import React from "react";
import PropTypes from "prop-types";
import { vtypeToString } from "../../vtypes/utils";
import classes from "./mycomponent.module.css";
import { PVComponent, PVWidget, PVWidgetPropType } from "../Widget/widget";

const MyExtraProps = {
  MyProp: PropTypes.oneOfType([PropTypes.string, PropTypes.number]).isRequired
};

export const MyPVComponent = (
  props: PropTypes.InferProps<typeof MyExtraProps> & PVComponent
): JSX.Element => (
  <div className={`MyComponent ${classes.MyComponent}`} style={props.style}>
    {props.MyProp}: {vtypeToString(props.value)}
  </div>
);

const MyWidgetProps = {
  ...MyExtraProps,
  ...PVWidgetPropType
};

export const MyPVWidget = (
  props: PropTypes.InferProps<typeof MyWidgetProps>
): JSX.Element => <Widget baseWidget={MyPVComponent} {...props} />;

MyPVWidget.propTypes = MyWidgetProps;

When instantiating your PVWidget, you only need to supply the pvName parameter and the PVWidget will provide the rest of the information via some Higher Order Components which are working behind the scenes:

<MyPVWidget 
  pvName={"loc://pv1"}
  MyProp={"PV Value"}
  containerStyling={{
    position: "relative",
    height: "",
    width: "",
    margin: "",
    padding: ""
  }}
/>

Adding my widget from JSON

The component which converts JSON into a screen fromJson. It maintains a dictionary of text: widgets. To add your widget, import your widget into this file and add it to the dictionary with a useful name.

At Runtime

At runtime, the function which turns the JSON description into Widgets on the screen will compare the provided parameters against the PropType for the widget. It will raise an error if an incorrect type is applied, if a required prop is not provided or if unexpected properties are provided. This will cause an error widget to be displayed on the screen (it should be relatively obvious).

To get more information, open your browser console (Ctrl + Shift + I or similar) and you will see a message specifying the offending file and object. In the future this capability may be enhanced to provide greater utility.

Testing

Whether you are creating a Widget from scratch or making a small change to an existing Widget, testing is important. Creating stateless components is a good step towards testability. We are using [Jest] and [Enzyme] to test our components and testing takes place on any push with Travis CI. You can also perform testing yourself with npm run test. Press a to run all available tests and u to update a snapshot if it changed in a way you were expecting and it needs to be updated. The snapshot file should also be committed.

Many of our components already have tests so you can observe what we have done. Tests should be included with the filename component.test.tsx

Prop Types on their own

The above example have combined TypeScript and PropTypes, but it may be that you just want to use PropTypes as you are more familiar with how they work. In this case, simply avoid the references to TypeScript, types, InferProps etc. You will still need to import Widget and WidgetPropType or their PV equivalents and attach some PropTypes to the widget you export. Note that for PV widgets this will make things a little difficult as the type system supporting values is complex and has not yet been ported to PropTypes. This is not a particular priority - PropTypes is good for runtime checking of widget types and TypeScript is better for all of the behind the scenes work.

Clone this wiki locally