Player.java 45.2 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
18
import emu.grasscutter.game.home.GameHome;
import emu.grasscutter.game.managers.DeforestationManager.DeforestationManager;
19
import emu.grasscutter.game.entity.EntityGadget;
Melledy's avatar
Melledy committed
20
import emu.grasscutter.game.entity.EntityItem;
21
import emu.grasscutter.game.entity.GameEntity;
Kinesis's avatar
Kinesis committed
22
import emu.grasscutter.game.expedition.ExpeditionInfo;
Melledy's avatar
Melledy committed
23
24
25
import emu.grasscutter.game.friends.FriendsList;
import emu.grasscutter.game.friends.PlayerProfile;
import emu.grasscutter.game.gacha.PlayerGachaInfo;
26
import emu.grasscutter.game.inventory.GameItem;
Melledy's avatar
Melledy committed
27
import emu.grasscutter.game.inventory.Inventory;
Melledy's avatar
Melledy committed
28
import emu.grasscutter.game.mail.Mail;
29
import emu.grasscutter.game.mail.MailHandler;
30
import emu.grasscutter.game.managers.InsectCaptureManager;
ImmuState's avatar
ImmuState committed
31
import emu.grasscutter.game.managers.ResinManager;
Melledy's avatar
Melledy committed
32
33
34
35
36
37
import emu.grasscutter.game.managers.deforestation.DeforestationManager;
import emu.grasscutter.game.managers.energy.EnergyManager;
import emu.grasscutter.game.managers.forging.ActiveForgeData;
import emu.grasscutter.game.managers.forging.ForgingManager;
import emu.grasscutter.game.managers.mapmark.*;
import emu.grasscutter.game.managers.stamina.StaminaManager;
38
import emu.grasscutter.game.managers.SotSManager;
Melledy's avatar
Melledy committed
39
import emu.grasscutter.game.props.ActionReason;
40
import emu.grasscutter.game.props.EntityType;
Melledy's avatar
Melledy committed
41
import emu.grasscutter.game.props.PlayerProperty;
42
import emu.grasscutter.game.props.SceneType;
Melledy's avatar
Melledy committed
43
import emu.grasscutter.game.quest.QuestManager;
Kengxxiao's avatar
Kengxxiao committed
44
import emu.grasscutter.game.shop.ShopLimit;
45
import emu.grasscutter.game.tower.TowerData;
Akka's avatar
Akka committed
46
import emu.grasscutter.game.tower.TowerManager;
Melledy's avatar
Melledy committed
47
48
import emu.grasscutter.game.world.Scene;
import emu.grasscutter.game.world.World;
49
import emu.grasscutter.net.packet.BasePacket;
50
import emu.grasscutter.net.proto.*;
Melledy's avatar
Melledy committed
51
import emu.grasscutter.net.proto.AbilityInvokeEntryOuterClass.AbilityInvokeEntry;
52
import emu.grasscutter.net.proto.AttackResultOuterClass.AttackResult;
Melledy's avatar
Melledy committed
53
import emu.grasscutter.net.proto.CombatInvokeEntryOuterClass.CombatInvokeEntry;
ImmuState's avatar
ImmuState committed
54
import emu.grasscutter.net.proto.GadgetInteractReqOuterClass.GadgetInteractReq;
Melledy's avatar
Melledy committed
55
56
57
58
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
59
import emu.grasscutter.net.proto.ProfilePictureOuterClass.ProfilePicture;
Melledy's avatar
Melledy committed
60
import emu.grasscutter.net.proto.SocialDetailOuterClass.SocialDetail;
61

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

Asnxthaony's avatar
Asnxthaony committed
75
import java.util.*;
76
import java.util.concurrent.LinkedBlockingQueue;
Asnxthaony's avatar
Asnxthaony committed
77

78
79
import static emu.grasscutter.Configuration.*;

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

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

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

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

104
105
	private Integer widgetId;

106
107
108
	private Set<Integer> realmList;
	private Integer currentRealmId;

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

Melledy's avatar
Melledy committed
125
	private TeamManager teamManager;
Akka's avatar
Akka committed
126

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

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

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

Yazawazi's avatar
Yazawazi committed
148
149
150
	private List<Integer> showAvatarList;
	private boolean showAvatars;

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

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

gentlespoon's avatar
gentlespoon committed
163
	@Transient private MapMarksManager mapMarksManager;
164
	@Transient private StaminaManager staminaManager;
165
	@Transient private EnergyManager energyManager;
ImmuState's avatar
ImmuState committed
166
	@Transient private ResinManager resinManager;
167
	@Transient private ForgingManager forgingManager;
168
	@Transient private DeforestationManager deforestationManager;
169
	@Transient private GameHome home;
170

171
	private long springLastUsed;
gentlespoon's avatar
gentlespoon committed
172
	private HashMap<String, MapMark> mapMarks;
ImmuState's avatar
ImmuState committed
173
	private int nextResinRefresh;
174

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

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

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

Melledy's avatar
Melledy committed
207
208
209
		this.setSceneId(3);
		this.setRegionId(1);
		this.sceneState = SceneLoadState.NONE;
Asnxthaony's avatar
Asnxthaony committed
210

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

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

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

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

266
	public int getUid() {
Melledy's avatar
Melledy committed
267
268
269
		return id;
	}

270
	public void setUid(int id) {
Melledy's avatar
Melledy committed
271
272
		this.id = id;
	}
Asnxthaony's avatar
Asnxthaony committed
273

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

	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
294

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

Melledy's avatar
Melledy committed
299
300
301
	public GameServer getServer() {
		return this.getSession().getServer();
	}
Asnxthaony's avatar
Asnxthaony committed
302

Melledy's avatar
Melledy committed
303
304
305
	public synchronized World getWorld() {
		return this.world;
	}
Asnxthaony's avatar
Asnxthaony committed
306

Melledy's avatar
Melledy committed
307
308
309
	public synchronized void setWorld(World world) {
		this.world = world;
	}
Asnxthaony's avatar
Asnxthaony committed
310

311
	public synchronized Scene getScene() {
312
313
314
		return scene;
	}

315
	public synchronized void setScene(Scene scene) {
316
317
318
		this.scene = scene;
	}

Melledy's avatar
Melledy committed
319
320
321
322
323
324
325
326
327
328
329
330
	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
331

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

350
351
352
353
354
355
356
357
	public Integer getWidgetId() {
		return widgetId;
	}

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

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

Melledy's avatar
Melledy committed
389
390
391
	public Position getRotation() {
		return rotation;
	}
Asnxthaony's avatar
Asnxthaony committed
392

Melledy's avatar
Melledy committed
393
394
395
	public int getLevel() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_LEVEL);
	}
Asnxthaony's avatar
Asnxthaony committed
396

Melledy's avatar
Melledy committed
397
398
399
	public int getExp() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_EXP);
	}
Asnxthaony's avatar
Asnxthaony committed
400

Melledy's avatar
Melledy committed
401
402
403
	public int getWorldLevel() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_WORLD_LEVEL);
	}
404
405
406
407
408
	
	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
409

Melledy's avatar
Melledy committed
410
411
412
	public int getPrimogems() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_HCOIN);
	}
Asnxthaony's avatar
Asnxthaony committed
413

Melledy's avatar
Melledy committed
414
415
416
417
	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
418

Melledy's avatar
Melledy committed
419
420
421
	public int getMora() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_SCOIN);
	}
Asnxthaony's avatar
Asnxthaony committed
422

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

437
438
439
440
441
442
443
444
	public int getHomeCoin() {
		return this.getProperty(PlayerProperty.PROP_PLAYER_HOME_COIN);
	}

	public void setHomeCoin(int coin) {
		this.setProperty(PlayerProperty.PROP_PLAYER_HOME_COIN, coin);
		this.sendPacket(new PacketPlayerPropNotify(this, PlayerProperty.PROP_PLAYER_HOME_COIN));
	}
Melledy's avatar
Melledy committed
445
	private int getExpRequired(int level) {
446
		PlayerLevelData levelData = GameData.getPlayerLevelDataMap().get(level);
Asnxthaony's avatar
Asnxthaony committed
447
		return levelData != null ? levelData.getExp() : 0;
Melledy's avatar
Melledy committed
448
	}
Asnxthaony's avatar
Asnxthaony committed
449

Melledy's avatar
Melledy committed
450
	private float getExpModifier() {
451
		return GAME_OPTIONS.rates.adventureExp;
Melledy's avatar
Melledy committed
452
	}
Asnxthaony's avatar
Asnxthaony committed
453

Melledy's avatar
Melledy committed
454
455
456
457
	// Affected by exp rate
	public void earnExp(int exp) {
		addExpDirectly((int) (exp * getExpModifier()));
	}
Asnxthaony's avatar
Asnxthaony committed
458

Melledy's avatar
Melledy committed
459
460
461
462
463
464
	// 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
465

Melledy's avatar
Melledy committed
466
		exp += gain;
Asnxthaony's avatar
Asnxthaony committed
467

Melledy's avatar
Melledy committed
468
469
470
471
472
473
		while (exp >= reqExp && reqExp > 0) {
			exp -= reqExp;
			level += 1;
			reqExp = getExpRequired(level);
			hasLeveledUp = true;
		}
Asnxthaony's avatar
Asnxthaony committed
474

Melledy's avatar
Melledy committed
475
476
477
478
479
480
481
482
		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
483

Melledy's avatar
Melledy committed
484
485
		// Set exp
		this.setProperty(PlayerProperty.PROP_PLAYER_EXP, exp);
Asnxthaony's avatar
Asnxthaony committed
486

Melledy's avatar
Melledy committed
487
488
489
		// Update player with packet
		this.sendPacket(new PacketPlayerPropNotify(this, PlayerProperty.PROP_PLAYER_EXP));
	}
Asnxthaony's avatar
Asnxthaony committed
490

Melledy's avatar
Melledy committed
491
492
493
	private void updateProfile() {
		getProfile().syncWithCharacter(this);
	}
Asnxthaony's avatar
Asnxthaony committed
494

Melledy's avatar
Melledy committed
495
496
497
498
499
500
501
	public boolean isFirstLoginEnterScene() {
		return !this.hasSentAvatarDataNotify;
	}

	public TeamManager getTeamManager() {
		return this.teamManager;
	}
502
	
Akka's avatar
Akka committed
503
504
505
	public TowerManager getTowerManager() {
		return towerManager;
	}
506
507
	
	public TowerData getTowerData() {
508
509
510
511
		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();
		}
512
513
514
		return towerData;
	}
	
Melledy's avatar
Melledy committed
515
516
517
518
519
520
521
522
	public QuestManager getQuestManager() {
		return questManager;
	}

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

Melledy's avatar
Melledy committed
523
524
525
526
527
528
529
530
531
532
533
	public PlayerGachaInfo getGachaInfo() {
		return gachaInfo;
	}

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

534
535
	// 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
536
537
538
	public Map<Integer, Integer> getProperties() {
		return properties;
	}
Asnxthaony's avatar
Asnxthaony committed
539

540
541
	public boolean setProperty(PlayerProperty prop, int value) {
		return setPropertyWithSanityCheck(prop, value);
Melledy's avatar
Melledy committed
542
	}
Asnxthaony's avatar
Asnxthaony committed
543

Melledy's avatar
Melledy committed
544
545
546
547
548
549
550
	public int getProperty(PlayerProperty prop) {
		return getProperties().get(prop.getId());
	}

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

Melledy's avatar
Melledy committed
552
553
554
	public Set<Integer> getCostumeList() {
		return costumeList;
	}
Asnxthaony's avatar
Asnxthaony committed
555

Melledy's avatar
Melledy committed
556
557
558
559
	public Set<Integer> getNameCardList() {
		return this.nameCardList;
	}

560
	public Set<Integer> getUnlockedForgingBlueprints() {
561
562
563
		return this.unlockedForgingBlueprints;
	}

ImmuState's avatar
ImmuState committed
564
565
566
567
	public Set<Integer> getUnlockedCombines() {
		return this.unlockedCombines;
	}

568
	public List<ActiveForgeData> getActiveForges() {
569
		return this.activeForges;
570
571
	}

Melledy's avatar
Melledy committed
572
	public MpSettingType getMpSetting() {
573
		return MpSettingType.MP_SETTING_TYPE_ENTER_AFTER_APPLY; // TEMP
Melledy's avatar
Melledy committed
574
	}
575
576
577
578
	
	public Queue<AttackResult> getAttackResults() {
		return this.attackResults;
	}
Asnxthaony's avatar
Asnxthaony committed
579

Melledy's avatar
Melledy committed
580
581
582
583
584
585
586
	public synchronized Int2ObjectMap<CoopRequest> getCoopRequests() {
		return coopRequests;
	}

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

Melledy's avatar
Melledy committed
588
589
590
591
	public InvokeHandler<AbilityInvokeEntry> getAbilityInvokeHandler() {
		return this.abilityInvokeHandler;
	}

592
593
594
595
	public InvokeHandler<AbilityInvokeEntry> getClientAbilityInitFinishHandler() {
		return clientAbilityInitFinishHandler;
	}

Melledy's avatar
Melledy committed
596
597
598
	public AvatarStorage getAvatars() {
		return avatars;
	}
Asnxthaony's avatar
Asnxthaony committed
599

Melledy's avatar
Melledy committed
600
601
602
	public Inventory getInventory() {
		return inventory;
	}
Asnxthaony's avatar
Asnxthaony committed
603

Melledy's avatar
Melledy committed
604
605
606
607
	public FriendsList getFriendsList() {
		return this.friendsList;
	}

608
609
610
611
	public MailHandler getMailHandler() {
		return mailHandler;
	}

Melledy's avatar
Melledy committed
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
647
648
649
650
651
652
653
654
655
656
657
658
659
	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
660

Melledy's avatar
Melledy committed
661
662
663
664
665
666
667
		if (newPauseState && !oldPauseState) {
			this.onPause();
		} else if (oldPauseState && !newPauseState) {
			this.onUnpause();
		}
	}

668
669
670
671
672
673
674
675
	public long getSpringLastUsed() {
		return springLastUsed;
	}

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

ImmuState's avatar
ImmuState committed
676
677
678
679
680
681
682
683
	public int getNextResinRefresh() {
		return nextResinRefresh;
	}

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

Melledy's avatar
Melledy committed
684
685
686
687
688
689
690
	public SceneLoadState getSceneLoadState() {
		return sceneState;
	}

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

Melledy's avatar
Melledy committed
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
	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
711

Yazawazi's avatar
Yazawazi committed
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
	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
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
756
757
758
759
760
761
762
763
764
765
766
767
768
	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
769
		Date now = DateHelper.onlyYearMonthDay(new Date());
Yazawazi's avatar
Yazawazi committed
770
771
772
773
		return (int) ((theLastDay.getTime() - now.getTime()) / (24 * 60 * 60 * 1000)); // By copilot 
	}

	public void rechargeMoonCard() {
774
		inventory.addItem(new GameItem(203, 300));
Yazawazi's avatar
Yazawazi committed
775
776
777
		if (!moonCard) {
			moonCard = true;
			Date now = new Date();
Yazawazi's avatar
Yazawazi committed
778
			moonCardStartTime = DateHelper.onlyYearMonthDay(now);
Yazawazi's avatar
Yazawazi committed
779
780
781
782
783
784
785
786
787
788
789
790
791
			moonCardDuration = 30;
		} else {
			moonCardDuration += 30;
		}
		if (!moonCardGetTimes.contains(moonCardStartTime)) {
			moonCardGetTimes.add(moonCardStartTime);
		}
	}

	public void getTodayMoonCard() {
		if (!moonCard) {
			return;
		}
Yazawazi's avatar
Yazawazi committed
792
		Date now = DateHelper.onlyYearMonthDay(new Date());
Yazawazi's avatar
Yazawazi committed
793
794
795
796
797
798
799
800
801
802
803
804
805
806
		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);
807
808
		GameItem item = new GameItem(201, 90);
		getInventory().addItem(item, ActionReason.BlessingRedeemReward);
Yazawazi's avatar
Yazawazi committed
809
810
811
		session.send(new PacketCardProductRewardNotify(getMoonCardRemainDays()));
	}

Kinesis's avatar
Kinesis committed
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
	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
833
834
835
836
	public List<ShopLimit> getShopLimit() {
		return shopLimit;
	}

Kengxxiao's avatar
Kengxxiao committed
837
838
839
840
841
	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
842
843
	}

Kengxxiao's avatar
Kengxxiao committed
844
845
846
847
848
849
850
	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
851
852
853
			ShopLimit sl = new ShopLimit();
			sl.setShopGoodId(goodsId);
			sl.setHasBought(boughtCount);
Kengxxiao's avatar
Kengxxiao committed
854
855
856
			sl.setHasBoughtInPeriod(boughtCount);
			sl.setNextRefreshTime(nextRefreshTime);
			getShopLimit().add(sl);
Kengxxiao's avatar
Kengxxiao committed
857
858
859
		}
		this.save();
	}
860
861
862
863
864
865
866
867
	public boolean getStamina() {
		// Get Stamina
		return stamina;
	}
	public void setStamina(boolean stamina) {
		// Set Stamina
		this.stamina = stamina;
	}
KingRainbow44's avatar
KingRainbow44 committed
868
	public boolean inGodmode() {
Melledy's avatar
Melledy committed
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
		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
884
	public void addAvatar(Avatar avatar, boolean addToCurrentTeam) {
Melledy's avatar
Melledy committed
885
		boolean result = getAvatars().addAvatar(avatar);
Asnxthaony's avatar
Asnxthaony committed
886

Melledy's avatar
Melledy committed
887
888
889
		if (result) {
			// Add starting weapon
			getAvatars().addStartingWeapon(avatar);
Asnxthaony's avatar
Asnxthaony committed
890

Melledy's avatar
Melledy committed
891
892
893
894
			// Done
			if (hasSentAvatarDataNotify()) {
				// Recalc stats
				avatar.recalcStats();
ProxyismGH's avatar
ProxyismGH committed
895
896
897
898
899
900
				// 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
901
902
903
904
905
			}
		} else {
			// Failed adding avatar
		}
	}
Asnxthaony's avatar
Asnxthaony committed
906

ProxyismGH's avatar
ProxyismGH committed
907
908
909
910
	public void addAvatar(Avatar avatar) {
		addAvatar(avatar, true);
	}

Melledy's avatar
Melledy committed
911
912
913
914
	public void addFlycloak(int flycloakId) {
		this.getFlyCloakList().add(flycloakId);
		this.sendPacket(new PacketAvatarGainFlycloakNotify(flycloakId));
	}
Asnxthaony's avatar
Asnxthaony committed
915

Melledy's avatar
Melledy committed
916
917
918
919
	public void addCostume(int costumeId) {
		this.getCostumeList().add(costumeId);
		this.sendPacket(new PacketAvatarGainCostumeNotify(costumeId));
	}
Asnxthaony's avatar
Asnxthaony committed
920

Melledy's avatar
Melledy committed
921
922
923
924
	public void addNameCard(int nameCardId) {
		this.getNameCardList().add(nameCardId);
		this.sendPacket(new PacketUnlockNameCardNotify(nameCardId));
	}
Asnxthaony's avatar
Asnxthaony committed
925

Melledy's avatar
Melledy committed
926
927
928
929
	public void setNameCard(int nameCardId) {
		if (!this.getNameCardList().contains(nameCardId)) {
			return;
		}
Asnxthaony's avatar
Asnxthaony committed
930

Melledy's avatar
Melledy committed
931
		this.setNameCardId(nameCardId);
Asnxthaony's avatar
Asnxthaony committed
932

Melledy's avatar
Melledy committed
933
934
		this.sendPacket(new PacketSetNameCardRsp(nameCardId));
	}
Asnxthaony's avatar
Asnxthaony committed
935

Melledy's avatar
Melledy committed
936
	public void dropMessage(Object message) {
937
938
939
940
		if (this.messageHandler != null) {
			this.messageHandler.append(message.toString());
			return;
		}
941
		this.sendPacket(new PacketPrivateChatNotify(GameConstants.SERVER_CONSOLE_UID, getUid(), message.toString()));
Melledy's avatar
Melledy committed
942
	}
Melledy's avatar
Melledy committed
943
944
945

	/**
	 * Sends a message to another player.
Asnxthaony's avatar
Asnxthaony committed
946
947
	 *
	 * @param sender  The sender of the message.
Melledy's avatar
Melledy committed
948
949
	 * @param message The message to send.
	 */
950
	public void sendMessage(Player sender, Object message) {
951
		this.sendPacket(new PacketPrivateChatNotify(sender.getUid(), this.getUid(), message.toString()));
Melledy's avatar
Melledy committed
952
	}
Asnxthaony's avatar
Asnxthaony committed
953

Benjamin Elsdon's avatar
Benjamin Elsdon committed
954
	// ---------------------MAIL------------------------
955

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

Benjamin Elsdon's avatar
Benjamin Elsdon committed
958
	public void sendMail(Mail message) {
959
		this.getMailHandler().sendMail(message);
960
	}
Benjamin Elsdon's avatar
Benjamin Elsdon committed
961
962

	public boolean deleteMail(int mailId) {
963
		return this.getMailHandler().deleteMail(mailId);
Benjamin Elsdon's avatar
Benjamin Elsdon committed
964
965
	}

966
967
	public Mail getMail(int index) { return this.getMailHandler().getMailById(index); }
	
968
	public int getMailId(Mail message) {
969
		return this.getMailHandler().getMailIndex(message);
Benjamin Elsdon's avatar
Benjamin Elsdon committed
970
971
	}

972
	public boolean replaceMailByIndex(int index, Mail message) {
973
		return this.getMailHandler().replaceMailByIndex(index, message);
Benjamin Elsdon's avatar
Benjamin Elsdon committed
974
	}
975
	
ImmuState's avatar
ImmuState committed
976
	public void interactWith(int gadgetEntityId, GadgetInteractReq request) {
977
		GameEntity entity = getScene().getEntityById(gadgetEntityId);
Melledy's avatar
Melledy committed
978
979
980
		if (entity == null) {
			return;
		}
Asnxthaony's avatar
Asnxthaony committed
981

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

Melledy's avatar
Melledy committed
1020
	public void onPause() {
1021
		getStaminaManager().stopSustainedStaminaHandler();
Melledy's avatar
Melledy committed
1022
	}
Asnxthaony's avatar
Asnxthaony committed
1023

Melledy's avatar
Melledy committed
1024
	public void onUnpause() {
1025
		getStaminaManager().startSustainedStaminaHandler();
Melledy's avatar
Melledy committed
1026
	}
Asnxthaony's avatar
Asnxthaony committed
1027

1028
	public void sendPacket(BasePacket packet) {
Melledy's avatar
Melledy committed
1029
		this.getSession().send(packet);
Melledy's avatar
Melledy committed
1030
	}
Asnxthaony's avatar
Asnxthaony committed
1031

Melledy's avatar
Melledy committed
1032
1033
	public OnlinePlayerInfo getOnlinePlayerInfo() {
		OnlinePlayerInfo.Builder onlineInfo = OnlinePlayerInfo.newBuilder()
Asnxthaony's avatar
Asnxthaony committed
1034
1035
1036
1037
1038
1039
				.setUid(this.getUid())
				.setNickname(this.getNickname())
				.setPlayerLevel(this.getLevel())
				.setMpSettingType(this.getMpSetting())
				.setNameCardId(this.getNameCardId())
				.setSignature(this.getSignature())
Melledy's avatar
Melledy committed
1040
				.setProfilePicture(ProfilePicture.newBuilder().setAvatarId(this.getHeadImage()));
Asnxthaony's avatar
Asnxthaony committed
1041

Melledy's avatar
Melledy committed
1042
		if (this.getWorld() != null) {
Melledy's avatar
Melledy committed
1043
			onlineInfo.setCurPlayerNumInWorld(getWorld().getPlayerCount());
Melledy's avatar
Melledy committed
1044
1045
1046
		} else {
			onlineInfo.setCurPlayerNumInWorld(1);
		}
Asnxthaony's avatar
Asnxthaony committed
1047

Melledy's avatar
Melledy committed
1048
1049
1050
		return onlineInfo.build();
	}

Asnxthaony's avatar
Asnxthaony committed
1051
	public PlayerBirthday getBirthday() {
Miyucchi's avatar
Miyucchi committed
1052
1053
1054
1055
1056
1057
1058
1059
		return this.birthday;
	}

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

Asnxthaony's avatar
Asnxthaony committed
1060
1061
1062
1063
	public boolean hasBirthday() {
		return this.birthday.getDay() > 0;
	}

1064
1065
	public PlayerCodex getCodex(){ return this.codex; }

1066
1067
1068
1069
1070
1071
1072
1073
	public Set<Integer> getRewardedLevels() {
		return rewardedLevels;
	}

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

Melledy's avatar
Melledy committed
1074
	public SocialDetail.Builder getSocialDetail() {
Yazawazi's avatar
Yazawazi committed
1075
		List<SocialShowAvatarInfoOuterClass.SocialShowAvatarInfo> socialShowAvatarInfoList = new ArrayList<>();
Yazawazi's avatar
Yazawazi committed
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
		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 {
1090
1091
			List<Integer> showAvatarList = DatabaseHelper.getPlayerByUid(id).getShowAvatarList();
			AvatarStorage avatars = DatabaseHelper.getPlayerByUid(id).getAvatars();
Yazawazi's avatar
Yazawazi committed
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
			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
1105
1106
		}

Asnxthaony's avatar
Asnxthaony committed
1107
1108
		SocialDetail.Builder social = SocialDetail.newBuilder()
				.setUid(this.getUid())
Yazawazi's avatar
Yazawazi committed
1109
				.setProfilePicture(ProfilePicture.newBuilder().setAvatarId(this.getHeadImage()))
Asnxthaony's avatar
Asnxthaony committed
1110
1111
1112
1113
1114
1115
				.setNickname(this.getNickname())
				.setSignature(this.getSignature())
				.setLevel(this.getLevel())
				.setBirthday(this.getBirthday().getFilledProtoWhenNotEmpty())
				.setWorldLevel(this.getWorldLevel())
				.setNameCardId(this.getNameCardId())
Yazawazi's avatar
Yazawazi committed
1116
1117
				.setIsShowAvatar(this.isShowAvatars())
				.addAllShowAvatarInfoList(socialShowAvatarInfoList)
Asnxthaony's avatar
Asnxthaony committed
1118
				.setFinishAchievementNum(0);
Melledy's avatar
Melledy committed
1119
1120
		return social;
	}
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130

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

		Player player;
		boolean shouldRecalc;
		if (this.isOnline()) {
			player = this;
			shouldRecalc = false;
		} else {
1131
			player = DatabaseHelper.getPlayerByUid(id);
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
			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;
	}
1150
1151
1152
	
	public PlayerWorldLocationInfoOuterClass.PlayerWorldLocationInfo getWorldPlayerLocationInfo() {
		return PlayerWorldLocationInfoOuterClass.PlayerWorldLocationInfo.newBuilder()
Asnxthaony's avatar
Asnxthaony committed
1153
1154
1155
				.setSceneId(this.getSceneId())
				.setPlayerLoc(this.getPlayerLocationInfo())
				.build();
Melledy's avatar
Melledy committed
1156
	}
Asnxthaony's avatar
Asnxthaony committed
1157

Melledy's avatar
Melledy committed
1158
	public PlayerLocationInfo getPlayerLocationInfo() {
Asnxthaony's avatar
Asnxthaony committed
1159
1160
1161
1162
1163
		return PlayerLocationInfo.newBuilder()
				.setUid(this.getUid())
				.setPos(this.getPos().toProto())
				.setRot(this.getRotation().toProto())
				.build();
Melledy's avatar
Melledy committed
1164
	}
Asnxthaony's avatar
Asnxthaony committed
1165

Angda Song's avatar
Angda Song committed
1166
1167
1168
1169
	public MapMarksManager getMapMarksManager() {
		return mapMarksManager;
	}

1170
	public StaminaManager getStaminaManager() { return staminaManager; }
1171

1172
1173
	public SotSManager getSotSManager() { return sotsManager; }

1174
1175
1176
1177
	public EnergyManager getEnergyManager() {
		return this.energyManager;
	}

ImmuState's avatar
ImmuState committed
1178
1179
1180
1181
	public ResinManager getResinManager() {
		return this.resinManager;
	}

1182
1183
1184
1185
	public ForgingManager getForgingManager() {
		return this.forgingManager;
	}

Melledy's avatar
Melledy committed
1186
1187
1188
1189
	public AbilityManager getAbilityManager() {
		return abilityManager;
	}

1190
1191
1192
1193
	public DeforestationManager getDeforestationManager() {
		return deforestationManager;
	}

gentlespoon's avatar
gentlespoon committed
1194
1195
1196
1197
	public HashMap<String, MapMark> getMapMarks() { return mapMarks; }

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

Melledy's avatar
Melledy committed
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
	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()) {
1209
1210
1211
1212
				req.getRequester().sendPacket(new PacketPlayerApplyEnterMpResultNotify(
						this,
						false,
						PlayerApplyEnterMpResultNotifyOuterClass.PlayerApplyEnterMpResultNotify.Reason.REASON_SYSTEM_JUDGE));
Melledy's avatar
Melledy committed
1213
1214
1215
1216
1217
				it.remove();
			}
		}
		// Ping
		if (this.getWorld() != null) {
Melledy's avatar
Melledy committed
1218
1219
			// RTT notify - very important to send this often
			this.sendPacket(new PacketWorldPlayerRTTNotify(this.getWorld()));
Asnxthaony's avatar
Asnxthaony committed
1220

Melledy's avatar
Melledy committed
1221
1222
1223
1224
1225
1226
1227
			// 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
1228
		}
Kinesis's avatar
Kinesis committed
1229
1230
1231
1232
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
		// 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));
		}
1245
1246
1247

		// Send updated forge queue data, if necessary.
		this.getForgingManager().sendPlayerForgingUpdate();
ImmuState's avatar
ImmuState committed
1248
1249
1250

		// Recharge resin.
		this.getResinManager().rechargeResin();
1251
1252
1253
	}


Asnxthaony's avatar
Asnxthaony committed
1254

1255

Melledy's avatar
Melledy committed
1256
1257
1258
	public void resetSendPlayerLocTime() {
		this.nextSendPlayerLocTime = System.currentTimeMillis() + 5000;
	}
Melledy's avatar
Melledy committed
1259
1260
1261

	@PostLoad
	private void onLoad() {
1262
		this.getCodex().setPlayer(this);
Melledy's avatar
Melledy committed
1263
		this.getTeamManager().setPlayer(this);
1264
		this.getTowerManager().setPlayer(this);
Melledy's avatar
Melledy committed
1265
	}
Asnxthaony's avatar
Asnxthaony committed
1266

Melledy's avatar
Melledy committed
1267
1268
1269
	public void save() {
		DatabaseHelper.savePlayer(this);
	}
1270
1271
1272
	
	// Called from tokenrsp
	public void loadFromDatabase() {
Melledy's avatar
Melledy committed
1273
		// Make sure these exist
1274
1275
1276
		if (this.getTowerManager() == null) {
			this.towerManager = new TowerManager(this);
		}
Melledy's avatar
Melledy committed
1277
1278
		if (this.getTeamManager() == null) {
			this.teamManager = new TeamManager(this);
Asnxthaony's avatar
Asnxthaony committed
1279
		}
1280
1281
1282
		if (this.getCodex() == null) {
			this.codex = new PlayerCodex(this);
		}
Asnxthaony's avatar
Asnxthaony committed
1283
		if (this.getProfile().getUid() == 0) {
1284
			this.getProfile().syncWithCharacter(this);
Melledy's avatar
Melledy committed
1285
		}
1286
1287
		//Make sure towerManager's player is online player
		this.getTowerManager().setPlayer(this);
Melledy's avatar
Melledy committed
1288
1289
1290
1291
		// Load from db
		this.getAvatars().loadFromDatabase();
		this.getInventory().loadFromDatabase();
		this.getAvatars().postLoad();
Asnxthaony's avatar
Asnxthaony committed
1292

Melledy's avatar
Melledy committed
1293
		this.getFriendsList().loadFromDatabase();
1294
		this.getMailHandler().loadFromDatabase();
Melledy's avatar
Melledy committed
1295
		this.getQuestManager().loadFromDatabase();
1296

1297
1298
1299
	}

	public void onLogin() {
Melledy's avatar
Melledy committed
1300
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
		// 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
1314
1315
1316
		// Create world
		World world = new World(this);
		world.addPlayer(this);
Asnxthaony's avatar
Asnxthaony committed
1317
1318

		// Multiplayer setting
Melledy's avatar
Melledy committed
1319
1320
		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
1321

Melledy's avatar
Melledy committed
1322
1323
1324
1325
1326
		// 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
1327
1328
		session.send(new PacketFinishedParentQuestNotify(this));
		session.send(new PacketQuestListNotify(this));
1329
		session.send(new PacketCodexDataFullNotify(this));
1330
1331
		session.send(new PacketAllWidgetDataNotify(this));
		session.send(new PacketWidgetGadgetAllDataNotify());
ImmuState's avatar
ImmuState committed
1332
		session.send(new PacketCombineDataNotify(this.unlockedCombines));
1333
		this.forgingManager.sendForgeDataNotify();
ImmuState's avatar
ImmuState committed
1334
		this.resinManager.onPlayerLogin();
1335

Yazawazi's avatar
Yazawazi committed
1336
1337
		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.

1338
1339
1340
1341
		// Home
		home = GameHome.getByUid(getUid());
		home.onOwnerLogin(this);

Melledy's avatar
Melledy committed
1342
		session.send(new PacketPlayerEnterSceneNotify(this)); // Enter game world
1343
		session.send(new PacketPlayerLevelRewardUpdateNotify(rewardedLevels));
Melledy's avatar
Melledy committed
1344
1345
1346
1347
		session.send(new PacketOpenStateUpdateNotify());

		// First notify packets sent
		this.setHasSentAvatarDataNotify(true);
1348
1349
1350
		
		// Set session state
		session.setState(SessionState.ACTIVE);
1351
1352
1353

		// Call join event.
		PlayerJoinEvent event = new PlayerJoinEvent(this); event.call();
1354
		if(event.isCanceled()){ // If event is not cancelled, continue.
1355
			session.close();
1356
1357
1358
1359
1360
1361
			return;
		}
		
		// register
		getServer().registerPlayer(this);
		getProfile().setPlayer(this); // Set online
Melledy's avatar
Melledy committed
1362
	}
Asnxthaony's avatar
Asnxthaony committed
1363

Melledy's avatar
Melledy committed
1364
	public void onLogout() {
1365
1366
1367
		try{
			// stop stamina calculation
			getStaminaManager().stopSustainedStaminaHandler();
1368

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

1372
1373
1374
1375
1376
1377
1378
1379
			// 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
1380

1381
			this.getCoopRequests().clear();
Asnxthaony's avatar
Asnxthaony committed
1382

1383
1384
1385
1386
1387
1388
1389
1390
1391
1392
1393
1394
1395
1396
1397
1398
1399
1400
			// 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();
		}
	}
1401

1402
1403
1404
1405
1406
	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
1407
	}
Asnxthaony's avatar
Asnxthaony committed
1408

Melledy's avatar
Melledy committed
1409
	public enum SceneLoadState {
Asnxthaony's avatar
Asnxthaony committed
1410
1411
		NONE(0), LOADING(1), INIT(2), LOADED(3);

Melledy's avatar
Melledy committed
1412
		private final int value;
Asnxthaony's avatar
Asnxthaony committed
1413

Melledy's avatar
Melledy committed
1414
1415
1416
		private SceneLoadState(int value) {
			this.value = value;
		}
Asnxthaony's avatar
Asnxthaony committed
1417

Melledy's avatar
Melledy committed
1418
1419
1420
1421
		public int getValue() {
			return this.value;
		}
	}
1422
1423
1424
1425

	public void setMessageHandler(MessageHandler messageHandler) {
		this.messageHandler = messageHandler;
	}
1426
1427
1428
1429
1430
1431
1432
1433
1434
1435
1436
1437
1438
1439
1440
1441
1442
1443
1444

	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
1445
			if (!(value >= 0 && value <= SotSManager.GlobalMaximumSpringVolume)) { return false; }
1446
1447
1448
1449
1450
1451
1452
1453
1454
1455
1456
1457
1458
1459
1460
1461
		} 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
1462
			if (!(value >= 0 && value <= StaminaManager.GlobalCharacterMaximumStamina)) { return false; }
1463
1464
1465
1466
1467
1468
1469
1470
1471
1472
		} 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
1473
			// see PlayerProperty.PROP_PLAYER_HCOIN comments
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
1509
1510
1511
1512
1513
1514
1515
1516
1517
1518
1519
1520
1521
1522
1523
		} 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
1524
}