mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-17 22:57:21 +01:00
commit3aba0abeddAuthor: SabreCat <sabe@habitica.com> Date: Mon Oct 2 20:51:20 2023 -0500 fix(router): use state to pass modal launch info commit541eadd319Merge:c0bb56c8c289fff49d02Author: SabreCat <sabe@habitica.com> Date: Mon Oct 2 20:12:40 2023 -0500 Merge branch 'release' into report-profile-modal commitc0bb56c8c2Author: SabreCat <sabe@habitica.com> Date: Wed Sep 27 16:15:28 2023 -0500 test(profiles): add integrations commit9b644e9ad8Author: SabreCat <sabe@habitica.com> Date: Tue Sep 26 17:17:22 2023 -0500 fix(profile): adjust margin commitbfefe5dfa9Author: SabreCat <sabe@habitica.com> Date: Tue Sep 26 17:12:24 2023 -0500 fix(profiles): moar layout fixes commit8f211ee3e2Author: SabreCat <sabe@habitica.com> Date: Mon Sep 25 17:32:04 2023 -0500 fix(profile): fix admin actions Correct "user is banned" banner Fix bouncing modal Add "Days" smart plural Fix leaky CSS on Market page Refactor some redundant functions commitb1d23ec88bMerge:ee9709a9e1a63cc84779Author: SabreCat <sabe@habitica.com> Date: Mon Sep 25 15:37:54 2023 -0500 Merge branch 'release' into report-profile-modal commitee9709a9e1Author: CuriousMagpie <eilatan@gmail.com> Date: Mon Sep 18 16:30:30 2023 -0400 WIP(profile): add banned banner, toggle switches now toggle, add "days" to Next Login Reward commitf80928a895Author: CuriousMagpie <eilatan@gmail.com> Date: Mon Sep 18 13:43:34 2023 -0400 update(node): update node modules commit1d552f7e80Author: SabreCat <sabe@habitica.com> Date: Fri Sep 15 16:52:22 2023 -0500 fix(import): remove empty import commitf55d74a95dAuthor: SabreCat <sabe@habitica.com> Date: Fri Sep 15 16:39:50 2023 -0500 refactor(profiles): remove email feature also still more visual cleanup of profile modal commit311c743284Author: SabreCat <sabe@habitica.com> Date: Fri Sep 15 15:44:56 2023 -0500 refactor(profile): remove page view commitf8632bf50dMerge:ec85159c659e25360102Author: SabreCat <sabe@habitica.com> Date: Fri Sep 15 15:23:21 2023 -0500 Merge branch 'release' into report-profile-modal commitec85159c65Author: SabreCat <sabe@habitica.com> Date: Mon Sep 11 22:53:14 2023 -0500 feat(profiles): load modal instead of page? commit9986082914Author: CuriousMagpie <eilatan@gmail.com> Date: Fri Sep 8 14:49:57 2023 -0400 WIP(profile): fixed a comment, woohoo commit6262a9ba0cMerge:ae2b614df2ea2b007b1aAuthor: CuriousMagpie <eilatan@gmail.com> Date: Fri Sep 8 13:40:23 2023 -0400 Merge remote-tracking branch 'origin/report-profile-modal' into report-profile-modal commitea2b007b1aAuthor: SabreCat <sabe@habitica.com> Date: Thu Sep 7 16:54:19 2023 -0500 fix(profile): focus behavior commitae2b614df2Author: CuriousMagpie <eilatan@gmail.com> Date: Thu Sep 7 17:47:08 2023 -0400 WIP(profile): styling updates commit2e0723f1b9Author: SabreCat <sabe@habitica.com> Date: Thu Sep 7 15:37:59 2023 -0500 feat(moderation): unflag profile Also a few stylistic tweaks commitedcf8113deAuthor: SabreCat <sabe@habitica.com> Date: Wed Sep 6 16:39:02 2023 -0500 WIP(profile): dropdown draft commit0691483d63Author: CuriousMagpie <eilatan@gmail.com> Date: Wed Sep 6 16:33:30 2023 -0400 WIP(profile): Styling and string updates commit7e9d57d10aAuthor: SabreCat <sabe@habitica.com> Date: Wed Sep 6 11:40:31 2023 -0500 feat(profile): functional dropdown buttons commita2989b2833Merge:af6575e40ce072d7c09cAuthor: SabreCat <sabe@habitica.com> Date: Wed Sep 6 10:04:57 2023 -0500 Merge branch 'release' into report-profile-modal commitaf6575e40cAuthor: CuriousMagpie <eilatan@gmail.com> Date: Wed Sep 6 11:01:05 2023 -0400 WIP(profile): comment cleanup commit7b1de37202Author: CuriousMagpie <eilatan@gmail.com> Date: Tue Sep 5 17:22:14 2023 -0400 WIP(profile): remove shadowban tooltip commitd1177c32b9Merge:321a01b08131f821021bAuthor: CuriousMagpie <eilatan@gmail.com> Date: Tue Sep 5 17:02:40 2023 -0400 Merge branch 'sabrecat/report-profile' into report-profile-modal commit321a01b081Author: CuriousMagpie <eilatan@gmail.com> Date: Fri Sep 1 16:14:36 2023 -0400 WIP(profile): close button finally workinating commite143d36d28Author: CuriousMagpie <eilatan@gmail.com> Date: Fri Sep 1 15:52:38 2023 -0400 WIP(profile): close icon moved to profile.vue commit31f821021bMerge:a8f5e25d388957c5c009Author: SabreCat <sabe@habitica.com> Date: Fri Sep 1 14:52:31 2023 -0500 Merge branch 'report-profile-modal' into sabrecat/report-profile commit8957c5c009Merge:d340f06a220aec3866a4Author: CuriousMagpie <eilatan@gmail.com> Date: Fri Sep 1 15:38:12 2023 -0400 Merge remote-tracking branch 'origin/report-profile-modal' into report-profile-modal commitd340f06a22Author: CuriousMagpie <eilatan@gmail.com> Date: Fri Sep 1 15:37:57 2023 -0400 WIP(profile): fixed user not found error commit0aec3866a4Merge:b01f323b14ac7c8e0eb6Author: Natalie <78037386+CuriousMagpie@users.noreply.github.com> Date: Fri Sep 1 15:28:58 2023 -0400 Merge branch 'HabitRPG:develop' into report-profile-modal commita8f5e25d38Author: SabreCat <sabe@habitica.com> Date: Thu Aug 31 17:02:07 2023 -0500 feat(community): basic "report profile" commitb01f323b14Author: CuriousMagpie <eilatan@gmail.com> Date: Thu Aug 31 17:42:12 2023 -0400 WIP(profile): removed refactoring crud, located where close icon should be (profileModal.vue) commitce7d51a20cMerge:010f2299f0ac7c8e0eb6Author: SabreCat <sabe@habitica.com> Date: Thu Aug 31 14:20:37 2023 -0500 Merge branch 'release' into sabrecat/report-profile commit18b41acd94Author: CuriousMagpie <eilatan@gmail.com> Date: Thu Aug 31 12:23:41 2023 -0400 WIP(profile): moar buttonz commit9387b3a6bcAuthor: CuriousMagpie <eilatan@gmail.com> Date: Wed Aug 30 17:21:36 2023 -0400 WIP(profile): buttons commitb3ea48c4f5Author: CuriousMagpie <eilatan@gmail.com> Date: Fri Aug 25 15:52:41 2023 -0400 WIP(profile): work on achievement component commita1ceb2ea75Author: CuriousMagpie <eilatan@gmail.com> Date: Fri Aug 25 14:39:12 2023 -0400 WIP(profile): create achievements component commit4a24d9b80bMerge:8fe263a3771e05297e96Author: CuriousMagpie <eilatan@gmail.com> Date: Wed Aug 23 13:14:39 2023 -0400 Merge branch 'develop' into report-profile-modal commit1e05297e96Author: CuriousMagpie <eilatan@gmail.com> Date: Wed Aug 23 13:12:52 2023 -0400 package updates commit8fe263a377Author: CuriousMagpie <eilatan@gmail.com> Date: Wed Aug 23 12:12:36 2023 -0400 update(dependencies): ran npm install to update dependencies commit190fe048a1Merge:3ea48ab5cbfa83d1a9cfAuthor: CuriousMagpie <eilatan@gmail.com> Date: Wed Aug 23 11:52:08 2023 -0400 Merge branch 'develop' into report-profile-modal commit3ea48ab5cbAuthor: CuriousMagpie <eilatan@gmail.com> Date: Fri Aug 11 17:12:31 2023 -0400 WIP(user profile): dropdown menu and toggles and colors oh my commitc301a2b460Merge:1da6af11b5647b27c55fAuthor: CuriousMagpie <eilatan@gmail.com> Date: Fri Aug 11 12:40:07 2023 -0400 Merge branch 'develop' into report-profile-modal commit1da6af11b5Author: CuriousMagpie <eilatan@gmail.com> Date: Thu Aug 10 16:50:07 2023 -0400 WIP(user profile): moved some CSS classes out of unscoped and into the scoped section, started on toggle buttons commitdd55cbc928Author: CuriousMagpie <eilatan@gmail.com> Date: Wed Aug 9 15:38:46 2023 -0400 WIP(user profile): workin on the hamburger (kebab?) menu commit3834093207Author: CuriousMagpie <eilatan@gmail.com> Date: Tue Aug 8 14:14:40 2023 -0400 WIP(user profiles): working on the drop down menu commitf2be588195Author: CuriousMagpie <eilatan@gmail.com> Date: Mon Aug 7 16:10:30 2023 -0400 WIP(user profile): options menu commit010f2299f0Author: SabreCat <sabe@habitica.com> Date: Mon Aug 7 11:49:04 2023 -0500 fix(lint): eof and const commit4551dbf4b3Author: CuriousMagpie <eilatan@gmail.com> Date: Fri Aug 4 15:34:05 2023 -0400 WIP(user profile): styling the top portion of the modal commit19a9fe3644Author: CuriousMagpie <eilatan@gmail.com> Date: Thu Aug 3 15:06:51 2023 -0400 WIP(user profile): adding buttons commitdfdb305b1cAuthor: CuriousMagpie <eilatan@gmail.com> Date: Wed Aug 2 14:41:20 2023 -0400 WIP(user profile): layout commitded4eee693Author: CuriousMagpie <eilatan@gmail.com> Date: Wed Aug 2 12:04:02 2023 -0400 WIP(user profile): start flex grid & tidy up CSS commitaaca48be32Author: CuriousMagpie <eilatan@gmail.com> Date: Fri Jul 28 16:44:06 2023 -0400 WIP(user profile): mostly css updates commite531985b87Author: CuriousMagpie <eilatan@gmail.com> Date: Thu Jul 27 16:49:44 2023 -0400 WIP(user profile): one infinitesimal change that's hardly worth the electricity it's made from commiteb4021fcc7Author: CuriousMagpie <eilatan@gmail.com> Date: Wed Jul 26 16:33:05 2023 -0400 feat(content): upgrade profile page commit1b25394f3eMerge:c50cee0d888558dcc3a8Author: CuriousMagpie <eilatan@gmail.com> Date: Wed Jul 26 11:50:12 2023 -0400 Merge branch 'develop' into report-profile-modal commitc50cee0d88Author: SabreCat <sabe@habitica.com> Date: Wed Jul 12 16:32:25 2023 -0500 fix(flagging): debug params issue Also add and document the "source" body param commit55848c58beAuthor: SabreCat <sabe@habitica.com> Date: Mon Jul 10 16:24:20 2023 -0500 WIP(members): basic report a user API commitdda6180792Author: SabreCat <sabe@habitica.com> Date: Thu Jul 6 10:05:07 2023 -0500 fix(lint): remove console.info
421 lines
14 KiB
JavaScript
421 lines
14 KiB
JavaScript
import _ from 'lodash';
|
|
import moment from 'moment';
|
|
import common from '../../../common';
|
|
import baseModel from '../../libs/baseModel';
|
|
import * as Tasks from '../task';
|
|
import {
|
|
model as UserNotification,
|
|
} from '../userNotification';
|
|
import {
|
|
model as PushDevice,
|
|
} from '../pushDevice';
|
|
import {
|
|
model as Tag,
|
|
} from '../tag';
|
|
import {
|
|
model as NewsPost,
|
|
} from '../newsPost';
|
|
import { // eslint-disable-line import/no-cycle
|
|
userActivityWebhook,
|
|
} from '../../libs/webhook';
|
|
import schema from './schema'; // eslint-disable-line import/no-cycle
|
|
|
|
schema.plugin(baseModel, {
|
|
// noSet is not used as updating uses a whitelist and creating only accepts
|
|
// specific params (password, email, username, ...)
|
|
noSet: [],
|
|
private: ['auth.local.hashed_password', 'auth.local.passwordHashMethod', 'auth.local.salt', '_cronSignature', '_ABtests', 'secret', 'profile.flags'],
|
|
toJSONTransform: function userToJSON (plainObj, originalDoc) {
|
|
plainObj._tmp = originalDoc._tmp; // be sure to send down drop notifs
|
|
|
|
if (plainObj._tmp && plainObj._tmp.leveledUp) {
|
|
delete plainObj._tmp.leveledUp;
|
|
}
|
|
|
|
delete plainObj.filters;
|
|
|
|
if (plainObj.flags && originalDoc.isSelected('flags.lastNewStuffRead')) {
|
|
plainObj.flags.newStuff = originalDoc.checkNewStuff();
|
|
}
|
|
|
|
if (plainObj.auth && plainObj.auth.local && originalDoc.auth.local.hashed_password) {
|
|
plainObj.auth.local.has_password = true;
|
|
} else if (plainObj.auth && plainObj.auth.local && originalDoc.auth.local.email) {
|
|
plainObj.auth.local.has_password = false;
|
|
}
|
|
|
|
return plainObj;
|
|
},
|
|
});
|
|
|
|
function findTag (user, tagName) {
|
|
const tagID = _.find(user.tags, userTag => userTag.name === tagName(user.preferences.language));
|
|
return tagID.id;
|
|
}
|
|
|
|
function _populateDefaultTasks (user, taskTypes) {
|
|
let defaultsData;
|
|
if (user.registeredThrough === 'habitica-android' || user.registeredThrough === 'habitica-ios') {
|
|
defaultsData = common.content.userDefaultsMobile;
|
|
} else {
|
|
defaultsData = common.content.userDefaults;
|
|
}
|
|
const tagsI = taskTypes.indexOf('tag');
|
|
|
|
if (tagsI !== -1) {
|
|
user.tags = _.map(defaultsData.tags, tag => {
|
|
const newTag = _.cloneDeep(tag);
|
|
|
|
// tasks automatically get _id=helpers.uuid() from TaskSchema id.default,
|
|
// but tags are Schema.Types.Mixed - so we need to manually invoke here
|
|
newTag.id = common.uuid();
|
|
// Render tag's name in user's language
|
|
newTag.name = newTag.name(user.preferences.language);
|
|
return newTag;
|
|
});
|
|
}
|
|
|
|
// @TODO: default tasks are handled differently now, and not during registration.
|
|
// We should move this code
|
|
|
|
// TODO why isn't this using createTasks from libs/tasksManager?
|
|
|
|
const tasksToCreate = [];
|
|
if (user.registeredThrough === 'habitica-web') return Promise.all(tasksToCreate);
|
|
|
|
if (tagsI !== -1) {
|
|
taskTypes = _.clone(taskTypes); // eslint-disable-line no-param-reassign
|
|
taskTypes.splice(tagsI, 1);
|
|
}
|
|
|
|
_.each(taskTypes, taskType => {
|
|
const tasksOfType = _.map(defaultsData[`${taskType}s`], taskDefaults => {
|
|
const newTask = new Tasks[taskType](taskDefaults);
|
|
|
|
newTask.userId = user._id;
|
|
newTask.text = taskDefaults.text(user.preferences.language);
|
|
if (newTask.notes) newTask.notes = taskDefaults.notes(user.preferences.language);
|
|
if (taskDefaults.checklist) {
|
|
newTask.checklist = _.map(taskDefaults.checklist, checklistItem => {
|
|
checklistItem.text = checklistItem.text(user.preferences.language);
|
|
return checklistItem;
|
|
});
|
|
}
|
|
|
|
if (taskDefaults.tags) {
|
|
newTask.tags = _.compact(_.map(taskDefaults.tags, _.partial(findTag, user)));
|
|
}
|
|
|
|
return newTask.save();
|
|
});
|
|
|
|
tasksToCreate.push(...tasksOfType);
|
|
});
|
|
|
|
return Promise.all(tasksToCreate)
|
|
.then(tasksCreated => {
|
|
_.each(tasksCreated, task => {
|
|
user.tasksOrder[`${task.type}s`].push(task._id);
|
|
});
|
|
});
|
|
}
|
|
|
|
function pinBaseItems (user) {
|
|
const itemsPaths = [
|
|
'weapon_warrior_0', 'armor_warrior_1',
|
|
'shield_warrior_1', 'head_warrior_1',
|
|
];
|
|
|
|
itemsPaths.map(p => user.pinnedItems.push({
|
|
type: 'marketGear',
|
|
path: `gear.flat.${p}`,
|
|
}));
|
|
|
|
user.pinnedItems.push(
|
|
{ type: 'potion', path: 'potion' },
|
|
{ type: 'armoire', path: 'armoire' },
|
|
);
|
|
}
|
|
|
|
function _setUpNewUser (user) {
|
|
// Mark the last news post as read
|
|
const lastNewsPost = NewsPost.lastNewsPost();
|
|
if (lastNewsPost) {
|
|
user.flags.lastNewStuffRead = lastNewsPost._id;
|
|
}
|
|
|
|
let taskTypes;
|
|
const iterableFlags = user.flags.toObject();
|
|
|
|
user.items.quests.dustbunnies = 1;
|
|
user.purchased.background.violet = true;
|
|
user.preferences.background = 'violet';
|
|
if (moment().isBefore('2023-03-15T12:00-05:00')) {
|
|
user.migration = '20230314_pi_day';
|
|
user.items.gear.owned.head_special_piDay = true;
|
|
user.items.gear.equipped.head = 'head_special_piDay';
|
|
user.items.gear.owned.shield_special_piDay = true;
|
|
user.items.gear.equipped.shield = 'shield_special_piDay';
|
|
user.items.food.Pie_Skeleton = 1;
|
|
user.items.food.Pie_Base = 1;
|
|
user.items.food.Pie_CottonCandyBlue = 1;
|
|
user.items.food.Pie_CottonCandyPink = 1;
|
|
user.items.food.Pie_Shade = 1;
|
|
user.items.food.Pie_White = 1;
|
|
user.items.food.Pie_Golden = 1;
|
|
user.items.food.Pie_Zombie = 1;
|
|
user.items.food.Pie_Desert = 1;
|
|
user.items.food.Pie_Red = 1;
|
|
}
|
|
|
|
user.markModified('items achievements');
|
|
|
|
if (user.registeredThrough === 'habitica-web') {
|
|
taskTypes = ['habit', 'daily', 'todo', 'reward', 'tag'];
|
|
|
|
_.each(iterableFlags.tutorial.common, (val, section) => {
|
|
user.flags.tutorial.common[section] = true;
|
|
});
|
|
} else {
|
|
user.flags.showTour = false;
|
|
|
|
_.each(iterableFlags.tour, (val, section) => {
|
|
user.flags.tour[section] = -2;
|
|
});
|
|
|
|
if (user.registeredThrough === 'habitica-android' || user.registeredThrough === 'habitica-ios') {
|
|
taskTypes = ['habit', 'daily', 'todo', 'reward', 'tag'];
|
|
} else {
|
|
taskTypes = ['todo', 'tag'];
|
|
}
|
|
}
|
|
|
|
pinBaseItems(user);
|
|
return _populateDefaultTasks(user, taskTypes);
|
|
}
|
|
|
|
function _setProfileName (user) {
|
|
const localUsername = user.auth.local && user.auth.local.username;
|
|
const anonymous = 'profile name not found';
|
|
|
|
return localUsername || anonymous;
|
|
}
|
|
|
|
schema.post('init', function postInitUser () {
|
|
// Cleanup any corrupt data that could have ended up inside the user schema.
|
|
// In particular:
|
|
// - tags https://github.com/HabitRPG/habitica/issues/10688
|
|
// - notifications https://github.com/HabitRPG/habitica/issues/9923
|
|
// - push devices https://github.com/HabitRPG/habitica/issues/11805
|
|
// and https://github.com/HabitRPG/habitica/issues/11868
|
|
|
|
// Make sure notifications are loaded
|
|
if (this.isDirectSelected('notifications')) {
|
|
this.notifications = UserNotification.cleanupCorruptData(this.notifications);
|
|
}
|
|
|
|
// Make sure pushDevices are loaded
|
|
if (this.isDirectSelected('pushDevices')) {
|
|
this.pushDevices = PushDevice.cleanupCorruptData(this.pushDevices);
|
|
}
|
|
|
|
// Make sure tags are loaded
|
|
if (this.isDirectSelected('tags')) {
|
|
this.tags = Tag.cleanupCorruptData(this.tags);
|
|
}
|
|
|
|
return true;
|
|
});
|
|
|
|
schema.pre('validate', function preValidateUser (next) {
|
|
// Populate new user with profile name, not running in pre('save') because the field
|
|
// is required and validation fails if it doesn't exists like for new users
|
|
if (this.isNew && !this.profile.name) {
|
|
this.profile.name = _setProfileName(this);
|
|
}
|
|
|
|
next();
|
|
});
|
|
|
|
schema.pre('save', true, function preSaveUser (next, done) {
|
|
next();
|
|
|
|
// VERY IMPORTANT NOTE: when only some fields from an user document are selected
|
|
// using `.select('field1 field2')` when the user is saved we must make sure that
|
|
// these hooks do not run using default data. For example if user.items is missing
|
|
// we do not want to run any hook that relies on user.items because it will
|
|
// use the default values defined in the user schema and not the real ones.
|
|
//
|
|
// To check if a field was selected Document.isDirectSelected('field') can be used.
|
|
// more info on its usage can be found at https://mongoosejs.com/docs/api.html#document_Document-isDirectSelected
|
|
|
|
// do not calculate achievements if items or achievements are not selected
|
|
if (this.isDirectSelected('items') && this.isDirectSelected('achievements')) {
|
|
// Determines if Beast Master should be awarded
|
|
const beastMasterProgress = common.count.beastMasterProgress(this.items.pets);
|
|
|
|
if (
|
|
(beastMasterProgress >= 90 || this.achievements.beastMasterCount > 0)
|
|
&& this.achievements.beastMaster !== true
|
|
) {
|
|
this.achievements.beastMaster = true;
|
|
this.addNotification(
|
|
'ACHIEVEMENT_STABLE',
|
|
{
|
|
achievement: 'beastMaster',
|
|
achievementNotification: 'beastAchievement',
|
|
},
|
|
);
|
|
}
|
|
|
|
// Determines if Mount Master should be awarded
|
|
const mountMasterProgress = common.count.mountMasterProgress(this.items.mounts);
|
|
|
|
if (
|
|
(mountMasterProgress >= 90 || this.achievements.mountMasterCount > 0)
|
|
&& this.achievements.mountMaster !== true
|
|
) {
|
|
this.achievements.mountMaster = true;
|
|
this.addNotification(
|
|
'ACHIEVEMENT_STABLE',
|
|
{
|
|
achievement: 'mountMaster',
|
|
achievementNotification: 'mountAchievement',
|
|
},
|
|
);
|
|
}
|
|
|
|
// Determines if Triad Bingo should be awarded
|
|
const dropPetCount = common.count.dropPetsCurrentlyOwned(this.items.pets);
|
|
const qualifiesForTriad = dropPetCount >= 90 && mountMasterProgress >= 90;
|
|
|
|
if (
|
|
(qualifiesForTriad || this.achievements.triadBingoCount > 0)
|
|
&& this.achievements.triadBingo !== true
|
|
) {
|
|
this.achievements.triadBingo = true;
|
|
this.addNotification(
|
|
'ACHIEVEMENT_STABLE',
|
|
{
|
|
achievement: 'triadBingo',
|
|
achievementNotification: 'triadBingoAchievement',
|
|
},
|
|
);
|
|
}
|
|
|
|
// EXAMPLE CODE for allowing all existing and new players to be
|
|
// automatically granted an item during a certain time period:
|
|
// if (!this.items.pets['JackOLantern-Base'] && moment().isBefore('2014-11-01'))
|
|
// this.items.pets['JackOLantern-Base'] = 5;
|
|
// this.markModified('items.pets');
|
|
}
|
|
|
|
// Filter notifications, remove unvalid and not necessary,
|
|
// handle the ones that have special requirements
|
|
if ( // Make sure all the data is loaded
|
|
this.isDirectSelected('notifications')
|
|
&& this.isDirectSelected('stats')
|
|
&& this.isDirectSelected('flags')
|
|
&& this.isDirectSelected('preferences')
|
|
) {
|
|
const unallocatedPointsNotifications = [];
|
|
|
|
this.notifications = this.notifications.filter(notification => {
|
|
// Remove all unallocated stats points
|
|
if (notification.type === 'UNALLOCATED_STATS_POINTS') {
|
|
unallocatedPointsNotifications.push(notification);
|
|
return false;
|
|
}
|
|
// Keep all the others
|
|
return true;
|
|
});
|
|
|
|
// Handle unallocated stats points notifications (keep only one and up to date)
|
|
const pointsToAllocate = this.stats.points;
|
|
const classNotEnabled = !this.flags.classSelected || this.preferences.disableClasses;
|
|
|
|
// Take the most recent notification
|
|
const unallLengh = unallocatedPointsNotifications.length;
|
|
const lastExistingNotification = unallocatedPointsNotifications[unallLengh - 1];
|
|
|
|
// Decide if it's outdated or not
|
|
const outdatedNotification = !lastExistingNotification
|
|
|| lastExistingNotification.data.points !== pointsToAllocate;
|
|
|
|
// If there are points to allocate and the notification is outdated, add a new notifications
|
|
if (pointsToAllocate > 0 && !classNotEnabled) {
|
|
if (outdatedNotification) {
|
|
this.addNotification('UNALLOCATED_STATS_POINTS', { points: pointsToAllocate });
|
|
} else { // otherwise add back the last one
|
|
this.notifications.push(lastExistingNotification);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (this.isDirectSelected('flags')) {
|
|
// Enable weekly recap emails for old users who sign in
|
|
if (this.flags.lastWeeklyRecapDiscriminator) {
|
|
// Enable weekly recap emails in 24 hours
|
|
this.flags.lastWeeklyRecap = moment().subtract(6, 'days').toDate();
|
|
// Unset the field so this is run only once
|
|
this.flags.lastWeeklyRecapDiscriminator = undefined;
|
|
}
|
|
}
|
|
|
|
if (this.isDirectSelected('preferences')) {
|
|
if (
|
|
_.isNaN(this.preferences.dayStart)
|
|
|| this.preferences.dayStart < 0
|
|
|| this.preferences.dayStart > 23
|
|
) {
|
|
this.preferences.dayStart = 0;
|
|
}
|
|
}
|
|
|
|
// our own version incrementer
|
|
if (this.isDirectSelected('_v')) {
|
|
if (_.isNaN(this._v) || !_.isNumber(this._v)) this._v = 0;
|
|
this._v += 1;
|
|
}
|
|
|
|
// Populate new users with default content
|
|
if (this.isNew) {
|
|
_setUpNewUser(this)
|
|
.then(() => done())
|
|
.catch(done);
|
|
} else {
|
|
done();
|
|
}
|
|
});
|
|
|
|
schema.pre('update', function preUpdateUser () {
|
|
this.update({}, { $inc: { _v: 1 } });
|
|
});
|
|
|
|
schema.pre('updateOne', function preUpdateUser () {
|
|
this.updateOne({}, { $inc: { _v: 1 } });
|
|
});
|
|
schema.pre('updateMany', function preUpdateUser () {
|
|
this.updateMany({}, { $inc: { _v: 1 } });
|
|
});
|
|
|
|
schema.post('save', function postSaveUser () {
|
|
// Send a webhook notification when the user has leveled up
|
|
if (this._tmp && this._tmp.leveledUp && this._tmp.leveledUp.length > 0) {
|
|
const lvlUpNotifications = this._tmp.leveledUp;
|
|
const firstLvlNotification = lvlUpNotifications[0];
|
|
const lastLvlNotification = lvlUpNotifications[lvlUpNotifications.length - 1];
|
|
|
|
const { initialLvl } = firstLvlNotification;
|
|
const finalLvl = lastLvlNotification.newLvl;
|
|
|
|
userActivityWebhook.send(this, {
|
|
type: 'leveledUp',
|
|
initialLvl,
|
|
finalLvl,
|
|
});
|
|
|
|
this._tmp.leveledUp = [];
|
|
}
|
|
});
|