Work with prices 
In this chapter you will learn how
- The price object is structured
- To format and indicate pricing tiers
- Display the correct prices depending on the context
- Use useProductPricecomposable to handle the most common cases
Structure of a price 
A product in Shopware can have multiple prices. All these prices are defined in a CalculatedPrice object, which contains the following fields:
| Field | Description | 
|---|---|
| unitPrice | The price of a single product item (e.g. only selling toilet rolls in packs of 12) | 
| quantity | The quantity of units that the price applies to (usually 1) | 
| totalPrice | The price for one product | 
| calculatedTaxes | The calculated tax rates and their proportions of the total price | 
| taxRules | Composition of the underlying tax rates | 
| referencePrice | The price per unit (e.g. 1.99€ per 100g) | 
| listPrice | The list price of the product | 
| regulationPrice | Some local laws enforce showing the cheapest price within the last 30 days. This is the price that is used for that | 
Example of a CalculatedPrice object
{
  "unitPrice": 58,
  "quantity": 1,
  "totalPrice": 58,
  "calculatedTaxes": [
    {
      "tax": 9.26,
      "taxRate": 19,
      "price": 58,
      "apiAlias": "cart_tax_calculated"
    }
  ],
  "taxRules": [
    {
      "taxRate": 19,
      "percentage": 100,
      "apiAlias": "cart_tax_rule"
    }
  ],
  "referencePrice": null,
  "listPrice": {
    "price": 83.3,
    "discount": -25.3,
    "percentage": 30.37,
    "apiAlias": "cart_list_price"
  },
  "regulationPrice": {
    "price": 54,
    "apiAlias": "cart_regulation_price"
  },
  "apiAlias": "calculated_price"
}Each product has at least one CalculatedPrice object assigned to it, which can be accessed through product.calculatedPrice. It contains the product's default price, which applies when no other prices are defined.
Display a default price 
All prices are passed as floating point numbers, rounded to the decimals which are specified in your stores currency settings. You can use the getFormattedPrice helper method to apply the the correct formatting including currency symbol to the price.
<script setup>
import { useProductSearch } from '@shopware-pwa/composables-next';
const { getFormattedPrice } = usePrice();
const { search } = useProductSearch();
const { product } = await search('some-product-id');
const { unitPrice, price, tierPrices, isListPrice } = useProductPrice(ref(product));
</script>
<template>
  <div>
    <div>
      <b>{{ product.name }}</b>
    </div>
    <div>
      {{ getFormattedPrice(unitPrice) }}
      <small>
        incl. {{ price.taxRules[0].taxRate }}% tax
      </small>
    </div>
    <div v-if="isListPrice">
      <small>
        <del>
          {{ getFormattedPrice(price.listPrice.price) }}
        </del>
        (-{{ price.listPrice.percentage }}%)
      </small>
    </div>
  </div>
</template>Pricing tiers and quantity prices 
Pricing tiers add one layer of complexity to the pricing model. In Shopware, you can define multiple pricing tiers for a product. Each tier has a quantity and a price.
These pricing tiers are passed through a product's calculatedPrices field. The calculatedPrices field is an array of CalculatedPrice objects sorted by the quantity field, which defines the bounds of a pricing range.
[
	{
		"unitPrice": 58,
		"quantity": 5, // quantity from 1 to 5
		"totalPrice": 290,
		/* ... */
	},
	{
		"unitPrice": 50,
		"quantity": 10, // quantity from 6 to 10
		"totalPrice": 500,
		/* ... */
	},
	{
		"unitPrice": 44,
		"quantity": 11, // quantity from 11 to max
		"totalPrice": 484,
		/* ... */
]Display tier prices 
Displaying tier prices is fairly straightforward. You can just iterate through the calculatedPrices array and display the quantity limit and price for each tier.
<script setup>
import { useProduct } from "@shopware-pwa/composables-next";
Advanced prices 
For more complex pricing models, Shopware also supports advanced prices. The API automatically determines the correct prices for a product based on the user's context, so you don't have to deal with it in your frontend.
Display the correct price 
A product's calculatedPrice field is not always the default price since there may be tier prices or a single advanced price defined for the product. Therefore, you should create a switch within your template that differs correctly. You can use the following sudo-code as a starting point:
if product.calculatedPrices.length is more than 1
else
if product.calculatedPrices.length equals 1
product.calculatedPrice to product.calculatedPrices[0]Full example 
See a full example of displaying the default price or pricing tiers depending on the product's pricing structure below:
Click to expand full example
<script setup>
import { useProduct } from "@shopware-pwa/composables-next";
const { getFormattedPrice } = usePrice();
const { product, search } = useProduct();
await search("some-product-id");
// If there is exactly one pricing tier, use it as the default price.
// Otherwise use the calculatedPrice
const defaultPrice = computed(() => {
  if (product.value?.calculatedPrices?.length === 1) {
    return product.value.calculatedPrices[0];
  }
  return product.value?.calculatedPrice;
});
</script>
<template>
  <ul v-if="product.calculatedPrices.length > 1">
    <!-- Show pricing tiers -->
    <li
      v-for="(tierPrice, index) in product.calculatedPrices"
      :key="tierPrice.quantity"
    >
      <!-- Display "from" or "to" depending on quantity level -->
      {{ index == product.calculatedPrices.length - 1 ? "from" : "to" }}
      {{ tierPrice.quantity }} -
      {{ getFormattedPrice(tierPrice.unitPrice) }}
    </li>
  </ul>
  <div v-else>
    <!-- Show default price -->
    <div>
      {{ getFormattedPrice(defaultPrice.totalPrice) }}
      <small> incl. {{ defaultPrice.taxRules[0].taxRate }}% tax </small>
    </div>
    <div v-if="!!defaultPrice.listPrice">
      <small>
        <del>
          {{ getFormattedPrice(defaultPrice.listPrice.price) }}
        </del>
        (-{{ defaultPrice.listPrice.percentage }}%)
      </small>
    </div>
  </div>
</template>useProductPrice composable 
See dedicated Composables > useProductPrice page to check the details of the helper function.
Product listing 
Price for non-variant product (also for having tier pricing):
<script setup lang="ts">
const { totalPrice, displayFrom } = useProductPrice(/** argument omitted - Product object */);
</script>
<template>
<div>
  <span v-if="displayFrom">from</span>{{ totalPrice }} $
</div>
</template>If there is a range of prices available, you can point this out by adding from prefix, using the displayFrom indicator. The result will be a total price, prefixed by from phrase. In this case, unit price is equal to the lowest price available.
In order to ensure if the variant prices are available, you can utilize the displayVariantsFrom computed property, that contains the value in current currency:
<script setup lang="ts">
const { totalPrice, displayVariantsFrom } =
  useProductPrice(/** argument omitted - Product object */);
</script>
<template>
  <div>
    {{ totalPrice }} $
    <span v-if="displayVariantsFrom">
      Variants from {{ displayVariantsFrom }} $
    </span>
  </div>
</template>Product details page 
In this case, there are few options to display:
- Regular price
- Product with list price (kind of discount)
- Tier prices
const { totalPrice, price, tierPrices, isListPrice } = useProductPrice(product);
const { getFormattedPrice } = usePrice();Regular price, with list price included (in case of manufacturer's suggested retail price):
<template>
  <div v-if="isListPrice" class="old-price line-through">
    {{ price?.listPrice?.price }} $
    <!-- old price before discount -->
  </div>
  <div v-if="totalPrice">
    {{ totalPrice }} $
    <!-- actual price after discount -->
  </div>
</template>Tier prices presented as a table with range labeled by "to" and "from":
<template>
  <div>
    <table v-if="tierPrices.length">
      <!-- check if tierPrices array is not empty -->
      <tr v-for="(tierPrice, index) in tierPrices" :key="tierPrice.label">
        <td>
          <span v-if="index < tierPrices.length - 1"> To </span>
          <span v-else> From </span>
          {{ tierPrice.quantity }}
        </td>
        <td>{{ tierPrice.totalPrice }} $</td>
      </tr>
    </table>
    <div v-else>
      <!-- show the regular unit price instead -->
      {{ totalPrice }} $
    </div>
  </div>
</template>Format price according to current context 
There are additional metadata available in the current API context. One of them is the current currency. In order to display the price together with the currency symbol applied to the current context, use getFormattedPrice helper.
const price = 12.95;
const { getFormattedPrice } = usePrice();
const priceWithCurrency = getFormattedPrice(price);
// output: 12.95 $Thanks to this, the priceWithCurrency will have the current currency symbol prefixed or suffixed, according to the configuration.