mirror of
https://github.com/HabitRPG/habitica.git
synced 2025-12-18 23:27:26 +01:00
163 lines
5.3 KiB
JavaScript
163 lines
5.3 KiB
JavaScript
import isArray from 'lodash/isArray';
|
|
|
|
// @TODO: Let's separate some of the business logic out of Vue if possible
|
|
export default {
|
|
methods: {
|
|
handleCastCancelKeyUp (keyEvent) {
|
|
if (keyEvent.keyCode !== 27) return;
|
|
this.castCancel();
|
|
},
|
|
async castStart (spell, member) {
|
|
if (this.$store.state.spellOptions.castingSpell) {
|
|
this.castCancel();
|
|
return;
|
|
}
|
|
|
|
if (this.user.stats.mp < spell.mana) return this.text(this.$t('notEnoughMana'));
|
|
|
|
if (spell.immediateUse && this.user.stats.gp < spell.value) {
|
|
return this.text('Not enough gold.');
|
|
}
|
|
|
|
this.potionClickMode = true;
|
|
this.applyingAction = true;
|
|
this.$store.state.spellOptions.castingSpell = true;
|
|
this.spell = spell;
|
|
|
|
if (spell.target === 'self') {
|
|
return this.castEnd(null, spell.target);
|
|
} else if (spell.target === 'party') {
|
|
if (!this.user.party._id) {
|
|
let party = [this.user];
|
|
return this.castEnd(party, spell.target);
|
|
}
|
|
|
|
let partyMembers = this.$store.state.partyMembers.data;
|
|
if (!isArray(partyMembers)) {
|
|
partyMembers = [this.user];
|
|
}
|
|
this.castEnd(partyMembers, spell.target);
|
|
} else if (spell.target === 'tasks') {
|
|
let userTasks = this.$store.state.tasks.data;
|
|
// exclude rewards
|
|
let tasks = userTasks.habits
|
|
.concat(userTasks.dailys)
|
|
.concat(userTasks.todos);
|
|
// exclude challenge and group plan tasks
|
|
tasks = tasks.filter((task) => {
|
|
if (task.challenge && task.challenge.id && !task.challenge.broken) return false;
|
|
if (task.group && task.group.id && !task.group.broken) return false;
|
|
return true;
|
|
});
|
|
return this.castEnd(tasks, spell.target);
|
|
} else if (spell.target === 'user') {
|
|
let result = await this.castEnd(member, spell.target);
|
|
|
|
if (member.id === this.user.id) {
|
|
this.$set(this.$store.state.user.data, 'stats', member.stats);
|
|
}
|
|
|
|
return result;
|
|
} else {
|
|
// If the cast target has to be selected (and can be cancelled)
|
|
document.addEventListener('keyup', this.handleCastCancelKeyUp);
|
|
|
|
this.$root.$on('castEnd', (target, type, $event) => {
|
|
this.castEnd(target, type, $event);
|
|
});
|
|
}
|
|
},
|
|
async castEnd (target, type) {
|
|
if (!this.$store.state.spellOptions.castingSpell) return;
|
|
let beforeQuestProgress;
|
|
if (this.spell.target === 'party') beforeQuestProgress = this.questProgress();
|
|
|
|
if (!this.applyingAction) return 'No applying action';
|
|
|
|
if (this.spell.target !== type) return this.text(this.$t('invalidTarget'));
|
|
if (target && target.type && target.type === 'reward') return this.text(this.$t('invalidTarget'));
|
|
if (target && target.challenge && target.challenge.id) return this.text(this.$t('invalidTarget'));
|
|
if (target && target.group && target.group.id) return this.text(this.$t('invalidTarget'));
|
|
|
|
let spell = this.spell;
|
|
|
|
this.castCancel();
|
|
|
|
// the selected member doesn't have the flags property which sets `cardReceived`
|
|
if (spell.pinType !== 'card') {
|
|
spell.cast(this.user, target);
|
|
}
|
|
|
|
let targetId = target ? target._id : null;
|
|
|
|
let spellText = typeof spell.text === 'function' ? spell.text() : spell.text;
|
|
|
|
let apiResult = await this.$store.dispatch('user:castSpell', {
|
|
key: spell.key,
|
|
targetId,
|
|
pinType: spell.pinType,
|
|
});
|
|
let msg = '';
|
|
|
|
switch (type) {
|
|
case 'task':
|
|
msg = this.$t('youCastTarget', {
|
|
spell: spellText,
|
|
target: target.text,
|
|
});
|
|
break;
|
|
case 'user':
|
|
msg = spell.pinType === 'card' ?
|
|
this.$t('sentCardToUser', { profileName: target.profile.name }) :
|
|
this.$t('youCastTarget', {
|
|
spell: spellText,
|
|
target: target.profile.name,
|
|
});
|
|
break;
|
|
case 'party':
|
|
msg = this.$t('youCastParty', {
|
|
spell: spellText,
|
|
});
|
|
break;
|
|
default:
|
|
msg = this.$t('youCast', {
|
|
spell: spellText,
|
|
});
|
|
break;
|
|
}
|
|
|
|
if (spell.pinType === 'card') {
|
|
const newUserGp = apiResult.data.data.user.stats.gp;
|
|
this.$store.state.user.data.stats.gp = newUserGp;
|
|
}
|
|
|
|
|
|
this.markdown(msg); // @TODO: mardown directive?
|
|
if (!beforeQuestProgress) return;
|
|
let questProgress = this.questProgress() - beforeQuestProgress;
|
|
if (questProgress > 0) {
|
|
let userQuest = this.quests[this.user.party.quest.key];
|
|
if (userQuest.boss) {
|
|
this.quest('questDamage', questProgress.toFixed(1));
|
|
} else if (userQuest.collection && userQuest.collect) {
|
|
this.quest('questCollection', questProgress);
|
|
}
|
|
}
|
|
|
|
return;
|
|
// @TOOD: User.sync();
|
|
},
|
|
castCancel () {
|
|
this.potionClickMode = false;
|
|
this.applyingAction = false;
|
|
this.spell = null;
|
|
document.querySelector('body').style.cursor = 'initial';
|
|
this.$store.state.spellOptions.castingSpell = false;
|
|
|
|
// Remove listeners
|
|
this.$root.$off('castEnd');
|
|
document.removeEventListener('keyup', this.handleCastCancelKeyUp);
|
|
},
|
|
},
|
|
};
|