Player.java 42.8 KB
Newer Older
Melledy's avatar
Melledy committed
1
package emu.grasscutter.game.player;
Melledy's avatar
Melledy committed
2
3

import dev.morphia.annotations.*;
4
import emu.grasscutter.GameConstants;
5
import emu.grasscutter.Grasscutter;
6
import emu.grasscutter.data.GameData;
Melledy's avatar
Melledy committed
7
import emu.grasscutter.data.excels.PlayerLevelData;
Melledy's avatar
Melledy committed
8
import emu.grasscutter.database.DatabaseHelper;
Melledy's avatar
Melledy committed
9
10
import emu.grasscutter.game.Account;
import emu.grasscutter.game.CoopRequest;
Melledy's avatar
Melledy committed
11
import emu.grasscutter.game.ability.AbilityManager;
12
import emu.grasscutter.game.avatar.Avatar;
Kengxxiao's avatar
Kengxxiao committed
13
14
import emu.grasscutter.game.avatar.AvatarProfileData;
import emu.grasscutter.game.avatar.AvatarStorage;
15
16
import emu.grasscutter.game.entity.EntityMonster;
import emu.grasscutter.game.entity.EntityVehicle;
17
import emu.grasscutter.game.managers.DeforestationManager.DeforestationManager;
18
import emu.grasscutter.game.entity.EntityGadget;
Melledy's avatar
Melledy committed
19
import emu.grasscutter.game.entity.EntityItem;
20
import emu.grasscutter.game.entity.GameEntity;
Kinesis's avatar
Kinesis committed
21
import emu.grasscutter.game.expedition.ExpeditionInfo;
Melledy's avatar
Melledy committed
22
23
24
import emu.grasscutter.game.friends.FriendsList;
import emu.grasscutter.game.friends.PlayerProfile;
import emu.grasscutter.game.gacha.PlayerGachaInfo;
25
import emu.grasscutter.game.inventory.GameItem;
Melledy's avatar
Melledy committed
26
import emu.grasscutter.game.inventory.Inventory;
Melledy's avatar
Melledy committed
27
import emu.grasscutter.game.mail.Mail;
28
import emu.grasscutter.game.mail.MailHandler;
29
import emu.grasscutter.game.managers.InsectCaptureManager;
30
import emu.grasscutter.game.managers.StaminaManager.StaminaManager;
31
import emu.grasscutter.game.managers.SotSManager;
32
import emu.grasscutter.game.managers.EnergyManager.EnergyManager;
Melledy's avatar
Melledy committed
33
import emu.grasscutter.game.props.ActionReason;
34
import emu.grasscutter.game.props.EntityType;
Melledy's avatar
Melledy committed
35
import emu.grasscutter.game.props.PlayerProperty;
36
import emu.grasscutter.game.props.SceneType;
Melledy's avatar
Melledy committed
37
import emu.grasscutter.game.quest.QuestManager;
Kengxxiao's avatar
Kengxxiao committed
38
import emu.grasscutter.game.shop.ShopLimit;
Angda Song's avatar
Angda Song committed
39
import emu.grasscutter.game.managers.MapMarkManager.*;
Akka's avatar
Akka committed
40
import emu.grasscutter.game.tower.TowerManager;
Melledy's avatar
Melledy committed
41
42
import emu.grasscutter.game.world.Scene;
import emu.grasscutter.game.world.World;
43
import emu.grasscutter.net.packet.BasePacket;
44
import emu.grasscutter.net.proto.*;
Melledy's avatar
Melledy committed
45
import emu.grasscutter.net.proto.AbilityInvokeEntryOuterClass.AbilityInvokeEntry;
46
import emu.grasscutter.net.proto.AttackResultOuterClass.AttackResult;
Melledy's avatar
Melledy committed
47
48
49
50
51
import emu.grasscutter.net.proto.CombatInvokeEntryOuterClass.CombatInvokeEntry;
import emu.grasscutter.net.proto.InteractTypeOuterClass.InteractType;
import emu.grasscutter.net.proto.MpSettingTypeOuterClass.MpSettingType;
import emu.grasscutter.net.proto.OnlinePlayerInfoOuterClass.OnlinePlayerInfo;
import emu.grasscutter.net.proto.PlayerLocationInfoOuterClass.PlayerLocationInfo;
Melledy's avatar
Melledy committed
52
import emu.grasscutter.net.proto.ProfilePictureOuterClass.ProfilePicture;
Melledy's avatar
Melledy committed
53
import emu.grasscutter.net.proto.SocialDetailOuterClass.SocialDetail;
54

55
56
import emu.grasscutter.server.event.player.PlayerJoinEvent;
import emu.grasscutter.server.event.player.PlayerQuitEvent;
Melledy's avatar
Melledy committed
57
58
import emu.grasscutter.server.game.GameServer;
import emu.grasscutter.server.game.GameSession;
59
import emu.grasscutter.server.game.GameSession.SessionState;
60
import emu.grasscutter.server.packet.send.*;
Yazawazi's avatar
utils    
Yazawazi committed
61
import emu.grasscutter.utils.DateHelper;
Kengxxiao's avatar
Kengxxiao committed
62
import emu.grasscutter.utils.Position;
63
import emu.grasscutter.utils.MessageHandler;
Kinesis's avatar
Kinesis committed
64
import emu.grasscutter.utils.Utils;
Melledy's avatar
Melledy committed
65
66
67
import it.unimi.dsi.fastutil.ints.Int2ObjectMap;
import it.unimi.dsi.fastutil.ints.Int2ObjectOpenHashMap;

Asnxthaony's avatar
Asnxthaony committed
68
import java.util.*;
69
import java.util.concurrent.LinkedBlockingQueue;
Asnxthaony's avatar
Asnxthaony committed
70

71
72
import static emu.grasscutter.Configuration.*;

KingRainbow44's avatar
KingRainbow44 committed
73
@Entity(value = "players", useDiscriminator = false)
74
public class Player {
Angda Song's avatar
Angda Song committed
75

Melledy's avatar
Melledy committed
76
77
	@Id private int id;
	@Indexed(options = @IndexOptions(unique = true)) private String accountId;
Asnxthaony's avatar
Asnxthaony committed
78

Melledy's avatar
Melledy committed
79
80
81
82
83
84
85
	@Transient private Account account;
	private String nickname;
	private String signature;
	private int headImage;
	private int nameCardId = 210001;
	private Position pos;
	private Position rotation;
Miyucchi's avatar
Miyucchi committed
86
	private PlayerBirthday birthday;
87
	private PlayerCodex codex;
Asnxthaony's avatar
Asnxthaony committed
88

Melledy's avatar
Melledy committed
89
90
91
92
	private Map<Integer, Integer> properties;
	private Set<Integer> nameCardList;
	private Set<Integer> flyCloakList;
	private Set<Integer> costumeList;
93
	private Set<Integer> unlockedForgingBlueprints;
Asnxthaony's avatar
Asnxthaony committed
94

95
96
	private Integer widgetId;

97
98
99
	private Set<Integer> realmList;
	private Integer currentRealmId;

Melledy's avatar
Melledy committed
100
101
102
	@Transient private long nextGuid = 0;
	@Transient private int peerId;
	@Transient private World world;
103
	@Transient private Scene scene;
Melledy's avatar
Melledy committed
104
105
106
107
	@Transient private GameSession session;
	@Transient private AvatarStorage avatars;
	@Transient private Inventory inventory;
	@Transient private FriendsList friendsList;
108
	@Transient private MailHandler mailHandler;
109
	@Transient private MessageHandler messageHandler;
Melledy's avatar
Melledy committed
110
	@Transient private AbilityManager abilityManager;
Melledy's avatar
Melledy committed
111
	@Transient private QuestManager questManager;
Melledy's avatar
Melledy committed
112
	
113
	@Transient private SotSManager sotsManager;
114
	@Transient private InsectCaptureManager insectCaptureManager;
115

Melledy's avatar
Melledy committed
116
	private TeamManager teamManager;
Akka's avatar
Akka committed
117
118

	private TowerManager towerManager;
Melledy's avatar
Melledy committed
119
120
121
122
	private PlayerGachaInfo gachaInfo;
	private PlayerProfile playerProfile;
	private boolean showAvatar;
	private ArrayList<AvatarProfileData> shownAvatars;
123
	private Set<Integer> rewardedLevels;
Kengxxiao's avatar
Kengxxiao committed
124
	private ArrayList<ShopLimit> shopLimit;
Kinesis's avatar
Kinesis committed
125
	private Map<Long, ExpeditionInfo> expeditionInfo;
Asnxthaony's avatar
Asnxthaony committed
126

Melledy's avatar
Melledy committed
127
128
129
130
	private int sceneId;
	private int regionId;
	private int mainCharacterId;
	private boolean godmode;
Asnxthaony's avatar
Asnxthaony committed
131

132
	private boolean stamina;
Yazawazi's avatar
Yazawazi committed
133
134
135
136
137
	private boolean moonCard;
	private Date moonCardStartTime;
	private int moonCardDuration;
	private Set<Date> moonCardGetTimes;

Yazawazi's avatar
Yazawazi committed
138
139
140
	private List<Integer> showAvatarList;
	private boolean showAvatars;

Melledy's avatar
Melledy committed
141
142
143
144
	@Transient private boolean paused;
	@Transient private int enterSceneToken;
	@Transient private SceneLoadState sceneState;
	@Transient private boolean hasSentAvatarDataNotify;
Melledy's avatar
Melledy committed
145
	@Transient private long nextSendPlayerLocTime = 0;
Asnxthaony's avatar
Asnxthaony committed
146

Melledy's avatar
Melledy committed
147
	@Transient private final Int2ObjectMap<CoopRequest> coopRequests;
148
	@Transient private final Queue<AttackResult> attackResults;
Melledy's avatar
Melledy committed
149
150
	@Transient private final InvokeHandler<CombatInvokeEntry> combatInvokeHandler;
	@Transient private final InvokeHandler<AbilityInvokeEntry> abilityInvokeHandler;
151
	@Transient private final InvokeHandler<AbilityInvokeEntry> clientAbilityInitFinishHandler;
Asnxthaony's avatar
Asnxthaony committed
152

gentlespoon's avatar
gentlespoon committed
153
	@Transient private MapMarksManager mapMarksManager;
154
	@Transient private StaminaManager staminaManager;
155
	@Transient private EnergyManager energyManager;
156
	@Transient private DeforestationManager deforestationManager;
157

158
	private long springLastUsed;
gentlespoon's avatar
gentlespoon committed
159
	private HashMap<String, MapMark> mapMarks;
160

Asnxthaony's avatar
Asnxthaony committed
161
162
	@Deprecated
	@SuppressWarnings({"rawtypes", "unchecked"}) // Morphia only!
163
	public Player() {
Melledy's avatar
Melledy committed
164
165
166
		this.inventory = new Inventory(this);
		this.avatars = new AvatarStorage(this);
		this.friendsList = new FriendsList(this);
167
		this.mailHandler = new MailHandler(this);
168
		this.towerManager = new TowerManager(this);
Melledy's avatar
Melledy committed
169
		this.abilityManager = new AbilityManager(this);
170
		this.deforestationManager = new DeforestationManager(this);
171
		this.insectCaptureManager = new InsectCaptureManager(this);
172

Melledy's avatar
Melledy committed
173
		this.setQuestManager(new QuestManager(this));
Melledy's avatar
Melledy committed
174
175
176
177
178
179
180
181
182
		this.pos = new Position();
		this.rotation = new Position();
		this.properties = new HashMap<>();
		for (PlayerProperty prop : PlayerProperty.values()) {
			if (prop.getId() < 10000) {
				continue;
			}
			this.properties.put(prop.getId(), 0);
		}
Asnxthaony's avatar
Asnxthaony committed
183

184
185
186
187
		this.gachaInfo = new PlayerGachaInfo();
		this.nameCardList = new HashSet<>();
		this.flyCloakList = new HashSet<>();
		this.costumeList = new HashSet<>();
188
		this.unlockedForgingBlueprints = new HashSet<>();
Asnxthaony's avatar
Asnxthaony committed
189

Melledy's avatar
Melledy committed
190
191
192
		this.setSceneId(3);
		this.setRegionId(1);
		this.sceneState = SceneLoadState.NONE;
Asnxthaony's avatar
Asnxthaony committed
193

194
		this.attackResults = new LinkedBlockingQueue<>();
Melledy's avatar
Melledy committed
195
196
197
		this.coopRequests = new Int2ObjectOpenHashMap<>();
		this.combatInvokeHandler = new InvokeHandler(PacketCombatInvocationsNotify.class);
		this.abilityInvokeHandler = new InvokeHandler(PacketAbilityInvocationsNotify.class);
198
		this.clientAbilityInitFinishHandler = new InvokeHandler(PacketClientAbilityInitFinishNotify.class);
Miyucchi's avatar
Miyucchi committed
199
200

		this.birthday = new PlayerBirthday();
201
		this.rewardedLevels = new HashSet<>();
Yazawazi's avatar
Yazawazi committed
202
		this.moonCardGetTimes = new HashSet<>();
203
		this.codex = new PlayerCodex(this);
Kengxxiao's avatar
Kengxxiao committed
204
205

		this.shopLimit = new ArrayList<>();
Kinesis's avatar
Kinesis committed
206
		this.expeditionInfo = new HashMap<>();
207
		this.messageHandler = null;
gentlespoon's avatar
gentlespoon committed
208
		this.mapMarksManager = new MapMarksManager(this);
209
		this.staminaManager = new StaminaManager(this);
210
		this.sotsManager = new SotSManager(this);
211
		this.energyManager = new EnergyManager(this);
Melledy's avatar
Melledy committed
212
	}
Asnxthaony's avatar
Asnxthaony committed
213

Melledy's avatar
Melledy committed
214
	// On player creation
215
	public Player(GameSession session) {
Melledy's avatar
Melledy committed
216
217
218
219
220
221
222
		this();
		this.account = session.getAccount();
		this.accountId = this.getAccount().getId();
		this.session = session;
		this.nickname = "Traveler";
		this.signature = "";
		this.teamManager = new TeamManager(this);
Miyucchi's avatar
Miyucchi committed
223
		this.birthday = new PlayerBirthday();
224
		this.codex = new PlayerCodex(this);
Melledy's avatar
Melledy committed
225
226
227
228
229
230
231
232
233
234
		this.setProperty(PlayerProperty.PROP_PLAYER_LEVEL, 1);
		this.setProperty(PlayerProperty.PROP_IS_SPRING_AUTO_USE, 1);
		this.setProperty(PlayerProperty.PROP_SPRING_AUTO_USE_PERCENT, 50);
		this.setProperty(PlayerProperty.PROP_IS_FLYABLE, 1);
		this.setProperty(PlayerProperty.PROP_IS_TRANSFERABLE, 1);
		this.setProperty(PlayerProperty.PROP_MAX_STAMINA, 24000);
		this.setProperty(PlayerProperty.PROP_CUR_PERSIST_STAMINA, 24000);
		this.setProperty(PlayerProperty.PROP_PLAYER_RESIN, 160);
		this.getFlyCloakList().add(140001);
		this.getNameCardList().add(210001);
235
		this.getPos().set(GameConstants.START_POSITION);
Melledy's avatar
Melledy committed
236
		this.getRotation().set(0, 307, 0);
237
		this.messageHandler = null;
gentlespoon's avatar
gentlespoon committed
238
		this.mapMarksManager = new MapMarksManager(this);
239
		this.staminaManager = new StaminaManager(this);
240
		this.sotsManager = new SotSManager(this);
241
		this.energyManager = new EnergyManager(this);
242
		this.deforestationManager = new DeforestationManager(this);
Melledy's avatar
Melledy committed
243
244
	}

245
	public int getUid() {
Melledy's avatar
Melledy committed
246
247
248
		return id;
	}

249
	public void setUid(int id) {
Melledy's avatar
Melledy committed
250
251
		this.id = id;
	}
Asnxthaony's avatar
Asnxthaony committed
252

253
	public long getNextGameGuid() {
Melledy's avatar
Melledy committed
254
		long nextId = ++this.nextGuid;
255
		return ((long) this.getUid() << 32) + nextId;
Melledy's avatar
Melledy committed
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
	}

	public Account getAccount() {
		return account;
	}

	public void setAccount(Account account) {
		this.account = account;
	}

	public GameSession getSession() {
		return session;
	}

	public void setSession(GameSession session) {
		this.session = session;
	}
Asnxthaony's avatar
Asnxthaony committed
273

Melledy's avatar
Melledy committed
274
275
276
	public boolean isOnline() {
		return this.getSession() != null && this.getSession().isActive();
	}
Asnxthaony's avatar
Asnxthaony committed
277

Melledy's avatar
Melledy committed
278
279
280
	public GameServer getServer() {
		return this.getSession().getServer();
	}
Asnxthaony's avatar
Asnxthaony committed
281

Melledy's avatar
Melledy committed
282
283
284
	public synchronized World getWorld() {
		return this.world;
	}
Asnxthaony's avatar
Asnxthaony committed
285

Melledy's avatar
Melledy committed
286
287
288
	public synchronized void setWorld(World world) {
		this.world = world;
	}
Asnxthaony's avatar
Asnxthaony committed
289

290
	public synchronized Scene getScene() {
291
292
293
		return scene;
	}

294
	public synchronized void setScene(Scene scene) {
295
296
297
		this.scene = scene;
	}

Melledy's avatar
Melledy committed
298
299
300
301
302
303
304
305
306
307
308
309
	public int getGmLevel() {
		return 1;
	}

	public String getNickname() {
		return nickname;
	}

	public void setNickname(String nickName) {
		this.nickname = nickName;
		this.updateProfile();
	}
Asnxthaony's avatar
Asnxthaony committed
310

Melledy's avatar
Melledy committed
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
	public int getHeadImage() {
		return headImage;
	}

	public void setHeadImage(int picture) {
		this.headImage = picture;
		this.updateProfile();
	}

	public String getSignature() {
		return signature;
	}

	public void setSignature(String signature) {
		this.signature = signature;
		this.updateProfile();
	}

329
330
331
332
333
334
335
336
	public Integer getWidgetId() {
		return widgetId;
	}

	public void setWidgetId(Integer widgetId) {
		this.widgetId = widgetId;
	}

337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
	public Set<Integer> getRealmList() {
		return realmList;
	}

	public void setRealmList(Set<Integer> realmList) {
		this.realmList = realmList;
	}

	public void addRealmList(int realmId) {
		if (this.realmList == null) {
			this.realmList = new HashSet<>();
		} else if (this.realmList.contains(realmId)) {
			return;
		}
		this.realmList.add(realmId);
	}

	public Integer getCurrentRealmId() {
		return currentRealmId;
	}

	public void setCurrentRealmId(Integer currentRealmId) {
		this.currentRealmId = currentRealmId;
	}

Melledy's avatar
Melledy committed
362
363
364
	public Position getPos() {
		return pos;
	}
Asnxthaony's avatar
Asnxthaony committed
365

Melledy's avatar
Melledy committed
366
367
368
	public Position getRotation() {
		return rotation;
	}
Asnxthaony's avatar
Asnxthaony committed
369

Melledy's avatar
Melledy committed
370
371
372
	public int getLevel() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_LEVEL);
	}
Asnxthaony's avatar
Asnxthaony committed
373

Melledy's avatar
Melledy committed
374
375
376
	public int getExp() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_EXP);
	}
Asnxthaony's avatar
Asnxthaony committed
377

Melledy's avatar
Melledy committed
378
379
380
	public int getWorldLevel() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_WORLD_LEVEL);
	}
381
382
383
384
385
	
	public void setWorldLevel(int level) {
		this.setProperty(PlayerProperty.PROP_PLAYER_WORLD_LEVEL, level);
		this.sendPacket(new PacketPlayerPropNotify(this, PlayerProperty.PROP_PLAYER_WORLD_LEVEL));
	}
Asnxthaony's avatar
Asnxthaony committed
386

Melledy's avatar
Melledy committed
387
388
389
	public int getPrimogems() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_HCOIN);
	}
Asnxthaony's avatar
Asnxthaony committed
390

Melledy's avatar
Melledy committed
391
392
393
394
	public void setPrimogems(int primogem) {
		this.setProperty(PlayerProperty.PROP_PLAYER_HCOIN, primogem);
		this.sendPacket(new PacketPlayerPropNotify(this, PlayerProperty.PROP_PLAYER_HCOIN));
	}
Asnxthaony's avatar
Asnxthaony committed
395

Melledy's avatar
Melledy committed
396
397
398
	public int getMora() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_SCOIN);
	}
Asnxthaony's avatar
Asnxthaony committed
399

Melledy's avatar
Melledy committed
400
401
402
403
	public void setMora(int mora) {
		this.setProperty(PlayerProperty.PROP_PLAYER_SCOIN, mora);
		this.sendPacket(new PacketPlayerPropNotify(this, PlayerProperty.PROP_PLAYER_SCOIN));
	}
404
405
406
407
408
409
410
411
412
	
	public int getCrystals() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_MCOIN);
	}

	public void setCrystals(int crystals) {
		this.setProperty(PlayerProperty.PROP_PLAYER_MCOIN, crystals);
		this.sendPacket(new PacketPlayerPropNotify(this, PlayerProperty.PROP_PLAYER_MCOIN));
	}
Asnxthaony's avatar
Asnxthaony committed
413

Melledy's avatar
Melledy committed
414
	private int getExpRequired(int level) {
415
		PlayerLevelData levelData = GameData.getPlayerLevelDataMap().get(level);
Asnxthaony's avatar
Asnxthaony committed
416
		return levelData != null ? levelData.getExp() : 0;
Melledy's avatar
Melledy committed
417
	}
Asnxthaony's avatar
Asnxthaony committed
418

Melledy's avatar
Melledy committed
419
	private float getExpModifier() {
420
		return GAME_OPTIONS.rates.adventureExp;
Melledy's avatar
Melledy committed
421
	}
Asnxthaony's avatar
Asnxthaony committed
422

Melledy's avatar
Melledy committed
423
424
425
426
	// Affected by exp rate
	public void earnExp(int exp) {
		addExpDirectly((int) (exp * getExpModifier()));
	}
Asnxthaony's avatar
Asnxthaony committed
427

Melledy's avatar
Melledy committed
428
429
430
431
432
433
	// Directly give player exp
	public void addExpDirectly(int gain) {
		boolean hasLeveledUp = false;
		int level = getLevel();
		int exp = getExp();
		int reqExp = getExpRequired(level);
Asnxthaony's avatar
Asnxthaony committed
434

Melledy's avatar
Melledy committed
435
		exp += gain;
Asnxthaony's avatar
Asnxthaony committed
436

Melledy's avatar
Melledy committed
437
438
439
440
441
442
		while (exp >= reqExp && reqExp > 0) {
			exp -= reqExp;
			level += 1;
			reqExp = getExpRequired(level);
			hasLeveledUp = true;
		}
Asnxthaony's avatar
Asnxthaony committed
443

Melledy's avatar
Melledy committed
444
445
446
447
448
449
450
451
		if (hasLeveledUp) {
			// Set level property
			this.setProperty(PlayerProperty.PROP_PLAYER_LEVEL, level);
			// Update social status
			this.updateProfile();
			// Update player with packet
			this.sendPacket(new PacketPlayerPropNotify(this, PlayerProperty.PROP_PLAYER_LEVEL));
		}
Asnxthaony's avatar
Asnxthaony committed
452

Melledy's avatar
Melledy committed
453
454
		// Set exp
		this.setProperty(PlayerProperty.PROP_PLAYER_EXP, exp);
Asnxthaony's avatar
Asnxthaony committed
455

Melledy's avatar
Melledy committed
456
457
458
		// Update player with packet
		this.sendPacket(new PacketPlayerPropNotify(this, PlayerProperty.PROP_PLAYER_EXP));
	}
Asnxthaony's avatar
Asnxthaony committed
459

Melledy's avatar
Melledy committed
460
461
462
	private void updateProfile() {
		getProfile().syncWithCharacter(this);
	}
Asnxthaony's avatar
Asnxthaony committed
463

Melledy's avatar
Melledy committed
464
465
466
467
468
469
470
471
	public boolean isFirstLoginEnterScene() {
		return !this.hasSentAvatarDataNotify;
	}

	public TeamManager getTeamManager() {
		return this.teamManager;
	}

Akka's avatar
Akka committed
472
473
474
475
	public TowerManager getTowerManager() {
		return towerManager;
	}

Melledy's avatar
Melledy committed
476
477
478
479
480
481
482
483
	public QuestManager getQuestManager() {
		return questManager;
	}

	public void setQuestManager(QuestManager questManager) {
		this.questManager = questManager;
	}

Melledy's avatar
Melledy committed
484
485
486
487
488
489
490
491
492
493
494
	public PlayerGachaInfo getGachaInfo() {
		return gachaInfo;
	}

	public PlayerProfile getProfile() {
		if (this.playerProfile == null) {
			this.playerProfile = new PlayerProfile(this);
		}
		return playerProfile;
	}

495
496
	// TODO: Based on the proto, property value could be int or float.
	//  Although there's no float value at this moment, our code should be prepared for float values.
Melledy's avatar
Melledy committed
497
498
499
	public Map<Integer, Integer> getProperties() {
		return properties;
	}
Asnxthaony's avatar
Asnxthaony committed
500

501
502
	public boolean setProperty(PlayerProperty prop, int value) {
		return setPropertyWithSanityCheck(prop, value);
Melledy's avatar
Melledy committed
503
	}
Asnxthaony's avatar
Asnxthaony committed
504

Melledy's avatar
Melledy committed
505
506
507
508
509
510
511
	public int getProperty(PlayerProperty prop) {
		return getProperties().get(prop.getId());
	}

	public Set<Integer> getFlyCloakList() {
		return flyCloakList;
	}
Asnxthaony's avatar
Asnxthaony committed
512

Melledy's avatar
Melledy committed
513
514
515
	public Set<Integer> getCostumeList() {
		return costumeList;
	}
Asnxthaony's avatar
Asnxthaony committed
516

Melledy's avatar
Melledy committed
517
518
519
520
	public Set<Integer> getNameCardList() {
		return this.nameCardList;
	}

521
522
523
524
	public Set<Integer> getUnlockedForgingBlueprints() {
		return unlockedForgingBlueprints;
	}

Melledy's avatar
Melledy committed
525
	public MpSettingType getMpSetting() {
526
		return MpSettingType.MP_SETTING_TYPE_ENTER_AFTER_APPLY; // TEMP
Melledy's avatar
Melledy committed
527
	}
528
529
530
531
	
	public Queue<AttackResult> getAttackResults() {
		return this.attackResults;
	}
Asnxthaony's avatar
Asnxthaony committed
532

Melledy's avatar
Melledy committed
533
534
535
536
537
538
539
	public synchronized Int2ObjectMap<CoopRequest> getCoopRequests() {
		return coopRequests;
	}

	public InvokeHandler<CombatInvokeEntry> getCombatInvokeHandler() {
		return this.combatInvokeHandler;
	}
Asnxthaony's avatar
Asnxthaony committed
540

Melledy's avatar
Melledy committed
541
542
543
544
	public InvokeHandler<AbilityInvokeEntry> getAbilityInvokeHandler() {
		return this.abilityInvokeHandler;
	}

545
546
547
548
	public InvokeHandler<AbilityInvokeEntry> getClientAbilityInitFinishHandler() {
		return clientAbilityInitFinishHandler;
	}

Melledy's avatar
Melledy committed
549
550
551
	public AvatarStorage getAvatars() {
		return avatars;
	}
Asnxthaony's avatar
Asnxthaony committed
552

Melledy's avatar
Melledy committed
553
554
555
	public Inventory getInventory() {
		return inventory;
	}
Asnxthaony's avatar
Asnxthaony committed
556

Melledy's avatar
Melledy committed
557
558
559
560
	public FriendsList getFriendsList() {
		return this.friendsList;
	}

561
562
563
564
	public MailHandler getMailHandler() {
		return mailHandler;
	}

Melledy's avatar
Melledy committed
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
	public int getEnterSceneToken() {
		return enterSceneToken;
	}

	public void setEnterSceneToken(int enterSceneToken) {
		this.enterSceneToken = enterSceneToken;
	}

	public int getNameCardId() {
		return nameCardId;
	}

	public void setNameCardId(int nameCardId) {
		this.nameCardId = nameCardId;
		this.updateProfile();
	}

	public int getMainCharacterId() {
		return mainCharacterId;
	}

	public void setMainCharacterId(int mainCharacterId) {
		this.mainCharacterId = mainCharacterId;
	}

	public int getPeerId() {
		return peerId;
	}

	public void setPeerId(int peerId) {
		this.peerId = peerId;
	}

	public int getClientTime() {
		return session.getClientTime();
	}

	public long getLastPingTime() {
		return session.getLastPingTime();
	}

	public boolean isPaused() {
		return paused;
	}

	public void setPaused(boolean newPauseState) {
		boolean oldPauseState = this.paused;
		this.paused = newPauseState;
Asnxthaony's avatar
Asnxthaony committed
613

Melledy's avatar
Melledy committed
614
615
616
617
618
619
620
		if (newPauseState && !oldPauseState) {
			this.onPause();
		} else if (oldPauseState && !newPauseState) {
			this.onUnpause();
		}
	}

621
622
623
624
625
626
627
628
	public long getSpringLastUsed() {
		return springLastUsed;
	}

	public void setSpringLastUsed(long val) {
		springLastUsed = val;
	}

Melledy's avatar
Melledy committed
629
630
631
632
633
634
635
	public SceneLoadState getSceneLoadState() {
		return sceneState;
	}

	public void setSceneLoadState(SceneLoadState sceneState) {
		this.sceneState = sceneState;
	}
Asnxthaony's avatar
Asnxthaony committed
636

Melledy's avatar
Melledy committed
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
	public boolean isInMultiplayer() {
		return this.getWorld() != null && this.getWorld().isMultiplayer();
	}

	public int getSceneId() {
		return sceneId;
	}

	public void setSceneId(int sceneId) {
		this.sceneId = sceneId;
	}

	public int getRegionId() {
		return regionId;
	}

	public void setRegionId(int regionId) {
		this.regionId = regionId;
	}
Asnxthaony's avatar
Asnxthaony committed
656

Yazawazi's avatar
Yazawazi committed
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
	public void setShowAvatars(boolean showAvatars) {
		this.showAvatars = showAvatars;
	}

	public boolean isShowAvatars() {
		return showAvatars;
	}

	public void setShowAvatarList(List<Integer> showAvatarList) {
		this.showAvatarList = showAvatarList;
	}

	public List<Integer> getShowAvatarList() {
		return showAvatarList;
	}

Yazawazi's avatar
Yazawazi committed
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
	public boolean inMoonCard() {
		return moonCard;
	}

	public void setMoonCard(boolean moonCard) {
		this.moonCard = moonCard;
	}

	public void addMoonCardDays(int days) {
		this.moonCardDuration += days;
	}

	public int getMoonCardDuration() {
		return moonCardDuration;
	}

	public void setMoonCardDuration(int moonCardDuration) {
		this.moonCardDuration = moonCardDuration;
	}

	public Date getMoonCardStartTime() {
		return moonCardStartTime;
	}

	public void setMoonCardStartTime(Date moonCardStartTime) {
		this.moonCardStartTime = moonCardStartTime;
	}

	public Set<Date> getMoonCardGetTimes() {
		return moonCardGetTimes;
	}

	public void setMoonCardGetTimes(Set<Date> moonCardGetTimes) {
		this.moonCardGetTimes = moonCardGetTimes;
	}

	public int getMoonCardRemainDays() {
		Calendar remainCalendar = Calendar.getInstance();
		remainCalendar.setTime(moonCardStartTime);
		remainCalendar.add(Calendar.DATE, moonCardDuration);
		Date theLastDay = remainCalendar.getTime();
Yazawazi's avatar
Yazawazi committed
714
		Date now = DateHelper.onlyYearMonthDay(new Date());
Yazawazi's avatar
Yazawazi committed
715
716
717
718
		return (int) ((theLastDay.getTime() - now.getTime()) / (24 * 60 * 60 * 1000)); // By copilot 
	}

	public void rechargeMoonCard() {
719
		inventory.addItem(new GameItem(203, 300));
Yazawazi's avatar
Yazawazi committed
720
721
722
		if (!moonCard) {
			moonCard = true;
			Date now = new Date();
Yazawazi's avatar
Yazawazi committed
723
			moonCardStartTime = DateHelper.onlyYearMonthDay(now);
Yazawazi's avatar
Yazawazi committed
724
725
726
727
728
729
730
731
732
733
734
735
736
			moonCardDuration = 30;
		} else {
			moonCardDuration += 30;
		}
		if (!moonCardGetTimes.contains(moonCardStartTime)) {
			moonCardGetTimes.add(moonCardStartTime);
		}
	}

	public void getTodayMoonCard() {
		if (!moonCard) {
			return;
		}
Yazawazi's avatar
Yazawazi committed
737
		Date now = DateHelper.onlyYearMonthDay(new Date());
Yazawazi's avatar
Yazawazi committed
738
739
740
741
742
743
744
745
746
747
748
749
750
751
		if (moonCardGetTimes.contains(now)) {
			return;
		}
		Date stopTime = new Date();
		Calendar stopCalendar = Calendar.getInstance();
		stopCalendar.setTime(stopTime);
		stopCalendar.add(Calendar.DATE, moonCardDuration);
		stopTime = stopCalendar.getTime();
		if (now.after(stopTime)) {
			moonCard = false;
			return;
		}
		moonCardGetTimes.add(now);
		addMoonCardDays(1);
752
753
		GameItem item = new GameItem(201, 90);
		getInventory().addItem(item, ActionReason.BlessingRedeemReward);
Yazawazi's avatar
Yazawazi committed
754
755
756
		session.send(new PacketCardProductRewardNotify(getMoonCardRemainDays()));
	}

Kinesis's avatar
Kinesis committed
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
	public Map<Long, ExpeditionInfo> getExpeditionInfo() {
		return expeditionInfo;
	}

	public void addExpeditionInfo(long avaterGuid, int expId, int hourTime, int startTime){
		ExpeditionInfo exp = new ExpeditionInfo();
		exp.setExpId(expId);
		exp.setHourTime(hourTime);
		exp.setState(1);
		exp.setStartTime(startTime);
		expeditionInfo.put(avaterGuid, exp);
	}

	public void removeExpeditionInfo(long avaterGuid){
		expeditionInfo.remove(avaterGuid);
	}

	public ExpeditionInfo getExpeditionInfo(long avaterGuid){
		return expeditionInfo.get(avaterGuid);
	}

Kengxxiao's avatar
Kengxxiao committed
778
779
780
781
	public List<ShopLimit> getShopLimit() {
		return shopLimit;
	}

Kengxxiao's avatar
Kengxxiao committed
782
783
784
785
786
	public ShopLimit getGoodsLimit(int goodsId) {
		Optional<ShopLimit> shopLimit = this.shopLimit.stream().filter(x -> x.getShopGoodId() == goodsId).findFirst();
		if (shopLimit.isEmpty())
			return null;
		return shopLimit.get();
Kengxxiao's avatar
Kengxxiao committed
787
788
	}

Kengxxiao's avatar
Kengxxiao committed
789
790
791
792
793
794
795
	public void addShopLimit(int goodsId, int boughtCount, int nextRefreshTime) {
		ShopLimit target = getGoodsLimit(goodsId);
		if (target != null) {
			target.setHasBought(target.getHasBought() + boughtCount);
			target.setHasBoughtInPeriod(target.getHasBoughtInPeriod() + boughtCount);
			target.setNextRefreshTime(nextRefreshTime);
		} else {
Kengxxiao's avatar
Kengxxiao committed
796
797
798
			ShopLimit sl = new ShopLimit();
			sl.setShopGoodId(goodsId);
			sl.setHasBought(boughtCount);
Kengxxiao's avatar
Kengxxiao committed
799
800
801
			sl.setHasBoughtInPeriod(boughtCount);
			sl.setNextRefreshTime(nextRefreshTime);
			getShopLimit().add(sl);
Kengxxiao's avatar
Kengxxiao committed
802
803
804
		}
		this.save();
	}
805
806
807
808
809
810
811
812
	public boolean getStamina() {
		// Get Stamina
		return stamina;
	}
	public void setStamina(boolean stamina) {
		// Set Stamina
		this.stamina = stamina;
	}
KingRainbow44's avatar
KingRainbow44 committed
813
	public boolean inGodmode() {
Melledy's avatar
Melledy committed
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
		return godmode;
	}

	public void setGodmode(boolean godmode) {
		this.godmode = godmode;
	}

	public boolean hasSentAvatarDataNotify() {
		return hasSentAvatarDataNotify;
	}

	public void setHasSentAvatarDataNotify(boolean hasSentAvatarDataNotify) {
		this.hasSentAvatarDataNotify = hasSentAvatarDataNotify;
	}

ProxyismGH's avatar
ProxyismGH committed
829
	public void addAvatar(Avatar avatar, boolean addToCurrentTeam) {
Melledy's avatar
Melledy committed
830
		boolean result = getAvatars().addAvatar(avatar);
Asnxthaony's avatar
Asnxthaony committed
831

Melledy's avatar
Melledy committed
832
833
834
		if (result) {
			// Add starting weapon
			getAvatars().addStartingWeapon(avatar);
Asnxthaony's avatar
Asnxthaony committed
835

Melledy's avatar
Melledy committed
836
837
838
839
			// Done
			if (hasSentAvatarDataNotify()) {
				// Recalc stats
				avatar.recalcStats();
ProxyismGH's avatar
ProxyismGH committed
840
841
842
843
844
845
				// Packet, show notice on left if the avatar will be added to the team
				sendPacket(new PacketAvatarAddNotify(avatar, addToCurrentTeam && this.getTeamManager().canAddAvatarToCurrentTeam()));
				if (addToCurrentTeam) {
					// If space in team, add
					this.getTeamManager().addAvatarToCurrentTeam(avatar);
				}
Melledy's avatar
Melledy committed
846
847
848
849
850
			}
		} else {
			// Failed adding avatar
		}
	}
Asnxthaony's avatar
Asnxthaony committed
851

ProxyismGH's avatar
ProxyismGH committed
852
853
854
855
	public void addAvatar(Avatar avatar) {
		addAvatar(avatar, true);
	}

Melledy's avatar
Melledy committed
856
857
858
859
	public void addFlycloak(int flycloakId) {
		this.getFlyCloakList().add(flycloakId);
		this.sendPacket(new PacketAvatarGainFlycloakNotify(flycloakId));
	}
Asnxthaony's avatar
Asnxthaony committed
860

Melledy's avatar
Melledy committed
861
862
863
864
	public void addCostume(int costumeId) {
		this.getCostumeList().add(costumeId);
		this.sendPacket(new PacketAvatarGainCostumeNotify(costumeId));
	}
Asnxthaony's avatar
Asnxthaony committed
865

Melledy's avatar
Melledy committed
866
867
868
869
	public void addNameCard(int nameCardId) {
		this.getNameCardList().add(nameCardId);
		this.sendPacket(new PacketUnlockNameCardNotify(nameCardId));
	}
Asnxthaony's avatar
Asnxthaony committed
870

Melledy's avatar
Melledy committed
871
872
873
874
	public void setNameCard(int nameCardId) {
		if (!this.getNameCardList().contains(nameCardId)) {
			return;
		}
Asnxthaony's avatar
Asnxthaony committed
875

Melledy's avatar
Melledy committed
876
		this.setNameCardId(nameCardId);
Asnxthaony's avatar
Asnxthaony committed
877

Melledy's avatar
Melledy committed
878
879
		this.sendPacket(new PacketSetNameCardRsp(nameCardId));
	}
Asnxthaony's avatar
Asnxthaony committed
880

Melledy's avatar
Melledy committed
881
	public void dropMessage(Object message) {
882
883
884
885
		if (this.messageHandler != null) {
			this.messageHandler.append(message.toString());
			return;
		}
886
		this.sendPacket(new PacketPrivateChatNotify(GameConstants.SERVER_CONSOLE_UID, getUid(), message.toString()));
Melledy's avatar
Melledy committed
887
	}
Melledy's avatar
Melledy committed
888
889
890

	/**
	 * Sends a message to another player.
Asnxthaony's avatar
Asnxthaony committed
891
892
	 *
	 * @param sender  The sender of the message.
Melledy's avatar
Melledy committed
893
894
	 * @param message The message to send.
	 */
895
	public void sendMessage(Player sender, Object message) {
896
		this.sendPacket(new PacketPrivateChatNotify(sender.getUid(), this.getUid(), message.toString()));
Melledy's avatar
Melledy committed
897
	}
Asnxthaony's avatar
Asnxthaony committed
898

Benjamin Elsdon's avatar
Benjamin Elsdon committed
899
	// ---------------------MAIL------------------------
900

901
	public List<Mail> getAllMail() { return this.getMailHandler().getMail(); }
902

Benjamin Elsdon's avatar
Benjamin Elsdon committed
903
	public void sendMail(Mail message) {
904
		this.getMailHandler().sendMail(message);
905
	}
Benjamin Elsdon's avatar
Benjamin Elsdon committed
906
907

	public boolean deleteMail(int mailId) {
908
		return this.getMailHandler().deleteMail(mailId);
Benjamin Elsdon's avatar
Benjamin Elsdon committed
909
910
	}

911
912
	public Mail getMail(int index) { return this.getMailHandler().getMailById(index); }
	
913
	public int getMailId(Mail message) {
914
		return this.getMailHandler().getMailIndex(message);
Benjamin Elsdon's avatar
Benjamin Elsdon committed
915
916
	}

917
	public boolean replaceMailByIndex(int index, Mail message) {
918
		return this.getMailHandler().replaceMailByIndex(index, message);
Benjamin Elsdon's avatar
Benjamin Elsdon committed
919
	}
920
	
Melledy's avatar
Melledy committed
921
	public void interactWith(int gadgetEntityId) {
922
		GameEntity entity = getScene().getEntityById(gadgetEntityId);
Melledy's avatar
Melledy committed
923
924
925
		if (entity == null) {
			return;
		}
Asnxthaony's avatar
Asnxthaony committed
926

Melledy's avatar
Melledy committed
927
		// Handle
928
		if (entity instanceof EntityItem drop) {
Melledy's avatar
Melledy committed
929
			// Pick item
Kengxxiao's avatar
Kengxxiao committed
930
931
932
			if (!drop.isShare()) // check drop owner to avoid someone picked up item in others' world
			{
				int dropOwner = (int)(drop.getGuid() >> 32);
933
				if (dropOwner != getUid()) {
Kengxxiao's avatar
Kengxxiao committed
934
					return;
935
				}
Kengxxiao's avatar
Kengxxiao committed
936
937
			}
			entity.getScene().removeEntity(entity);
938
			GameItem item = new GameItem(drop.getItemData(), drop.getCount());
Melledy's avatar
Melledy committed
939
			// Add to inventory
940
			boolean success = getInventory().addItem(item, ActionReason.SubfieldDrop);
Melledy's avatar
Melledy committed
941
			if (success) {
942
				if (!drop.isShare()) { // not shared drop
943
					this.sendPacket(new PacketGadgetInteractRsp(drop, InteractType.INTERACT_TYPE_PICK_ITEM));
944
				}else{
945
					this.getScene().broadcastPacket(new PacketGadgetInteractRsp(drop, InteractType.INTERACT_TYPE_PICK_ITEM));
946
				}
Melledy's avatar
Melledy committed
947
			}
948
		} else if (entity instanceof EntityGadget gadget) {
949
950
951
952
			if (gadget.getGadgetData().getType() == EntityType.RewardStatue) {
				if (scene.getChallenge() != null) {
					scene.getChallenge().getStatueDrops(this);
				}
953
				this.sendPacket(new PacketGadgetInteractRsp(gadget, InteractType.INTERACT_TYPE_OPEN_STATUE));
954
			}
955
956
957
958
		} else if (entity instanceof EntityMonster monster) {
			insectCaptureManager.arrestSmallCreature(monster);
		} else if (entity instanceof EntityVehicle vehicle) {// try to arrest it, example: glowworm
			insectCaptureManager.arrestSmallCreature(vehicle);
Kengxxiao's avatar
Kengxxiao committed
959
960
961
		} else {
			// Delete directly
			entity.getScene().removeEntity(entity);
Melledy's avatar
Melledy committed
962
963
		}
	}
Asnxthaony's avatar
Asnxthaony committed
964

Melledy's avatar
Melledy committed
965
	public void onPause() {
966
		getStaminaManager().stopSustainedStaminaHandler();
Melledy's avatar
Melledy committed
967
	}
Asnxthaony's avatar
Asnxthaony committed
968

Melledy's avatar
Melledy committed
969
	public void onUnpause() {
970
		getStaminaManager().startSustainedStaminaHandler();
Melledy's avatar
Melledy committed
971
	}
Asnxthaony's avatar
Asnxthaony committed
972

973
	public void sendPacket(BasePacket packet) {
Melledy's avatar
Melledy committed
974
		this.getSession().send(packet);
Melledy's avatar
Melledy committed
975
	}
Asnxthaony's avatar
Asnxthaony committed
976

Melledy's avatar
Melledy committed
977
978
	public OnlinePlayerInfo getOnlinePlayerInfo() {
		OnlinePlayerInfo.Builder onlineInfo = OnlinePlayerInfo.newBuilder()
Asnxthaony's avatar
Asnxthaony committed
979
980
981
982
983
984
				.setUid(this.getUid())
				.setNickname(this.getNickname())
				.setPlayerLevel(this.getLevel())
				.setMpSettingType(this.getMpSetting())
				.setNameCardId(this.getNameCardId())
				.setSignature(this.getSignature())
Melledy's avatar
Melledy committed
985
				.setProfilePicture(ProfilePicture.newBuilder().setAvatarId(this.getHeadImage()));
Asnxthaony's avatar
Asnxthaony committed
986

Melledy's avatar
Melledy committed
987
		if (this.getWorld() != null) {
Melledy's avatar
Melledy committed
988
			onlineInfo.setCurPlayerNumInWorld(getWorld().getPlayerCount());
Melledy's avatar
Melledy committed
989
990
991
		} else {
			onlineInfo.setCurPlayerNumInWorld(1);
		}
Asnxthaony's avatar
Asnxthaony committed
992

Melledy's avatar
Melledy committed
993
994
995
		return onlineInfo.build();
	}

Asnxthaony's avatar
Asnxthaony committed
996
	public PlayerBirthday getBirthday() {
Miyucchi's avatar
Miyucchi committed
997
998
999
1000
1001
1002
1003
1004
		return this.birthday;
	}

	public void setBirthday(int d, int m) {
		this.birthday = new PlayerBirthday(d, m);
		this.updateProfile();
	}

Asnxthaony's avatar
Asnxthaony committed
1005
1006
1007
1008
	public boolean hasBirthday() {
		return this.birthday.getDay() > 0;
	}

1009
1010
	public PlayerCodex getCodex(){ return this.codex; }

1011
1012
1013
1014
1015
1016
1017
1018
	public Set<Integer> getRewardedLevels() {
		return rewardedLevels;
	}

	public void setRewardedLevels(Set<Integer> rewardedLevels) {
		this.rewardedLevels = rewardedLevels;
	}

Melledy's avatar
Melledy committed
1019
	public SocialDetail.Builder getSocialDetail() {
Yazawazi's avatar
Yazawazi committed
1020
		List<SocialShowAvatarInfoOuterClass.SocialShowAvatarInfo> socialShowAvatarInfoList = new ArrayList<>();
Yazawazi's avatar
Yazawazi committed
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
		if (this.isOnline()) {
			if (this.getShowAvatarList() != null) {
				for (int avatarId : this.getShowAvatarList()) {
					socialShowAvatarInfoList.add(
							socialShowAvatarInfoList.size(),
							SocialShowAvatarInfoOuterClass.SocialShowAvatarInfo.newBuilder()
									.setAvatarId(avatarId)
									.setLevel(getAvatars().getAvatarById(avatarId).getLevel())
									.setCostumeId(getAvatars().getAvatarById(avatarId).getCostume())
									.build()
					);
				}
			}
		} else {
1035
1036
			List<Integer> showAvatarList = DatabaseHelper.getPlayerByUid(id).getShowAvatarList();
			AvatarStorage avatars = DatabaseHelper.getPlayerByUid(id).getAvatars();
Yazawazi's avatar
Yazawazi committed
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
			avatars.loadFromDatabase();
			if (showAvatarList != null) {
				for (int avatarId : showAvatarList) {
					socialShowAvatarInfoList.add(
							socialShowAvatarInfoList.size(),
							SocialShowAvatarInfoOuterClass.SocialShowAvatarInfo.newBuilder()
									.setAvatarId(avatarId)
									.setLevel(avatars.getAvatarById(avatarId).getLevel())
									.setCostumeId(avatars.getAvatarById(avatarId).getCostume())
									.build()
					);
				}
			}
Yazawazi's avatar
Yazawazi committed
1050
1051
		}

Asnxthaony's avatar
Asnxthaony committed
1052
1053
		SocialDetail.Builder social = SocialDetail.newBuilder()
				.setUid(this.getUid())
Yazawazi's avatar
Yazawazi committed
1054
				.setProfilePicture(ProfilePicture.newBuilder().setAvatarId(this.getHeadImage()))
Asnxthaony's avatar
Asnxthaony committed
1055
1056
1057
1058
1059
1060
				.setNickname(this.getNickname())
				.setSignature(this.getSignature())
				.setLevel(this.getLevel())
				.setBirthday(this.getBirthday().getFilledProtoWhenNotEmpty())
				.setWorldLevel(this.getWorldLevel())
				.setNameCardId(this.getNameCardId())
Yazawazi's avatar
Yazawazi committed
1061
1062
				.setIsShowAvatar(this.isShowAvatars())
				.addAllShowAvatarInfoList(socialShowAvatarInfoList)
Asnxthaony's avatar
Asnxthaony committed
1063
				.setFinishAchievementNum(0);
Melledy's avatar
Melledy committed
1064
1065
		return social;
	}
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075

	public List<ShowAvatarInfoOuterClass.ShowAvatarInfo> getShowAvatarInfoList() {
		List<ShowAvatarInfoOuterClass.ShowAvatarInfo> showAvatarInfoList = new ArrayList<>();

		Player player;
		boolean shouldRecalc;
		if (this.isOnline()) {
			player = this;
			shouldRecalc = false;
		} else {
1076
			player = DatabaseHelper.getPlayerByUid(id);
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
			player.getAvatars().loadFromDatabase();
			player.getInventory().loadFromDatabase();
			shouldRecalc = true;
		}

		List<Integer> showAvatarList = player.getShowAvatarList();
		AvatarStorage avatars = player.getAvatars();
		if (showAvatarList != null) {
			for (int avatarId : showAvatarList) {
				Avatar avatar = avatars.getAvatarById(avatarId);
				if (shouldRecalc) {
					avatar.recalcStats();
				}
				showAvatarInfoList.add(avatar.toShowAvatarInfoProto());
			}
		}
		return showAvatarInfoList;
	}
1095
1096
1097
	
	public PlayerWorldLocationInfoOuterClass.PlayerWorldLocationInfo getWorldPlayerLocationInfo() {
		return PlayerWorldLocationInfoOuterClass.PlayerWorldLocationInfo.newBuilder()
Asnxthaony's avatar
Asnxthaony committed
1098
1099
1100
				.setSceneId(this.getSceneId())
				.setPlayerLoc(this.getPlayerLocationInfo())
				.build();
Melledy's avatar
Melledy committed
1101
	}
Asnxthaony's avatar
Asnxthaony committed
1102

Melledy's avatar
Melledy committed
1103
	public PlayerLocationInfo getPlayerLocationInfo() {
Asnxthaony's avatar
Asnxthaony committed
1104
1105
1106
1107
1108
		return PlayerLocationInfo.newBuilder()
				.setUid(this.getUid())
				.setPos(this.getPos().toProto())
				.setRot(this.getRotation().toProto())
				.build();
Melledy's avatar
Melledy committed
1109
	}
Asnxthaony's avatar
Asnxthaony committed
1110

Angda Song's avatar
Angda Song committed
1111
1112
1113
1114
	public MapMarksManager getMapMarksManager() {
		return mapMarksManager;
	}

1115
	public StaminaManager getStaminaManager() { return staminaManager; }
1116

1117
1118
	public SotSManager getSotSManager() { return sotsManager; }

1119
1120
1121
1122
	public EnergyManager getEnergyManager() {
		return this.energyManager;
	}

Melledy's avatar
Melledy committed
1123
1124
1125
1126
	public AbilityManager getAbilityManager() {
		return abilityManager;
	}

1127
1128
1129
1130
	public DeforestationManager getDeforestationManager() {
		return deforestationManager;
	}

gentlespoon's avatar
gentlespoon committed
1131
1132
1133
1134
	public HashMap<String, MapMark> getMapMarks() { return mapMarks; }

	public void setMapMarks(HashMap<String, MapMark> newMarks) { mapMarks = newMarks; }

Melledy's avatar
Melledy committed
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
	public synchronized void onTick() {
		// Check ping
		if (this.getLastPingTime() > System.currentTimeMillis() + 60000) {
			this.getSession().close();
			return;
		}
		// Check co-op requests
		Iterator<CoopRequest> it = this.getCoopRequests().values().iterator();
		while (it.hasNext()) {
			CoopRequest req = it.next();
			if (req.isExpired()) {
1146
1147
1148
1149
				req.getRequester().sendPacket(new PacketPlayerApplyEnterMpResultNotify(
						this,
						false,
						PlayerApplyEnterMpResultNotifyOuterClass.PlayerApplyEnterMpResultNotify.Reason.REASON_SYSTEM_JUDGE));
Melledy's avatar
Melledy committed
1150
1151
1152
1153
1154
				it.remove();
			}
		}
		// Ping
		if (this.getWorld() != null) {
Melledy's avatar
Melledy committed
1155
1156
			// RTT notify - very important to send this often
			this.sendPacket(new PacketWorldPlayerRTTNotify(this.getWorld()));
Asnxthaony's avatar
Asnxthaony committed
1157

Melledy's avatar
Melledy committed
1158
1159
1160
1161
1162
1163
1164
			// Update player locations if in multiplayer every 5 seconds
			long time = System.currentTimeMillis();
			if (this.getWorld().isMultiplayer() && this.getScene() != null && time > nextSendPlayerLocTime) {
				this.sendPacket(new PacketWorldPlayerLocationNotify(this.getWorld()));
				this.sendPacket(new PacketScenePlayerLocationNotify(this.getScene()));
				this.resetSendPlayerLocTime();
			}
Melledy's avatar
Melledy committed
1165
		}
Kinesis's avatar
Kinesis committed
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
		// Expedition
		var timeNow = Utils.getCurrentSeconds();
		var needNotify = false;
		for (Long key : expeditionInfo.keySet()) {
			ExpeditionInfo e = expeditionInfo.get(key);
			if(e.getState() == 1){
				if(timeNow - e.getStartTime() >= e.getHourTime() * 60 * 60){
					e.setState(2);
					needNotify = true;
				}
			}
		}
		if(needNotify){
			this.save();
			this.sendPacket(new PacketAvatarExpeditionDataNotify(this));
		}
1182
1183
1184
	}


Asnxthaony's avatar
Asnxthaony committed
1185

1186

Melledy's avatar
Melledy committed
1187
1188
1189
	public void resetSendPlayerLocTime() {
		this.nextSendPlayerLocTime = System.currentTimeMillis() + 5000;
	}
Melledy's avatar
Melledy committed
1190
1191
1192

	@PostLoad
	private void onLoad() {
1193
		this.getCodex().setPlayer(this);
Melledy's avatar
Melledy committed
1194
		this.getTeamManager().setPlayer(this);
1195
		this.getTowerManager().setPlayer(this);
Melledy's avatar
Melledy committed
1196
	}
Asnxthaony's avatar
Asnxthaony committed
1197

Melledy's avatar
Melledy committed
1198
1199
1200
	public void save() {
		DatabaseHelper.savePlayer(this);
	}
1201
1202
1203
	
	// Called from tokenrsp
	public void loadFromDatabase() {
Melledy's avatar
Melledy committed
1204
1205
1206
		// Make sure these exist
		if (this.getTeamManager() == null) {
			this.teamManager = new TeamManager(this);
Asnxthaony's avatar
Asnxthaony committed
1207
		}
1208
1209
1210
		if (this.getCodex() == null) {
			this.codex = new PlayerCodex(this);
		}
Asnxthaony's avatar
Asnxthaony committed
1211
		if (this.getProfile().getUid() == 0) {
1212
			this.getProfile().syncWithCharacter(this);
Melledy's avatar
Melledy committed
1213
		}
Asnxthaony's avatar
Asnxthaony committed
1214

Melledy's avatar
Melledy committed
1215
1216
1217
1218
		// Load from db
		this.getAvatars().loadFromDatabase();
		this.getInventory().loadFromDatabase();
		this.getAvatars().postLoad();
Asnxthaony's avatar
Asnxthaony committed
1219

Melledy's avatar
Melledy committed
1220
		this.getFriendsList().loadFromDatabase();
1221
		this.getMailHandler().loadFromDatabase();
Melledy's avatar
Melledy committed
1222
		this.getQuestManager().loadFromDatabase();
1223

1224
1225
1226
1227
1228
		// Add to gameserver (Always handle last)
		if (getSession().isActive()) {
			getServer().registerPlayer(this);
			getProfile().setPlayer(this); // Set online
		}
1229

1230
1231
1232
	}

	public void onLogin() {
Melledy's avatar
Melledy committed
1233
1234
1235
1236
1237
1238
1239
		// Quest - Commented out because a problem is caused if you log out while this quest is active
		/*
		if (getQuestManager().getMainQuestById(351) == null) {
			GameQuest quest = getQuestManager().addQuest(35104);
			if (quest != null) {
				quest.finish();
			}
Asnxthaony's avatar
Asnxthaony committed
1240

Melledy's avatar
Melledy committed
1241
1242
1243
1244
1245
1246
1247
			getQuestManager().addQuest(35101);
			
			this.setSceneId(3);
			this.getPos().set(GameConstants.START_POSITION);
		}
		*/
		
Melledy's avatar
Melledy committed
1248
1249
1250
		// Create world
		World world = new World(this);
		world.addPlayer(this);
Asnxthaony's avatar
Asnxthaony committed
1251
1252

		// Multiplayer setting
Melledy's avatar
Melledy committed
1253
1254
		this.setProperty(PlayerProperty.PROP_PLAYER_MP_SETTING_TYPE, this.getMpSetting().getNumber());
		this.setProperty(PlayerProperty.PROP_IS_MP_MODE_AVAILABLE, 1);
Asnxthaony's avatar
Asnxthaony committed
1255

Melledy's avatar
Melledy committed
1256
1257
1258
1259
1260
		// Packets
		session.send(new PacketPlayerDataNotify(this)); // Player data
		session.send(new PacketStoreWeightLimitNotify());
		session.send(new PacketPlayerStoreNotify(this));
		session.send(new PacketAvatarDataNotify(this));
Melledy's avatar
Melledy committed
1261
1262
		session.send(new PacketFinishedParentQuestNotify(this));
		session.send(new PacketQuestListNotify(this));
1263
		session.send(new PacketCodexDataFullNotify(this));
1264
1265
		session.send(new PacketAllWidgetDataNotify(this));
		session.send(new PacketWidgetGadgetAllDataNotify());
1266
1267
		session.send(new PacketPlayerHomeCompInfoNotify(this));
		session.send(new PacketHomeComfortInfoNotify(this));
ImmuState's avatar
ImmuState committed
1268
		session.send(new PacketForgeDataNotify(this));
1269

Yazawazi's avatar
Yazawazi committed
1270
1271
		getTodayMoonCard(); // The timer works at 0:0, some users log in after that, use this method to check if they have received a reward today or not. If not, send the reward.

Melledy's avatar
Melledy committed
1272
		session.send(new PacketPlayerEnterSceneNotify(this)); // Enter game world
1273
		session.send(new PacketPlayerLevelRewardUpdateNotify(rewardedLevels));
Melledy's avatar
Melledy committed
1274
1275
1276
1277
		session.send(new PacketOpenStateUpdateNotify());

		// First notify packets sent
		this.setHasSentAvatarDataNotify(true);
1278
1279
1280
		
		// Set session state
		session.setState(SessionState.ACTIVE);
1281
1282
1283
1284
1285

		// Call join event.
		PlayerJoinEvent event = new PlayerJoinEvent(this); event.call();
		if(event.isCanceled()) // If event is not cancelled, continue.
			session.close();
Melledy's avatar
Melledy committed
1286
	}
Asnxthaony's avatar
Asnxthaony committed
1287

Melledy's avatar
Melledy committed
1288
	public void onLogout() {
1289
1290
1291
		try{
			// stop stamina calculation
			getStaminaManager().stopSustainedStaminaHandler();
1292

1293
			// force to leave the dungeon (inside has a "if")
1294
			this.getServer().getDungeonManager().exitDungeon(this);
Asnxthaony's avatar
Asnxthaony committed
1295

1296
1297
1298
1299
1300
1301
1302
1303
			// Leave world
			if (this.getWorld() != null) {
				this.getWorld().removePlayer(this);
			}

			// Status stuff
			this.getProfile().syncWithCharacter(this);
			this.getProfile().setPlayer(null); // Set offline
Asnxthaony's avatar
Asnxthaony committed
1304

1305
			this.getCoopRequests().clear();
Asnxthaony's avatar
Asnxthaony committed
1306

1307
1308
1309
1310
1311
1312
1313
1314
1315
1316
1317
1318
1319
1320
1321
1322
1323
1324
			// Save to db
			this.save();
			this.getTeamManager().saveAvatars();
			this.getFriendsList().save();

			// Call quit event.
			PlayerQuitEvent event = new PlayerQuitEvent(this); event.call();

			//reset wood
			getDeforestationManager().resetWood();

		}catch (Throwable e){
			e.printStackTrace();
			Grasscutter.getLogger().warn("Player (UID {}) save failure", getUid());
		}finally {
			removeFromServer();
		}
	}
1325

1326
1327
1328
1329
1330
	public void removeFromServer() {
		// Remove from server.
		//Note: DON'T DELETE BY UID,BECAUSE THERE ARE MULTIPLE SAME UID PLAYERS WHEN DUPLICATED LOGIN!
		//so I decide to delete by object rather than uid
		getServer().getPlayers().values().removeIf(player1 -> player1 == this);
Melledy's avatar
Melledy committed
1331
	}
Asnxthaony's avatar
Asnxthaony committed
1332

Melledy's avatar
Melledy committed
1333
	public enum SceneLoadState {
Asnxthaony's avatar
Asnxthaony committed
1334
1335
		NONE(0), LOADING(1), INIT(2), LOADED(3);

Melledy's avatar
Melledy committed
1336
		private final int value;
Asnxthaony's avatar
Asnxthaony committed
1337

Melledy's avatar
Melledy committed
1338
1339
1340
		private SceneLoadState(int value) {
			this.value = value;
		}
Asnxthaony's avatar
Asnxthaony committed
1341

Melledy's avatar
Melledy committed
1342
1343
1344
1345
		public int getValue() {
			return this.value;
		}
	}
1346
1347
1348
1349

	public void setMessageHandler(MessageHandler messageHandler) {
		this.messageHandler = messageHandler;
	}
1350
1351
1352
1353
1354
1355
1356
1357
1358
1359
1360
1361
1362
1363
1364
1365
1366
1367
1368

	private void saveSanityCheckedProperty(PlayerProperty prop, int value) {
		getProperties().put(prop.getId(), value);
	}

	private boolean setPropertyWithSanityCheck(PlayerProperty prop, int value) {
		if (prop == PlayerProperty.PROP_EXP) { // 1001
			if (!(value >= 0)) { return false; }
		} else if (prop == PlayerProperty.PROP_BREAK_LEVEL) { // 1002
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_SATIATION_VAL) { // 1003
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_SATIATION_PENALTY_TIME) { // 1004
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_LEVEL) { // 4001
			if (!(value >= 0 && value <= 90)) { return false; }
		} else if (prop == PlayerProperty.PROP_LAST_CHANGE_AVATAR_TIME) { // 10001
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_MAX_SPRING_VOLUME) { // 10002
1369
			if (!(value >= 0 && value <= SotSManager.GlobalMaximumSpringVolume)) { return false; }
1370
1371
1372
1373
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
1384
1385
		} else if (prop == PlayerProperty.PROP_CUR_SPRING_VOLUME) { // 10003
			int playerMaximumSpringVolume = getProperty(PlayerProperty.PROP_MAX_SPRING_VOLUME);
			if (!(value >= 0 && value <= playerMaximumSpringVolume)) { return false; }
		} else if (prop == PlayerProperty.PROP_IS_SPRING_AUTO_USE) { // 10004
			if (!(value >= 0 && value <= 1)) { return false; }
		} else if (prop == PlayerProperty.PROP_SPRING_AUTO_USE_PERCENT) { // 10005
			if (!(value >= 0 && value <= 100)) { return false; }
		} else if (prop == PlayerProperty.PROP_IS_FLYABLE) { // 10006
			if (!(0 <= value && value <= 1)) { return false; }
		} else if (prop == PlayerProperty.PROP_IS_WEATHER_LOCKED) { // 10007
			if (!(0 <= value && value <= 1)) { return false; }
		} else if (prop == PlayerProperty.PROP_IS_GAME_TIME_LOCKED) { // 10008
			if (!(0 <= value && value <= 1)) { return false; }
		} else if (prop == PlayerProperty.PROP_IS_TRANSFERABLE) { // 10009
			if (!(0 <= value && value <= 1)) { return false; }
		} else if (prop == PlayerProperty.PROP_MAX_STAMINA) { // 10010
gentlespoon's avatar
gentlespoon committed
1386
			if (!(value >= 0 && value <= StaminaManager.GlobalCharacterMaximumStamina)) { return false; }
1387
1388
1389
1390
1391
1392
1393
1394
1395
1396
		} else if (prop == PlayerProperty.PROP_CUR_PERSIST_STAMINA) { // 10011
			int playerMaximumStamina = getProperty(PlayerProperty.PROP_MAX_STAMINA);
			if (!(value >= 0 && value <= playerMaximumStamina)) { return false; }
		} else if (prop == PlayerProperty.PROP_CUR_TEMPORARY_STAMINA) { // 10012
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_PLAYER_LEVEL) { // 10013
			if (!(0 < value && value <= 90)) { return false; }
		} else if (prop == PlayerProperty.PROP_PLAYER_EXP) { // 10014
			if (!(0 <= value)) { return false; }
		} else if (prop == PlayerProperty.PROP_PLAYER_HCOIN) { // 10015
1397
			// see PlayerProperty.PROP_PLAYER_HCOIN comments
1398
1399
1400
1401
1402
1403
1404
1405
1406
1407
1408
1409
1410
1411
1412
1413
1414
1415
1416
1417
1418
1419
1420
1421
1422
1423
1424
1425
1426
1427
1428
1429
1430
1431
1432
1433
1434
1435
1436
1437
1438
1439
1440
1441
1442
1443
1444
1445
1446
1447
		} else if (prop == PlayerProperty.PROP_PLAYER_SCOIN) { // 10016
			// See 10015
		} else if (prop == PlayerProperty.PROP_PLAYER_MP_SETTING_TYPE) { // 10017
			if (!(0 <= value && value <= 2)) { return false; }
		} else if (prop == PlayerProperty.PROP_IS_MP_MODE_AVAILABLE) { // 10018
			if (!(0 <= value && value <= 1)) { return false; }
		} else if (prop == PlayerProperty.PROP_PLAYER_WORLD_LEVEL) { // 10019
			if (!(0 <= value && value <= 8)) { return false; }
		} else if (prop == PlayerProperty.PROP_PLAYER_RESIN) { // 10020
			// Do not set 160 as a cap, because player can have more than 160 when they use fragile resin.
			if (!(0 <= value)) { return false; }
		} else if (prop == PlayerProperty.PROP_PLAYER_WAIT_SUB_HCOIN) { // 10022
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_PLAYER_WAIT_SUB_SCOIN) { // 10023
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_IS_ONLY_MP_WITH_PS_PLAYER) { // 10024
			if (!(0 <= value && value <= 1)) { return false; }
		} else if (prop == PlayerProperty.PROP_PLAYER_MCOIN) { // 10025
			// see 10015
		} else if (prop == PlayerProperty.PROP_PLAYER_WAIT_SUB_MCOIN) { // 10026
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_PLAYER_LEGENDARY_KEY) { // 10027
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_IS_HAS_FIRST_SHARE) { // 10028
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_PLAYER_FORGE_POINT) { // 10029
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_CUR_CLIMATE_METER) { // 10035
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_CUR_CLIMATE_TYPE) { // 10036
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_CUR_CLIMATE_AREA_ID) { // 10037
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_CUR_CLIMATE_AREA_CLIMATE_TYPE) { // 10038
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_PLAYER_WORLD_LEVEL_LIMIT) { // 10039
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_PLAYER_WORLD_LEVEL_ADJUST_CD) { // 10040
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_PLAYER_LEGENDARY_DAILY_TASK_NUM) { // 10041
			// TODO: implement sanity check
		} else if (prop == PlayerProperty.PROP_PLAYER_HOME_COIN) { // 10042
			if (!(0 <= value)) { return false; }
		} else if (prop == PlayerProperty.PROP_PLAYER_WAIT_SUB_HOME_COIN) { // 10043
			// TODO: implement sanity check
		}
		saveSanityCheckedProperty(prop, value);
		return false;
	}

Melledy's avatar
Melledy committed
1448
}