123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464 |
- /* global
- AT: false
- */
- "use strict";
- // Allowed Internal (client-side) States
- AT.prototype.STATES = [
- "changePwd", // Change Password
- "enrollAccount", // Account Enrollment
- "forgotPwd", // Forgot Password
- "hide", // Nothing displayed
- "resetPwd", // Reset Password
- "signIn", // Sign In
- "signUp", // Sign Up
- "verifyEmail", // Email verification
- "resendVerificationEmail", // Resend verification email
- ];
- AT.prototype._loginType = "";
- // Flag telling whether the whole form should appear disabled
- AT.prototype._disabled = false;
- // State validation
- AT.prototype._isValidState = function(value) {
- return _.contains(this.STATES, value);
- };
- // Flags used to avoid clearing errors and redirecting to previous route when
- // signing in/up as a results of a call to ensureSignedIn
- AT.prototype.avoidRedirect = false;
- AT.prototype.avoidClearError = false;
- // Token to be provided for routes like reset-password and enroll-account
- AT.prototype.paramToken = null;
- AT.prototype.loginType = function () {
- return this._loginType;
- };
- AT.prototype.getparamToken = function() {
- return this.paramToken;
- };
- // Getter for current state
- AT.prototype.getState = function() {
- return this.state.form.get("state");
- };
- // Getter for disabled state
- AT.prototype.disabled = function() {
- return this.state.form.equals("disabled", true) ? "disabled" : undefined;
- };
- // Setter for disabled state
- AT.prototype.setDisabled = function(value) {
- check(value, Boolean);
- return this.state.form.set("disabled", value);
- };
- // Setter for current state
- AT.prototype.setState = function(state, callback) {
- check(state, String);
- if (!this._isValidState(state) || (this.options.forbidClientAccountCreation && state === 'signUp')) {
- throw new Meteor.Error(500, "Internal server error", "accounts-templates-core package got an invalid state value!");
- }
- this.state.form.set("state", state);
- if (!this.avoidClearError) {
- this.clearState();
- }
- this.avoidClearError = false;
- if (_.isFunction(callback)) {
- callback();
- }
- };
- AT.prototype.clearState = function() {
- _.each(this._fields, function(field) {
- field.clearStatus();
- });
- var form = this.state.form;
- form.set("error", null);
- form.set("result", null);
- form.set("message", null);
- AccountsTemplates.setDisabled(false);
- };
- AT.prototype.clearError = function() {
- this.state.form.set("error", null);
- };
- AT.prototype.clearResult = function() {
- this.state.form.set("result", null);
- };
- AT.prototype.clearMessage = function() {
- this.state.form.set("message", null);
- };
- // Initialization
- AT.prototype.init = function() {
- console.warn("[AccountsTemplates] There is no more need to call AccountsTemplates.init()! Simply remove the call ;-)");
- };
- AT.prototype._init = function() {
- if (this._initialized) {
- return;
- }
- var usernamePresent = this.hasField("username");
- var emailPresent = this.hasField("email");
- if (usernamePresent && emailPresent) {
- this._loginType = "username_and_email";
- } else {
- this._loginType = usernamePresent ? "username" : "email";
- }
- if (this._loginType === "username_and_email") {
- // Possibly adds the field username_and_email in case
- // it was not configured
- if (!this.hasField("username_and_email")) {
- this.addField({
- _id: "username_and_email",
- type: "text",
- displayName: "usernameOrEmail",
- placeholder: "usernameOrEmail",
- required: true,
- });
- }
- }
- // Only in case password confirmation is required
- if (this.options.confirmPassword) {
- // Possibly adds the field password_again in case
- // it was not configured
- if (!this.hasField("password_again")) {
- var pwdAgain = _.clone(this.getField("password"));
- pwdAgain._id = "password_again";
- pwdAgain.displayName = {
- "default": "passwordAgain",
- changePwd: "newPasswordAgain",
- resetPwd: "newPasswordAgain",
- };
- pwdAgain.placeholder = {
- "default": "passwordAgain",
- changePwd: "newPasswordAgain",
- resetPwd: "newPasswordAgain",
- };
- this.addField(pwdAgain);
- }
- } else {
- if (this.hasField("password_again")) {
- throw new Error("AccountsTemplates: a field password_again was added but confirmPassword is set to false!");
- }
- }
- // Possibly adds the field current_password in case
- // it was not configured
- if (this.options.enablePasswordChange) {
- if (!this.hasField("current_password")) {
- this.addField({
- _id: "current_password",
- type: "password",
- displayName: "currentPassword",
- placeholder: "currentPassword",
- required: true,
- });
- }
- }
- // Ensuser the right order of special fields
- var moveFieldAfter = function(fieldName, referenceFieldName) {
- var fieldIds = AccountsTemplates.getFieldIds();
- var refFieldId = _.indexOf(fieldIds, referenceFieldName);
- // In case the reference field is not present, just return...
- if (refFieldId === -1) {
- return;
- }
- var fieldId = _.indexOf(fieldIds, fieldName);
- // In case the sought field is not present, just return...
- if (fieldId === -1) {
- return;
- }
- if (fieldId !== -1 && fieldId !== (refFieldId + 1)) {
- // removes the field
- var field = AccountsTemplates._fields.splice(fieldId, 1)[0];
- // push the field right after the reference field position
- var newFieldIds = AccountsTemplates.getFieldIds();
- var newReferenceFieldId = _.indexOf(newFieldIds, referenceFieldName);
- AccountsTemplates._fields.splice(newReferenceFieldId + 1, 0, field);
- }
- };
- // Ensuser the right order of special fields
- var moveFieldBefore = function(fieldName, referenceFieldName) {
- var fieldIds = AccountsTemplates.getFieldIds();
- var refFieldId = _.indexOf(fieldIds, referenceFieldName);
- // In case the reference field is not present, just return...
- if (refFieldId === -1) {
- return;
- }
- var fieldId = _.indexOf(fieldIds, fieldName);
- // In case the sought field is not present, just return...
- if (fieldId === -1) {
- return;
- }
- if (fieldId !== -1 && fieldId !== (refFieldId - 1)) {
- // removes the field
- var field = AccountsTemplates._fields.splice(fieldId, 1)[0];
- // push the field right after the reference field position
- var newFieldIds = AccountsTemplates.getFieldIds();
- var newReferenceFieldId = _.indexOf(newFieldIds, referenceFieldName);
- AccountsTemplates._fields.splice(newReferenceFieldId, 0, field);
- }
- };
- // The final order should be something like:
- // - username
- // - email
- // - username_and_email
- // - password
- // - password_again
- //
- // ...so lets do it in reverse order...
- moveFieldAfter("username_and_email", "username");
- moveFieldAfter("username_and_email", "email");
- moveFieldBefore("current_password", "password");
- moveFieldAfter("password", "current_password");
- moveFieldAfter("password_again", "password");
- // Sets visibility condition and validation flags for each field
- var gPositiveValidation = !!AccountsTemplates.options.positiveValidation;
- var gNegativeValidation = !!AccountsTemplates.options.negativeValidation;
- var gShowValidating = !!AccountsTemplates.options.showValidating;
- var gContinuousValidation = !!AccountsTemplates.options.continuousValidation;
- var gNegativeFeedback = !!AccountsTemplates.options.negativeFeedback;
- var gPositiveFeedback = !!AccountsTemplates.options.positiveFeedback;
- _.each(this._fields, function(field) {
- // Visibility
- switch(field._id) {
- case "current_password":
- field.visible = ["changePwd"];
- break;
- case "email":
- field.visible = ["forgotPwd", "signUp", "resendVerificationEmail"];
- if (AccountsTemplates.loginType() === "email") {
- field.visible.push("signIn");
- }
- break;
- case "password":
- field.visible = ["changePwd", "enrollAccount", "resetPwd", "signIn", "signUp"];
- break;
- case "password_again":
- field.visible = ["changePwd", "enrollAccount", "resetPwd", "signUp"];
- break;
- case "username":
- field.visible = ["signUp"];
- if (AccountsTemplates.loginType() === "username") {
- field.visible.push("signIn");
- }
- break;
- case "username_and_email":
- field.visible = [];
- if (AccountsTemplates.loginType() === "username_and_email") {
- field.visible.push("signIn");
- }
- break;
- default:
- field.visible = ["signUp"];
- }
- // Validation
- var positiveValidation = field.positiveValidation;
- if (_.isUndefined(positiveValidation)) {
- field.positiveValidation = gPositiveValidation;
- }
- var negativeValidation = field.negativeValidation;
- if (_.isUndefined(negativeValidation)) {
- field.negativeValidation = gNegativeValidation;
- }
- field.validation = field.positiveValidation || field.negativeValidation;
- if (_.isUndefined(field.continuousValidation)) {
- field.continuousValidation = gContinuousValidation;
- }
- field.continuousValidation = field.validation && field.continuousValidation;
- if (_.isUndefined(field.negativeFeedback)) {
- field.negativeFeedback = gNegativeFeedback;
- }
- if (_.isUndefined(field.positiveFeedback)) {
- field.positiveFeedback = gPositiveFeedback;
- }
- field.feedback = field.negativeFeedback || field.positiveFeedback;
- // Validating icon
- var showValidating = field.showValidating;
- if (_.isUndefined(showValidating)) {
- field.showValidating = gShowValidating;
- }
- // Custom Template
- if (field.template) {
- if (field.template in Template) {
- Template[field.template].helpers(AccountsTemplates.atInputHelpers);
- } else {
- console.warn(
- "[UserAccounts] Warning no template " + field.template + " found!"
- );
- }
- }
- });
- // Initializes reactive states
- var form = new ReactiveDict();
- form.set("disabled", false);
- form.set("state", "signIn");
- form.set("result", null);
- form.set("error", null);
- form.set("message", null);
- this.state = {
- form: form,
- };
- // Possibly subscribes to extended user data (to get the list of registered services...)
- if (this.options.showAddRemoveServices) {
- Meteor.subscribe("userRegisteredServices");
- }
- //Check that reCaptcha site keys are available and no secret keys visible
- if (this.options.showReCaptcha) {
- var atSiteKey = null;
- var atSecretKey = null;
- var settingsSiteKey = null;
- var settingsSecretKey = null;
- if (AccountsTemplates.options.reCaptcha) {
- atSiteKey = AccountsTemplates.options.reCaptcha.siteKey;
- atSecretKey = AccountsTemplates.options.reCaptcha.secretKey;
- }
- if (Meteor.settings && Meteor.settings.public && Meteor.settings.public.reCaptcha) {
- settingsSiteKey = Meteor.settings.public.reCaptcha.siteKey;
- settingsSecretKey = Meteor.settings.public.reCaptcha.secretKey;
- }
- if (atSecretKey || settingsSecretKey) {
- //erase the secret key
- if (atSecretKey) {
- AccountsTemplates.options.reCaptcha.secretKey = null;
- }
- if (settingsSecretKey) {
- Meteor.settings.public.reCaptcha.secretKey = null;
- }
- var loc = atSecretKey ? "User Accounts configuration!" : "Meteor settings!";
- throw new Meteor.Error(401, "User Accounts: DANGER - reCaptcha private key leaked to client from " + loc
- + " Provide the key in server settings ONLY.");
- }
- if (!atSiteKey && !settingsSiteKey) {
- throw new Meteor.Error(401, "User Accounts: reCaptcha site key not found! Please provide it or set showReCaptcha to false.");
- }
- }
- // Marks AccountsTemplates as initialized
- this._initialized = true;
- };
- AT.prototype.linkClick = function(route) {
- if (AccountsTemplates.disabled()) {
- return;
- }
- AccountsTemplates.setState(route);
- if (AccountsTemplates.options.focusFirstInput) {
- var firstVisibleInput = _.find(this.getFields(), function(f) {
- return _.contains(f.visible, route);
- });
- if (firstVisibleInput) {
- $("input#at-field-" + firstVisibleInput._id).focus();
- }
- }
- };
- AT.prototype.logout = function() {
- var onLogoutHook = AccountsTemplates.options.onLogoutHook;
- Meteor.logout(function() {
- if (onLogoutHook) {
- onLogoutHook();
- }
- });
- };
- AT.prototype.submitCallback = function(error, state, onSuccess) {
- var onSubmitHook = AccountsTemplates.options.onSubmitHook;
- if (onSubmitHook) {
- onSubmitHook(error, state);
- }
- if (error) {
- if (_.isObject(error.details)) {
- // If error.details is an object, we may try to set fields errors from it
- _.each(error.details, function(error, fieldId) {
- AccountsTemplates.getField(fieldId).setError(error);
- });
- } else {
- var err = "error.accounts.Unknown error";
- if (error.reason) {
- err = error.reason;
- }
- if (err.substring(0, 15) !== "error.accounts.") {
- err = "error.accounts." + err;
- }
- AccountsTemplates.state.form.set("error", [err]);
- }
- AccountsTemplates.setDisabled(false);
- // Possibly resets reCaptcha form
- if (state === "signUp" && AccountsTemplates.options.showReCaptcha) {
- grecaptcha.reset();
- }
- } else {
- if (onSuccess) {
- onSuccess();
- }
- if (state) {
- AccountsTemplates.setDisabled(false);
- }
- }
- };
- AccountsTemplates = new AT();
- // Initialization
- Meteor.startup(function() {
- AccountsTemplates._init();
- });
|