Shopify App - Step by Step
Hi there, i am going to create a shopify app for subscriptions and especially for cancelling those according to new german law Therefore I created a personal step by step plan to create an shopify app. I justed copied most of the stuff from different source, hope it helps :)
Vocabulary:
There are three key areas where apps interact with the Shopify platform: Shopify admin, online store, and checkout.
Shopify admin
After logging in to Shopify, merchants set up their store, configure settings, and manage their business using the Shopify admin.
The Shopify admin includes core aspects of the merchant's Shopify business, including orders, products, and customers. Merchants also install apps in the Shopify admin.
Online store
The online store is an online home for a merchant's business. Merchants use the online store to create webpages, publish a blog, and sell their products.
As an app developer, you can build integrations in all the places where merchants want to sell, and where their customers want to buy.
Checkout
Merchants use the secure Shopify checkout to accept orders and take payments wherever they sell online. After a customer adds products to a cart, they use Shopify checkout to enter their shipping information and payment details before placing the order.
Apps can integrate with Shopify checkout to offer additional functionality to merchants and customers. For example, merchants can use apps that support post-purchase offers to show a cross-sell or upsell offer to customers at checkout.
Create your first shopify app
Step 1: Register Shopify Partner Account
Create First an Partner Account here: partners.shopify.com/signup
Step 2: Create an test store
A development store is a free Shopify account that comes with a few limitations. As a Shopify Partner, you can create an unlimited number of development stores.
You can use a development store to test any themes or apps that you create, or to set up a Shopify store for a client. Setting up a development store and transferring it to a client gives you a recurring commission. 2.1. Log in to your Partner Dashboard. 2.2. Click Stores. 2.3. Click Add store. 2.4. In the Store type section, select Development store. 2.5. In the Login information section, enter a name for your store and a password that you can use to log in. By default, the email associated with your Partner Dashboard is used as the username, but you can change it if you want to. 2.6. Optional: Enable a developer preview by checking Create a non-transferrable store that uses a developer preview. Select a developer preview version from the drop-down list. 2.7. In the Store address section, enter your address. 2.8. Optional: In the Store purpose section, select the reason why you're creating this development store. 2.9. Click Save.
Step 3: Install latest Node.js
Step 4: Install Shopify CLI
If you want to use Shopify CLI natively on Windows 10, then first make sure that you've installed Ruby+Devkit using RubyInstaller for Windows (version 2.7 or higher). Alternatively, you can use Shopify CLI using Windows Subsystem for Linux, in which case you need to install the following:
After you install the prerequisites, you can install Shopify CLI as a Ruby gem using the RubyGems.org package manager. In a new terminal window, navigate to your home directory and run the following command:
gem install shopify-cli
To verify that Shopify CLI is installed properly, run the following command:
shopify version
Step 5: Login to Shopify with the CLI
Before we create the project with the help of the CLI we need to login to shopify. Therefore we type in in the terminal:
shopify login
A browser tab will be opened, log in to your partner account. You should see a message in your terminal, after successfull logged in
Logged into store xxxxxx.myshopify.com in partner organization xxxx
Step 6: Create a new project
After you've installed Shopify CLI, you're ready to create a new project.
Navigate to the directory where you want to create your project and run shopify node create
. This command scaffolds a new Node.js app in a subdirectory and creates your app in the Partner Dashboard.
Step 7: Start a local development server
After your app is created, you can work with it by navigating to your project directory and running shopify node serve
to start a local development server.
Shopify CLI uses ngrok to create a tunnel that allows your app to be accessed using a unique HTTPS URL, which is mandatory when creating an app.
Step 8: Install your app on your development store
With the server running, open the URL that your terminal printed out in the previous step. When you open the URL, you're prompted to install the app on your development store.
If you follow this link, you should see a message containing your app and the scopes of it. Press install and your good to go further.
Step 9: Start building the app
Previously, you created a new app with Shopify CLI. You're now ready to start building your app.
In this tutorial, you'll accomplish a series of tasks to add some specific functionality to your app. Your final app will be simple, but you’ll learn where to find resources to build more complex features on your own. After you've finished this tutorial, you'll have accomplished the following:
- populated products in your development store to test your app against
- built the beginnings of your user interface with Polaris
- set up a GraphQL query to retrieve products
- set up a GraphQL mutation to update the prices of products
Step 10 : Populate products
Shopify CLI helps with the process of adding example data to test your app's behavior. You can use Shopify CLI to create records for products, customers, and draft orders.
Because your app needs to interact with product data, start by populating products in your development store:
- Open a new terminal window.
- Navigate to your project directory.
- Run
shopify populate products
Step 11: Add an empty state
Now that you can run your app in Shopify, you can view and test your frontend components as you build them. You can use Polaris, Shopify’s React component library and design system, to build your user interface.
Use Polaris to add an empty state to your app. The Polaris Empty state component helps to communicate the value of your app and its primary action when merchants first add it to their Shopify admin.
- In your code editor, navigate to your
pages/index.js
file. - Replace the contents of the file with an
EmptyState
component:
import { Heading, Page, TextStyle, Layout, EmptyState} from "@shopify/polaris";
const img = 'https://cdn.shopify.com/s/files/1/0757/9955/files/empty-state.svg';
const Index = () => (
<Page>
<Layout>
<EmptyState // Empty state component
heading="Discount your products temporarily"
action={{
content: 'Select products',
onAction: () => this.setState({ open: true }),
}}
image={img}
>
<p>Select products to change their price temporarily.</p>
</EmptyState>
</Layout>
</Page>
);
export default Index;
When you preview your embedded app, it displays the empty state.
Step 12: Add a resource picker
Next, add a resource picker so that you can select products from your app. You can use App Bridge, Shopify's standalone vanilla JavaScript library, to add a resource picker to your app.
The App Bridge ResourcePicker
action set provides a search-based interface to help you find and select one or more products and then returns the selected resources to your app.
In your pages/index.js
file, add a class that sets a state for the resource picker. Then, add the ResourcePicker
component to the primary action button on the EmptyState
component:
import React from 'react';
import { Heading, Page, TextStyle, Layout, EmptyState} from "@shopify/polaris";
import { ResourcePicker, TitleBar } from '@shopify/app-bridge-react';
const img = 'https://cdn.shopify.com/s/files/1/0757/9955/files/empty-state.svg';
// Sets the state for the resource picker
class Index extends React.Component {
state = { open: false };
render() {
return (
<Page>
<TitleBar
primaryAction={{
content: 'Select products',
onAction: () => this.setState({ open: true }),
}}
/>
<ResourcePicker // Resource picker component
resourceType="Product"
showVariants={false}
open={this.state.open}
onSelection={(resources) => this.handleSelection(resources)}
onCancel={() => this.setState({ open: false })}
/>
<Layout>
<EmptyState
heading="Discount your products temporarily"
action={{
content: 'Select products',
onAction: () => this.setState({ open: true }),
}}
image={img}
>
<p>Select products to change their price temporarily.</p>
</EmptyState>
</Layout>
</Page>
);
}
handleSelection = (resources) => {
this.setState({ open: false });
console.log(resources);
};
}
export default Index;
In your embedded app, when you click Select products, the Add products modal opens.
Step 13: Add a resource list
Now that you've set up your resource picker, you need a way to retrieve products. You can retrieve products using the GraphQL Admin API. Ultimately, you want to display these products in a resource list.
To allow your app to query data with GraphQL, create a new ResourceList.js
file and include the graphql-tag
and react-apollo
imports in the file.
Then, set up a GraphQL query called getProducts
to retrieve a list of products and their prices.
Run
npm install store-js
Create a new
components
folder in thepages
folder in your project, and create a newResourceList.js
file in the folder.Add imports to your
ResourceList.js
file and set up your GraphQL query to retrieve products and their prices:import React from 'react'; import gql from 'graphql-tag'; import { Query } from 'react-apollo'; import { Card, ResourceList, Stack, TextStyle, Thumbnail, } from '@shopify/polaris'; import store from 'store-js'; import { Redirect } from '@shopify/app-bridge/actions'; import { Context } from '@shopify/app-bridge-react'; // GraphQL query to retrieve products by IDs. // The price field belongs to the variants object because // variations of a product can have different prices. const GET_PRODUCTS_BY_ID = gql` query getProducts($ids: [ID!]!) { nodes(ids: $ids) { ... on Product { title handle descriptionHtml id images(first: 1) { edges { node { originalSrc altText } } } variants(first: 1) { edges { node { price id } } } } } } `;
In your
ResourceList.js
file, after your GraphQL query, set up a class calledResourceListWithProducts
that extends theResourceList
component and returns products and prices. Then, define yourResourceList
component:class ResourceListWithProducts extends React.Component { static contextType = Context; render() { const app = this.context; return ( // GraphQL query to retrieve products and their prices <Query query={GET_PRODUCTS_BY_ID} variables={{ ids: store.get('ids') }}> {({ data, loading, error }) => { if (loading) return <div>Loading…</div>; if (error) return <div>{error.message}</div>; return ( <Card> <ResourceList // Defines your resource list component showHeader resourceName={{ singular: 'Product', plural: 'Products' }} items={data.nodes} renderItem={item => { const media = ( <Thumbnail source={ item.images.edges[0] ? item.images.edges[0].node.originalSrc : '' } alt={ item.images.edges[0] ? item.images.edges[0].node.altText : '' } /> ); const price = item.variants.edges[0].node.price; return ( <ResourceList.Item id={item.id} media={media} accessibilityLabel={`View details for ${item.title}`} onClick={() => { store.set('item', item); }} > <Stack> <Stack.Item fill> <h3> <TextStyle variation="strong"> {item.title} </TextStyle> </h3> </Stack.Item> <Stack.Item> <p>${price}</p> </Stack.Item> </Stack> </ResourceList.Item> ); }} /> </Card> ); }} </Query> ); } } export default ResourceListWithProducts;
In
pages/index.js
file, add your imports and define a constant for your app's empty state. Then, update the code that controls the layout of your empty state, and specify using your new resource list with products:import React from 'react'; import { Page, Layout, EmptyState} from "@shopify/polaris"; import { ResourcePicker, TitleBar } from '@shopify/app-bridge-react'; import store from 'store-js'; import ResourceListWithProducts from './components/ResourceList'; const img = 'https://cdn.shopify.com/s/files/1/0757/9955/files/empty-state.svg'; class Index extends React.Component { state = { open: false }; render() { // A constant that defines your app's empty state const emptyState = !store.get('ids'); return ( <Page> <TitleBar primaryAction={{ content: 'Select products', onAction: () => this.setState({ open: true }), }} /> <ResourcePicker resourceType="Product" showVariants={false} open={this.state.open} onSelection={(resources) => this.handleSelection(resources)} onCancel={() => this.setState({ open: false })} /> {emptyState ? ( // Controls the layout of your app's empty state <Layout> <EmptyState heading="Discount your products temporarily" action={{ content: 'Select products', onAction: () => this.setState({ open: true }), }} image={img} > <p>Select products to change their price temporarily.</p> </EmptyState> </Layout> ) : ( // Uses the new resource list that retrieves products by IDs <ResourceListWithProducts /> )} </Page> ); } handleSelection = (resources) => { const idsFromResources = resources.selection.map((product) => product.id); this.setState({ open: false }); store.set('ids', idsFromResources); }; } export default Index;
Now, when you click Select products, and add products from the Add products modal, a list of products displays.
Step 14: Update product prices
You've implemented a GraphQL query to read product data, and added the functionality to display the retrieved products in a resource list. Next, you'll use GraphQL to modify product data.
Set up a GraphQL mutation called ProductVariantUpdate
to update the prices of products in your app.
- Create a new
ApplyRandomPrices.js
file in yourcomponents
folder. Add imports to your
ApplyRandomPrices.js
file and set up a GraphQL mutation that allows your app to update the prices of products:pages/components/ApplyRandomPrices.js
import React, { useState } from 'react'; import gql from 'graphql-tag'; import { Mutation } from 'react-apollo'; import { Layout, Button, Banner, Toast, Stack, Frame } from '@shopify/polaris'; import { Context } from '@shopify/app-bridge-react'; // GraphQL mutation that updates the prices of products const UPDATE_PRICE = gql` mutation productVariantUpdate($input: ProductVariantInput!) { productVariantUpdate(input: $input) { product { title } productVariant { id price } } } `;
After your mutation in
ApplyRandomPrices.js
, set up a class calledApplyRandomPrices
which takes your mutation's input and applies a random price to the selected product:pages/components/ApplyRandomPrices.js
class ApplyRandomPrices extends React.Component { static contextType = Context; render() { return ( // Uses mutation's input to update product prices <Mutation mutation={UPDATE_PRICE}> {(handleSubmit, {error, data}) => { const [hasResults, setHasResults] = useState(false); const showError = error && ( <Banner status="critical">{error.message}</Banner> ); const showToast = hasResults && ( <Toast content="Successfully updated" onDismiss={() => setHasResults(false)} /> ); return ( <Frame> {showToast} <Layout.Section> {showError} </Layout.Section> <Layout.Section> <Stack distribution={"center"}> <Button primary textAlign={"center"} onClick={() => { let promise = new Promise((resolve) => resolve()); for (const variantId in this.props.selectedItems) { const price = Math.random().toPrecision(3) * 10; const productVariableInput = { id: this.props.selectedItems[variantId].variants.edges[0].node.id, price: price, }; promise = promise.then(() => handleSubmit({ variables: { input: productVariableInput }})); } if (promise) { promise.then(() => this.props.onUpdate().then(() => setHasResults(true))); }} } > Randomize prices </Button> </Stack> </Layout.Section> </Frame> ); }} </Mutation> ); } } export default ApplyRandomPrices;
Update your
pages/index.js
file to include the following imports:import React from 'react'; import gql from 'graphql-tag'; import { Mutation } from 'react-apollo'; import { Page, Layout, EmptyState, Button, Card } from "@shopify/polaris"; import { ResourcePicker, TitleBar } from '@shopify/app-bridge-react'; import store from 'store-js'; import ResourceListWithProducts from './components/ResourceList';
In
ResourceList.js
, add theApplyRandomPrices
import. Implement a constructor on yourResourceListWithProducts
class and update your GraphQL query to enable refetching products by ID. Finally, update yourResourceList
component:pages/components/ResourceList.js
import React from 'react'; import gql from 'graphql-tag'; import { Query } from 'react-apollo'; import { Card, ResourceList, Stack, TextStyle, Thumbnail, } from '@shopify/polaris'; import store from 'store-js'; import { Redirect } from '@shopify/app-bridge/actions'; import { Context } from '@shopify/app-bridge-react'; import ApplyRandomPrices from './ApplyRandomPrices'; // GraphQL query that retrieves products by ID const GET_PRODUCTS_BY_ID = gql` query getProducts($ids: [ID!]!) { nodes(ids: $ids) { ... on Product { title handle descriptionHtml id images(first: 1) { edges { node { originalSrc altText } } } variants(first: 1) { edges { node { price id } } } } } } `; class ResourceListWithProducts extends React.Component { static contextType = Context; // A constructor that defines selected items and nodes constructor(props) { super(props); this.state = { selectedItems: [], selectedNodes: {}, }; } render() { const app = this.context; // Returns products by ID return ( <Query query={GET_PRODUCTS_BY_ID} variables={{ ids: store.get('ids') }}> {({ data, loading, error, refetch }) => { // Refetches products by ID if (loading) return <div>Loading…</div>; if (error) return <div>{error.message}</div>; const nodesById = {}; data.nodes.forEach(node => nodesById[node.id] = node); return ( <> <Card> <ResourceList showHeader resourceName={{ singular: 'Product', plural: 'Products' }} items={data.nodes} selectable selectedItems={this.state.selectedItems} onSelectionChange={selectedItems => { const selectedNodes = {}; selectedItems.forEach(item => selectedNodes[item] = nodesById[item]); return this.setState({ selectedItems: selectedItems, selectedNodes: selectedNodes, }); }} renderItem={item => { const media = ( <Thumbnail source={ item.images.edges[0] ? item.images.edges[0].node.originalSrc : '' } alt={ item.images.edges[0] ? item.images.edges[0].node.altText : '' } /> ); const price = item.variants.edges[0].node.price; return ( <ResourceList.Item id={item.id} media={media} accessibilityLabel={`View details for ${item.title}`} verticalAlignment="center" onClick={() => { let index = this.state.selectedItems.indexOf(item.id); const node = nodesById[item.id]; if (index === -1) { this.state.selectedItems.push(item.id); this.state.selectedNodes[item.id] = node; } else { this.state.selectedItems.splice(index, 1); delete this.state.selectedNodes[item.id]; } this.setState({ selectedItems: this.state.selectedItems, selectedNodes: this.state.selectedNodes, }); }} > <Stack alignment="center"> <Stack.Item fill> <h3> <TextStyle variation="strong"> {item.title} </TextStyle> </h3> </Stack.Item> <Stack.Item> <p>${price}</p> </Stack.Item> </Stack> </ResourceList.Item> ); }} /> </Card> <ApplyRandomPrices selectedItems={this.state.selectedNodes} onUpdate={refetch} /> </> ); }} </Query> ); } } export default ResourceListWithProducts;
In your app, you can now update the prices of products.
Next steps
- Use webhooks to stay in sync with Shopify or execute code after a specific event occurs in a shop.
- Identify your app business model and learn how to use the Billing API to bill customers with recurring monthly charges or one-time purchases.
- Learn how to use app extensions to a