Documentation
Feedback
Guides
VTEX IO Apps

VTEX IO Apps
Basic components
Product Price
Official extension
Version: 1.31.0
Latest version: 1.31.0

{"base64":"  ","img":{"width":110,"height":20,"type":"svg","mime":"image/svg+xml","wUnits":"px","hUnits":"px","url":"https://img.shields.io/badge/all_contributors-4-orange.svg?style=flat-square"}}

Product Price app is responsible for exporting blocks related to the product's price, such as list price, selling price and savings.

{"base64":"  ","img":{"width":2782,"height":1182,"type":"png","mime":"image/png","wUnits":"px","hUnits":"px","length":389736,"url":"https://user-images.githubusercontent.com/8443580/77692675-d5694180-6f85-11ea-8690-49db5be24b3d.png"}}

:information_source: Currently, the Product Price is the only app responsible for providing product price blocks for your theme. Both Product Summary Price and Product Price blocks, respectively from Product Summary and Store Components apps, were deprecated and therefore will no longer be evolved.

Configuration

Step 1 - Adding the Product Price app to your theme's dependencies

In your theme's manifest.json, add the Product Price app as a dependency:


_10
"dependencies": {
_10
"vtex.product-price": "1.x"
_10
}

Now, you can use all the blocks exported by the product-price app. Check out the full list below:

Block nameDescription
product-list-priceRenders the product list price. If it is equal or lower than the product selling price, this block will not be rendered.
product-selling-priceRenders the product selling price.
product-spot-priceRenders the product spot price (in case it equals the product selling price, the block is not rendered). This block finds the spot price by looking for the cheapest price of all installments options.
product-installmentsRenders the product installments. If more than one option is available, the one with the biggest number of installments will be displayed by default.
product-installments-listRenders all the installments of the payment system with the biggest amount of installments options by default.
product-installments-list-itemRenders an installments option of the product-installments-list-item
product-price-savingsRenders the product price savings, if there is any. It can show the percentage of the discount or the value of the absolute saving.
product-spot-price-savingsRenders the product spot price savings, if there is any. It can show the percentage of the discount or the value of the absolute saving.
product-list-price-rangeRenders the product list price range. It follows the same logic applied to the ListPrice: if its value is equal to the product selling price, this block is not rendered.
product-selling-price-rangeThe product selling price range.
product-seller-nameRenders the name of the product's seller.
product-price-suspenseRenders a fallback component when the price is loading and its children blocks when the price is not loading. This block is extremely useful when the store works with asynchronous prices.

All blocks listed above use product price data fetched from the store catalog. In order to understand further, please access the Pricing Module overview.

Step 2 - Adding the Product Price's blocks to your theme's templates

To add the Product Price's blocks in your theme, you just need to declare them as children of the product-summary.shelf, exported by the Product Summary app, or declare them in the theme's Product template (store.product).

Notice the following: these blocks necessarily need a Product context in order to work properly. Therefore, when declaring them as children of the product-summary.shelf, be sure that they are in a store template in which this context is available.

For example:


_12
"shelf#home": {
_12
"blocks": ["product-summary.shelf"]
_12
},
_12
_12
"product-summary.shelf": {
_12
"children": [
_12
"product-list-price",
_12
"product-selling-price#summary",
_12
"product-price-savings",
_12
"product-installments"
_12
]
_12
},

Except for the product-price-suspense, every block in this app only has three props in common:

Prop nameTypeDescriptionDefault value
markers[string]IDs of your choosing to identify the block's rendered message and customize it using the admin's Site Editor. Learn how to use them accessing the documentation on Using the Markers prop to customize a block's message. Notice the following: a block's message can also be customized in the Store Theme source code using the message prop.[]
blockClassstringBlock ID of your choosing to be used in CSS customization.undefined
messagestringDefines the block's default text i.e. the block message. You can also define which text message a block will render on the UI using the admin's Site Editor.undefined

For example:


_10
"product-selling-price#summary": {
_10
"props": {
_10
"markers": [
_10
"highlight"
_10
],
_10
"blockClass": "summary",
_10
"message": "Selling price: {sellingPriceValue}"
_10
}
_10
},

The block product-price-savings has two additional props:

Prop nameTypeDescriptionDefault value
percentageStylelocale or compactSet to compact if you want to remove the white space between the number and the percent sign. It uses pattern provided by the current locale as default.locale
minimumPercentagenumberSet the minimum value for the percentage value display. If not informed, it always appears. Example: 10, savings under or equal 10% will not show up.0

The following blocks product-list-price, product-selling-price, product-spot-price, product-spot-price-savings, product-price-savings, product-list-price-range, and product-selling-price-range have an additional prop:

Prop nameTypeDescriptionDefault value
alwaysShowbooleanRenders the block even when the product is unavailable.false

For example:


_10
"product-selling-price#summary": {
_10
"props": {
_10
"alwaysShow": true
_10
}
_10
},

The block product-installments-list has two additional props:

Prop nameTypeDescriptionDefault value
paymentSystemNamestringThis prop enables you to filter the listed installments options by a certain payment system. If not passed, the installments of the payment system with the biggest amount of installments options will be rendered.undefined
installmentsToShownumber[]Which installments options you want to show the user, in terms of the number of installments. For example, if [1, 3] is passed as a value for this prop, only the installments options with NumberOfInstallments equal to 1 and 3 will be rendered. If not passed, all options will be rendered.undefined

And the block product-installments also has two additional props:

Prop nameTypeDescriptionDefault value
installmentsCriteriamax-quantity or max-quantity-without-interestWhen set to max-quantity, the block will render the installments plan with the biggest number of installments. When set to max-quantity-without-interest, the block will render the installments plan with the biggest number of installments and zero interest. Notice that, if this prop is set to max-quantity-without-interest, and no installments plan matches the 'without interest' criteria, the component will fallback the default behavior.max-quantity
installmentOptionsFilter{ paymentSystemName?: string, installmentsQuantity?: number }Allows you to define two filtering rules that will narrow down the possible installments plans the component might render.undefined

If you are using the asynchronous price feature, you can take advantage of the product-price-suspense and its props:

Prop nameTypeDescriptionDefault value
FallbackblockName of the block that will be rendered when the price is loading.rich-text#loading

For example:


_27
{
_27
"rich-text#loading": {
_27
"props": {
_27
"text": "Loading..."
_27
}
_27
},
_27
"product-price-suspense": {
_27
"props": {
_27
"Fallback": "rich-text#loading"
_27
},
_27
"children": [
_27
"product-list-price#summary",
_27
"flex-layout.row#selling-price-savings",
_27
"product-installments#summary",
_27
"add-to-cart-button"
_27
]
_27
},
_27
"product-summary.shelf": {
_27
"children": [
_27
"stack-layout#prodsum",
_27
"product-summary-name",
_27
"product-rating-inline",
_27
"product-summary-space",
_27
"product-price-suspense"
_27
]
_27
}
_27
}

{"base64":"","img":{"src":"https://user-images.githubusercontent.com/40380674/97020006-69ed4f80-1527-11eb-8165-ff12389c7c81.gif","width":264,"height":487,"type":"gif"}}

Step 3 - Editing the block's messages

Every Product Price's block uses the ICU Message Format, making it possible to fully edit the text message and variables displayed by each block.

The variable values are rendered according to the context wrapping the block, meaning that this last one uses product data to render the price accordingly.

It is possible, however, to define which message texts a block will render on the UI using the message prop, as explained previously.

The markers prop, in turn, needs an extra configuration in the admin's Site Editor to properly work. When using it, do not forget to access the Using the Markers prop to customize a block's message documentation.

{"base64":"  ","img":{"width":2288,"height":1454,"type":"gif","mime":"image/gif","wUnits":"px","hUnits":"px","length":3874251,"url":"https://user-images.githubusercontent.com/52087100/78073694-bdbffd80-7377-11ea-9262-40854dccdd53.gif"}}

In addition to that, keep in mind the message variables for each block since you will need them to edit the desired messages using the admin's Site Editor:

  • product-list-price
Message variableTypeDescription
listPriceValuestringList price value.
listPriceWithTaxstringList price value with tax.
listPriceWithUnitMultiplierstringList price multiplied by unit multiplier.
taxPercentagestringTax percentage.
taxValuestringTax value.
hasMeasurementUnitbooleanWhether the product has a measurement unit (true) or not (false).
measurementUnitstringMeasure unit text.
hasUnitMultiplierbooleanWhether the product has a unit multiplier different than 1 (true) or not (false).
unitMultiplierstringValue of the unit multiplier.
  • product-selling-price
Message variableTypeDescription
sellingPriceValuestringSelling price value.
sellingPriceWithTaxstringSelling price with tax.
sellingPriceWithUnitMultiplierstringSelling price multiplied by unit multiplier.
taxPercentagestringTax percentage.
hasListPricebooleanWhether the product has a list price (true) or not (false).
taxValuestringTax value.
hasMeasurementUnitbooleanWhether the product has a measurement unit (true) or not (false).
measurementUnitstringMeasure unit text.
hasUnitMultiplierbooleanWhether the product has a unit multiplier different than 1 (true) or not (false).
unitMultiplierstringValue of the unit multiplier.

You can use the product-selling-price's sellingPriceWithUnitMultiplier, hasMeasurementUnit, unitMultiplier, and measurementUnit variables together to render measurement unit and unit multiplier on products that have it.

For example:


_10
{
_10
"product-selling-price": {
_10
"props": {
_10
"message": "{sellingPriceWithUnitMultiplier}{hasMeasurementUnit, select, true { / {hasUnitMultiplier, select, true {{unitMultiplier}} false {}} {measurementUnit}} false {}}"
_10
}
_10
}
_10
}

According to the example above, products with measurement unit and unit multiplier would be rendered as follows:

$24.00 / 2 oz

Notice that the measurement unit and unit multiplier would be properly rendered alongside their price

Still, according to the example, products that don't have a measurement unit and a unit multiplier will render only their price:

$24.00

  • product-spot-price
Message variableTypeDescription
spotPriceValuestringSpot price value.
  • product-installments and product-installments-list-item
Message variablesTypeDescription
spotPriceValuestringSpot price value.
installmentsNumberstringNumber of installments.
installmentValuestringValue of each installment.
installmentsTotalValuestringTotal value of installments.
interestRatestringInterest rate.
paymentSystemNamestringPayment System of the installments.
hasInterestbooleanWhether the installments have interest (true) or not (false).
hasMoreThanOnebooleanWhether there're more than 1 installments (true) or not (false).
  • product-price-savings
Message variablesTypeDescription
previousPriceValuestringPrevious price value (list price).
newPriceValuestringNew price value (selling price).
savingsValuestringDifference between previous product price and the new one.
savingsWithTaxstringDifference between previous product price and the new one with taxes.
savingsPercentagestringPercentage of savings.
  • product-spot-price-savings
Message variablesTypeDescription
previousPriceValuestringPrevious price value (list price).
newSpotPriceValuestringNew price value (spot price).
spotPriceSavingsValuestringDifference between previous product price and the spot price.
spotPriceSavingsWithTaxstringDifference between previous product price and the spot price with taxes.
spotPriceSavingsPercentagestringPercentage of savings.
  • product-list-price-range
Message variablesTypeDescription
minPriceValuestringValue of the cheapest list price SKU.
maxPriceValuestringValue of the most expensive list price SKU.
minPriceWithTaxstringValue of the cheapest list price SKU with tax.
maxPriceWithTaxstringValue of the most expensive list price SKU with tax.
listPriceValuestringValue of the list price of the only SKU available.
listPriceWithTaxstringValue of the list price of the only SKU available with tax.
  • product-selling-price-range
Message variablesTypeDescription
minPriceValuestringValue of the cheapest selling price SKU.
maxPriceValuestringValue of the most expensive selling price SKU.
minPriceWithTaxstringValue of the cheapest selling price SKU with tax.
maxPriceWithTaxstringValue of the most expensive selling price SKU with tax.
sellingPriceValuestringValue of the selling price of the only SKU available.
sellingPriceWithTaxstringValue of the selling price of the only SKU available with tax.
  • product-seller-name
Message variableTypeDescription
sellerNamestringThe name of the product's seller.

In the gif example above, the block was firstly displaying a Save $224.40 message. By editing the message exported, it now renders a You are saving: $224.40 (37%) message thanks to the changes performed through the admin's Site Editor:

{"base64":"  ","img":{"width":2878,"height":1538,"type":"png","mime":"image/png","wUnits":"px","hUnits":"px","length":394651,"url":"https://user-images.githubusercontent.com/52087100/78073688-bc8ed080-7377-11ea-9a7a-53c36d9a9fe2.png"}}

Customization

To apply CSS customization in this and other blocks, follow the instructions given in the recipe on Using CSS Handles for store customization.

CSS Handles
installmentValue
installmentsListContainer
installmentsNumber
installmentsTotalValue
installments
interestRate
listPrice'
listPrice--isUnavailable
listPriceRangeMaxValue
listPriceRangeMaxWithTax
listPriceRangeMinValue
listPriceRangeMinWithTax
listPriceRangeUniqueValue
listPriceRangeUniqueWithTax
listPriceRange
listPriceRange--isUnavailable
listPriceValue
listPriceWithTax
newPriceValue
newSpotPriceValue
paymentSystemName
previousPriceValue
savingsPercentage
savingsValue
savingsWithTax
savings
savings--isUnavailable
sellerName
sellerNameContainer
sellerNameContainer--isDefaultSeller
sellingPrice--hasListPrice
sellingPrice--isUnavailable
sellingPriceRangeMaxValue
sellingPriceRangeMaxWithTax
sellingPriceRangeMinValue
sellingPriceRangeMinWithTax
sellingPriceRangeUniqueValue
sellingPriceRangeUniqueWithTax
sellingPriceRange
sellingPriceRange--isUnavailable
sellingPriceRange--hasListPrice
sellingPriceValue
sellingPriceWithTax
sellingPrice
spotPriceSavingsPercentage
spotPriceSavingsValue
spotPriceSavingsWithTax
spotPriceSavings
spotPriceSavings--isUnavailable
spotPriceValue
spotPrice
spotPrice--isUnavailable
taxPercentage
See also
VTEX App Store
VTEX IO Apps