mirror of
https://github.com/wekan/wekan.git
synced 2025-12-17 07:50:12 +01:00
Include to Wekan packages directory contents, so that meteor command would build all directly.
This also simplifies build scripts. Thanks to xet7 !
This commit is contained in:
parent
6117097a93
commit
73e265d8fd
354 changed files with 36977 additions and 106 deletions
464
packages/meteor-useraccounts-core/lib/client.js
Normal file
464
packages/meteor-useraccounts-core/lib/client.js
Normal file
|
|
@ -0,0 +1,464 @@
|
|||
/* 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();
|
||||
});
|
||||
Loading…
Add table
Add a link
Reference in a new issue