From e08a29e73ea4f7e6d78e8e7f5a6e7033dbc1f542 Mon Sep 17 00:00:00 2001 From: Daniel Weipert Date: Mon, 30 Dec 2024 15:15:01 +0100 Subject: next commit --- .gitignore | 5 +- client.gd | 33 ++++---- client.tscn | 2 +- data_models/login_character_list.gd | 16 ++++ extractor/gat_format.gd | 18 ++++- extractor/gnd_format.gd | 11 ++- extractor/grf.gd | 5 ++ extractor/rsm_format.gd | 87 +++++++++++++++++++++ packets/map_server/initial_status_packet.gd | 2 +- packets/map_server/update_hotkeys_packet.gd | 2 +- ui/chat_window.tscn | 2 +- ui/login.gd | 29 ++++--- ui/login.tscn | 37 ++++++--- ui/login/character_selection_item.gd | 54 +++++++++---- ui/login/character_selection_item.tscn | 19 ++--- ui/login/character_selection_status.gd | 18 +++++ ui/login/character_selection_status.tscn | 36 ++++----- ui/login/login_character_selection_list.gd | 111 +++++++++++++++++++++++++++ ui/login/login_character_selection_list.tscn | 61 +++++++++++++++ ui/theme_clear.tres | 4 +- ui/window.gd | 1 + ui/window.tscn | 5 +- 22 files changed, 457 insertions(+), 101 deletions(-) create mode 100644 data_models/login_character_list.gd create mode 100644 extractor/rsm_format.gd create mode 100644 ui/login/login_character_selection_list.gd create mode 100644 ui/login/login_character_selection_list.tscn diff --git a/.gitignore b/.gitignore index 653681f..375f419 100644 --- a/.gitignore +++ b/.gitignore @@ -3,9 +3,6 @@ *.gd.uid /android/ -/client_data/*.grf -/client_data/data/ -/client_data/BGM/ -/client_data/System/ +/client_data/ /extractor/test/ diff --git a/client.gd b/client.gd index 5025c8c..3a69e9d 100644 --- a/client.gd +++ b/client.gd @@ -21,19 +21,26 @@ func _ready() -> void: #FileAccess.get_file_as_bytes("res://client_data/data/sprite/cursors.act") #) #) - GATFormat.from_bytes( - ByteStream.from_bytes( - FileAccess.get_file_as_bytes("res://client_data/data/int_land02.gat") - ) - ) - GNDFormat.from_bytes( - ByteStream.from_bytes( - FileAccess.get_file_as_bytes("res://client_data/data/int_land02.gnd") - ) - ) - RSWFormat.from_bytes( + #GATFormat.from_bytes( + #ByteStream.from_bytes( + #FileAccess.get_file_as_bytes("res://client_data/data/int_land02.gat") + #) + #) + #GNDFormat.from_bytes( + #ByteStream.from_bytes( + #FileAccess.get_file_as_bytes("res://client_data/data/int_land02.gnd") + #) + #) + #RSWFormat.from_bytes( + #ByteStream.from_bytes( + ##FileAccess.get_file_as_bytes("res://client_data/data/int_land02.rsw") + #FileAccess.get_file_as_bytes("res://client_data/data/pay_dun00.rsw") + #) + #) + RSMFormat.from_bytes( ByteStream.from_bytes( - #FileAccess.get_file_as_bytes("res://client_data/data/int_land02.rsw") - FileAccess.get_file_as_bytes("res://client_data/data/pay_dun00.rsw") + #FileAccess.get_file_as_bytes("res://client_data/data/model/prontera/chair_01.rsm") + FileAccess.get_file_as_bytes("res://client_data/data/model/izlude/iz_academy.rsm") + # TODO: find newer version rsm to test ) ) diff --git a/client.tscn b/client.tscn index 9578f9d..c9d2006 100644 --- a/client.tscn +++ b/client.tscn @@ -1,6 +1,6 @@ [gd_scene load_steps=3 format=3 uid="uid://cuylx656oarpy"] -[ext_resource type="Script" uid="uid://dsiqqaly54ms0" path="res://client.gd" id="1_e4txq"] +[ext_resource type="Script" uid="uid://u2h0eahwiv1u" path="res://client.gd" id="1_e4txq"] [ext_resource type="PackedScene" uid="uid://n0y3fpb8j820" path="res://ui/cursor.tscn" id="2_abx5m"] [node name="Client" type="Node"] diff --git a/data_models/login_character_list.gd b/data_models/login_character_list.gd new file mode 100644 index 0000000..f34d61e --- /dev/null +++ b/data_models/login_character_list.gd @@ -0,0 +1,16 @@ +class_name LoginCharacterList +extends Resource + + +var slot_count: int + +var character_information: Array[CharacterInformation] + + +static func from_character_list_packet(packet: CharacterServerLoginSuccessCharacterListPacket) -> LoginCharacterList: + var resource := LoginCharacterList.new() + + resource.slot_count = packet.maximum_slot_count + resource.character_information = packet.character_information + + return resource diff --git a/extractor/gat_format.gd b/extractor/gat_format.gd index 712657b..7192cbc 100644 --- a/extractor/gat_format.gd +++ b/extractor/gat_format.gd @@ -42,6 +42,14 @@ static func from_bytes(bytes: ByteStream) -> GATFormat: return gat_format +enum TileFlags { + Walkable = 0b00000001, + Water = 0b00000010, + Snipable = 0b00000100, + Cliff = 0b00001000, +} + + class Tile: ## Byte Type: f32 [br] ## Byte Length: 4 [br] @@ -63,12 +71,12 @@ class Tile: ## Orignal Coordinates: (1, 1) var top_right_altitude: int - ## Byte Type: u32 ?[br] - ## Byte Length: 4 ? + ## Byte Type: u8 [br] + ## Byte Length: 1 var terrain_type: int - func get_height_map() -> Dictionary: # Dictionary[Vector2, int] + func get_height_map() -> Dictionary[Vector2, int]: return { Vector2(0, 0): top_left_altitude, Vector2(1, 0): top_right_altitude, @@ -84,6 +92,8 @@ class Tile: tile.bottom_right_altitude = bytes.decode_float() tile.top_left_altitude = bytes.decode_float() tile.top_right_altitude = bytes.decode_float() - tile.terrain_type = bytes.decode_u32() + tile.terrain_type = bytes.decode_u8() + + bytes.advance(3) # unused return tile diff --git a/extractor/gnd_format.gd b/extractor/gnd_format.gd index f13956e..ea8d965 100644 --- a/extractor/gnd_format.gd +++ b/extractor/gnd_format.gd @@ -211,7 +211,7 @@ class Surface: var vertex_color_alpha: int - func get_uvs() -> Dictionary: # Dictionary[Vector2, Vector2] + func get_uvs() -> Dictionary[Vector2, Vector2]: return { Vector2(0, 0): Vector2(u_top_left, v_top_left), Vector2(1, 0): Vector2(u_top_right, v_top_right), @@ -220,6 +220,15 @@ class Surface: } + func get_vertex_color() -> Color: + return Color8( + vertex_color_red, + vertex_color_green, + vertex_color_blue, + vertex_color_alpha + ) + + static func from_bytes(bytes: ByteStream) -> Surface: var surface = Surface.new() diff --git a/extractor/grf.gd b/extractor/grf.gd index e2a3c82..1d19046 100644 --- a/extractor/grf.gd +++ b/extractor/grf.gd @@ -446,6 +446,11 @@ func convert(destination: String = "res://client_data"): scene_root.add_child(audio, true) audio.owner = scene_root + var surfrace_tool := SurfaceTool.new() + for surface: GNDFormat.Surface in gnd.surfaces: + pass + #surfrace_tool.add_vertex() + scene.pack(scene_root) ResourceSaver.save(scene, "%s/%s/%s.tscn" % [destination, base_directory_path, file_name]) diff --git a/extractor/rsm_format.gd b/extractor/rsm_format.gd new file mode 100644 index 0000000..d3dabb0 --- /dev/null +++ b/extractor/rsm_format.gd @@ -0,0 +1,87 @@ +class_name RSMFormat + + +## Byte Length: 4 [br] +## GRAT +var signature: String = "GRSM" + +## Byte Type: u8 [br] +## Byte Length: 2 +var version: Version + +## Byte Type: u32 [br] +## Byte Length: 4 +var animation_length: int + +## Byte Type: u32 [br] +## Byte Length: 4 +var shade_type: int + +## Byte Type: u8 [br] +## Byte Length: 1 +## Versions: [>=1.4] +var alpha: int + +## Byte Type: u8 [br] +## Byte Length: 16 +## Versions: [<2.2] +var reserved: PackedByteArray + +## Byte Type: f32 [br] +## Byte Length: 4 +## Versions: [>=2.2] +var frames_per_second: float + +## Byte Type: u32 [br] +## Byte Length: 4 +## Versions: [<2.3] +var texture_count: int + +## Length: [member texture_count] +## Byte Length: 40 +var texture_names: Array[String] + +## Byte Type: u8 +## Byte Length: 40 +var root_node_name: String + +## Byte Type: u32 [br] +## Byte Length: 4 +## Versions: [>=2.2] +var root_node_count: int + +## Length: [member root_node_count] +## Byte Length: 40 +var root_node_names: Array[String] + +## Byte Type: u32 [br] +## Byte Length: 4 +var node_count: int + +## Length: [member node_count] +var nodes: Array[ModelNode] + + +static func from_bytes(bytes: ByteStream) -> RSMFormat: + var rsm_format = RSMFormat.new() + + bytes.advance(rsm_format.signature.length()) + + @warning_ignore("shadowed_variable") + var version = Version.new() + version.major = bytes.decode_u8() + version.minor = bytes.decode_u8() + rsm_format.version = version + + print(inst_to_dict(rsm_format)) + return rsm_format + + +class ModelNode: + ## Byte Type: u8 [br] + ## Byte Length: 40 + var node_name: String + + ## Byte Type: u8 [br] + ## Byte Length: 40 + var parent_node_name: String diff --git a/packets/map_server/initial_status_packet.gd b/packets/map_server/initial_status_packet.gd index 9c41ef9..43b65fe 100644 --- a/packets/map_server/initial_status_packet.gd +++ b/packets/map_server/initial_status_packet.gd @@ -12,7 +12,7 @@ const BYTE_LENGTH := 2 + 14 + 28 # TODO -static func from_bytes(bytes: PackedByteArray) -> InitialStatusPacket: +static func from_bytes(_bytes: PackedByteArray) -> InitialStatusPacket: var packet = InitialStatusPacket.new() return packet diff --git a/packets/map_server/update_hotkeys_packet.gd b/packets/map_server/update_hotkeys_packet.gd index f991a7e..3788d1d 100644 --- a/packets/map_server/update_hotkeys_packet.gd +++ b/packets/map_server/update_hotkeys_packet.gd @@ -11,7 +11,7 @@ const BYTE_LENGTH := 2 + 1 + 2 + 7*38 # TODO -static func from_bytes(bytes: PackedByteArray) -> UpdateHotkeysPacket: +static func from_bytes(_bytes: PackedByteArray) -> UpdateHotkeysPacket: var packet = UpdateHotkeysPacket.new() return packet diff --git a/ui/chat_window.tscn b/ui/chat_window.tscn index 5b4e8aa..e67d360 100644 --- a/ui/chat_window.tscn +++ b/ui/chat_window.tscn @@ -1,6 +1,6 @@ [gd_scene load_steps=3 format=3 uid="uid://c8uqw08hxfqlu"] -[ext_resource type="Script" uid="uid://bdjmcp6ewhq54" path="res://ui/chat_window.gd" id="1_vovuq"] +[ext_resource type="Script" uid="uid://cy5bwkc4gokw1" path="res://ui/chat_window.gd" id="1_vovuq"] [sub_resource type="StyleBoxFlat" id="StyleBoxFlat_ku06j"] bg_color = Color(0.133333, 0.133333, 0.133333, 0.784314) diff --git a/ui/login.gd b/ui/login.gd index 3180024..8cabec8 100644 --- a/ui/login.gd +++ b/ui/login.gd @@ -88,23 +88,20 @@ func _on_character_server_login_pressed(character_server_info: CharacterServerIn var character_list: CharacterServerLoginSuccessCharacterListPacket = await Network.character_server.logged_in_character_list - for node in %CharacterList.get_children(): - node.queue_free() - - for slot_idx in character_list.character_information.size(): - var info: CharacterInformation = character_list.character_information[slot_idx] - var character = preload("res://ui/login/character_selection_item.tscn").instantiate() - character.initialize_with_info(info) - character.selected.connect(func(): - current_character_slot_idx = slot_idx - current_character_information = info - %CharacterSelectionStatus.set_info(info) - ) - %CharacterList.add_child(character) + var login_character_list := LoginCharacterList.from_character_list_packet(character_list) + %CharacterSelectionList.login_character_list = login_character_list + %CharacterSelectionList.selected.connect(func(slot_idx: int): + current_character_slot_idx = slot_idx + %CharacterSelectionSlotLabel.text = "%s/%s" % [str(slot_idx + 1), login_character_list.slot_count] - # pre-select first character - if slot_idx == 0: - character.selected.emit() + if slot_idx < character_list.character_information.size(): + current_character_information = character_list.character_information[slot_idx] + else: + current_character_information = null + ) + + # pre-select first character + %CharacterSelectionList.select(0) switch_screen(%CharacterSelection) diff --git a/ui/login.tscn b/ui/login.tscn index 9cba3ca..4842c26 100644 --- a/ui/login.tscn +++ b/ui/login.tscn @@ -1,7 +1,7 @@ [gd_scene load_steps=20 format=4 uid="uid://dser74lcd3a4g"] [ext_resource type="Script" uid="uid://dqswsdaamfhbq" path="res://ui/login.gd" id="1_1m5cv"] -[ext_resource type="Texture2D" uid="uid://cxd6dnc7s17vg" path="res://ui/login/backround.jpg" id="2_elmti"] +[ext_resource type="Texture2D" uid="uid://cxd6dnc7s17vg" path="res://client_data/skin/login_background.jpg" id="2_elmti"] [ext_resource type="AudioStream" uid="uid://br8ujl4uxv14a" path="res://client_data/BGM/01.mp3" id="3_2nukd"] [ext_resource type="Theme" uid="uid://c5sm3yvuakj3b" path="res://ui/theme.tres" id="3_7ogdv"] [ext_resource type="PackedScene" uid="uid://cjcm2mai50thr" path="res://ui/bmp_texture_button.tscn" id="3_qemc0"] @@ -16,7 +16,7 @@ [ext_resource type="Texture2D" uid="uid://b13jh48hyjyrt" path="res://client_data/data/texture/À¯ÀúÀÎÅÍÆäÀ̽º/btn_next.bmp" id="14_pfdi0"] [ext_resource type="Texture2D" uid="uid://bo277gh8uyw1w" path="res://client_data/data/texture/À¯ÀúÀÎÅÍÆäÀ̽º/btn_next_b.bmp" id="15_38pxr"] [ext_resource type="Texture2D" uid="uid://qvmuk8xa3yej" path="res://client_data/data/texture/À¯ÀúÀÎÅÍÆäÀ̽º/btn_next_a.bmp" id="16_hqeko"] -[ext_resource type="PackedScene" uid="uid://uo7qgv3ydh0q" path="res://ui/login/character_selection_status.tscn" id="16_tihdy"] +[ext_resource type="PackedScene" uid="uid://bxbprntny8duj" path="res://ui/login/login_character_selection_list.tscn" id="16_tihdy"] [ext_resource type="PackedScene" uid="uid://swtqlba1wi3o" path="res://ui/window.tscn" id="17_mma3j"] [sub_resource type="AudioStreamWAV" id="AudioStreamWAV_f4kp2"] @@ -72,6 +72,7 @@ theme_override_constants/margin_bottom = 15 [node name="VBoxContainer" type="VBoxContainer" parent="Login/Window/VBoxContainer/Body/MarginContainer"] layout_mode = 2 +theme_override_constants/separation = 6 [node name="Username" type="HBoxContainer" parent="Login/Window/VBoxContainer/Body/MarginContainer/VBoxContainer"] layout_mode = 2 @@ -249,23 +250,35 @@ text = "Cancel" [node name="Window" parent="CharacterSelection" instance=ExtResource("17_mma3j")] layout_mode = 2 -[node name="MarginContainer" type="MarginContainer" parent="CharacterSelection/Window/VBoxContainer/Body" index="0"] +[node name="HBoxContainer" type="HBoxContainer" parent="CharacterSelection/Window/VBoxContainer/TitleBar/TitleBarElements" index="0"] layout_mode = 2 -theme_override_constants/margin_left = 64 -theme_override_constants/margin_top = 16 -theme_override_constants/margin_right = 64 -theme_override_constants/margin_bottom = 16 -[node name="VBoxContainer" type="VBoxContainer" parent="CharacterSelection/Window/VBoxContainer/Body/MarginContainer"] +[node name="Label" type="Label" parent="CharacterSelection/Window/VBoxContainer/TitleBar/TitleBarElements/HBoxContainer"] +texture_filter = 1 +layout_mode = 2 +theme = ExtResource("9_toei2") +text = "Character Select" + +[node name="-" type="Label" parent="CharacterSelection/Window/VBoxContainer/TitleBar/TitleBarElements/HBoxContainer"] +texture_filter = 1 layout_mode = 2 -theme_override_constants/separation = 32 +theme = ExtResource("9_toei2") +text = "-" -[node name="CharacterList" type="HBoxContainer" parent="CharacterSelection/Window/VBoxContainer/Body/MarginContainer/VBoxContainer"] +[node name="CharacterSelectionSlotLabel" type="Label" parent="CharacterSelection/Window/VBoxContainer/TitleBar/TitleBarElements/HBoxContainer"] unique_name_in_owner = true +texture_filter = 1 +layout_mode = 2 +theme = ExtResource("9_toei2") + +[node name="MarginContainer" type="MarginContainer" parent="CharacterSelection/Window/VBoxContainer/Body" index="0"] layout_mode = 2 -theme_override_constants/separation = 32 +theme_override_constants/margin_left = 64 +theme_override_constants/margin_top = 16 +theme_override_constants/margin_right = 64 +theme_override_constants/margin_bottom = 16 -[node name="CharacterSelectionStatus" parent="CharacterSelection/Window/VBoxContainer/Body/MarginContainer/VBoxContainer" instance=ExtResource("16_tihdy")] +[node name="CharacterSelectionList" parent="CharacterSelection/Window/VBoxContainer/Body/MarginContainer" instance=ExtResource("16_tihdy")] unique_name_in_owner = true layout_mode = 2 diff --git a/ui/login/character_selection_item.gd b/ui/login/character_selection_item.gd index 7ad77b6..fa0fad8 100644 --- a/ui/login/character_selection_item.gd +++ b/ui/login/character_selection_item.gd @@ -4,14 +4,22 @@ extends PanelContainer signal selected -static var selected_item: CharacterSelectionItem: +var is_hovered := false: set(value): - var previous_item = selected_item - selected_item = value - if previous_item: - previous_item.draw_highlight(false) + is_hovered = value + draw_highlight(value) + +var is_selected := false: + set(value): + is_selected = value + draw_highlight(false) + if value: + selected.emit() -var is_hovered := false + +func _ready() -> void: + clear() + %SelectionBorder.visible = false func initialize_with_info(info: CharacterInformation): @@ -21,6 +29,7 @@ func initialize_with_info(info: CharacterInformation): Constants.FilePaths.get_player_head(info.gender, info.head), ] ) + %Head.visible = true %Body.texture = load( "%s/%s/000.png" % [ @@ -28,30 +37,43 @@ func initialize_with_info(info: CharacterInformation): Constants.FilePaths.get_player_body(info.gender, info.job), ] ) - - %Name.text = info.name + %Body.visible = true + + +func clear(): + %Head.visible = false + %Body.visible = false func draw_highlight(is_highlighted: bool): - if selected_item == self: - is_highlighted = true - var style_box: StyleBoxFlat = get_theme_stylebox("panel").duplicate() style_box.draw_center = is_highlighted add_theme_stylebox_override("panel", style_box) + + if is_selected: + %SelectionBorder.visible = true + var style_box2: StyleBoxFlat = get_theme_stylebox("panel").duplicate() + style_box2.border_color.a = 0.0 + add_theme_stylebox_override("panel", style_box2) + + # spacing "fix" + #%SelectionBorder.top_level = true + #await get_tree().process_frame + #%SelectionBorder.global_position = global_position + else: + %SelectionBorder.visible = false + var style_box2: StyleBoxFlat = get_theme_stylebox("panel").duplicate() + style_box2.border_color.a = 1.0 + add_theme_stylebox_override("panel", style_box2) func _on_gui_input(event: InputEvent) -> void: if event.is_pressed(): - selected_item = self - draw_highlight(true) - selected.emit() + is_selected = true func _on_mouse_entered() -> void: is_hovered = true - draw_highlight(is_hovered) func _on_mouse_exited() -> void: is_hovered = false - draw_highlight(is_hovered) diff --git a/ui/login/character_selection_item.tscn b/ui/login/character_selection_item.tscn index adc37bb..4b8a249 100644 --- a/ui/login/character_selection_item.tscn +++ b/ui/login/character_selection_item.tscn @@ -1,9 +1,10 @@ -[gd_scene load_steps=6 format=3 uid="uid://rrd131rq74n5"] +[gd_scene load_steps=7 format=3 uid="uid://rrd131rq74n5"] [ext_resource type="Script" uid="uid://bsglhorusc7ug" path="res://ui/login/character_selection_item.gd" id="1_25yur"] [ext_resource type="Texture2D" uid="uid://danymuvfjf4o1" path="res://client_data/data/sprite/Àΰ£Á·/¸Ó¸®Åë/³²/16_³²/000.png" id="2_aqbfs"] [ext_resource type="Texture2D" uid="uid://cwqgdd00sf7pu" path="res://client_data/data/sprite/Àΰ£Á·/¸öÅë/³²/Ãʺ¸ÀÚ_³²/000.png" id="3_xv3pn"] -[ext_resource type="Theme" uid="uid://c6y6r8kcnbb10" path="res://ui/theme_clear.tres" id="4_u21ok"] +[ext_resource type="PackedScene" uid="uid://knmmuhon34rh" path="res://ui/bmp_texture_rect.tscn" id="4_vhrt2"] +[ext_resource type="Texture2D" uid="uid://g2cx2uuvffe" path="res://client_data/data/texture/À¯ÀúÀÎÅÍÆäÀ̽º/login_interface/box_select.bmp" id="5_vhrt2"] [sub_resource type="StyleBoxFlat" id="StyleBoxFlat_u21ok"] bg_color = Color(0, 0, 0, 0.0980392) @@ -15,9 +16,9 @@ border_width_bottom = 1 border_color = Color(0.760784, 0.760784, 0.760784, 1) [node name="CharacterSelectionItem" type="PanelContainer"] -custom_minimum_size = Vector2(150, 150) -offset_right = 150.0 -offset_bottom = 163.0 +custom_minimum_size = Vector2(141, 146) +offset_right = 141.0 +offset_bottom = 146.0 mouse_default_cursor_shape = 2 theme_override_styles/panel = SubResource("StyleBoxFlat_u21ok") script = ExtResource("1_25yur") @@ -52,12 +53,12 @@ layout_mode = 2 texture = ExtResource("3_xv3pn") stretch_mode = 3 -[node name="Name" type="Label" parent="MarginContainer/VBoxContainer"] +[node name="SelectionBorder" parent="." instance=ExtResource("4_vhrt2")] unique_name_in_owner = true +show_behind_parent = true layout_mode = 2 -theme = ExtResource("4_u21ok") -text = "Name" -horizontal_alignment = 1 +texture = ExtResource("5_vhrt2") +stretch_mode = 4 [connection signal="gui_input" from="." to="." method="_on_gui_input"] [connection signal="mouse_entered" from="." to="." method="_on_mouse_entered"] diff --git a/ui/login/character_selection_status.gd b/ui/login/character_selection_status.gd index 2c0e450..5bbd200 100644 --- a/ui/login/character_selection_status.gd +++ b/ui/login/character_selection_status.gd @@ -17,3 +17,21 @@ func set_info(info: CharacterInformation): %Luk.text = str(info.luck) %Map.text = info.map_name.substr(0, info.map_name.length() - 4) + + +func clear(): + %Name.text = "" + %Job.text = "" + %Level.text = "" + %Exp.text = "" + %Hp.text = "" + %Sp.text = "" + + %Str.text = "" + %Agi.text = "" + %Vit.text = "" + %Int.text = "" + %Dex.text = "" + %Luk.text = "" + + %Map.text = "" diff --git a/ui/login/character_selection_status.tscn b/ui/login/character_selection_status.tscn index 0de1865..31c5ab0 100644 --- a/ui/login/character_selection_status.tscn +++ b/ui/login/character_selection_status.tscn @@ -6,15 +6,15 @@ [sub_resource type="StyleBoxEmpty" id="StyleBoxEmpty_qjr2b"] [node name="CharacterSelectionStatus" type="PanelContainer"] -offset_right = 360.0 -offset_bottom = 158.0 +offset_right = 352.0 +offset_bottom = 152.0 theme = ExtResource("1_o7mxo") theme_override_styles/panel = SubResource("StyleBoxEmpty_qjr2b") script = ExtResource("2_nhn0f") [node name="VBoxContainer" type="VBoxContainer" parent="."] layout_mode = 2 -theme_override_constants/separation = 8 +theme_override_constants/separation = 2 [node name="VBoxContainer" type="VBoxContainer" parent="VBoxContainer"] layout_mode = 2 @@ -32,11 +32,11 @@ custom_minimum_size = Vector2(45, 0) layout_mode = 2 theme = ExtResource("1_o7mxo") theme_type_variation = &"CharacterSelectionStatusLabel" -text = "NAME" +text = "Name" [node name="Name" type="Label" parent="VBoxContainer/VBoxContainer/HBoxContainer/HBoxContainer"] unique_name_in_owner = true -custom_minimum_size = Vector2(130, 0) +custom_minimum_size = Vector2(120, 0) layout_mode = 2 size_flags_horizontal = 3 theme = ExtResource("1_o7mxo") @@ -55,7 +55,7 @@ text = "STR" [node name="Str" type="Label" parent="VBoxContainer/VBoxContainer/HBoxContainer/HBoxContainer2"] unique_name_in_owner = true -custom_minimum_size = Vector2(130, 0) +custom_minimum_size = Vector2(120, 0) layout_mode = 2 size_flags_horizontal = 3 theme = ExtResource("1_o7mxo") @@ -74,11 +74,11 @@ custom_minimum_size = Vector2(45, 0) layout_mode = 2 theme = ExtResource("1_o7mxo") theme_type_variation = &"CharacterSelectionStatusLabel" -text = "JOB" +text = "Job" [node name="Job" type="Label" parent="VBoxContainer/VBoxContainer/HBoxContainer2/HBoxContainer"] unique_name_in_owner = true -custom_minimum_size = Vector2(130, 0) +custom_minimum_size = Vector2(120, 0) layout_mode = 2 size_flags_horizontal = 3 theme = ExtResource("1_o7mxo") @@ -97,7 +97,7 @@ text = "AGI" [node name="Agi" type="Label" parent="VBoxContainer/VBoxContainer/HBoxContainer2/HBoxContainer2"] unique_name_in_owner = true -custom_minimum_size = Vector2(130, 0) +custom_minimum_size = Vector2(120, 0) layout_mode = 2 size_flags_horizontal = 3 theme = ExtResource("1_o7mxo") @@ -120,7 +120,7 @@ text = "Lv." [node name="Level" type="Label" parent="VBoxContainer/VBoxContainer/HBoxContainer3/HBoxContainer"] unique_name_in_owner = true -custom_minimum_size = Vector2(130, 0) +custom_minimum_size = Vector2(120, 0) layout_mode = 2 size_flags_horizontal = 3 theme = ExtResource("1_o7mxo") @@ -139,7 +139,7 @@ text = "VIT" [node name="Vit" type="Label" parent="VBoxContainer/VBoxContainer/HBoxContainer3/HBoxContainer2"] unique_name_in_owner = true -custom_minimum_size = Vector2(130, 0) +custom_minimum_size = Vector2(120, 0) layout_mode = 2 size_flags_horizontal = 3 theme = ExtResource("1_o7mxo") @@ -162,7 +162,7 @@ text = "EXP" [node name="Exp" type="Label" parent="VBoxContainer/VBoxContainer/HBoxContainer4/HBoxContainer"] unique_name_in_owner = true -custom_minimum_size = Vector2(130, 0) +custom_minimum_size = Vector2(120, 0) layout_mode = 2 size_flags_horizontal = 3 theme = ExtResource("1_o7mxo") @@ -181,7 +181,7 @@ text = "INT" [node name="Int" type="Label" parent="VBoxContainer/VBoxContainer/HBoxContainer4/HBoxContainer2"] unique_name_in_owner = true -custom_minimum_size = Vector2(130, 0) +custom_minimum_size = Vector2(120, 0) layout_mode = 2 size_flags_horizontal = 3 theme = ExtResource("1_o7mxo") @@ -204,7 +204,7 @@ text = "HP" [node name="Hp" type="Label" parent="VBoxContainer/VBoxContainer/HBoxContainer5/HBoxContainer"] unique_name_in_owner = true -custom_minimum_size = Vector2(130, 0) +custom_minimum_size = Vector2(120, 0) layout_mode = 2 size_flags_horizontal = 3 theme = ExtResource("1_o7mxo") @@ -223,7 +223,7 @@ text = "DEX" [node name="Dex" type="Label" parent="VBoxContainer/VBoxContainer/HBoxContainer5/HBoxContainer2"] unique_name_in_owner = true -custom_minimum_size = Vector2(130, 0) +custom_minimum_size = Vector2(120, 0) layout_mode = 2 size_flags_horizontal = 3 theme = ExtResource("1_o7mxo") @@ -246,7 +246,7 @@ text = "SP" [node name="Sp" type="Label" parent="VBoxContainer/VBoxContainer/HBoxContainer6/HBoxContainer"] unique_name_in_owner = true -custom_minimum_size = Vector2(130, 0) +custom_minimum_size = Vector2(120, 0) layout_mode = 2 size_flags_horizontal = 3 theme = ExtResource("1_o7mxo") @@ -265,7 +265,7 @@ text = "LUK" [node name="Luk" type="Label" parent="VBoxContainer/VBoxContainer/HBoxContainer6/HBoxContainer2"] unique_name_in_owner = true -custom_minimum_size = Vector2(130, 0) +custom_minimum_size = Vector2(120, 0) layout_mode = 2 size_flags_horizontal = 3 theme = ExtResource("1_o7mxo") @@ -284,7 +284,7 @@ text = "MAP" [node name="Map" type="Label" parent="VBoxContainer/HBoxContainer"] unique_name_in_owner = true -custom_minimum_size = Vector2(307, 0) +custom_minimum_size = Vector2(287, 0) layout_mode = 2 theme = ExtResource("1_o7mxo") theme_type_variation = &"CharacterSelectionStatusLabel2" diff --git a/ui/login/login_character_selection_list.gd b/ui/login/login_character_selection_list.gd new file mode 100644 index 0000000..221c47f --- /dev/null +++ b/ui/login/login_character_selection_list.gd @@ -0,0 +1,111 @@ +class_name CharacterSelectionList +extends Control + + +signal selected(slot_idx: int) + +var item_scene := preload("res://ui/login/character_selection_item.tscn") + +var slot_offset := 0 + +@export var displayed_slots_count: int = 3: + set = set_displayed_slots_count + +@export var selected_slot_index: int = -1: + set = set_selected_slot_index + +@export var login_character_list: LoginCharacterList: + set = set_login_character_list + + +func _ready() -> void: + if %CharacterList.get_child_count() != displayed_slots_count: + set_displayed_slots_count(displayed_slots_count) + + +func set_displayed_slots_count(value: int) -> void: + displayed_slots_count = value + + for node in %CharacterList.get_children(): + node.queue_free() + + for _idx in displayed_slots_count: + var item = preload("res://ui/login/character_selection_item.tscn").instantiate() + %CharacterList.add_child(item) + + draw() + + +func set_selected_slot_index(value: int) -> void: + if selected_slot_index == value: + return + + var character_list := login_character_list.character_information + var maximum_slot_count := login_character_list.slot_count + + %CharacterList.get_child(selected_slot_index - slot_offset).is_selected = false + + if value >= maximum_slot_count: + selected_slot_index = 0 + elif value < 0: + selected_slot_index = maximum_slot_count - 1 + else: + selected_slot_index = value + + if selected_slot_index > slot_offset + (displayed_slots_count - 1): + #slot_offset = selected_slot_index - (displayed_slots_count - 1) + slot_offset = min(selected_slot_index, maximum_slot_count - displayed_slots_count) + elif selected_slot_index < slot_offset: + #slot_offset = selected_slot_index + slot_offset = max(0, selected_slot_index - displayed_slots_count) + + draw() + %CharacterList.get_child(selected_slot_index - slot_offset).is_selected = true + + if selected_slot_index < character_list.size(): + %CharacterSelectionStatus.set_info(character_list[selected_slot_index]) + else: + $CharacterSelectionStatus.clear() + + selected.emit(selected_slot_index) + + +func set_login_character_list(value: LoginCharacterList) -> void: + login_character_list = value + draw() + + +func draw(): + if not login_character_list: + return + + var character_information: Array[CharacterInformation] = login_character_list.character_information + + for display_slot_idx in displayed_slots_count: + var item: CharacterSelectionItem = %CharacterList.get_child(display_slot_idx) + var slot_idx = slot_offset + display_slot_idx + + if slot_idx < character_information.size(): + var info: CharacterInformation = character_information[slot_idx] + item.initialize_with_info(info) + else: + item.clear() + + if item.selected.is_connected(_on_item_selected): + item.selected.disconnect(_on_item_selected) + item.selected.connect(_on_item_selected.bind(slot_idx)) + + +func select(slot_idx: int): + selected_slot_index = slot_idx + + +func _on_item_selected(slot_idx: int): + selected_slot_index = slot_idx + + +func _on_button_left_pressed() -> void: + selected_slot_index -= 1 + +func _on_button_right_pressed() -> void: + selected_slot_index += 1 diff --git a/ui/login/login_character_selection_list.tscn b/ui/login/login_character_selection_list.tscn new file mode 100644 index 0000000..1a5aba7 --- /dev/null +++ b/ui/login/login_character_selection_list.tscn @@ -0,0 +1,61 @@ +[gd_scene load_steps=11 format=3 uid="uid://bxbprntny8duj"] + +[ext_resource type="PackedScene" uid="uid://uo7qgv3ydh0q" path="res://ui/login/character_selection_status.tscn" id="1_5anyi"] +[ext_resource type="Script" uid="uid://dmch4gi1khn2r" path="res://ui/login/login_character_selection_list.gd" id="1_togb6"] +[ext_resource type="PackedScene" uid="uid://rrd131rq74n5" path="res://ui/login/character_selection_item.tscn" id="2_k142l"] +[ext_resource type="PackedScene" uid="uid://cjcm2mai50thr" path="res://ui/bmp_texture_button.tscn" id="2_s7n6r"] +[ext_resource type="Texture2D" uid="uid://s5uc6sa6y88f" path="res://client_data/data/texture/À¯ÀúÀÎÅÍÆäÀ̽º/select_character/chr_arrow_l_out.bmp" id="3_c5a25"] +[ext_resource type="Texture2D" uid="uid://b8n4w8a4yumax" path="res://client_data/data/texture/À¯ÀúÀÎÅÍÆäÀ̽º/select_character/chr_arrow_l_press.bmp" id="4_ntkas"] +[ext_resource type="Texture2D" uid="uid://e3k5tdqv2jft" path="res://client_data/data/texture/À¯ÀúÀÎÅÍÆäÀ̽º/select_character/chr_arrow_l_over.bmp" id="5_606nc"] +[ext_resource type="Texture2D" uid="uid://bhap22qcv1mga" path="res://client_data/data/texture/À¯ÀúÀÎÅÍÆäÀ̽º/select_character/chr_arrow_r_out.bmp" id="7_c8nb3"] +[ext_resource type="Texture2D" uid="uid://dvf7yx7b3dj4m" path="res://client_data/data/texture/À¯ÀúÀÎÅÍÆäÀ̽º/select_character/chr_arrow_r_press.bmp" id="8_p8h60"] +[ext_resource type="Texture2D" uid="uid://4ngcs00s5cnn" path="res://client_data/data/texture/À¯ÀúÀÎÅÍÆäÀ̽º/select_character/chr_arrow_r_over.bmp" id="9_7p86f"] + +[node name="LoginCharacterSelectionList" type="VBoxContainer"] +theme_override_constants/separation = 32 +script = ExtResource("1_togb6") + +[node name="MarginContainer" type="MarginContainer" parent="."] +layout_mode = 2 +theme_override_constants/margin_left = -30 +theme_override_constants/margin_top = 0 +theme_override_constants/margin_right = 0 + +[node name="HBoxContainer" type="HBoxContainer" parent="MarginContainer"] +layout_mode = 2 +theme_override_constants/separation = 16 +alignment = 1 + +[node name="ButtonLeft" parent="MarginContainer/HBoxContainer" instance=ExtResource("2_s7n6r")] +texture_filter = 0 +layout_mode = 2 +texture_normal = ExtResource("3_c5a25") +texture_pressed = ExtResource("4_ntkas") +texture_hover = ExtResource("5_606nc") +stretch_mode = 5 + +[node name="CharacterList" type="HBoxContainer" parent="MarginContainer/HBoxContainer"] +unique_name_in_owner = true +layout_mode = 2 +theme_override_constants/separation = 32 + +[node name="CharacterSelectionItem" parent="MarginContainer/HBoxContainer/CharacterList" instance=ExtResource("2_k142l")] +layout_mode = 2 + +[node name="CharacterSelectionItem2" parent="MarginContainer/HBoxContainer/CharacterList" instance=ExtResource("2_k142l")] +layout_mode = 2 + +[node name="ButtonRight" parent="MarginContainer/HBoxContainer" instance=ExtResource("2_s7n6r")] +texture_filter = 0 +layout_mode = 2 +texture_normal = ExtResource("7_c8nb3") +texture_pressed = ExtResource("8_p8h60") +texture_hover = ExtResource("9_7p86f") +stretch_mode = 5 + +[node name="CharacterSelectionStatus" parent="." instance=ExtResource("1_5anyi")] +unique_name_in_owner = true +layout_mode = 2 + +[connection signal="pressed" from="MarginContainer/HBoxContainer/ButtonLeft" to="." method="_on_button_left_pressed"] +[connection signal="pressed" from="MarginContainer/HBoxContainer/ButtonRight" to="." method="_on_button_right_pressed"] diff --git a/ui/theme_clear.tres b/ui/theme_clear.tres index a02f1b4..5805499 100644 --- a/ui/theme_clear.tres +++ b/ui/theme_clear.tres @@ -65,9 +65,9 @@ texture = SubResource("GradientTexture2D_je15x") [sub_resource type="StyleBoxEmpty" id="StyleBoxEmpty_f5aki"] [sub_resource type="StyleBoxFlat" id="StyleBoxFlat_f5aki"] -content_margin_left = 2.0 +content_margin_left = 3.0 content_margin_top = 0.0 -content_margin_right = 2.0 +content_margin_right = 3.0 content_margin_bottom = 0.0 bg_color = Color(0.94902, 0.94902, 0.94902, 1) border_width_left = 1 diff --git a/ui/window.gd b/ui/window.gd index cd6fade..25b0988 100644 --- a/ui/window.gd +++ b/ui/window.gd @@ -8,6 +8,7 @@ var drag_anchor := Vector2.ZERO func _process(_delta: float) -> void: if is_dragging: global_position += get_global_mouse_position() - drag_anchor + #%TitleBar.size # TODO: use title bar size as min/max. dont let the bar vanish drag_anchor = get_global_mouse_position() diff --git a/ui/window.tscn b/ui/window.tscn index 6208682..5a0ecc9 100644 --- a/ui/window.tscn +++ b/ui/window.tscn @@ -17,8 +17,8 @@ border_width_right = 1 border_color = Color(0.760784, 0.760784, 0.760784, 1) [node name="Window" type="PanelContainer"] -offset_right = 209.0 -offset_bottom = 136.0 +offset_right = 432.0 +offset_bottom = 188.0 script = ExtResource("1_hfgic") [node name="VBoxContainer" type="VBoxContainer" parent="."] @@ -26,6 +26,7 @@ layout_mode = 2 theme_override_constants/separation = 0 [node name="TitleBar" type="PanelContainer" parent="VBoxContainer"] +unique_name_in_owner = true layout_mode = 2 [node name="TitleBarBackground" type="HBoxContainer" parent="VBoxContainer/TitleBar"] -- cgit v1.2.3