Starting item and inventory tools, sketched spear animation templates.

This commit is contained in:
2025-08-28 08:36:20 -04:00
parent 2a236ea041
commit c41cac58c2
11 changed files with 183 additions and 7 deletions

View File

@@ -2,4 +2,7 @@ class_name Item extends Resource
@export var image : Texture2D
@export var name : StringName
@export var brief : String
@export var quantity : int = 1
@export var max_quantity : int = 1

Binary file not shown.

View File

@@ -25,9 +25,14 @@ var armor : Armor
var accessory : Accessory
var inventory : Dictionary[Vector2, Item] = {}
var inventory_size : Vector2i = Vector2i(6,1)
var inventory_count : int = 0
var quest_sprite : QuestSprite
signal changed()
signal levelled()
signal died()
func _init() -> void:
stats = StatBlock.new()
@@ -45,10 +50,12 @@ func generate() -> void:
max_energy = stats.INT * 10 + stats.FAI * 10
life = max_life
energy = max_energy
changed.emit()
func assign_quest(quest : Quest) -> void:
self.quest = quest
quest.initiate(self)
changed.emit()
func full_name() -> String:
return given_name + " " + surname
@@ -57,18 +64,65 @@ func gain_level() -> void:
level += 1
#TODO: Make stats improve based on job
Game.notice("%s has reached level %d!" % [full_name(), level])
#TODO: Make Sideview display level up
changed.emit()
levelled.emit()
func gain_exp(amount : int) -> void:
exp += amount
while exp >= get_tnl():
exp -= get_tnl()
gain_level()
changed.emit()
func gain_gold(amount :int) -> void:
gold += amount
changed.emit()
func get_tnl() -> int:
if job:
return job.get_tnl(level)
else:
return level * 10
func find_open_inventory_slot(item : Item) -> Vector2:
var first_open : Vector2 = Vector2(-1,-1)
for j in range(inventory_size.y):
for i in range(inventory_size.x):
var curr_slot : Item = inventory.get(Vector2(i,j))
if curr_slot != null:
#TODO: Figure out how to handle split stacking later?
if curr_slot.name == item.name and curr_slot.max_quantity < curr_slot.quantity + item.quantity:
return Vector2(i,j)
elif first_open == Vector2(-1,-1):
first_open = Vector2(i,j)
return first_open
func try_pickup_item(item : Item) -> bool:
if inventory_count >= inventory_size.x * inventory_size.y:
return false
pickup_item(item)
return true
func pickup_item(item : Item) -> void:
var open_slot = find_open_inventory_slot(item)
if inventory.has(open_slot):
inventory[open_slot].quantity += item.quantity
else:
if inventory_count >= inventory_size.x * inventory_size.y:
printerr("Cannot fit additional item!")
inventory[open_slot] = item
inventory_count+=1
changed.emit()
func move_item(from : Vector2, to: Vector2) -> void:
if !inventory.has(from):
printerr("Cannot move item from %s to %s, %s is empty!" % [from, to, from])
var itm = inventory[from]
if inventory.has(to):
var itm2 = inventory[to]
inventory[from] = itm2
else:
inventory.erase(from)
inventory[to] = itm
changed.emit()

View File

@@ -15,9 +15,19 @@ var data : Adventurer
@onready var faiLabel :Label = %FAILabel
@onready var lukLabel :Label = %LUKLabel
var item_slots : Array[ItemSlot]
func setup(adv : Adventurer) -> void:
data = adv
data.changed.connect(_on_data_changed)
update_stats()
item_slots = []
for child : ItemSlot in %InventoryGrid.get_children():
item_slots.append(child)
#TODO: Show equipment
update_items()
func update_stats() -> void:
nameLabel.text = data.full_name()
if data.job:
jobLabel.text = data.job.name
@@ -33,8 +43,21 @@ func setup(adv : Adventurer) -> void:
chaLabel.text = str(data.stats.CHA)
faiLabel.text = str(data.stats.FAI)
lukLabel.text = str(data.stats.LUK)
#TODO: Show equipment
populate_items()
func populate_items() -> void:
pass
func update_items() -> void:
var isize = data.inventory_size.x * data.inventory_size.y
var islots = len(item_slots)
%InventoryGrid.columns = data.inventory_size.x
for crd in data.inventory.keys():
var idx = crd.y * data.inventory_size.x + crd.x
var item = data.inventory[crd]
item_slots[idx].update(item)
item_slots[idx].visible = true
for idx in range(isize, islots):
item_slots[idx].visible = false
func _on_data_changed() -> void:
update_stats()
update_items()

View File

@@ -0,0 +1 @@
extends Window

View File

@@ -0,0 +1 @@
uid://cpiucydmov1qj

View File

@@ -7,7 +7,8 @@ var item : Item
@onready var item_sprite : TextureRect = $Item
func assign(itm : Item) -> void:
item = item
item = itm
if item.image != null:
item_sprite.texture = item.image
func swap(item_slot : ItemSlot) -> void:
@@ -15,6 +16,16 @@ func swap(item_slot : ItemSlot) -> void:
assign(itm)
item_slot.assign(itm)
func update(itm : Item) -> void:
if itm == null:
clear()
elif itm != item:
assign(itm)
func clear() -> void:
item = null
item_sprite.texture = null
func _on_gui_input(event: InputEvent) -> void:
var mmevt = event as InputEventMouseMotion
var mbevt = event as InputEventMouseButton

View File

@@ -12,6 +12,7 @@ var data : Adventurer
func _ready() -> void:
Game.player = self
data = Adventurer.new()
data.inventory_size = Vector2(4,2)
data.name = "Player"
setup.call_deferred()

View File

@@ -1,6 +1,6 @@
extends Node2D
var test_adv = preload("res://templates/test_adventurer.tscn")
var test_item = preload("res://data/items/pitchfork.tres")
@onready var door_player : AudioStreamPlayer = $AudioStreamPlayer
func _ready() -> void:
#var adv : Adventurer = test_adv.instantiate() as Adventurer
@@ -13,6 +13,9 @@ func _ready() -> void:
quest.events.append(evt)
Guild.add_quest(quest)
Guild.assign_quest(Game.player.data, quest)
var itm = test_item.duplicate()
Game.player.data.pickup_item(itm)
Game.player.data.move_item(Vector2(0,0), Vector2(1,0))
#Game.end_shift()
#var tween = create_tween()
#tween.tween_interval(3)

View File

@@ -0,0 +1,78 @@
[gd_scene load_steps=4 format=3 uid="uid://dggjyxoa33r7y"]
[ext_resource type="Script" uid="uid://cpiucydmov1qj" path="res://scripts/item_display_window.gd" id="1_n2g4t"]
[ext_resource type="Texture2D" uid="uid://cjjtgrr56guj0" path="res://graphics/ui/item-slot.png" id="2_dxjvd"]
[ext_resource type="Texture2D" uid="uid://clrvwaqb61lpv" path="res://graphics/items/pitchfork.png" id="3_tsl8r"]
[node name="Item Display Window" type="Window"]
oversampling_override = 1.0
size = Vector2i(300, 300)
borderless = true
popup_window = true
script = ExtResource("1_n2g4t")
[node name="Panel" type="PanelContainer" parent="."]
custom_minimum_size = Vector2(300, 300)
anchors_preset = -1
offset_right = 300.0
offset_bottom = 300.0
[node name="MarginContainer" type="MarginContainer" parent="Panel"]
layout_mode = 2
theme_override_constants/margin_left = 2
theme_override_constants/margin_top = 2
theme_override_constants/margin_right = 0
theme_override_constants/margin_bottom = 0
[node name="VBoxContainer" type="VBoxContainer" parent="Panel/MarginContainer"]
layout_mode = 2
[node name="HBoxContainer" type="HBoxContainer" parent="Panel/MarginContainer/VBoxContainer"]
layout_mode = 2
[node name="MarginContainer" type="CenterContainer" parent="Panel/MarginContainer/VBoxContainer/HBoxContainer"]
custom_minimum_size = Vector2(71, 71)
layout_mode = 2
[node name="TextureRect2" type="TextureRect" parent="Panel/MarginContainer/VBoxContainer/HBoxContainer/MarginContainer"]
custom_minimum_size = Vector2(71, 71)
layout_mode = 2
texture = ExtResource("2_dxjvd")
[node name="TextureRect" type="TextureRect" parent="Panel/MarginContainer/VBoxContainer/HBoxContainer/MarginContainer"]
layout_mode = 2
texture = ExtResource("3_tsl8r")
stretch_mode = 3
[node name="VBoxContainer" type="VBoxContainer" parent="Panel/MarginContainer/VBoxContainer/HBoxContainer"]
layout_mode = 2
theme_override_constants/separation = 0
[node name="Label" type="Label" parent="Panel/MarginContainer/VBoxContainer/HBoxContainer/VBoxContainer"]
layout_mode = 2
text = "ITEM NAME"
[node name="Label2" type="Label" parent="Panel/MarginContainer/VBoxContainer/HBoxContainer/VBoxContainer"]
layout_mode = 2
text = "ITEM TYPE"
[node name="Label3" type="Label" parent="Panel/MarginContainer/VBoxContainer/HBoxContainer/VBoxContainer"]
layout_mode = 2
text = "ITEM GRADE"
[node name="Label2" type="Label" parent="Panel/MarginContainer/VBoxContainer"]
layout_mode = 2
text = "Primary Stat Goes Here"
[node name="Label3" type="Label" parent="Panel/MarginContainer/VBoxContainer"]
custom_minimum_size = Vector2(0, 60)
layout_mode = 2
text = "Secondary Stats Go Here"
[node name="HSeparator" type="HSeparator" parent="Panel/MarginContainer/VBoxContainer"]
layout_mode = 2
[node name="Label" type="Label" parent="Panel/MarginContainer/VBoxContainer"]
custom_minimum_size = Vector2(0, 150)
layout_mode = 2
text = "Brief Description Goes Here"

View File

@@ -522,6 +522,7 @@ offset_bottom = 509.0
texture = ExtResource("10_rldp4")
[node name="InventoryGrid" type="GridContainer" parent="."]
unique_name_in_owner = true
anchors_preset = -1
offset_left = 325.0
offset_top = 513.0