viable :) #3
12 changed files with 250 additions and 27 deletions
BIN
assets/PunchVFX.aseprite
Normal file
BIN
assets/PunchVFX.aseprite
Normal file
Binary file not shown.
14
assets/PunchVFX.aseprite.import
Normal file
14
assets/PunchVFX.aseprite.import
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
[remap]
|
||||||
|
|
||||||
|
importer="aseprite_wizard.plugin.noop"
|
||||||
|
type="PackedDataContainer"
|
||||||
|
uid="uid://sg2ms6cssj01"
|
||||||
|
path="res://.godot/imported/PunchVFX.aseprite-b06de0b1aa18d48e800d94ff6cf92f62.res"
|
||||||
|
|
||||||
|
[deps]
|
||||||
|
|
||||||
|
source_file="res://assets/PunchVFX.aseprite"
|
||||||
|
dest_files=["res://.godot/imported/PunchVFX.aseprite-b06de0b1aa18d48e800d94ff6cf92f62.res"]
|
||||||
|
|
||||||
|
[params]
|
||||||
|
|
BIN
assets/export/punch-.png
Normal file
BIN
assets/export/punch-.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 707 B |
34
assets/export/punch-.png.import
Normal file
34
assets/export/punch-.png.import
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
[remap]
|
||||||
|
|
||||||
|
importer="texture"
|
||||||
|
type="CompressedTexture2D"
|
||||||
|
uid="uid://cv731yjx5v71a"
|
||||||
|
path="res://.godot/imported/punch-.png-0d4c585a346b088b7f81847a43a045e4.ctex"
|
||||||
|
metadata={
|
||||||
|
"vram_texture": false
|
||||||
|
}
|
||||||
|
|
||||||
|
[deps]
|
||||||
|
|
||||||
|
source_file="res://assets/export/punch-.png"
|
||||||
|
dest_files=["res://.godot/imported/punch-.png-0d4c585a346b088b7f81847a43a045e4.ctex"]
|
||||||
|
|
||||||
|
[params]
|
||||||
|
|
||||||
|
compress/mode=0
|
||||||
|
compress/high_quality=false
|
||||||
|
compress/lossy_quality=0.7
|
||||||
|
compress/hdr_compression=1
|
||||||
|
compress/normal_map=0
|
||||||
|
compress/channel_pack=0
|
||||||
|
mipmaps/generate=false
|
||||||
|
mipmaps/limit=-1
|
||||||
|
roughness/mode=0
|
||||||
|
roughness/src_normal=""
|
||||||
|
process/fix_alpha_border=true
|
||||||
|
process/premult_alpha=false
|
||||||
|
process/normal_map_invert_y=false
|
||||||
|
process/hdr_as_srgb=false
|
||||||
|
process/hdr_clamp_exposure=false
|
||||||
|
process/size_limit=0
|
||||||
|
detect_3d/compress_to=1
|
BIN
assets/weegee.aseprite
Normal file
BIN
assets/weegee.aseprite
Normal file
Binary file not shown.
14
assets/weegee.aseprite.import
Normal file
14
assets/weegee.aseprite.import
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
[remap]
|
||||||
|
|
||||||
|
importer="aseprite_wizard.plugin.noop"
|
||||||
|
type="PackedDataContainer"
|
||||||
|
uid="uid://boc2fqbk18m6"
|
||||||
|
path="res://.godot/imported/weegee.aseprite-3b2165eb9806715fac46d106c4ddf9ee.res"
|
||||||
|
|
||||||
|
[deps]
|
||||||
|
|
||||||
|
source_file="res://assets/weegee.aseprite"
|
||||||
|
dest_files=["res://.godot/imported/weegee.aseprite-3b2165eb9806715fac46d106c4ddf9ee.res"]
|
||||||
|
|
||||||
|
[params]
|
||||||
|
|
|
@ -1,7 +1,8 @@
|
||||||
[gd_scene load_steps=35 format=3 uid="uid://n65kl0jalqdt"]
|
[gd_scene load_steps=36 format=3 uid="uid://n65kl0jalqdt"]
|
||||||
|
|
||||||
[ext_resource type="Texture2D" uid="uid://b447sysxt4rhp" path="res://assets/export/crawler-.png" id="1_bdup0"]
|
[ext_resource type="Texture2D" uid="uid://b447sysxt4rhp" path="res://assets/export/crawler-.png" id="1_bdup0"]
|
||||||
[ext_resource type="Script" path="res://scripts/crawler.gd" id="1_dcux3"]
|
[ext_resource type="Script" path="res://scripts/crawler.gd" id="1_dcux3"]
|
||||||
|
[ext_resource type="AudioStream" uid="uid://dfrc2luquqqlw" path="res://assets/Explode.wav" id="3_qvljj"]
|
||||||
|
|
||||||
[sub_resource type="AtlasTexture" id="AtlasTexture_6iqis"]
|
[sub_resource type="AtlasTexture" id="AtlasTexture_6iqis"]
|
||||||
atlas = ExtResource("1_bdup0")
|
atlas = ExtResource("1_bdup0")
|
||||||
|
@ -263,7 +264,7 @@ script = ExtResource("1_dcux3")
|
||||||
texture_filter = 1
|
texture_filter = 1
|
||||||
position = Vector2(-34, 0)
|
position = Vector2(-34, 0)
|
||||||
sprite_frames = SubResource("SpriteFrames_e3bx4")
|
sprite_frames = SubResource("SpriteFrames_e3bx4")
|
||||||
animation = &"jump away"
|
animation = &"Whip"
|
||||||
metadata/_aseprite_wizard_config_ = {
|
metadata/_aseprite_wizard_config_ = {
|
||||||
"layer": "",
|
"layer": "",
|
||||||
"o_ex_p": "",
|
"o_ex_p": "",
|
||||||
|
@ -288,3 +289,24 @@ shape = SubResource("RectangleShape2D_k4vrw")
|
||||||
position = Vector2(-42, 1)
|
position = Vector2(-42, 1)
|
||||||
shape = SubResource("RectangleShape2D_s2a15")
|
shape = SubResource("RectangleShape2D_s2a15")
|
||||||
debug_color = Color(0.949698, 0.110143, 0.492661, 0.42)
|
debug_color = Color(0.949698, 0.110143, 0.492661, 0.42)
|
||||||
|
|
||||||
|
[node name="ExplosionSFX" type="AudioStreamPlayer2D" parent="."]
|
||||||
|
position = Vector2(-34, 0)
|
||||||
|
stream = ExtResource("3_qvljj")
|
||||||
|
volume_db = 10.0
|
||||||
|
|
||||||
|
[node name="DeathTimer" type="Timer" parent="."]
|
||||||
|
wait_time = 1.1
|
||||||
|
one_shot = true
|
||||||
|
|
||||||
|
[node name="HurtTimer" type="Timer" parent="."]
|
||||||
|
wait_time = 0.6
|
||||||
|
one_shot = true
|
||||||
|
|
||||||
|
[node name="AttackTimer" type="Timer" parent="."]
|
||||||
|
wait_time = 0.5
|
||||||
|
one_shot = true
|
||||||
|
|
||||||
|
[node name="Recharge" type="Timer" parent="."]
|
||||||
|
wait_time = 0.5
|
||||||
|
one_shot = true
|
||||||
|
|
|
@ -1,8 +1,9 @@
|
||||||
[gd_scene load_steps=79 format=3 uid="uid://cjiuycwqqxaxn"]
|
[gd_scene load_steps=88 format=3 uid="uid://cjiuycwqqxaxn"]
|
||||||
|
|
||||||
[ext_resource type="Script" path="res://scripts/player.gd" id="1_oy25y"]
|
[ext_resource type="Script" path="res://scripts/player.gd" id="1_oy25y"]
|
||||||
[ext_resource type="Texture2D" uid="uid://d4mjdwhmho5vs" path="res://assets/export/player-Layer 5.png" id="2_g0lnn"]
|
[ext_resource type="Texture2D" uid="uid://d4mjdwhmho5vs" path="res://assets/export/player-Layer 5.png" id="2_g0lnn"]
|
||||||
[ext_resource type="AudioStream" uid="uid://cir5d1qi0hcev" path="res://assets/PlayerHurt.wav" id="3_ggoam"]
|
[ext_resource type="AudioStream" uid="uid://cir5d1qi0hcev" path="res://assets/PlayerHurt.wav" id="3_ggoam"]
|
||||||
|
[ext_resource type="Texture2D" uid="uid://cv731yjx5v71a" path="res://assets/export/punch-.png" id="3_jp05m"]
|
||||||
[ext_resource type="AudioStream" uid="uid://du0f57xgcxuxf" path="res://assets/PlayerDeathSound.wav" id="4_8skcp"]
|
[ext_resource type="AudioStream" uid="uid://du0f57xgcxuxf" path="res://assets/PlayerDeathSound.wav" id="4_8skcp"]
|
||||||
[ext_resource type="Texture2D" uid="uid://cliy3s4yykb3k" path="res://assets/export/health-.png" id="5_1x6di"]
|
[ext_resource type="Texture2D" uid="uid://cliy3s4yykb3k" path="res://assets/export/health-.png" id="5_1x6di"]
|
||||||
|
|
||||||
|
@ -612,6 +613,63 @@ size = Vector2(28, 64)
|
||||||
[sub_resource type="RectangleShape2D" id="RectangleShape2D_mst3b"]
|
[sub_resource type="RectangleShape2D" id="RectangleShape2D_mst3b"]
|
||||||
size = Vector2(27, 41)
|
size = Vector2(27, 41)
|
||||||
|
|
||||||
|
[sub_resource type="AtlasTexture" id="AtlasTexture_1n7yn"]
|
||||||
|
atlas = ExtResource("3_jp05m")
|
||||||
|
region = Rect2(0, 0, 32, 32)
|
||||||
|
|
||||||
|
[sub_resource type="AtlasTexture" id="AtlasTexture_dtbdr"]
|
||||||
|
atlas = ExtResource("3_jp05m")
|
||||||
|
region = Rect2(32, 0, 32, 32)
|
||||||
|
|
||||||
|
[sub_resource type="AtlasTexture" id="AtlasTexture_dhtio"]
|
||||||
|
atlas = ExtResource("3_jp05m")
|
||||||
|
region = Rect2(64, 0, 32, 32)
|
||||||
|
|
||||||
|
[sub_resource type="AtlasTexture" id="AtlasTexture_t4yk0"]
|
||||||
|
atlas = ExtResource("3_jp05m")
|
||||||
|
region = Rect2(0, 32, 32, 32)
|
||||||
|
|
||||||
|
[sub_resource type="AtlasTexture" id="AtlasTexture_1emd7"]
|
||||||
|
atlas = ExtResource("3_jp05m")
|
||||||
|
region = Rect2(32, 32, 32, 32)
|
||||||
|
|
||||||
|
[sub_resource type="AtlasTexture" id="AtlasTexture_ccqcb"]
|
||||||
|
atlas = ExtResource("3_jp05m")
|
||||||
|
region = Rect2(64, 32, 32, 32)
|
||||||
|
|
||||||
|
[sub_resource type="AtlasTexture" id="AtlasTexture_g0kbm"]
|
||||||
|
atlas = ExtResource("3_jp05m")
|
||||||
|
region = Rect2(0, 64, 32, 32)
|
||||||
|
|
||||||
|
[sub_resource type="SpriteFrames" id="SpriteFrames_sbboq"]
|
||||||
|
animations = [{
|
||||||
|
"frames": [{
|
||||||
|
"duration": 1.0,
|
||||||
|
"texture": SubResource("AtlasTexture_1n7yn")
|
||||||
|
}, {
|
||||||
|
"duration": 1.0,
|
||||||
|
"texture": SubResource("AtlasTexture_dtbdr")
|
||||||
|
}, {
|
||||||
|
"duration": 1.0,
|
||||||
|
"texture": SubResource("AtlasTexture_dhtio")
|
||||||
|
}, {
|
||||||
|
"duration": 1.0,
|
||||||
|
"texture": SubResource("AtlasTexture_t4yk0")
|
||||||
|
}, {
|
||||||
|
"duration": 1.0,
|
||||||
|
"texture": SubResource("AtlasTexture_1emd7")
|
||||||
|
}, {
|
||||||
|
"duration": 1.0,
|
||||||
|
"texture": SubResource("AtlasTexture_ccqcb")
|
||||||
|
}, {
|
||||||
|
"duration": 1.0,
|
||||||
|
"texture": SubResource("AtlasTexture_g0kbm")
|
||||||
|
}],
|
||||||
|
"loop": true,
|
||||||
|
"name": &"default",
|
||||||
|
"speed": 10.0
|
||||||
|
}]
|
||||||
|
|
||||||
[sub_resource type="AtlasTexture" id="AtlasTexture_5bnwc"]
|
[sub_resource type="AtlasTexture" id="AtlasTexture_5bnwc"]
|
||||||
atlas = ExtResource("5_1x6di")
|
atlas = ExtResource("5_1x6di")
|
||||||
region = Rect2(128, 16, 64, 16)
|
region = Rect2(128, 16, 64, 16)
|
||||||
|
@ -723,6 +781,24 @@ position = Vector2(29, -3)
|
||||||
shape = SubResource("RectangleShape2D_mst3b")
|
shape = SubResource("RectangleShape2D_mst3b")
|
||||||
debug_color = Color(2.11775e-06, 0.631094, 0.465774, 0.42)
|
debug_color = Color(2.11775e-06, 0.631094, 0.465774, 0.42)
|
||||||
|
|
||||||
|
[node name="AnimatedSprite2D" type="AnimatedSprite2D" parent="PunchHitbox"]
|
||||||
|
texture_filter = 1
|
||||||
|
position = Vector2(26, -6)
|
||||||
|
sprite_frames = SubResource("SpriteFrames_sbboq")
|
||||||
|
metadata/_aseprite_wizard_config_ = {
|
||||||
|
"layer": "",
|
||||||
|
"o_ex_p": "",
|
||||||
|
"o_folder": "res://assets/export",
|
||||||
|
"o_name": "punch-",
|
||||||
|
"only_visible": true,
|
||||||
|
"slice": "",
|
||||||
|
"source": "res://assets/PunchVFX.aseprite"
|
||||||
|
}
|
||||||
|
metadata/_aseprite_wizard_interface_config_ = {
|
||||||
|
"output_section": true
|
||||||
|
}
|
||||||
|
metadata/_aseprite_wizard_source_file_hash_ = "4c71a20217ed605bc54873905f5606e3"
|
||||||
|
|
||||||
[node name="AttackTimer" type="Timer" parent="."]
|
[node name="AttackTimer" type="Timer" parent="."]
|
||||||
editor_description = "
|
editor_description = "
|
||||||
|
|
||||||
|
|
|
@ -37,7 +37,7 @@ scale = Vector2(1.01667, 1)
|
||||||
texture = ExtResource("4_d2h5p")
|
texture = ExtResource("4_d2h5p")
|
||||||
|
|
||||||
[node name="Player" parent="." instance=ExtResource("5_kc5l1")]
|
[node name="Player" parent="." instance=ExtResource("5_kc5l1")]
|
||||||
position = Vector2(-40, -49)
|
position = Vector2(-154, -43)
|
||||||
|
|
||||||
[node name="Borders" parent="." instance=ExtResource("6_47554")]
|
[node name="Borders" parent="." instance=ExtResource("6_47554")]
|
||||||
position = Vector2(-272, -240)
|
position = Vector2(-272, -240)
|
||||||
|
@ -71,10 +71,10 @@ one_shot = true
|
||||||
script = ExtResource("9_2n06p")
|
script = ExtResource("9_2n06p")
|
||||||
|
|
||||||
[node name="Crawler2" parent="Level2Spawner" instance=ExtResource("9_sqnor")]
|
[node name="Crawler2" parent="Level2Spawner" instance=ExtResource("9_sqnor")]
|
||||||
position = Vector2(-200, -41)
|
position = Vector2(-306, -39)
|
||||||
|
|
||||||
[node name="Crawler" parent="Level2Spawner" instance=ExtResource("9_sqnor")]
|
[node name="Crawler" parent="Level2Spawner" instance=ExtResource("9_sqnor")]
|
||||||
position = Vector2(168, -41)
|
position = Vector2(307, -42)
|
||||||
|
|
||||||
[node name="Floor" type="StaticBody2D" parent="."]
|
[node name="Floor" type="StaticBody2D" parent="."]
|
||||||
|
|
||||||
|
|
|
@ -4,18 +4,29 @@ class_name Crawler extends Entity
|
||||||
@onready var sprite = $AnimatedSprite2D
|
@onready var sprite = $AnimatedSprite2D
|
||||||
@onready var main_collider = $CollisionShape2D
|
@onready var main_collider = $CollisionShape2D
|
||||||
@onready var whip = $WhipArea
|
@onready var whip = $WhipArea
|
||||||
|
@onready var speed_mult = 0
|
||||||
|
@onready var is_dying = false
|
||||||
|
@onready var explosion_sfx = $ExplosionSFX
|
||||||
|
@onready var death_timer = $DeathTimer
|
||||||
|
@onready var hurting = false
|
||||||
|
@onready var hurt_timer = $HurtTimer
|
||||||
|
@onready var attack_timer = $AttackTimer
|
||||||
|
@onready var recharge_timer = $Recharge
|
||||||
|
|
||||||
## Crawl towards the player but stay within striking distance and not super close
|
## Crawl towards the player but stay within striking distance and not super close
|
||||||
|
|
||||||
const HEALTH = 40
|
const HEALTH = 45
|
||||||
const SPEED_MULTIPLIER = 50
|
const SPEED_MULTIPLIER = 50
|
||||||
const ATTACK_DAMAGE = 20
|
const ATTACK_DAMAGE = 20
|
||||||
|
|
||||||
|
var death_timer_started = false
|
||||||
|
var attacking = true
|
||||||
|
|
||||||
func _init() -> void:
|
func _init() -> void:
|
||||||
super._init(HEALTH, SPEED_MULTIPLIER, ATTACK_DAMAGE)
|
super._init(HEALTH, SPEED_MULTIPLIER, ATTACK_DAMAGE)
|
||||||
|
|
||||||
#func _ready() -> void:
|
func _ready() -> void:
|
||||||
#whip.visible = false
|
whip.visible = false
|
||||||
|
|
||||||
func face_player():
|
func face_player():
|
||||||
var player_position = player.global_position
|
var player_position = player.global_position
|
||||||
|
@ -36,13 +47,19 @@ func _physics_process(delta: float) -> void:
|
||||||
var distanceToPlayer = global_position.distance_to(player.global_position)
|
var distanceToPlayer = global_position.distance_to(player.global_position)
|
||||||
var desiredDistance = 70
|
var desiredDistance = 70
|
||||||
|
|
||||||
|
if health <= 0 and !death_timer_started:
|
||||||
|
is_dying = true
|
||||||
|
death_timer_started = true
|
||||||
|
death_timer.start()
|
||||||
|
|
||||||
|
if !is_dying:
|
||||||
|
if !hurting:
|
||||||
if sprite.flip_h:
|
if sprite.flip_h:
|
||||||
desiredDistance -= 69
|
desiredDistance -= 69
|
||||||
face_player()
|
face_player()
|
||||||
print(distanceToPlayer)
|
|
||||||
|
|
||||||
if distanceToPlayer > desiredDistance:
|
if distanceToPlayer > desiredDistance:
|
||||||
velocity.x = speed
|
velocity.x = speed * speed_mult
|
||||||
face_player()
|
face_player()
|
||||||
|
|
||||||
if not sprite.flip_h:
|
if not sprite.flip_h:
|
||||||
|
@ -50,7 +67,45 @@ func _physics_process(delta: float) -> void:
|
||||||
|
|
||||||
sprite.play("walk")
|
sprite.play("walk")
|
||||||
else:
|
else:
|
||||||
|
if recharge_timer.is_stopped():
|
||||||
|
attacking = true
|
||||||
|
whip.visible = true
|
||||||
|
whip.connect("body_entered", whip_hit)
|
||||||
|
attack_timer.start()
|
||||||
|
|
||||||
|
if attacking and !attack_timer.is_stopped():
|
||||||
velocity.x = 0
|
velocity.x = 0
|
||||||
sprite.play("Whip")
|
sprite.play("Whip")
|
||||||
|
elif attacking and attack_timer.is_stopped():
|
||||||
|
recharge_timer.start()
|
||||||
|
whip.visible = false
|
||||||
|
whip.disconnect("body_entered", whip_hit)
|
||||||
|
attacking = false
|
||||||
|
|
||||||
|
|
||||||
|
if hurt_timer.is_stopped():
|
||||||
|
hurting = false
|
||||||
|
else:
|
||||||
|
sprite.play("jump back")
|
||||||
|
|
||||||
move_and_slide()
|
move_and_slide()
|
||||||
|
else:
|
||||||
|
sprite.play("death")
|
||||||
|
explosion_sfx.play()
|
||||||
|
speed_mult = 0
|
||||||
|
if death_timer.is_stopped():
|
||||||
|
queue_free()
|
||||||
|
|
||||||
|
func charge() -> void:
|
||||||
|
speed_mult = 1
|
||||||
|
|
||||||
|
func hurt_anim():
|
||||||
|
hurting = true
|
||||||
|
hurt_timer.start()
|
||||||
|
|
||||||
|
func whip_hit(node: Node) -> void:
|
||||||
|
if node is Player:
|
||||||
|
node.health -= ATTACK_DAMAGE
|
||||||
|
node.take_knockback(50)
|
||||||
|
if node.has_method("hurt_anim"):
|
||||||
|
node.call("hurt_anim")
|
||||||
|
|
|
@ -12,6 +12,7 @@ class_name Player extends Entity
|
||||||
@onready var player_death_sfx = $PlayerDeathSfx
|
@onready var player_death_sfx = $PlayerDeathSfx
|
||||||
@onready var hurt_timer = $HurtTimer
|
@onready var hurt_timer = $HurtTimer
|
||||||
@onready var health_bar = $HealthBar
|
@onready var health_bar = $HealthBar
|
||||||
|
@onready var punch_vfx = $PunchHitbox/AnimatedSprite2D
|
||||||
|
|
||||||
var jumping = false
|
var jumping = false
|
||||||
var facing_right = true
|
var facing_right = true
|
||||||
|
@ -20,7 +21,7 @@ var hurting = false
|
||||||
var is_dying = false
|
var is_dying = false
|
||||||
var is_death_sfx = false
|
var is_death_sfx = false
|
||||||
|
|
||||||
const ATTACK_DAMAGE = 25
|
const ATTACK_DAMAGE = 20
|
||||||
const INITIAL_HEALTH = 50
|
const INITIAL_HEALTH = 50
|
||||||
const SPEED_MULTIPLIER = 300
|
const SPEED_MULTIPLIER = 300
|
||||||
const ATTACK_KNOCKBACK = 5000
|
const ATTACK_KNOCKBACK = 5000
|
||||||
|
@ -103,6 +104,7 @@ func _process(delta: float) -> void:
|
||||||
|
|
||||||
if attack_timer.is_stopped() and Input.is_action_just_pressed("attack"):
|
if attack_timer.is_stopped() and Input.is_action_just_pressed("attack"):
|
||||||
attack_timer.start()
|
attack_timer.start()
|
||||||
|
punch_vfx.play("default")
|
||||||
|
|
||||||
if is_on_floor():
|
if is_on_floor():
|
||||||
animated_sprite.play("Right" if right_punch else "Left")
|
animated_sprite.play("Right" if right_punch else "Left")
|
||||||
|
|
|
@ -1,15 +1,21 @@
|
||||||
extends Node2D
|
extends Node2D
|
||||||
|
|
||||||
@onready var complete = false
|
@onready var complete = false
|
||||||
|
@onready var wave_1 = [$Crawler, $Crawler2]
|
||||||
|
|
||||||
# Called when the node enters the scene tree for the first time.
|
# Called when the node enters the scene tree for the first time.
|
||||||
func _ready() -> void:
|
func _ready() -> void:
|
||||||
pass # Replace with function body.
|
for crawler in wave_1:
|
||||||
|
crawler.charge()
|
||||||
|
|
||||||
# Called every frame. 'delta' is the elapsed time since the previous frame.
|
# Called every frame. 'delta' is the elapsed time since the previous frame.
|
||||||
func _process(delta: float) -> void:
|
func _process(delta: float) -> void:
|
||||||
pass
|
for crawler in wave_1:
|
||||||
|
if crawler.is_dying:
|
||||||
|
wave_1.erase(crawler)
|
||||||
|
|
||||||
|
if wave_1 == []:
|
||||||
|
complete = true
|
||||||
|
|
||||||
func is_complete() -> bool:
|
func is_complete() -> bool:
|
||||||
return complete
|
return complete
|
||||||
|
|
Reference in a new issue