Preventing users from buying already gifted subscriber items (#7734)

* Adding the unopened mystery items to the call to get not obtained subscriber items. closes #7712

* refactoring according to pr

* Refactoring according to pr. moved time-travelers to it's own file and added new tests.
This commit is contained in:
Travis
2016-12-15 08:00:49 -08:00
committed by Keith Holliday
parent 0817cf96e1
commit fa788f49fc
8 changed files with 79 additions and 31 deletions

View File

@@ -28,7 +28,10 @@ describe('Inventory Controller', function() {
}, },
preferences: { preferences: {
suppressModals: {} suppressModals: {}
} },
purchased: {
plan: {}
},
}); });
Shared.wrap(user); Shared.wrap(user);

View File

@@ -0,0 +1,29 @@
import _ from 'lodash';
import {
generateUser,
} from '../helpers/common.helper';
import timeTravelers from '../../website/common/script/content/time-travelers'
describe('time-travelers store', () => {
let user;
beforeEach(() => {
user = generateUser();
});
it('removes owned sets from the time travelers store', () => {
user.items.gear.owned['head_mystery_201602'] = true;
expect(timeTravelers.timeTravelerStore(user)['201602']).to.not.exist;
expect(timeTravelers.timeTravelerStore(user)['201603']).to.exist;
});
it('removes unopened mystery item sets from the time travelers store', () => {
user.purchased = {
plan: {
mysteryItems: ['head_mystery_201602'],
},
};
expect(timeTravelers.timeTravelerStore(user)['201602']).to.not.exist;
expect(timeTravelers.timeTravelerStore(user)['201603']).to.exist;
});
});

View File

@@ -371,9 +371,16 @@ habitrpg.controller("InventoryCtrl",
$scope.hasAllTimeTravelerItemsOfType('mounts')); $scope.hasAllTimeTravelerItemsOfType('mounts'));
}; };
$scope.shouldShowTimeTravelerItem = function(category, item) {
if (category.identifier === 'pets' || category.identifier === 'mounts') {
return !user.items[category.identifier][item.key];
}
return !user.items.gear.owned[item.key] && user.purchased.plan.mysteryItems.indexOf(item.key) === -1;
};
$scope.hasAllTimeTravelerItemsOfType = function(type) { $scope.hasAllTimeTravelerItemsOfType = function(type) {
if (type === 'mystery') { if (type === 'mystery') {
var itemsLeftInTimeTravelerStore = Content.timeTravelerStore(user.items.gear.owned); var itemsLeftInTimeTravelerStore = Content.timeTravelerStore(user);
var keys = Object.keys(itemsLeftInTimeTravelerStore); var keys = Object.keys(itemsLeftInTimeTravelerStore);
return keys.length === 0; return keys.length === 0;

View File

@@ -14,8 +14,6 @@ let api = module.exports;
import achievements from './achievements'; import achievements from './achievements';
import mysterySets from './mystery-sets';
import eggs from './eggs'; import eggs from './eggs';
import hatchingPotions from './hatching-potions'; import hatchingPotions from './hatching-potions';
import stable from './stable'; import stable from './stable';
@@ -25,39 +23,19 @@ import appearances from './appearance';
import backgrounds from './appearance/backgrounds.js' import backgrounds from './appearance/backgrounds.js'
import spells from './spells'; import spells from './spells';
import faq from './faq'; import faq from './faq';
import timeTravelers from './time-travelers';
import loginIncentives from './loginIncentives'; import loginIncentives from './loginIncentives';
api.achievements = achievements; api.achievements = achievements;
api.mystery = mysterySets;
api.itemList = ITEM_LIST; api.itemList = ITEM_LIST;
api.gear = gear; api.gear = gear;
api.spells = spells; api.spells = spells;
/* api.mystery = timeTravelers.mystery;
Time Traveler Store, mystery sets need their items mapped in api.timeTravelerStore = timeTravelers.timeTravelerStore;
*/
_.each(api.mystery, function(v, k) {
return v.items = _.where(api.gear.flat, {
mystery: k
});
});
api.timeTravelerStore = function(owned) {
var ownedKeys;
ownedKeys = _.keys((typeof owned.toObject === "function" ? owned.toObject() : void 0) || owned);
return _.reduce(api.mystery, function(m, v, k) {
if (k === 'wondercon' || ~ownedKeys.indexOf(v.items[0].key)) {
return m;
}
m[k] = v;
return m;
}, {});
};
/* /*
--------------------------------------------------------------- ---------------------------------------------------------------

View File

@@ -0,0 +1,31 @@
import _ from 'lodash';
import mysterySets from './mystery-sets';
import gear from './gear';
let mystery = mysterySets;
_.each(mystery, (v, k) => {
return v.items = _.where(gear.flat, {
mystery: k,
});
});
let timeTravelerStore = (user) => {
let ownedKeys;
let owned = user.items.gear.owned;
let unopenedGifts = user.purchased.plan.mysteryItems;
ownedKeys = _.keys((typeof owned.toObject === 'function' ? owned.toObject() : undefined) || owned);
ownedKeys = _.union(ownedKeys, unopenedGifts);
return _.reduce(mystery, (m, v, k) => {
if (k === 'wondercon' || ownedKeys.indexOf(v.items[0].key) !== -1) {
return m;
}
m[k] = v;
return m;
}, {});
};
module.exports = {
timeTravelerStore,
mystery,
};

View File

@@ -180,7 +180,7 @@ shops.getTimeTravelersCategories = function getTimeTravelersCategories (user, la
} }
} }
let sets = content.timeTravelerStore(user.items.gear.owned); let sets = content.timeTravelerStore(user);
for (let setKey in sets) { for (let setKey in sets) {
if (sets.hasOwnProperty(setKey)) { if (sets.hasOwnProperty(setKey)) {
let set = sets[setKey]; let set = sets[setKey];

View File

@@ -15,7 +15,7 @@ module.exports = function buyMysterySet (user, req = {}, analytics) {
throw new NotAuthorized(i18n.t('notEnoughHourglasses', req.language)); throw new NotAuthorized(i18n.t('notEnoughHourglasses', req.language));
} }
let ref = content.timeTravelerStore(user.items.gear.owned); let ref = content.timeTravelerStore(user);
let mysterySet = ref ? ref[key] : undefined; let mysterySet = ref ? ref[key] : undefined;
if (!mysterySet) { if (!mysterySet) {

View File

@@ -21,7 +21,7 @@
.col-md-12 .col-md-12
li.customize-menu.inventory-gear li.customize-menu.inventory-gear
menu.pets-menu(label='{{::category.text}}', ng-repeat='category in timeTravelersCategories') menu.pets-menu(label='{{::category.text}}', ng-repeat='category in timeTravelersCategories')
div(ng-repeat='item in category.items', ng-if='category.identifier === "pets" || category.identifier === "mounts" ? !user.items[category.identifier][item.key] : !user.items.gear.owned[item.key]') div(ng-repeat='item in category.items', ng-if='shouldShowTimeTravelerItem(category, item)')
button.customize-option(class='{{item.class ? item.class : "shop_"+item.key}}', button.customize-option(class='{{item.class ? item.class : "shop_"+item.key}}',
popover='{{item.notes}}', popover-title='{{item.text}}', popover='{{item.notes}}', popover-title='{{item.text}}',
popover-trigger='mouseenter', popover-placement='right', popover-trigger='mouseenter', popover-placement='right',