Player.java 44.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;
ImmuState's avatar
ImmuState committed
30
import emu.grasscutter.game.managers.ResinManager;
31
import emu.grasscutter.game.managers.StaminaManager.StaminaManager;
32
import emu.grasscutter.game.managers.SotSManager;
33
import emu.grasscutter.game.managers.EnergyManager.EnergyManager;
34
import emu.grasscutter.game.managers.ForgingManager.ActiveForgeData;
35
import emu.grasscutter.game.managers.ForgingManager.ForgingManager;
Melledy's avatar
Melledy committed
36
import emu.grasscutter.game.props.ActionReason;
37
import emu.grasscutter.game.props.EntityType;
Melledy's avatar
Melledy committed
38
import emu.grasscutter.game.props.PlayerProperty;
39
import emu.grasscutter.game.props.SceneType;
Melledy's avatar
Melledy committed
40
import emu.grasscutter.game.quest.QuestManager;
Kengxxiao's avatar
Kengxxiao committed
41
import emu.grasscutter.game.shop.ShopLimit;
Angda Song's avatar
Angda Song committed
42
import emu.grasscutter.game.managers.MapMarkManager.*;
43
import emu.grasscutter.game.tower.TowerData;
Akka's avatar
Akka committed
44
import emu.grasscutter.game.tower.TowerManager;
Melledy's avatar
Melledy committed
45
46
import emu.grasscutter.game.world.Scene;
import emu.grasscutter.game.world.World;
47
import emu.grasscutter.net.packet.BasePacket;
48
import emu.grasscutter.net.proto.*;
Melledy's avatar
Melledy committed
49
import emu.grasscutter.net.proto.AbilityInvokeEntryOuterClass.AbilityInvokeEntry;
50
import emu.grasscutter.net.proto.AttackResultOuterClass.AttackResult;
Melledy's avatar
Melledy committed
51
import emu.grasscutter.net.proto.CombatInvokeEntryOuterClass.CombatInvokeEntry;
ImmuState's avatar
ImmuState committed
52
import emu.grasscutter.net.proto.GadgetInteractReqOuterClass.GadgetInteractReq;
Melledy's avatar
Melledy committed
53
54
55
56
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
57
import emu.grasscutter.net.proto.ProfilePictureOuterClass.ProfilePicture;
Melledy's avatar
Melledy committed
58
import emu.grasscutter.net.proto.SocialDetailOuterClass.SocialDetail;
59

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

Asnxthaony's avatar
Asnxthaony committed
73
import java.util.*;
74
import java.util.concurrent.LinkedBlockingQueue;
Asnxthaony's avatar
Asnxthaony committed
75

76
77
import static emu.grasscutter.Configuration.*;

KingRainbow44's avatar
KingRainbow44 committed
78
@Entity(value = "players", useDiscriminator = false)
79
public class Player {
Angda Song's avatar
Angda Song committed
80

Melledy's avatar
Melledy committed
81
82
	@Id private int id;
	@Indexed(options = @IndexOptions(unique = true)) private String accountId;
Asnxthaony's avatar
Asnxthaony committed
83

Melledy's avatar
Melledy committed
84
85
86
87
88
89
90
	@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
91
	private PlayerBirthday birthday;
92
	private PlayerCodex codex;
Asnxthaony's avatar
Asnxthaony committed
93

Melledy's avatar
Melledy committed
94
95
96
97
	private Map<Integer, Integer> properties;
	private Set<Integer> nameCardList;
	private Set<Integer> flyCloakList;
	private Set<Integer> costumeList;
98
	private Set<Integer> unlockedForgingBlueprints;
ImmuState's avatar
ImmuState committed
99
	private Set<Integer> unlockedCombines;
100
	private List<ActiveForgeData> activeForges;
Asnxthaony's avatar
Asnxthaony committed
101

102
103
	private Integer widgetId;

104
105
106
	private Set<Integer> realmList;
	private Integer currentRealmId;

Melledy's avatar
Melledy committed
107
108
109
	@Transient private long nextGuid = 0;
	@Transient private int peerId;
	@Transient private World world;
110
	@Transient private Scene scene;
Melledy's avatar
Melledy committed
111
112
113
114
	@Transient private GameSession session;
	@Transient private AvatarStorage avatars;
	@Transient private Inventory inventory;
	@Transient private FriendsList friendsList;
115
	@Transient private MailHandler mailHandler;
116
	@Transient private MessageHandler messageHandler;
Melledy's avatar
Melledy committed
117
	@Transient private AbilityManager abilityManager;
Melledy's avatar
Melledy committed
118
	@Transient private QuestManager questManager;
Melledy's avatar
Melledy committed
119
	
120
	@Transient private SotSManager sotsManager;
121
	@Transient private InsectCaptureManager insectCaptureManager;
122

Melledy's avatar
Melledy committed
123
	private TeamManager teamManager;
Akka's avatar
Akka committed
124

125
126
	@Transient private TowerManager towerManager;
	private TowerData towerData;
Melledy's avatar
Melledy committed
127
128
129
130
	private PlayerGachaInfo gachaInfo;
	private PlayerProfile playerProfile;
	private boolean showAvatar;
	private ArrayList<AvatarProfileData> shownAvatars;
131
	private Set<Integer> rewardedLevels;
Kengxxiao's avatar
Kengxxiao committed
132
	private ArrayList<ShopLimit> shopLimit;
Kinesis's avatar
Kinesis committed
133
	private Map<Long, ExpeditionInfo> expeditionInfo;
Asnxthaony's avatar
Asnxthaony committed
134

Melledy's avatar
Melledy committed
135
136
137
138
	private int sceneId;
	private int regionId;
	private int mainCharacterId;
	private boolean godmode;
Asnxthaony's avatar
Asnxthaony committed
139

140
	private boolean stamina;
Yazawazi's avatar
Yazawazi committed
141
142
143
144
145
	private boolean moonCard;
	private Date moonCardStartTime;
	private int moonCardDuration;
	private Set<Date> moonCardGetTimes;

Yazawazi's avatar
Yazawazi committed
146
147
148
	private List<Integer> showAvatarList;
	private boolean showAvatars;

Melledy's avatar
Melledy committed
149
150
151
152
	@Transient private boolean paused;
	@Transient private int enterSceneToken;
	@Transient private SceneLoadState sceneState;
	@Transient private boolean hasSentAvatarDataNotify;
Melledy's avatar
Melledy committed
153
	@Transient private long nextSendPlayerLocTime = 0;
Asnxthaony's avatar
Asnxthaony committed
154

Melledy's avatar
Melledy committed
155
	@Transient private final Int2ObjectMap<CoopRequest> coopRequests;
156
	@Transient private final Queue<AttackResult> attackResults;
Melledy's avatar
Melledy committed
157
158
	@Transient private final InvokeHandler<CombatInvokeEntry> combatInvokeHandler;
	@Transient private final InvokeHandler<AbilityInvokeEntry> abilityInvokeHandler;
159
	@Transient private final InvokeHandler<AbilityInvokeEntry> clientAbilityInitFinishHandler;
Asnxthaony's avatar
Asnxthaony committed
160

gentlespoon's avatar
gentlespoon committed
161
	@Transient private MapMarksManager mapMarksManager;
162
	@Transient private StaminaManager staminaManager;
163
	@Transient private EnergyManager energyManager;
ImmuState's avatar
ImmuState committed
164
	@Transient private ResinManager resinManager;
165
	@Transient private ForgingManager forgingManager;
166
	@Transient private DeforestationManager deforestationManager;
167

168
	private long springLastUsed;
gentlespoon's avatar
gentlespoon committed
169
	private HashMap<String, MapMark> mapMarks;
ImmuState's avatar
ImmuState committed
170
	private int nextResinRefresh;
171

Asnxthaony's avatar
Asnxthaony committed
172
173
	@Deprecated
	@SuppressWarnings({"rawtypes", "unchecked"}) // Morphia only!
174
	public Player() {
Melledy's avatar
Melledy committed
175
176
177
		this.inventory = new Inventory(this);
		this.avatars = new AvatarStorage(this);
		this.friendsList = new FriendsList(this);
178
		this.mailHandler = new MailHandler(this);
179
		this.towerManager = new TowerManager(this);
Melledy's avatar
Melledy committed
180
		this.abilityManager = new AbilityManager(this);
181
		this.deforestationManager = new DeforestationManager(this);
182
		this.insectCaptureManager = new InsectCaptureManager(this);
183

Melledy's avatar
Melledy committed
184
		this.setQuestManager(new QuestManager(this));
Melledy's avatar
Melledy committed
185
186
187
188
189
190
191
192
193
		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
194

195
196
197
198
		this.gachaInfo = new PlayerGachaInfo();
		this.nameCardList = new HashSet<>();
		this.flyCloakList = new HashSet<>();
		this.costumeList = new HashSet<>();
199
		this.towerData = new TowerData();
200
		this.unlockedForgingBlueprints = new HashSet<>();
ImmuState's avatar
ImmuState committed
201
		this.unlockedCombines = new HashSet<>();
202
		this.activeForges = new ArrayList<>();
Asnxthaony's avatar
Asnxthaony committed
203

Melledy's avatar
Melledy committed
204
205
206
		this.setSceneId(3);
		this.setRegionId(1);
		this.sceneState = SceneLoadState.NONE;
Asnxthaony's avatar
Asnxthaony committed
207

208
		this.attackResults = new LinkedBlockingQueue<>();
Melledy's avatar
Melledy committed
209
210
211
		this.coopRequests = new Int2ObjectOpenHashMap<>();
		this.combatInvokeHandler = new InvokeHandler(PacketCombatInvocationsNotify.class);
		this.abilityInvokeHandler = new InvokeHandler(PacketAbilityInvocationsNotify.class);
212
		this.clientAbilityInitFinishHandler = new InvokeHandler(PacketClientAbilityInitFinishNotify.class);
Miyucchi's avatar
Miyucchi committed
213
214

		this.birthday = new PlayerBirthday();
215
		this.rewardedLevels = new HashSet<>();
Yazawazi's avatar
Yazawazi committed
216
		this.moonCardGetTimes = new HashSet<>();
217
		this.codex = new PlayerCodex(this);
Kengxxiao's avatar
Kengxxiao committed
218
219

		this.shopLimit = new ArrayList<>();
Kinesis's avatar
Kinesis committed
220
		this.expeditionInfo = new HashMap<>();
221
		this.messageHandler = null;
gentlespoon's avatar
gentlespoon committed
222
		this.mapMarksManager = new MapMarksManager(this);
223
		this.staminaManager = new StaminaManager(this);
224
		this.sotsManager = new SotSManager(this);
225
		this.energyManager = new EnergyManager(this);
ImmuState's avatar
ImmuState committed
226
		this.resinManager = new ResinManager(this);
227
		this.forgingManager = new ForgingManager(this);
Melledy's avatar
Melledy committed
228
	}
Asnxthaony's avatar
Asnxthaony committed
229

Melledy's avatar
Melledy committed
230
	// On player creation
231
	public Player(GameSession session) {
Melledy's avatar
Melledy committed
232
233
234
235
236
237
238
		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
239
		this.birthday = new PlayerBirthday();
240
		this.codex = new PlayerCodex(this);
Melledy's avatar
Melledy committed
241
242
243
244
245
246
247
248
249
250
		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);
251
		this.getPos().set(GameConstants.START_POSITION);
Melledy's avatar
Melledy committed
252
		this.getRotation().set(0, 307, 0);
253
		this.messageHandler = null;
gentlespoon's avatar
gentlespoon committed
254
		this.mapMarksManager = new MapMarksManager(this);
255
		this.staminaManager = new StaminaManager(this);
256
		this.sotsManager = new SotSManager(this);
257
		this.energyManager = new EnergyManager(this);
ImmuState's avatar
ImmuState committed
258
		this.resinManager = new ResinManager(this);
259
		this.deforestationManager = new DeforestationManager(this);
260
		this.forgingManager = new ForgingManager(this);
Melledy's avatar
Melledy committed
261
262
	}

263
	public int getUid() {
Melledy's avatar
Melledy committed
264
265
266
		return id;
	}

267
	public void setUid(int id) {
Melledy's avatar
Melledy committed
268
269
		this.id = id;
	}
Asnxthaony's avatar
Asnxthaony committed
270

271
	public long getNextGameGuid() {
Melledy's avatar
Melledy committed
272
		long nextId = ++this.nextGuid;
273
		return ((long) this.getUid() << 32) + nextId;
Melledy's avatar
Melledy committed
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
	}

	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
291

Melledy's avatar
Melledy committed
292
293
294
	public boolean isOnline() {
		return this.getSession() != null && this.getSession().isActive();
	}
Asnxthaony's avatar
Asnxthaony committed
295

Melledy's avatar
Melledy committed
296
297
298
	public GameServer getServer() {
		return this.getSession().getServer();
	}
Asnxthaony's avatar
Asnxthaony committed
299

Melledy's avatar
Melledy committed
300
301
302
	public synchronized World getWorld() {
		return this.world;
	}
Asnxthaony's avatar
Asnxthaony committed
303

Melledy's avatar
Melledy committed
304
305
306
	public synchronized void setWorld(World world) {
		this.world = world;
	}
Asnxthaony's avatar
Asnxthaony committed
307

308
	public synchronized Scene getScene() {
309
310
311
		return scene;
	}

312
	public synchronized void setScene(Scene scene) {
313
314
315
		this.scene = scene;
	}

Melledy's avatar
Melledy committed
316
317
318
319
320
321
322
323
324
325
326
327
	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
328

Melledy's avatar
Melledy committed
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
	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();
	}

347
348
349
350
351
352
353
354
	public Integer getWidgetId() {
		return widgetId;
	}

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

355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
	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
380
381
382
	public Position getPos() {
		return pos;
	}
Asnxthaony's avatar
Asnxthaony committed
383

Melledy's avatar
Melledy committed
384
385
386
	public Position getRotation() {
		return rotation;
	}
Asnxthaony's avatar
Asnxthaony committed
387

Melledy's avatar
Melledy committed
388
389
390
	public int getLevel() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_LEVEL);
	}
Asnxthaony's avatar
Asnxthaony committed
391

Melledy's avatar
Melledy committed
392
393
394
	public int getExp() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_EXP);
	}
Asnxthaony's avatar
Asnxthaony committed
395

Melledy's avatar
Melledy committed
396
397
398
	public int getWorldLevel() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_WORLD_LEVEL);
	}
399
400
401
402
403
	
	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
404

Melledy's avatar
Melledy committed
405
406
407
	public int getPrimogems() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_HCOIN);
	}
Asnxthaony's avatar
Asnxthaony committed
408

Melledy's avatar
Melledy committed
409
410
411
412
	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
413

Melledy's avatar
Melledy committed
414
415
416
	public int getMora() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_SCOIN);
	}
Asnxthaony's avatar
Asnxthaony committed
417

Melledy's avatar
Melledy committed
418
419
420
421
	public void setMora(int mora) {
		this.setProperty(PlayerProperty.PROP_PLAYER_SCOIN, mora);
		this.sendPacket(new PacketPlayerPropNotify(this, PlayerProperty.PROP_PLAYER_SCOIN));
	}
422
423
424
425
426
427
428
429
430
	
	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
431

Melledy's avatar
Melledy committed
432
	private int getExpRequired(int level) {
433
		PlayerLevelData levelData = GameData.getPlayerLevelDataMap().get(level);
Asnxthaony's avatar
Asnxthaony committed
434
		return levelData != null ? levelData.getExp() : 0;
Melledy's avatar
Melledy committed
435
	}
Asnxthaony's avatar
Asnxthaony committed
436

Melledy's avatar
Melledy committed
437
	private float getExpModifier() {
438
		return GAME_OPTIONS.rates.adventureExp;
Melledy's avatar
Melledy committed
439
	}
Asnxthaony's avatar
Asnxthaony committed
440

Melledy's avatar
Melledy committed
441
442
443
444
	// Affected by exp rate
	public void earnExp(int exp) {
		addExpDirectly((int) (exp * getExpModifier()));
	}
Asnxthaony's avatar
Asnxthaony committed
445

Melledy's avatar
Melledy committed
446
447
448
449
450
451
	// 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
452

Melledy's avatar
Melledy committed
453
		exp += gain;
Asnxthaony's avatar
Asnxthaony committed
454

Melledy's avatar
Melledy committed
455
456
457
458
459
460
		while (exp >= reqExp && reqExp > 0) {
			exp -= reqExp;
			level += 1;
			reqExp = getExpRequired(level);
			hasLeveledUp = true;
		}
Asnxthaony's avatar
Asnxthaony committed
461

Melledy's avatar
Melledy committed
462
463
464
465
466
467
468
469
		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
470

Melledy's avatar
Melledy committed
471
472
		// Set exp
		this.setProperty(PlayerProperty.PROP_PLAYER_EXP, exp);
Asnxthaony's avatar
Asnxthaony committed
473

Melledy's avatar
Melledy committed
474
475
476
		// Update player with packet
		this.sendPacket(new PacketPlayerPropNotify(this, PlayerProperty.PROP_PLAYER_EXP));
	}
Asnxthaony's avatar
Asnxthaony committed
477

Melledy's avatar
Melledy committed
478
479
480
	private void updateProfile() {
		getProfile().syncWithCharacter(this);
	}
Asnxthaony's avatar
Asnxthaony committed
481

Melledy's avatar
Melledy committed
482
483
484
485
486
487
488
	public boolean isFirstLoginEnterScene() {
		return !this.hasSentAvatarDataNotify;
	}

	public TeamManager getTeamManager() {
		return this.teamManager;
	}
489
	
Akka's avatar
Akka committed
490
491
492
	public TowerManager getTowerManager() {
		return towerManager;
	}
493
494
	
	public TowerData getTowerData() {
495
496
497
498
		if(towerData==null){
			// because of mistake, null may be saved as storage at some machine, this if can be removed in future
			towerData = new TowerData();
		}
499
500
501
		return towerData;
	}
	
Melledy's avatar
Melledy committed
502
503
504
505
506
507
508
509
	public QuestManager getQuestManager() {
		return questManager;
	}

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

Melledy's avatar
Melledy committed
510
511
512
513
514
515
516
517
518
519
520
	public PlayerGachaInfo getGachaInfo() {
		return gachaInfo;
	}

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

521
522
	// 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
523
524
525
	public Map<Integer, Integer> getProperties() {
		return properties;
	}
Asnxthaony's avatar
Asnxthaony committed
526

527
528
	public boolean setProperty(PlayerProperty prop, int value) {
		return setPropertyWithSanityCheck(prop, value);
Melledy's avatar
Melledy committed
529
	}
Asnxthaony's avatar
Asnxthaony committed
530

Melledy's avatar
Melledy committed
531
532
533
534
535
536
537
	public int getProperty(PlayerProperty prop) {
		return getProperties().get(prop.getId());
	}

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

Melledy's avatar
Melledy committed
539
540
541
	public Set<Integer> getCostumeList() {
		return costumeList;
	}
Asnxthaony's avatar
Asnxthaony committed
542

Melledy's avatar
Melledy committed
543
544
545
546
	public Set<Integer> getNameCardList() {
		return this.nameCardList;
	}

547
	public Set<Integer> getUnlockedForgingBlueprints() {
548
549
550
		return this.unlockedForgingBlueprints;
	}

ImmuState's avatar
ImmuState committed
551
552
553
554
	public Set<Integer> getUnlockedCombines() {
		return this.unlockedCombines;
	}

555
	public List<ActiveForgeData> getActiveForges() {
556
		return this.activeForges;
557
558
	}

Melledy's avatar
Melledy committed
559
	public MpSettingType getMpSetting() {
560
		return MpSettingType.MP_SETTING_TYPE_ENTER_AFTER_APPLY; // TEMP
Melledy's avatar
Melledy committed
561
	}
562
563
564
565
	
	public Queue<AttackResult> getAttackResults() {
		return this.attackResults;
	}
Asnxthaony's avatar
Asnxthaony committed
566

Melledy's avatar
Melledy committed
567
568
569
570
571
572
573
	public synchronized Int2ObjectMap<CoopRequest> getCoopRequests() {
		return coopRequests;
	}

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

Melledy's avatar
Melledy committed
575
576
577
578
	public InvokeHandler<AbilityInvokeEntry> getAbilityInvokeHandler() {
		return this.abilityInvokeHandler;
	}

579
580
581
582
	public InvokeHandler<AbilityInvokeEntry> getClientAbilityInitFinishHandler() {
		return clientAbilityInitFinishHandler;
	}

Melledy's avatar
Melledy committed
583
584
585
	public AvatarStorage getAvatars() {
		return avatars;
	}
Asnxthaony's avatar
Asnxthaony committed
586

Melledy's avatar
Melledy committed
587
588
589
	public Inventory getInventory() {
		return inventory;
	}
Asnxthaony's avatar
Asnxthaony committed
590

Melledy's avatar
Melledy committed
591
592
593
594
	public FriendsList getFriendsList() {
		return this.friendsList;
	}

595
596
597
598
	public MailHandler getMailHandler() {
		return mailHandler;
	}

Melledy's avatar
Melledy committed
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
	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
647

Melledy's avatar
Melledy committed
648
649
650
651
652
653
654
		if (newPauseState && !oldPauseState) {
			this.onPause();
		} else if (oldPauseState && !newPauseState) {
			this.onUnpause();
		}
	}

655
656
657
658
659
660
661
662
	public long getSpringLastUsed() {
		return springLastUsed;
	}

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

ImmuState's avatar
ImmuState committed
663
664
665
666
667
668
669
670
	public int getNextResinRefresh() {
		return nextResinRefresh;
	}

	public void setNextResinRefresh(int value) {
		this.nextResinRefresh = value;
	}

Melledy's avatar
Melledy committed
671
672
673
674
675
676
677
	public SceneLoadState getSceneLoadState() {
		return sceneState;
	}

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

Melledy's avatar
Melledy committed
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
	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
698

Yazawazi's avatar
Yazawazi committed
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
	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
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
	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
756
		Date now = DateHelper.onlyYearMonthDay(new Date());
Yazawazi's avatar
Yazawazi committed
757
758
759
760
		return (int) ((theLastDay.getTime() - now.getTime()) / (24 * 60 * 60 * 1000)); // By copilot 
	}

	public void rechargeMoonCard() {
761
		inventory.addItem(new GameItem(203, 300));
Yazawazi's avatar
Yazawazi committed
762
763
764
		if (!moonCard) {
			moonCard = true;
			Date now = new Date();
Yazawazi's avatar
Yazawazi committed
765
			moonCardStartTime = DateHelper.onlyYearMonthDay(now);
Yazawazi's avatar
Yazawazi committed
766
767
768
769
770
771
772
773
774
775
776
777
778
			moonCardDuration = 30;
		} else {
			moonCardDuration += 30;
		}
		if (!moonCardGetTimes.contains(moonCardStartTime)) {
			moonCardGetTimes.add(moonCardStartTime);
		}
	}

	public void getTodayMoonCard() {
		if (!moonCard) {
			return;
		}
Yazawazi's avatar
Yazawazi committed
779
		Date now = DateHelper.onlyYearMonthDay(new Date());
Yazawazi's avatar
Yazawazi committed
780
781
782
783
784
785
786
787
788
789
790
791
792
793
		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);
794
795
		GameItem item = new GameItem(201, 90);
		getInventory().addItem(item, ActionReason.BlessingRedeemReward);
Yazawazi's avatar
Yazawazi committed
796
797
798
		session.send(new PacketCardProductRewardNotify(getMoonCardRemainDays()));
	}

Kinesis's avatar
Kinesis committed
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
	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
820
821
822
823
	public List<ShopLimit> getShopLimit() {
		return shopLimit;
	}

Kengxxiao's avatar
Kengxxiao committed
824
825
826
827
828
	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
829
830
	}

Kengxxiao's avatar
Kengxxiao committed
831
832
833
834
835
836
837
	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
838
839
840
			ShopLimit sl = new ShopLimit();
			sl.setShopGoodId(goodsId);
			sl.setHasBought(boughtCount);
Kengxxiao's avatar
Kengxxiao committed
841
842
843
			sl.setHasBoughtInPeriod(boughtCount);
			sl.setNextRefreshTime(nextRefreshTime);
			getShopLimit().add(sl);
Kengxxiao's avatar
Kengxxiao committed
844
845
846
		}
		this.save();
	}
847
848
849
850
851
852
853
854
	public boolean getStamina() {
		// Get Stamina
		return stamina;
	}
	public void setStamina(boolean stamina) {
		// Set Stamina
		this.stamina = stamina;
	}
KingRainbow44's avatar
KingRainbow44 committed
855
	public boolean inGodmode() {
Melledy's avatar
Melledy committed
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
		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
871
	public void addAvatar(Avatar avatar, boolean addToCurrentTeam) {
Melledy's avatar
Melledy committed
872
		boolean result = getAvatars().addAvatar(avatar);
Asnxthaony's avatar
Asnxthaony committed
873

Melledy's avatar
Melledy committed
874
875
876
		if (result) {
			// Add starting weapon
			getAvatars().addStartingWeapon(avatar);
Asnxthaony's avatar
Asnxthaony committed
877

Melledy's avatar
Melledy committed
878
879
880
881
			// Done
			if (hasSentAvatarDataNotify()) {
				// Recalc stats
				avatar.recalcStats();
ProxyismGH's avatar
ProxyismGH committed
882
883
884
885
886
887
				// 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
888
889
890
891
892
			}
		} else {
			// Failed adding avatar
		}
	}
Asnxthaony's avatar
Asnxthaony committed
893

ProxyismGH's avatar
ProxyismGH committed
894
895
896
897
	public void addAvatar(Avatar avatar) {
		addAvatar(avatar, true);
	}

Melledy's avatar
Melledy committed
898
899
900
901
	public void addFlycloak(int flycloakId) {
		this.getFlyCloakList().add(flycloakId);
		this.sendPacket(new PacketAvatarGainFlycloakNotify(flycloakId));
	}
Asnxthaony's avatar
Asnxthaony committed
902

Melledy's avatar
Melledy committed
903
904
905
906
	public void addCostume(int costumeId) {
		this.getCostumeList().add(costumeId);
		this.sendPacket(new PacketAvatarGainCostumeNotify(costumeId));
	}
Asnxthaony's avatar
Asnxthaony committed
907

Melledy's avatar
Melledy committed
908
909
910
911
	public void addNameCard(int nameCardId) {
		this.getNameCardList().add(nameCardId);
		this.sendPacket(new PacketUnlockNameCardNotify(nameCardId));
	}
Asnxthaony's avatar
Asnxthaony committed
912

Melledy's avatar
Melledy committed
913
914
915
916
	public void setNameCard(int nameCardId) {
		if (!this.getNameCardList().contains(nameCardId)) {
			return;
		}
Asnxthaony's avatar
Asnxthaony committed
917

Melledy's avatar
Melledy committed
918
		this.setNameCardId(nameCardId);
Asnxthaony's avatar
Asnxthaony committed
919

Melledy's avatar
Melledy committed
920
921
		this.sendPacket(new PacketSetNameCardRsp(nameCardId));
	}
Asnxthaony's avatar
Asnxthaony committed
922

Melledy's avatar
Melledy committed
923
	public void dropMessage(Object message) {
924
925
926
927
		if (this.messageHandler != null) {
			this.messageHandler.append(message.toString());
			return;
		}
928
		this.sendPacket(new PacketPrivateChatNotify(GameConstants.SERVER_CONSOLE_UID, getUid(), message.toString()));
Melledy's avatar
Melledy committed
929
	}
Melledy's avatar
Melledy committed
930
931
932

	/**
	 * Sends a message to another player.
Asnxthaony's avatar
Asnxthaony committed
933
934
	 *
	 * @param sender  The sender of the message.
Melledy's avatar
Melledy committed
935
936
	 * @param message The message to send.
	 */
937
	public void sendMessage(Player sender, Object message) {
938
		this.sendPacket(new PacketPrivateChatNotify(sender.getUid(), this.getUid(), message.toString()));
Melledy's avatar
Melledy committed
939
	}
Asnxthaony's avatar
Asnxthaony committed
940

Benjamin Elsdon's avatar
Benjamin Elsdon committed
941
	// ---------------------MAIL------------------------
942

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

Benjamin Elsdon's avatar
Benjamin Elsdon committed
945
	public void sendMail(Mail message) {
946
		this.getMailHandler().sendMail(message);
947
	}
Benjamin Elsdon's avatar
Benjamin Elsdon committed
948
949

	public boolean deleteMail(int mailId) {
950
		return this.getMailHandler().deleteMail(mailId);
Benjamin Elsdon's avatar
Benjamin Elsdon committed
951
952
	}

953
954
	public Mail getMail(int index) { return this.getMailHandler().getMailById(index); }
	
955
	public int getMailId(Mail message) {
956
		return this.getMailHandler().getMailIndex(message);
Benjamin Elsdon's avatar
Benjamin Elsdon committed
957
958
	}

959
	public boolean replaceMailByIndex(int index, Mail message) {
960
		return this.getMailHandler().replaceMailByIndex(index, message);
Benjamin Elsdon's avatar
Benjamin Elsdon committed
961
	}
962
	
ImmuState's avatar
ImmuState committed
963
	public void interactWith(int gadgetEntityId, GadgetInteractReq request) {
964
		GameEntity entity = getScene().getEntityById(gadgetEntityId);
Melledy's avatar
Melledy committed
965
966
967
		if (entity == null) {
			return;
		}
Asnxthaony's avatar
Asnxthaony committed
968

Melledy's avatar
Melledy committed
969
		// Handle
970
		if (entity instanceof EntityItem drop) {
Melledy's avatar
Melledy committed
971
			// Pick item
Kengxxiao's avatar
Kengxxiao committed
972
973
974
			if (!drop.isShare()) // check drop owner to avoid someone picked up item in others' world
			{
				int dropOwner = (int)(drop.getGuid() >> 32);
975
				if (dropOwner != getUid()) {
Kengxxiao's avatar
Kengxxiao committed
976
					return;
977
				}
Kengxxiao's avatar
Kengxxiao committed
978
979
			}
			entity.getScene().removeEntity(entity);
980
			GameItem item = new GameItem(drop.getItemData(), drop.getCount());
Melledy's avatar
Melledy committed
981
			// Add to inventory
982
			boolean success = getInventory().addItem(item, ActionReason.SubfieldDrop);
Melledy's avatar
Melledy committed
983
			if (success) {
984
				if (!drop.isShare()) { // not shared drop
985
					this.sendPacket(new PacketGadgetInteractRsp(drop, InteractType.INTERACT_TYPE_PICK_ITEM));
986
				}else{
987
					this.getScene().broadcastPacket(new PacketGadgetInteractRsp(drop, InteractType.INTERACT_TYPE_PICK_ITEM));
988
				}
Melledy's avatar
Melledy committed
989
			}
990
		} else if (entity instanceof EntityGadget gadget) {
991
992
			if (gadget.getGadgetData().getType() == EntityType.RewardStatue) {
				if (scene.getChallenge() != null) {
ImmuState's avatar
ImmuState committed
993
					scene.getChallenge().getStatueDrops(this, request);
994
				}
995
				this.sendPacket(new PacketGadgetInteractRsp(gadget, InteractType.INTERACT_TYPE_OPEN_STATUE));
996
			}
997
998
999
1000
		} 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
1001
1002
1003
		} else {
			// Delete directly
			entity.getScene().removeEntity(entity);
Melledy's avatar
Melledy committed
1004
1005
		}
	}
Asnxthaony's avatar
Asnxthaony committed
1006

Melledy's avatar
Melledy committed
1007
	public void onPause() {
1008
		getStaminaManager().stopSustainedStaminaHandler();
Melledy's avatar
Melledy committed
1009
	}
Asnxthaony's avatar
Asnxthaony committed
1010

Melledy's avatar
Melledy committed
1011
	public void onUnpause() {
1012
		getStaminaManager().startSustainedStaminaHandler();
Melledy's avatar
Melledy committed
1013
	}
Asnxthaony's avatar
Asnxthaony committed
1014

1015
	public void sendPacket(BasePacket packet) {
Melledy's avatar
Melledy committed
1016
		this.getSession().send(packet);
Melledy's avatar
Melledy committed
1017
	}
Asnxthaony's avatar
Asnxthaony committed
1018

Melledy's avatar
Melledy committed
1019
1020
	public OnlinePlayerInfo getOnlinePlayerInfo() {
		OnlinePlayerInfo.Builder onlineInfo = OnlinePlayerInfo.newBuilder()
Asnxthaony's avatar
Asnxthaony committed
1021
1022
1023
1024
1025
1026
				.setUid(this.getUid())
				.setNickname(this.getNickname())
				.setPlayerLevel(this.getLevel())
				.setMpSettingType(this.getMpSetting())
				.setNameCardId(this.getNameCardId())
				.setSignature(this.getSignature())
Melledy's avatar
Melledy committed
1027
				.setProfilePicture(ProfilePicture.newBuilder().setAvatarId(this.getHeadImage()));
Asnxthaony's avatar
Asnxthaony committed
1028

Melledy's avatar
Melledy committed
1029
		if (this.getWorld() != null) {
Melledy's avatar
Melledy committed
1030
			onlineInfo.setCurPlayerNumInWorld(getWorld().getPlayerCount());
Melledy's avatar
Melledy committed
1031
1032
1033
		} else {
			onlineInfo.setCurPlayerNumInWorld(1);
		}
Asnxthaony's avatar
Asnxthaony committed
1034

Melledy's avatar
Melledy committed
1035
1036
1037
		return onlineInfo.build();
	}

Asnxthaony's avatar
Asnxthaony committed
1038
	public PlayerBirthday getBirthday() {
Miyucchi's avatar
Miyucchi committed
1039
1040
1041
1042
1043
1044
1045
1046
		return this.birthday;
	}

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

Asnxthaony's avatar
Asnxthaony committed
1047
1048
1049
1050
	public boolean hasBirthday() {
		return this.birthday.getDay() > 0;
	}

1051
1052
	public PlayerCodex getCodex(){ return this.codex; }

1053
1054
1055
1056
1057
1058
1059
1060
	public Set<Integer> getRewardedLevels() {
		return rewardedLevels;
	}

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

Melledy's avatar
Melledy committed
1061
	public SocialDetail.Builder getSocialDetail() {
Yazawazi's avatar
Yazawazi committed
1062
		List<SocialShowAvatarInfoOuterClass.SocialShowAvatarInfo> socialShowAvatarInfoList = new ArrayList<>();
Yazawazi's avatar
Yazawazi committed
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
		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 {
1077
1078
			List<Integer> showAvatarList = DatabaseHelper.getPlayerByUid(id).getShowAvatarList();
			AvatarStorage avatars = DatabaseHelper.getPlayerByUid(id).getAvatars();
Yazawazi's avatar
Yazawazi committed
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
			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
1092
1093
		}

Asnxthaony's avatar
Asnxthaony committed
1094
1095
		SocialDetail.Builder social = SocialDetail.newBuilder()
				.setUid(this.getUid())
Yazawazi's avatar
Yazawazi committed
1096
				.setProfilePicture(ProfilePicture.newBuilder().setAvatarId(this.getHeadImage()))
Asnxthaony's avatar
Asnxthaony committed
1097
1098
1099
1100
1101
1102
				.setNickname(this.getNickname())
				.setSignature(this.getSignature())
				.setLevel(this.getLevel())
				.setBirthday(this.getBirthday().getFilledProtoWhenNotEmpty())
				.setWorldLevel(this.getWorldLevel())
				.setNameCardId(this.getNameCardId())
Yazawazi's avatar
Yazawazi committed
1103
1104
				.setIsShowAvatar(this.isShowAvatars())
				.addAllShowAvatarInfoList(socialShowAvatarInfoList)
Asnxthaony's avatar
Asnxthaony committed
1105
				.setFinishAchievementNum(0);
Melledy's avatar
Melledy committed
1106
1107
		return social;
	}
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117

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

		Player player;
		boolean shouldRecalc;
		if (this.isOnline()) {
			player = this;
			shouldRecalc = false;
		} else {
1118
			player = DatabaseHelper.getPlayerByUid(id);
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
			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;
	}
1137
1138
1139
	
	public PlayerWorldLocationInfoOuterClass.PlayerWorldLocationInfo getWorldPlayerLocationInfo() {
		return PlayerWorldLocationInfoOuterClass.PlayerWorldLocationInfo.newBuilder()
Asnxthaony's avatar
Asnxthaony committed
1140
1141
1142
				.setSceneId(this.getSceneId())
				.setPlayerLoc(this.getPlayerLocationInfo())
				.build();
Melledy's avatar
Melledy committed
1143
	}
Asnxthaony's avatar
Asnxthaony committed
1144

Melledy's avatar
Melledy committed
1145
	public PlayerLocationInfo getPlayerLocationInfo() {
Asnxthaony's avatar
Asnxthaony committed
1146
1147
1148
1149
1150
		return PlayerLocationInfo.newBuilder()
				.setUid(this.getUid())
				.setPos(this.getPos().toProto())
				.setRot(this.getRotation().toProto())
				.build();
Melledy's avatar
Melledy committed
1151
	}
Asnxthaony's avatar
Asnxthaony committed
1152

Angda Song's avatar
Angda Song committed
1153
1154
1155
1156
	public MapMarksManager getMapMarksManager() {
		return mapMarksManager;
	}

1157
	public StaminaManager getStaminaManager() { return staminaManager; }
1158

1159
1160
	public SotSManager getSotSManager() { return sotsManager; }

1161
1162
1163
1164
	public EnergyManager getEnergyManager() {
		return this.energyManager;
	}

ImmuState's avatar
ImmuState committed
1165
1166
1167
1168
	public ResinManager getResinManager() {
		return this.resinManager;
	}

1169
1170
1171
1172
	public ForgingManager getForgingManager() {
		return this.forgingManager;
	}

Melledy's avatar
Melledy committed
1173
1174
1175
1176
	public AbilityManager getAbilityManager() {
		return abilityManager;
	}

1177
1178
1179
1180
	public DeforestationManager getDeforestationManager() {
		return deforestationManager;
	}

gentlespoon's avatar
gentlespoon committed
1181
1182
1183
1184
	public HashMap<String, MapMark> getMapMarks() { return mapMarks; }

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

Melledy's avatar
Melledy committed
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
	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()) {
1196
1197
1198
1199
				req.getRequester().sendPacket(new PacketPlayerApplyEnterMpResultNotify(
						this,
						false,
						PlayerApplyEnterMpResultNotifyOuterClass.PlayerApplyEnterMpResultNotify.Reason.REASON_SYSTEM_JUDGE));
Melledy's avatar
Melledy committed
1200
1201
1202
1203
1204
				it.remove();
			}
		}
		// Ping
		if (this.getWorld() != null) {
Melledy's avatar
Melledy committed
1205
1206
			// RTT notify - very important to send this often
			this.sendPacket(new PacketWorldPlayerRTTNotify(this.getWorld()));
Asnxthaony's avatar
Asnxthaony committed
1207

Melledy's avatar
Melledy committed
1208
1209
1210
1211
1212
1213
1214
			// 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
1215
		}
Kinesis's avatar
Kinesis committed
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
		// 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));
		}
1232
1233
1234

		// Send updated forge queue data, if necessary.
		this.getForgingManager().sendPlayerForgingUpdate();
ImmuState's avatar
ImmuState committed
1235
1236
1237

		// Recharge resin.
		this.getResinManager().rechargeResin();
1238
1239
1240
	}


Asnxthaony's avatar
Asnxthaony committed
1241

1242

Melledy's avatar
Melledy committed
1243
1244
1245
	public void resetSendPlayerLocTime() {
		this.nextSendPlayerLocTime = System.currentTimeMillis() + 5000;
	}
Melledy's avatar
Melledy committed
1246
1247
1248

	@PostLoad
	private void onLoad() {
1249
		this.getCodex().setPlayer(this);
Melledy's avatar
Melledy committed
1250
		this.getTeamManager().setPlayer(this);
1251
		this.getTowerManager().setPlayer(this);
Melledy's avatar
Melledy committed
1252
	}
Asnxthaony's avatar
Asnxthaony committed
1253

Melledy's avatar
Melledy committed
1254
1255
1256
	public void save() {
		DatabaseHelper.savePlayer(this);
	}
1257
1258
1259
	
	// Called from tokenrsp
	public void loadFromDatabase() {
Melledy's avatar
Melledy committed
1260
		// Make sure these exist
1261
1262
1263
		if (this.getTowerManager() == null) {
			this.towerManager = new TowerManager(this);
		}
Melledy's avatar
Melledy committed
1264
1265
		if (this.getTeamManager() == null) {
			this.teamManager = new TeamManager(this);
Asnxthaony's avatar
Asnxthaony committed
1266
		}
1267
1268
1269
		if (this.getCodex() == null) {
			this.codex = new PlayerCodex(this);
		}
Asnxthaony's avatar
Asnxthaony committed
1270
		if (this.getProfile().getUid() == 0) {
1271
			this.getProfile().syncWithCharacter(this);
Melledy's avatar
Melledy committed
1272
		}
1273
1274
		//Make sure towerManager's player is online player
		this.getTowerManager().setPlayer(this);
Melledy's avatar
Melledy committed
1275
1276
1277
1278
		// Load from db
		this.getAvatars().loadFromDatabase();
		this.getInventory().loadFromDatabase();
		this.getAvatars().postLoad();
Asnxthaony's avatar
Asnxthaony committed
1279

Melledy's avatar
Melledy committed
1280
		this.getFriendsList().loadFromDatabase();
1281
		this.getMailHandler().loadFromDatabase();
Melledy's avatar
Melledy committed
1282
		this.getQuestManager().loadFromDatabase();
1283

1284
1285
1286
	}

	public void onLogin() {
Melledy's avatar
Melledy committed
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
1298
1299
1300
		// 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();
			}
			getQuestManager().addQuest(35101);
			
			this.setSceneId(3);
			this.getPos().set(GameConstants.START_POSITION);
		}
		*/
		
Melledy's avatar
Melledy committed
1301
1302
1303
		// Create world
		World world = new World(this);
		world.addPlayer(this);
Asnxthaony's avatar
Asnxthaony committed
1304
1305

		// Multiplayer setting
Melledy's avatar
Melledy committed
1306
1307
		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
1308

Melledy's avatar
Melledy committed
1309
1310
1311
1312
1313
		// 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
1314
1315
		session.send(new PacketFinishedParentQuestNotify(this));
		session.send(new PacketQuestListNotify(this));
1316
		session.send(new PacketCodexDataFullNotify(this));
1317
1318
		session.send(new PacketAllWidgetDataNotify(this));
		session.send(new PacketWidgetGadgetAllDataNotify());
1319
1320
		session.send(new PacketPlayerHomeCompInfoNotify(this));
		session.send(new PacketHomeComfortInfoNotify(this));
ImmuState's avatar
ImmuState committed
1321
		session.send(new PacketCombineDataNotify(this.unlockedCombines));
1322
		this.forgingManager.sendForgeDataNotify();
ImmuState's avatar
ImmuState committed
1323
		this.resinManager.onPlayerLogin();
1324

Yazawazi's avatar
Yazawazi committed
1325
1326
		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
1327
		session.send(new PacketPlayerEnterSceneNotify(this)); // Enter game world
1328
		session.send(new PacketPlayerLevelRewardUpdateNotify(rewardedLevels));
Melledy's avatar
Melledy committed
1329
1330
1331
1332
		session.send(new PacketOpenStateUpdateNotify());

		// First notify packets sent
		this.setHasSentAvatarDataNotify(true);
1333
1334
1335
		
		// Set session state
		session.setState(SessionState.ACTIVE);
1336
1337
1338

		// Call join event.
		PlayerJoinEvent event = new PlayerJoinEvent(this); event.call();
1339
		if(event.isCanceled()){ // If event is not cancelled, continue.
1340
			session.close();
1341
1342
1343
1344
1345
1346
			return;
		}
		
		// register
		getServer().registerPlayer(this);
		getProfile().setPlayer(this); // Set online
Melledy's avatar
Melledy committed
1347
	}
Asnxthaony's avatar
Asnxthaony committed
1348

Melledy's avatar
Melledy committed
1349
	public void onLogout() {
1350
1351
1352
		try{
			// stop stamina calculation
			getStaminaManager().stopSustainedStaminaHandler();
1353

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

1357
1358
1359
1360
1361
1362
1363
1364
			// 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
1365

1366
			this.getCoopRequests().clear();
Asnxthaony's avatar
Asnxthaony committed
1367

1368
1369
1370
1371
1372
1373
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
1384
1385
			// 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();
		}
	}
1386

1387
1388
1389
1390
1391
	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
1392
	}
Asnxthaony's avatar
Asnxthaony committed
1393

Melledy's avatar
Melledy committed
1394
	public enum SceneLoadState {
Asnxthaony's avatar
Asnxthaony committed
1395
1396
		NONE(0), LOADING(1), INIT(2), LOADED(3);

Melledy's avatar
Melledy committed
1397
		private final int value;
Asnxthaony's avatar
Asnxthaony committed
1398

Melledy's avatar
Melledy committed
1399
1400
1401
		private SceneLoadState(int value) {
			this.value = value;
		}
Asnxthaony's avatar
Asnxthaony committed
1402

Melledy's avatar
Melledy committed
1403
1404
1405
1406
		public int getValue() {
			return this.value;
		}
	}
1407
1408
1409
1410

	public void setMessageHandler(MessageHandler messageHandler) {
		this.messageHandler = messageHandler;
	}
1411
1412
1413
1414
1415
1416
1417
1418
1419
1420
1421
1422
1423
1424
1425
1426
1427
1428
1429

	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
1430
			if (!(value >= 0 && value <= SotSManager.GlobalMaximumSpringVolume)) { return false; }
1431
1432
1433
1434
1435
1436
1437
1438
1439
1440
1441
1442
1443
1444
1445
1446
		} 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
1447
			if (!(value >= 0 && value <= StaminaManager.GlobalCharacterMaximumStamina)) { return false; }
1448
1449
1450
1451
1452
1453
1454
1455
1456
1457
		} 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
1458
			// see PlayerProperty.PROP_PLAYER_HCOIN comments
1459
1460
1461
1462
1463
1464
1465
1466
1467
1468
1469
1470
1471
1472
1473
1474
1475
1476
1477
1478
1479
1480
1481
1482
1483
1484
1485
1486
1487
1488
1489
1490
1491
1492
1493
1494
1495
1496
1497
1498
1499
1500
1501
1502
1503
1504
1505
1506
1507
1508
		} 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
1509
}