How to make a shop
This is a quick guide of how to make a shop where you can buy items with other items or costing a scoreboard value (money or coins).
Item shop
This method consists of buying items with other items, in this example, you will buy 1 netherite ingot with 5 diamonds.
Related: Count how much of X item the player has
[!NOTE] You can add any command you want to run (for example a playsound) before the last command, but the selector must be the same as the command before.
Java
Using commands
First we figure out how many diamonds the player has and store the value in a scoreboard. If the player has the required number of diamonds we give the netherite ingot and clear the diamonds from them.
Using command blocks we need to make sure we select the correct player every time, so with the first command we tag the relevant player to later limit our selector to that player.
To avoid the player getting unintentionally re-selected we remove the tag at the end. Using functions we can rely on @s
instead, assuming the function is executed as
the relevant player.
# Command blocks
tag @p add buyer.netherite
execute as @a[tag=buyer.netherite] store result score @s diamonds run clear @s diamond 0
give @a[tag=buyer.netherite,scores={diamonds=5..}] netherite_ingot 1
clear @a[tag=buyer.netherite,scores={diamonds=5..}] diamond 5
tellraw @a[tag=buyer.netherite,scores={diamonds=5..}] {"text":"You bought a netherite ingot for 5 diamonds","color":"green"}
tellraw @a[tag=buyer.netherite,scores={diamonds=..4}] {"text":"You don't have 5 diamonds","color":"dark_red"}
tag @a remove buyer.netherite
Or if you prefer a function:
# function example:load
scoreboard objectives add diamonds dummy
# function example:buy/netherite
execute store result score @s diamonds run clear @s diamond 0
execute if entity @s[scores={diamonds=5..}] run function example:buy/netherite/success
execute unless entity @s[scores={diamonds=5..}] run tellraw @s {"text":"You don't have 5 diamonds","color":"dark_red"}
# function example:buy/netheirte/success
give @s netherite_ingot 1
clear @s diamond 5
tellraw @s {"text":"You bought a netherite ingot for 5 diamonds","color":"green"}
[!NOTE] The function
example:buy/netherite
must be runas
the player
In 1.20.5 execute if items
was added, which allows to count items in a diferent way. To use this method, all command remains the same, as the previus example except the one for counting items that is this one:
execute as @a[tag=buyer.netherite] store result score @s diamonds run clear @s diamond 0
That we can replace it for this one:
execute as @p[tag=buyer.netherite] store result score @s diamonds if items entity @s container.* diamond
Or if you are using the example function, you need to replace this command
execute store result score @s diamonds run clear @s diamond 0
With this command:
execute store result score @s diamonds if items entity @s container.* diamond
[!NOTE] The method that uses the
/clear
command will work in 1.20.5+ but it is recomended to use the one specific for that versions (usingexecute if items
, that will not work below 1.20.5).
Using a villager
Villagers are a passive mob that can be interacted with, and will open the trading GUI.
We can use this vanilla functionality to add our custom trades.
This can be done by manually typing the command, but it is recomeneded to use an online generator to generate the command with ease.
This method has some advantages and disadvantages.
Advantages:
- Easy to set up, it only require a summon command.
- It does not cause performance issues, as we don’t run any commands.
Disadvantages:
- Need to be interacted. You can’t force the player to buy that item.
- Limited to give 2 items to get one (workarrounds are possible, like giving a shulker box with items in it).
- You can’t show a succes message (workarrounds are possible).
- It is linked to a place, unless you teleport the villager to the player.
- 2 players can’t acces the trading screen at the same time.
Here is an example command:
/summon villager ~ ~ ~ {Offers:{Recipes:[{rewardExp:0b,maxUses:2147483647,buy:{id:"minecraft:diamond",count:1},buyB:{id:"minecraft:emerald",count:1},sell:{id:"minecraft:netherite_ingot",count:1}}]}}
Let’s break it donw:
- Inside
Offers
we haveRecipes
, there will be all listed trades. - This trade has
rewardExp:0b
so the player will not get any experience when buying it (you can enable it if you want). maxUses:2147483647
is the maximum times it can be traded, this is the maximum value we can set.- Inside
buy
andbuyB
we are going to specify what items the player will spend. - In
sell
we specify what item the player will get.
We can use item components or nbt for both the item the player will spend and the item the player will get, here is an example for 1.20.5+, every item needs to have a custom_data
of special:true
/summon villager ~ ~ ~ {Offers:{Recipes:[{rewardExp:0b,maxUses:2147483647,buy:{id:"minecraft:diamond",count:1,components:{"minecraft:custom_data":{special:true}}},buyB:{id:"minecraft:emerald",count:1,components:{"minecraft:custom_data":{special:true}}},sell:{id:"minecraft:netherite_ingot",count:1,components:{"minecraft:custom_model_data":special:true}}}]}}
Here is the same command but for pre-1.20.5, using nbt data instead:
/summon villager ~ ~ ~ {Offers:{Recipes:[{rewardExp:0b,maxUses:2147483647,buy:{id:"minecraft:diamond",Count:1b,tag:{special:true}},buyB:{id:"minecraft:emerald",Count:1b,tag:{special:true}},sell:{id:"minecraft:netherite_ingot",Count:1b,tag:{special:true}}}]}}
Bedrock
In bedrock we have the hasitem
argument, so it uses less commands than in Java.
If you don’t want to use an NPC can use this method, it is very similar to Java but it uses the hasitem
argument instead.
tag @p add buyer.netherite
give @a[tag=buyer.netherite,hasitem={item=diamond,quantity=5..}] netherite_ingot
tellraw @a[tag=buyer.netherite,hasitem={item=diamond,quantity=5..}] {"rawtext":[{"text":"§2You bought a netherite ingot"}]}
tellraw @a[tag=buyer.netherite,hasitem={item=diamond,quantity=..5}] {"rawtext":[{"text":"§3You don't have 5 diamonds"}]}
clear @a[tag=buyer.netherite,hasitem={item=diamond,quantity=5..}] diamond 5
tag @a remove buyer.netherite
[!NOTE] in order for it to work with npc, change
@p
to@initiator
Related: How to setup a NPC?
[!NOTE] It is super important to clear the diamonds in the last step before removing the tag
Add more than one items
In this gide we will use just one item, but you can have multiples but it will require a second tag, that must be added if the player has both items. Then when we clear the items we are going to clear them for the player with that tag.
In this example we will buy a gold block with 2 emeralds and 5 diamonds. If you are in Java you will need one scoreboard for each item, assuming you already store the items result in each one.
# Example
tag @p add buyer.example
execute as @a[tag=buyer.example] run tag @s[scores={diamonds=5..,emeralds=2..}] add buy.example
# run any tellraw to the player with the tag buy.example
clear @a[tag=buy.example] diamond 5
clear @a[tag=buy.example] emerald 2
give @a[tag=buy.example] gold_block 1 And then we remove all the previus used tags:
tag @a remove buy.example
tag @a remove buyer.example
In bedrock use the hasitem
argument instead of scores
Score shop
This method uses a scoreboard as a currency (such as coins
for this example) and you can buy items with that currency. In this example, you can buy a diamond
with 10 coins
. You can adjust the items and the prices by editing the command item and quantity.
Java and Bedrock
When the player wants to buy the item, run this commands in order. It can be achieved with a button, an impulse command block and a chain of command blocks behind as an example. In this example the currency is a dummy
scoreboard called coins
.
With command blocks
[!NOTE] This example uses Java syntax for the message that appears when the player buys the item
/tag @p add buyer.diamond
/execute as @p[tag=buyer.diamond] run tag @s[scores={coins=10..}] add buy.diamond
/scoreboard players remove @a[tag=buy.diamond] coins 10
/tellraw @a[tag=buyer_diamond,tag=!buy.diamond] "You need at least 10 coins"
/tellraw @a[tag=buy.diamond] "You bought a diamond"
/tag @a remove buy.diamond
/tag @a remove buyer.diamond
In a function
This is more optimized compared to using command blocks as functions keep the context
# function example:buy/diamond
execute if entity @s[scores={coins=10..}] run function example:buy/diamond/success
tellraw @s[scores={coins=..9}] {"text":"You don't have 10 coins","color":"red"}
# function example:buy/diamond/success
give @s diamond
scoreboard players remove @s coins 10
tellraw @s {"text":"You bought a diamond","color":"green"}
[!NOTE] The function
example:buy/diamond
must be runas
the player