The Bundles feature package offers out-of-the-box functionality to sell collections of items to players at a discount. You can choose whether to allow players to purchase bundles using a custom in-experience currency or Robux, which bundle type you want to use, what set of items you want to sell, and how you want to prompt players during their gameplay.
Using the package's customization options, you can tailor your bundles to meet the design and monetization goals of your experiences.
Get package
The Creator Store is a tab of the Toolbox that you can use to find all assets that are made by Roblox and the Roblox community for use within your projects, including model, image, mesh, audio, plugin, video, and font assets. You can use the Creator Store to add one or more assets directly into an open experience, including feature packages!
Every feature package requires the Core feature package to function properly. Once the Core and Bundles feature package assets are within your inventory, you can reuse them in any project on the platform.
To get the packages from your inventory into your experience:
Add the Core and Bundles feature package to your inventory within Studio by clicking the Add to Inventory link in the following set of components.
In the menu bar, select the View tab.
In the Show section, click Toolbox. The Toolbox window displays.
In the Toolbox window, click the Inventory tab. The My Models sort displays.
Click the Feature Package Core tile, then the Bundle Feature Package tile. Both package folders display in the Explorer window.
Drag the package folders into ReplicatedStorage.
Allow DataStore calls to track player purchases with the packages.
In the Home tab of the menu bar, select Game Settings.
Navigate to the Security tab, then enable Enable Studio Access to API Services.
Define currencies
If your experience has its own currency system, you can register those with the Core feature package by defining them in ReplicatedStorage.FeaturePackagesCore.Configs.Currencies. There is a commented out example of a Gems currency already in this file; replace it with your own.
Currencies
Gems = {displayName = "Gems",symbol = "💎",icon = nil,},
The Currencies script tells the Core feature package some metadata about your currency:
- (required) displayName - The name of your currency. If you do not specify a symbol or icon, this name is used in purchase buttons (i.e. "100 Gems").
- (optional) symbol - If you have a text character to use as the icon for your currency, this is used instead of the displayName in purchase buttons (i.e. "💎100").
- (optional) icon - If you have an AssetId image icon for your currency, this is used instead of the displayName in purchase buttons (i.e. image will be placed to the left of the price "🖼️100")
Once your currency is set up, you need to manually specify the bundle's price, currency, and icon for the heads up display instead of that information being fetched from the bundle's associated developer product.
Bundles
-- If you want to use a dev product, you must provide a unique devProductId, only used by one bundle.-- We will fetch bundle price and icon from the developer productpricing = {priceType = CurrencyTypes.PriceType.Marketplace,devProductId = 1795621566,},-- Otherwise, if you want to use in-experience currency instead of a dev product, you can use the following instead:-- Price here is in the in-experience currency, not Robuxpricing = {priceType = CurrencyTypes.PriceType.InExperience,price = 79,currencyId = "Gems",icon = 18712203759,},
You also need to reference the BundlesExample script to call setInExperiencePurchaseHandler.
BundlesExample
local function awardInExperiencePurchase(
_player: Player,
_bundleId: Types.BundleId,
_currencyId: CurrencyTypes.CurrencyId,
_price: number
)
-- Check if the player has enough currency to purchase the bundle
-- Update player data, give items, etc.
-- Deduct the currency from the player
task.wait(2)
return true
end
local function initializePurchaseHandlers()
local bundles = Bundles.getBundles()
for bundleId, bundle in bundles do
-- Bundle is not associated with a developer product if it does not have marketplace price type
if not bundle or bundle.pricing.priceType ~= "Marketplace" then
continue
end
Bundles.setPurchaseHandler(bundleId, awardMarketplacePurchase)
receiptHandlers[bundle.pricing.devProductId] = receiptHandler
end
-- If you have any in-experience currencies that you are using for bundles, set the handler here
for currencyId, _ in Currencies do
Bundles.setInExperiencePurchaseHandler(currencyId, awardInExperiencePurchase)
end
end
Specifically, you need to fill out awardInExperiencePurchase, which is called by a loop through Currencies inside of the example initializePurchaseHandlers (i.e. each currencyId is connected to the handler through Bundles.setInExperiencePurchaseHandler(currencyId, awardInExperiencePurchase)).
Define bundles
All bundles offerable in your experience can be defined within ReplicatedStorage.Bundles.Configs.Bundles, with types exported from the Types script in the same folder.
If you're using a devProductId, you need to update the bundle's main devProductId to match the one in your experience. This is what will be prompted through MarketplaceService to purchase the bundle itself. It's strongly recommended to use a new developer product for the bundle to make it easier to track separate sales.
If you want a bundle with multiple items, and if these are already represented by developer products in your experience, you don't need to explicitly set item price/assetId/name, which will be fetched via product info:
README
{itemType = ItemTypes.ItemType.DevProduct,devProductId = <DEV_PRODUCT_ID>,metadata = {caption = {text = "x1",color = Color3.fromRGB(236, 201, 74),} -- Caption is optional! You can also omit this field}},
Otherwise, you can manually configure those item details:
README
{itemType = ItemTypes.ItemType.Robux,priceInRobux = 49,icon = <IMAGE_ASSET_ID>,metadata = {caption = {text = "x1",color = Color3.fromRGB(236, 201, 74),} -- Caption is optional! You can also leave omit this field}},
For example, your entire bundle will likely look like this:
README
local starterBundle: Types.RelativeTimeBundle = {bundleType = Types.BundleType.RelativeTime,-- If you want to use a dev product, you must provide a unique devProductId, only used by one bundle.-- We will fetch bundle price and icon from the developer productpricing = {priceType = CurrencyTypes.PriceType.Marketplace,devProductId = <DEV_PRODUCT_ID>,},-- Otherwise, if you want to use in-experience currency instead of a dev product, you can use the following instead:-- Price here is in the in-experience currency, not Robux-- pricing = {-- priceType = CurrencyTypes.PriceType.InExperience,-- price = 79,-- currencyId = <CURRENCY_ID>,-- icon = <IMAGE_ASSET_ID>,-- },includedItems = {[1] = {-- The item itself is not sold via a developer product, so indicate how much it is worth in Robux and give an icon-- The priceInRobux helps Bundles show relative value of the bundle price vs. the sum of its contentsitemType = ItemTypes.ItemType.Robux,priceInRobux = 49,icon = <IMAGE_ASSET_ID>,-- Alternatively, if this has a dev product leave off price and icon above and just set the devProductId-- The price and icon will be fetched from the developer product-- devProductId = <ITEM_DEV_PRODUCT_ID>-- There are more optional metadata fields that are UI-specific if neededmetadata = {caption = {text = "x1",color = Color3.fromRGB(236, 201, 74),},},},[2] = {itemType = ItemTypes.ItemType.Robux,priceInRobux = 99,icon = <IMAGE_ASSET_ID>,metadata = {caption = {text = "x1",color = Color3.fromRGB(236, 201, 74),},},},[3] = {itemType = ItemTypes.ItemType.Robux,priceInRobux = 149,icon = <IMAGE_ASSET_ID>,metadata = {caption = {text = "x1",color = Color3.fromRGB(236, 201, 74),},},},},singleUse = true, -- Once purchased or expired, no longer valid even if your experience tries to prompt (onPlayerAdded). You can make this false while testing in studio.durationInSeconds = 900, -- 15 minutesincludesOfflineTime = false, -- Only count time elapsed in the experiencemetadata = {displayName = "STARTER BUNDLE",description = "Save 75% and get a head start!",},}
Integrate server logic
Take a look at ReplicatedStorage.Bundles.Server.Examples.BundlesExample, which shows how your server will interact with the Bundles feature package and the above methods on the ModuleScript. The snippets below are from that script.
You mainly need to hook up four things once dragging the Bundles feature package into your experience:
Connect purchase handlers through Bundles.setPurchaseHandler to specify the functions to call to award items when a purchase is being processed.
BundlesExamplelocal function awardMarketplacePurchase(_player: Player, _bundleId: Types.BundleId, _receiptInfo: { [string]: any })-- Update player data, give items, etc.-- ... AND record receiptInfo.PurchaseId so we can check if user already has this bundletask.wait(2)return Enum.ProductPurchaseDecision.PurchaseGrantedendlocal function awardInExperiencePurchase(_player: Player,_bundleId: Types.BundleId,_currencyId: CurrencyTypes.CurrencyId,_price: number)-- Check if the player has enough currency to purchase the bundle-- Update player data, give items, etc.-- Deduct the currency from the playertask.wait(2)return trueendlocal function initializePurchaseHandlers()local bundles = Bundles.getBundles()for bundleId, bundle in bundles do-- Bundle is not associated with a developer product if it does not have marketplace price typeif not bundle or bundle.pricing.priceType ~= "Marketplace" thencontinueendBundles.setPurchaseHandler(bundleId, awardMarketplacePurchase)receiptHandlers[bundle.pricing.devProductId] = receiptHandlerend-- If you have any in-experience currencies that you are using for bundles, set the handler herefor currencyId, _ in Currencies doBundles.setInExperiencePurchaseHandler(currencyId, awardInExperiencePurchase)endendConnect your logic for MarketplaceService.ProcessReceipt, but this might be done elsewhere if your experience already has developer products for sale. Essentially, when a developer product receipt is being processed, they will now call Bundles.getBundleByDevProduct to check if the product belongs to a bundle. If it does, the script then calls Bundles.processReceipt.
BundlesExample-- Process receipt from marketplace to determine if player needs to be charged or notlocal function processReceipt(receiptInfo): Enum.ProductPurchaseDecisionlocal userId, productId = receiptInfo.PlayerId, receiptInfo.ProductIdlocal player = Players:GetPlayerByUserId(userId)if not player thenreturn Enum.ProductPurchaseDecision.NotProcessedYetendlocal handler = receiptHandlers[productId] -- Get the handler for the productlocal success, result = pcall(handler, receiptInfo, player) -- Call the handler to check if purchase logic is successfulif not success or not result thenwarn("Failed to process receipt:", receiptInfo, result)return Enum.ProductPurchaseDecision.NotProcessedYetendreturn Enum.ProductPurchaseDecision.PurchaseGrantedendlocal function receiptHandler(receiptInfo: { [string]: any }, player: Player)local bundleId, _bundle = Bundles.getBundleByProductId(receiptInfo.ProductId)if bundleId then-- This purchase belongs to a bundle, let Bundles handle itlocal purchaseDecision = Bundles.processReceiptAsync(player, bundleId, receiptInfo)return purchaseDecision == Enum.ProductPurchaseDecision.PurchaseGrantedend-- This purchase does not belong to a bundle,-- ... Handle all your existing logic here if you have anyreturn falseendConnect Players.PlayerAdded:Connect(Bundles.OnPlayerAdded) so that the Bundles feature package re-prompts any active bundles that have not yet expired for a player.
READMElocal function onPlayerAdded(player: Player)-- Tell Bundles when player joins so it can reload their dataBundles.onPlayerAdded(player)-- If you had some starter bundle that you wanted to offer to all new users, you could prompt that here-- ... Bundles will handle if player already has purchased it or if it's expired since it's not repeatable-- Bundles.promptIfValidAsync(player, "StarterBundle")-- Calling this here just for example, you can call this whenever or wherever you wantonPromptBundleXYZEvent(player)endPrompt bundles. While this depends on gameplay, the example prompt players with a StarterBundle onPlayerAdded.
The Bundles feature package logic ensures each player doesn't get a repeat offer if they have already purchased the bundle, or if they let the offer already expire (based on bundle config).
Whenever you want to prompt a bundle to a player, call Bundles.promptIfValidAsync(player, bundleId).
READMElocal function onPromptBundleXYZEvent(player: Player)-- Connect whatever experience event you want to use to determine when a player gets prompted the bundle-- ... This will be whenever you've met your elligibility criteria to prompt a player the bundle-- ... For example, if you want to prompt a bundle when a player joins, or when a player levels uptask.spawn(Bundles.promptIfValidAsync, player, <Some_Bundle_Id>)-- ... If creating multiple bundles, using task.spawn() to wrap the above function call will minimize discrepancies between countdownsend
Consider the following best practice guidance on redundant recordings of ReceiptIds:
While the Bundles feature package does record ReceiptIds to avoid processing the same receipt twice, you should also be recording ReceiptIds inside of your tables so that if the purchase flow fails after their purchase handler has already finished, you know on subsequent retry not to award items again.
The Bundles feature package will not record the ReceiptId if the purchase fails at any step, so you should ensure that you are recording the ReceiptId in your tables before processing the receipt as part of your purchaseHandler.
This redundancy helps ensure that all purchase logic has been appropriately handled and that your DataStore+Bundles Feature Package's DataStore reaches eventual consistency, with the your data store being the source of truth.
Configure constants
Constants for the Core feature package live in two spots:
Shared constants live in ReplicatedStorage.FeaturePackagesCore.Configs.SharedConstants.
Package-specific constants, in this case the Bundles feature package, live in ReplicatedStorage.Bundles.Configs.Constants.
The main things you might want to adjust to meet the design requirements of your experience:
- Sound assetIDs
- Purchase effect duration and particle colors
- Heads up display collapsibility
Additionally, you can find strings for translation broken out into one location: ReplicatedStorage.FeaturePackagesCore.Configs.TranslationStrings.
Customize UI components
By modifying the package objects, such as colors, font, and transparency, you can adjust the visual presentation of your bundle prompts. However, keep in mind that if you move any of the objects around hierarchically, the code will not be able to find them, and you'll need to make adjustments to your code.
A prompt is made up of two high-level components:
- PromptItem – The individual component repeated out for each item within a bundle (item image, caption, name, price).
- Prompt – The prompt window itself.
The heads up display is also made up of two components:
- HudItem – An individual component that represents each menu option in the heads up display.
- Hud – To be filled with programmatically with HudItems.
If you want to have greater control over the heads up display, instead of just using the existing HUD UI within ReplicatedStorage.Bundles.Objects.BundlesGui, you can move things around to meet your own design requirements. Just be sure to update client script behavior in the ReplicatedStorage.Bundles.Client.UIController script.
API reference
Types
RelativeTime
Once the RelativeTime bundle is offered to a player, it remains available until the time duration runs out. This type displays on the player's heads up display, and automatically prompts on future sessions until the bundle expires or the player purchases it.
A common example of this bundle type is a single-use starter pack offer that displays to all new players for 24 hours. For industry best practices on how implement starter pack bundles, see Starter Pack Design.
Name | Type | Description |
---|---|---|
includeOfflineTime | bool | (Optional) If not set, only time spent in experience will count towards the remaining offer duration. |
singleUse | bool | (Optional) If not set, the purchase can be reactivated after it's purchased or expired. If set, once purchased or expired the first time, it will not be promptable ever again, even if you call Bundles.promptIfValidAsync with the bundleId. |
FixedTime
Once the FixedTime bundle is offered to a player, it remains available until the end of the set coordinated universal time (UTC). This type displays on the player's heads up display, and automatically prompts on future sessions until the bundle expires or the player purchases it.
A common example of this bundle type is a holiday offer that's only available for a given month.
OneTime
A OneTime bundle is only available in the moment that it's offered to a player. It does not display on the player's heads up display, and once a player closes the prompt, it cannot be reopened until it's prompted by the server again.
A common example of this bundle type is an offer to purchase more in-experience currency the moment a player runs out.