npm package discovery and stats viewer.

Discover Tips

  • General search

    [free text search, go nuts!]

  • Package details

    pkg:[package-name]

  • User packages

    @[username]

Sponsor

Optimize Toolset

I’ve always been into building performant and accessible sites, but lately I’ve been taking it extremely seriously. So much so that I’ve been building a tool to help me optimize and monitor the sites that I build to make sure that I’m making an attempt to offer the best experience to those who visit them. If you’re into performant, accessible and SEO friendly sites, you might like it too! You can check it out at Optimize Toolset.

About

Hi, 👋, I’m Ryan Hefner  and I built this site for me, and you! The goal of this site was to provide an easy way for me to check the stats on my npm packages, both for prioritizing issues and updates, and to give me a little kick in the pants to keep up on stuff.

As I was building it, I realized that I was actually using the tool to build the tool, and figured I might as well put this out there and hopefully others will find it to be a fast and useful way to search and browse npm packages as I have.

If you’re interested in other things I’m working on, follow me on Twitter or check out the open source projects I’ve been publishing on GitHub.

I am also working on a Twitter bot for this site to tweet the most popular, newest, random packages from npm. Please follow that account now and it will start sending out packages soon–ish.

Open Software & Tools

This site wouldn’t be possible without the immense generosity and tireless efforts from the people who make contributions to the world and share their work via open source initiatives. Thank you 🙏

© 2024 – Pkg Stats / Ryan Hefner

octane-components

v0.19.4

Published

Ready-to-use React components that make it easy to integrate with Octane

Downloads

59

Readme

octane-components

NPM JavaScript Style Guide Octane

Ready-to-use React components that make it easy to integrate with Octane

Getting started

octane-components has a collection of React components and actions (utility methods) to make integrating Octane into your UI simple. It works like this:

  1. On your backend, use your Octane API key to request a customer token.
  2. Pass this token to your client-side app and provide it to one of our components.
  3. The components handle the rest.

TL;DR

To demonstrate octane-components, we have a fully-functional example app that lives in the example/ directory. But in short:

## Install using yarn:
yarn add octane-components

## or NPM:
npm install --save octane-components
// And then use octane-components in your app.
import React from 'react';

import { PlanPicker, Actions } from 'octane-components';
import 'octane-components/dist/components/globals.css';
import 'octane-components/dist/components/PlanPicker/PlanPicker.css';

// /token is your token endpoint. See "Getting a customer token" below.
fetch('/token')
  .then((resp) => resp.json())
  .then((data) => {
    const { token } = data;
    ReactDOM.render(
      <PlanPicker
        customerToken={token}
        onPlanSelect={(plan) => Actions.subscribeCustomer(token, plan))}
      />,
      document.getElementById('root')
    );
  });

Getting a customer token

A customer token is a short-lived token that allows our components to take action on behalf of a customer. You can fetch one by sending a POST request to the /ecp/token endpoint using your Octane API key (more on that here).

You should not send your Octane API key to the client. Your API key can make arbitrary changes to any part of your Octane account and should be kept secret. Instead, you will need to expose a token endpoint that your client-side app can hit in order to fetch a customer token.

Here's an example of what that might look like in a hypothetical Express.js web server:

const axios = require('axios');
const express = require('express');
const app = express();

app.get('/token', (req, res) => {
  // Get the ID of your currently signed-in user.
  const userId = getSignedInUser();
  axios
    .post(
      `https://api.cloud.getoctane.io/ecp/token`,
      {
        customer_name: userId,
      },
      {
        headers: {
          Authorization: `Bearer [YOUR API KEY HERE]`,
        },
      }
    )
    .then((apiRes) => {
      res.json(apiRes.data);
    })
    .catch((error) => {
      console.error(error);
      res.status(500);
      res.send('Something went wrong!');
    });
});

app.listen(3000);

Standard Components

Components are interactive building blocks that handle interacting with specific parts of Octane. Most of them only need a customer token in order to work. By default they are unstyled, although they all include default styles that you can use and customize.

PlanPicker

PlanPicker renders a list of plans available to a customer. If a customer is subscribed to an available plan, that plan is marked as the current subscription. Plans are selectable and fire a callback when they're selected.

import { PlanPicker } from 'octane-components';
// CSS variables
import 'octane-components/dist/components/globals.css';
// Default theme
import 'octane-components/dist/components/PlanPicker/PlanPicker.css';

<PlanPicker
  // (Required) Your customer token.
  customerToken={token}
  // (Optional) Callback to fire when a plan is selected.
  onPlanSelect={(planName, plan) => {
    /* callback body */
  }}
/>;

Screenshot of the PlanPicker component

PaymentSubmission

PaymentSubmission renders a form to accept payment information from a customer. If the customer has payment information on file, the component indicates as much. PaymentSubmission is a wrapper around Stripe's CardElement component, but takes care of interacting with Stripe, initializing Stripe's API, and setting up a Stripe payment intent.

import { PaymentSubmission } from 'octane-components';
// CSS variables
import 'octane-components/dist/components/globals.css';
// Default theme
import 'octane-components/dist/components/PaymentSubmission/PaymentSubmission.css';

<PaymentSubmission
  // (Required) Your customer token.
  customerToken={token}
  // (Optional) Callback to fire when a payment method is set.
  //   Fires on component init if the customer has payment info on file.
  onPaymentSet={() => {
    /* callback body */
  }}
  // (Optional) The text to show on the "save payment" button. Defaults to "Save"
  saveButtonText={'Save'}
  //
  // The following are props provided by Stripe's CardElement.
  //
  // (Optional) An options object to pass to the underlying CardElement
  //   This can be used to style the CardElement, among other things.
  //   See: https://stripe.com/docs/js/elements_object/create_element?type=card
  options={{}}
  // (Optional) Change event handler to pass to the underlying CardElement
  //   See: https://stripe.com/docs/js/element/events/on_change?type=cardElement
  onChange={(event) => {
    /* callback body */
  }}
  // (Optional) Event handler triggered when the CardElement is fully rendered.
  //   Called with a reference to the underlying Element instance.
  //   See: https://stripe.com/docs/js/element
  onReady={(element) => {
    /* callback body */
  }}
  // (Optional) Triggered whenever the escape key is pressed inside the CardElement.
  onEscape={() => {
    /* callback body */
  }}
/>;

Screenshot of the PlanPicker component

Advanced Components

While our basic components are flexible enough for most common use cases, we also provide access to a number of more specific components to allow for more fine-tuned customization.

All of our Advanced Components must be wrapped in a TokenProvider component. Unlike our basic components, our advanced components expect the customer token to be provided through a React context which we expose through the TokenProvider component.

TokenProvider

TokenProvider accepts a customer token as its value and makes it available to any of its children.

<TokenProvider token={customerToken}>
  <YourCustomOctaneComponent />
</TokenProvider>

StripeElements

StripeElements makes it easy to use any of Stripe's own Elements components. It acts as a drop-in replacement to their own <Elements> wrapper, but it instead uses your Octane credentials to initialize the Stripe SDK.

import { StripeElements } from 'octane-components';
import { CardElement } from '@stripe/react-stripe-js';

const MyComponent = ({ customerToken }) => (
  <TokenProvider token={customerToken}>
    <StripeElements>
      <CardElement />
    </StripeElements>
  </TokenProvider>
);

While it is initializing, StripeElements does not render anything. If you'd like to render a loading element, you can provide it via the loading prop:

const MyComponent = ({ customerToken }) => (
  <TokenProvider token={customerToken}>
    <StripeElements loading={<LoadingSpinner />}>
      <CardElement />
    </StripeElements>
  </TokenProvider>
);

Additionally, we also expose a useStripeClientSecret hook that can be called from within the StripeElements component. The hook provides access to the Stripe client secret, which allows you to make Stripe API calls. Because it is initialized and owned by StripeElements, it can only be used by components within it. Here's an example (for more details on using Stripe's components, please refer to their docs).

import { StripeElements, useStripeClientSecret } from 'octane-components';
import { useStripe, useElements, CardElement } from '@stripe/react-stripe-js';

const MyForm = () => {
  const stripe = useStripe();
  const elements = useElements();
  const secret = useStripeClientSecret();

  const handleSubmit = async (event) => {
    event.preventDefault();

    if (!stripe || !elements || !secret) {
      // Stripe.js has not yet loaded.
      // Make sure to disable form submission until Stripe.js has loaded.
      return;
    }

    const result = await stripe.confirmCardSetup(secret, {
      payment_method: { card: elements.getElement(CardElement) },
    });

    if (result.error) {
      /* show error */
    } else {
      /* the payment has been processed! */
    }
  };

  const isDisabled = !stripe || !elements || !secret;

  return (
    <form onSubmit={handleSubmit}>
      <CardElement />
      <button disabled={isDisabled}>Confirm order</button>
    </form>
  );
};

const MyComponent = ({ customerToken }) => (
  <TokenProvider token={customerToken}>
    <StripeElements>
      <MyForm />
    </StripeElements>
  </TokenProvider>
);

EmbeddedPortal

EmbeddedPortal allows you to embed a customer portal on your website. This is a simple wrapper around an iframe, which generates a link to customer's page and requires customerToken from you. The component also allows you to specify other iframe attributes.

import { EmbeddedPortal } from 'octane-components';

<EmbeddedPortal
  // (Required) Customer's token.
  customerToken="CUSTOMER_TOKEN"
  // (Optional) List of the iframe attributes
  width={1200}
  height={500}
  title="Customer's Page"
  referrerPolicy='no-referrer'
  allowFullScreen
/>

Styling components

By default, components are unstyled. They are decorated with classes that should make it easy to style them to match any sort of branding.

To provide a starting point, each component comes with its own default stylesheet. Each stylesheet depends on CSS variables scoped to the .octane-component class. To use a stylesheet, import both the variables and a component's stylesheet wherever you'd like to use them.

/* index.css */

/* variables */
@import 'octane-components/dist/components/globals.css';
/* PlanPicker styles */
@import 'octane-components/dist/components/PlanPicker/PlanPicker.css';
/* PaymentSubmission styles */
@import 'octane-components/dist/components/PaymentSubmission/PaymentSubmission.css';

You can tweak our styling without writing new styles from scratch by overwriting our variables:

/* variables */
@import 'octane-components/dist/components/globals.css';

/* overrides */
.octane-component {
  /* Light theme */
  --octane-bg-primary: #d4f5f54d;
  --octane-text-primary: #071420;
  --octane-border-primary: #c7dcd5;
  --octane-text-accent: #128475;
  --octane-bg-secondary: #becbd8;
  --octane-bg-lighter: #484e5980;
  --octane-text-secondary: #282c2c;
  --octane-border-primary: #d4f5f54d;
}

/* PlanPicker styles */
@import 'octane-components/dist/components/PlanPicker/PlanPicker.css';
/* PaymentSubmission styles */
@import 'octane-components/dist/components/PaymentSubmission/PaymentSubmission.css';

Actions

In addition to components, octane-components provides access to "actions", or asynchronous interactions with our API. While these complement our React components nicely, they can be used in any client-side app. They all return Promises and work well with await, when available.

getActiveSubscription(token): Promise<PricePlan | null>

getActiveSubscription checks for a customer's active subscription. If it's there, it resolves to the PricePlan details for that subscription. If the customer has no subscription, it resolves to null.

Example

import { Actions } from 'octane-components';
const { getActiveSubscription } = Actions;

const planName = 'enterprise_plan';

fetch('/token')
  .then((resp) => resp.json())
  .then(({ token }) => getActiveSubscription(token))
  .then((sub) =>
    sub !== null ? alert('yeah') : alert('customer needs a subscription')
  );

Params

  • token (required, string) — A customer token.

hasPaymentInfo(token): Promise<boolean>

hasPaymentInfo resolves to true if the customer has payment info, and false otherwise.

Example

import { Actions } from 'octane-components';
const { hasPaymentInfo } = Actions;

const planName = 'enterprise_plan';

fetch('/token')
  .then((resp) => resp.json())
  .then(({ token }) => hasPaymentInfo(token))
  .then((hasIt) => (hasIt ? alert('yeah') : alert('need payment info')));

Params

  • token (required, string) — A customer token.

subscribeCustomer(token, plan, options): Promise<ActiveSubscription>

subscribeCustomer subscribes a customer to a specific plan. It accepts a customer token and the name of a plan, and will subscribe that customer to the version of that plan visible to octane-components.

Example

import { Actions } from 'octane-components';
const { subscribeCustomer } = Actions;

const planName = 'enterprise_plan';

fetch('/token')
  .then((resp) => resp.json())
  .then(({ token }) => subscribeCustomer(token, planName));

Note that checkForBillingInfo is a convenience check and does not guarantee that valid billing info will be available in the future. For example, a customer could remove their payment information, or their payment details might expire.

Params

  • token (required, string) — A customer token.
  • plan (required, string) — The name of the plan to subscribe the customer to
  • options (optional, object) — Optional configuration options
    • options.checkForBillingInfo (optional, boolean) — Whether or not to verify that there is valid payment information for the customer before subscribing them. Defaults to false.

Hooks

If you use React Hooks, we've wrapped all of our actions in hooks to make things a little easier. Hooks accept a token (which can be omitted if you're using <TokenProvider />) and fetch any data whenever your component loads.

Query hooks

Starting from 0.15.0 version, all query hooks additionally return refetch function to get the data whenever it's needed.

All these hooks fetch and refetch data from our end-customer API:

  • useActiveSubscription - fetch data about the customer's current subscription
  • useContactInfo - fetch and update the customer's contact info
  • useUsage - fetch the customer's usage
  • useHasPaymentInfo - fetch whether or not the customer has payment info on file
  • useInvoices - fetch a list of invoice metadata for the customer
  • usePaymentMethodStatus - fetch the customer's payment method status
  • usePricePlans - fetch a list of price plans this customer could subscribe to
  • useVendorInfo - fetch information about this customer's vendor
  • useCustomerLink - fetch a link to customer's page

These ones allow you also to update data through the same API:

  • useContactInfo - use update function to update contact info
  • useActiveSubscription - use update function to update active subscription
const { loading, result: invoices, refetch } = useInvoices({ token });

// `refetch` function will update `result` accordingly.
useEffect(() => {
  refetch();
});

return loading ? <Spinner /> : <InvoicesTable invoices={invoices} />;

useActiveSubscription, useContactInfo hooks also provide update function, which updates result. It takes a payload which will update hook's data.

const { loading, result: activeSubscription, update } = useActiveSubscription({ token });

if (loading) {
  return <Spinner />;
}

return (
  <>
    <ComponentDisplayingData subscription={activeSubscription} />
    <button onClick={() => update({ price_plan_uuid: 'pp_uuid', add_ons: [] })}>
      Update active subscription
    </button>
  </>
);

Mutation hooks

A few hooks are meant for mutations only. These hooks won't fetch any data when instantiated - you have to call the returned mutate function manually.

  • useStripeSetupIntent - create a Stripe SetupIntent for the customer
  • useUpdateContactInfo - (deprecated) update the customer's contact info - use update function from useContactInfo hook
  • useUpdateSubscription - (deprecated) update the customer's subscription - use update function from useActiveSubscription hook
  • useCustomerUsage - (deprecated) fetch the customer's usage - use useUsage instead
const [ updateSetupIntent, { data }] = useStripeSetupIntent({ token });
const { result, loading, error } = data;

if (loading) {
  return <Spinner />;
}

return (
  <>
    <ComponentDisplayingData setupIntent={result} />
    <ConfirmationDialog onConfirm={updateSetupIntent}>
  </>
);

Types

octane-components is written entirely in TypeScript and makes heavy use of Octane's schema types throughout the code. To make accessing them easy, we've made these types accessible:

import type { SchemaTypes } from 'octane-components';
type PricePlan = SchemaTypes['PricePlan'];

These types are generated directly from our OpenAPI schema file.

Local development

The easiest thing to do is to run yarn storybook, which will let you see and edit all of our components.

It can also be useful to develop alongside the example app that lives in example/.

Here are the other commands to know about:

# Run a full build once. This checks types and puts built code in dist/
yarn build
# Start a build server, rebuilding any time a file changes
yarn start
# Run all of our tests
yarn test
# Run the storybook server in development / watch mode
yarn storybook
# Build storybook once
yarn build-storybook
# Fetch our openapi spec and generate fresh TypeScript types
yarn generate

Releasing

Releasing is as easy as bumping the version of package.json in this repo. Any changes on the main branch will be automatically published to NPM.

License

MIT © Octane Software Technology, Inc.