Files
habitica/website/public/js/services/userServices.js

423 lines
14 KiB
JavaScript

'use strict';
angular.module('habitrpg')
.service('ApiUrl', ['API_URL', function(currentApiUrl) {
this.setApiUrl = function(newUrl){
currentApiUrl = newUrl;
};
this.get = function(){
return currentApiUrl;
};
}])
/**
* Services that persists and retrieves user from localStorage.
*/
.factory('User', ['$rootScope', '$http', '$location', '$window', 'STORAGE_USER_ID', 'STORAGE_SETTINGS_ID', 'Notification', 'ApiUrl',
function($rootScope, $http, $location, $window, STORAGE_USER_ID, STORAGE_SETTINGS_ID, Notification, ApiUrl) {
var authenticated = false;
var defaultSettings = {
auth: { apiId: '', apiToken: ''},
sync: {
queue: [], //here OT will be queued up, this is NOT call-back queue!
sent: [] //here will be OT which have been sent, but we have not got reply from server yet.
},
fetching: false, // whether fetch() was called or no. this is to avoid race conditions
online: false
};
var settings = {}; //habit mobile settings (like auth etc.) to be stored here
var user = {}; // this is stored as a reference accessible to all controllers, that way updates propagate
var userNotifications = {
// "party.order" : env.t("updatedParty"),
// "party.orderAscending" : env.t("updatedParty")
// party.order notifications are not currently needed because the party avatars are resorted immediately now
}; // this is a list of notifications to send to the user when changes are made, along with the message.
//first we populate user with schema
user.apiToken = user._id = ''; // we use id / apitoken to determine if registered
//than we try to load localStorage
if (localStorage.getItem(STORAGE_USER_ID)) {
_.extend(user, JSON.parse(localStorage.getItem(STORAGE_USER_ID)));
}
user._wrapped = false;
function sync() {
$http({
method: "GET",
url: 'api/v3/user/',
})
.then(function (response) {
if (response.data.message) Notification.text(response.data.message);
_.extend(user, response.data.data);
if (!user._wrapped) {
// This wraps user with `ops`, which are functions shared both on client and mobile. When performed on client,
// they update the user in the browser and then send the request to the server, where the same operation is
// replicated. We need to wrap each op to provide a callback to send that operation
$window.habitrpgShared.wrap(user);
_.each(user.ops, function(op,k){
user.ops[k] = function(req,cb){
if (cb) return op(req,cb);
op(req,function(err,response) {
for(var updatedItem in req.body) {
var itemUpdateResponse = userNotifications[updatedItem];
if(itemUpdateResponse) Notification.text(itemUpdateResponse);
}
if (err) {
var message = err.code ? err.message : err;
Notification.text(message);
// In the case of 200s, they're friendly alert messages like "Your pet has hatched!" - still send the op
if ((err.code && err.code >= 400) || !err.code) return;
}
userServices.log({op:k, params: req.params, query:req.query, body:req.body});
});
}
});
}
save();
$rootScope.$emit('userSynced');
})
}
sync();
var save = function () {
localStorage.setItem(STORAGE_USER_ID, JSON.stringify(user));
localStorage.setItem(STORAGE_SETTINGS_ID, JSON.stringify(settings));
};
function callOpsFunctionAndRequest (opName, endPoint, method, paramString, opData) {
if (!opData) opData = {};
$window.habitrpgShared.ops[opName](user, opData);
var url = 'api/v3/user/' + endPoint;
if (paramString) {
url += '/' + paramString
}
var body = {};
if (opData.body) body = opData.body;
var queryString = '';
if (opData.query) queryString = '?' + $.param(opData.query)
$http({
method: method,
url: url + queryString,
body: body,
})
.then(function (response) {
if (response.data.message) Notification.text(response.data.message);
save();
})
}
function setUser(updates) {
for (var key in updates) {
user[key] = updates[key];
}
}
var userServices = {
user: user,
allocate: function (data) {
callOpsFunctionAndRequest('allocate', 'allocate', "POST",'', data);
},
changeClass: function (data) {
callOpsFunctionAndRequest('changeClass', 'change-class', "POST",'', data);
},
addTask: function (data) {
//@TODO: Should this been on habitrpgShared?
user.ops.addTask(data);
save();
//@TODO: Call task service when PR is merged
},
score: function (data) {
user.ops.scoreTask(data);
save();
//@TODO: Call task service when PR is merged
},
sortTask: function (data) {
user.ops.sortTask(data);
save();
//@TODO: Call task service when PR is merged
},
updateTask: function (data) {
user.ops.updateTask(data);
save();
//@TODO: Call task service when PR is merged
},
deleteTask: function (data) {
user.ops.deleteTask(data);
save();
//@TODO: Call task service when PR is merged
},
addTag: function(data) {
user.ops.addTag(data);
save();
//@TODO: Call task service when PR is merged
},
updateTag: function(data) {
user.ops.updateTag(data);
save();
//@TODO: Call task service when PR is merged
},
addTenGems: function () {
$http({
method: "POST",
url: 'api/v3/debug/add-ten-gems',
})
.then(function (response) {
Notification.text('+10 Gems!');
sync();
})
},
addHourglass: function () {
$http({
method: "POST",
url: 'api/v3/debug/add-hourglass',
})
.then(function (response) {
sync();
})
},
clearNewMessages: function () {
$http({
method: "POST",
url: 'api/v3/user/mark-pms-read',
})
.then(function (response) {
sync();
})
},
clearPMs: function () {
callOpsFunctionAndRequest('clearPMs', 'messages', "DELETE");
},
buy: function (data) {
callOpsFunctionAndRequest('buy', 'buy', "POST", data.params.key, data);
},
purchase: function (data) {
var type = data.params.type;
var key = data.params.key;
callOpsFunctionAndRequest('purchase', 'purchase', "POST", type + '/' + key, data);
},
buySpecialSpell: function (data) {
$window.habitrpgShared.ops['buySpecialSpell'](user, data);
var key = data.params.key;
$http({
method: "POST",
url: 'api/v3/user/' + 'buy-special-spell/' + key,
})
.then(function (response) {
Notification.text(response.data.message);
})
},
sell: function (data) {
var type = data.params.type;
var key = data.params.key;
callOpsFunctionAndRequest('sell', 'sell', "POST", type + '/' + key, data);
},
hatch: function (data) {
var egg = data.params.egg;
var hatchingPotion = data.params.hatchingPotion;
callOpsFunctionAndRequest('hatch', 'hatch', "POST", egg + '/' + hatchingPotion, data);
},
feed: function (data) {
var pet = data.params.pet;
var food = data.params.food;
callOpsFunctionAndRequest('feed', 'feed', "POST", pet + '/' + food, data);
},
equip: function (data) {
var type = data.params.type;
var key = data.params.key;
callOpsFunctionAndRequest('equip', 'equip', "POST", type + '/' + key, data);
},
hourglassPurchase: function (data) {
var type = data.params.type;
var key = data.params.key;
callOpsFunctionAndRequest('hourglassPurchase', 'purchase-hourglass', "POST", type + '/' + key, data);
},
unlock: function (data) {
callOpsFunctionAndRequest('unlock', 'unlock', "POST", '', data);
},
set: function(updates) {
setUser(updates);
$http({
method: "PUT",
url: 'api/v3/user',
data: updates,
})
.then(function (response) {
sync();
})
},
reroll: function () {
callOpsFunctionAndRequest('reroll', 'reroll', "POST");
},
rebirth: function () {
callOpsFunctionAndRequest('rebirth', 'rebirth', "POST");
},
reset: function () {
callOpsFunctionAndRequest('reset', 'reset', "POST");
},
releaseBoth: function () {
callOpsFunctionAndRequest('releaseBoth', 'releaseBoth', "POST");
},
releaseMounts: function () {
callOpsFunctionAndRequest('releaseMounts', 'releaseMounts', "POST");
},
releasePets: function () {
callOpsFunctionAndRequest('releasePets', 'releasePets', "POST");
},
addWebhook: function (data) {
callOpsFunctionAndRequest('addWebhook', 'webhook', "POST", '', data, data.body);
},
updateWebhook: function (data) {
callOpsFunctionAndRequest('updateWebhook', 'webhook', "PUT", data.params.id, data, data.body);
},
deleteWebhook: function (data) {
callOpsFunctionAndRequest('deleteWebhook', 'webhook', "DELETE", data.params.id, data, data.body);
},
sleep: function () {
callOpsFunctionAndRequest('sleep', 'sleep', "POST");
},
online: function (status) {
if (status===true) {
settings.online = true;
// syncQueue();
} else {
settings.online = false;
};
},
authenticate: function (uuid, token, cb) {
if (!!uuid && !!token) {
var offset = moment().zone(); // eg, 240 - this will be converted on server as -(offset/60)
$http.defaults.headers.common['x-api-user'] = uuid;
$http.defaults.headers.common['x-api-key'] = token;
$http.defaults.headers.common['x-user-timezoneOffset'] = offset;
authenticated = true;
settings.auth.apiId = uuid;
settings.auth.apiToken = token;
settings.online = true;
if (user && user._v) user._v--; // shortcut to always fetch new updates on page reload
userServices.log({}, function(){
// If they don't have timezone, set it
if (user.preferences.timezoneOffset !== offset)
userServices.set({'preferences.timezoneOffset': offset});
cb && cb();
});
} else {
alert('Please enter your ID and Token in settings.')
}
},
authenticated: function(){
return this.settings.auth.apiId !== "";
},
getBalanceInGems: function() {
var balance = user.balance || 0;
return balance * 4;
},
log: function (action, cb) {
//push by one buy one if an array passed in.
if (_.isArray(action)) {
action.forEach(function (a) {
settings.sync.queue.push(a);
});
} else {
settings.sync.queue.push(action);
}
save();
},
sync: function(){
userServices.log({});
sync();
},
save: save,
settings: settings
};
//load settings if we have them
if (localStorage.getItem(STORAGE_SETTINGS_ID)) {
//use extend here to make sure we keep object reference in other angular controllers
_.extend(settings, JSON.parse(localStorage.getItem(STORAGE_SETTINGS_ID)));
//if settings were saved while fetch was in process reset the flag.
settings.fetching = false;
//create and load if not
} else {
localStorage.setItem(STORAGE_SETTINGS_ID, JSON.stringify(defaultSettings));
_.extend(settings, defaultSettings);
}
//If user does not have ApiID that forward him to settings.
if (!settings.auth.apiId || !settings.auth.apiToken) {
//var search = $location.search(); // FIXME this should be working, but it's returning an empty object when at a root url /?_id=...
var search = $location.search($window.location.search.substring(1)).$$search; // so we use this fugly hack instead
if (search.err) return alert(search.err);
if (search._id && search.apiToken) {
userServices.authenticate(search._id, search.apiToken, function(){
$window.location.href='/';
});
} else {
var isStaticOrSocial = $window.location.pathname.match(/^\/(static|social)/);
if (!isStaticOrSocial){
localStorage.clear();
$window.location.href = '/logout';
}
}
} else {
userServices.authenticate(settings.auth.apiId, settings.auth.apiToken)
}
return userServices;
}
]);