mirror of
https://github.com/overleaf/overleaf.git
synced 2025-01-07 15:30:53 +00:00
168 lines
5 KiB
CoffeeScript
168 lines
5 KiB
CoffeeScript
define [
|
|
"base"
|
|
"libs/passfield"
|
|
], (App) ->
|
|
App.directive "asyncForm", ($http, validateCaptcha) ->
|
|
return {
|
|
controller: ['$scope', ($scope) ->
|
|
@getEmail = () ->
|
|
return $scope.email
|
|
return this
|
|
]
|
|
link: (scope, element, attrs) ->
|
|
formName = attrs.asyncForm
|
|
|
|
scope[attrs.name].response = response = {}
|
|
scope[attrs.name].inflight = false
|
|
|
|
validateCaptchaIfEnabled = (callback = (response) ->) ->
|
|
if attrs.captcha?
|
|
validateCaptcha callback
|
|
else
|
|
callback()
|
|
|
|
submitRequest = (grecaptchaResponse) ->
|
|
formData = {}
|
|
for data in element.serializeArray()
|
|
formData[data.name] = data.value
|
|
|
|
if grecaptchaResponse?
|
|
formData['g-recaptcha-response'] = grecaptchaResponse
|
|
|
|
scope[attrs.name].inflight = true
|
|
|
|
# for asyncForm prevent automatic redirect to /login if
|
|
# authentication fails, we will handle it ourselves
|
|
$http
|
|
.post(element.attr('action'), formData, {disableAutoLoginRedirect: true})
|
|
.then (httpResponse) ->
|
|
{ data, status, headers, config } = httpResponse
|
|
scope[attrs.name].inflight = false
|
|
response.success = true
|
|
response.error = false
|
|
|
|
onSuccessHandler = scope[attrs.onSuccess]
|
|
if onSuccessHandler
|
|
onSuccessHandler(httpResponse)
|
|
return
|
|
|
|
if data.redir?
|
|
ga('send', 'event', formName, 'success')
|
|
window.location = data.redir
|
|
else if data.message?
|
|
response.message = data.message
|
|
|
|
if data.message.type == "error"
|
|
response.success = false
|
|
response.error = true
|
|
ga('send', 'event', formName, 'failure', data.message)
|
|
else
|
|
ga('send', 'event', formName, 'success')
|
|
|
|
.catch (httpResponse) ->
|
|
{ data, status, headers, config } = httpResponse
|
|
scope[attrs.name].inflight = false
|
|
response.success = false
|
|
response.error = true
|
|
|
|
onErrorHandler = scope[attrs.onError]
|
|
if onErrorHandler
|
|
onErrorHandler(httpResponse)
|
|
return
|
|
|
|
if status == 400 # Bad Request
|
|
response.message =
|
|
text: "Invalid Request. Please correct the data and try again."
|
|
type: 'error'
|
|
else if status == 403 # Forbidden
|
|
response.message =
|
|
text: "Session error. Please check you have cookies enabled. If the problem persists, try clearing your cache and cookies."
|
|
type: "error"
|
|
else
|
|
response.message =
|
|
text: data.message?.text or data.message or "Something went wrong talking to the server :(. Please try again."
|
|
type: 'error'
|
|
ga('send', 'event', formName, 'failure', data.message)
|
|
|
|
submit = () ->
|
|
validateCaptchaIfEnabled (response) ->
|
|
submitRequest response
|
|
|
|
element.on "submit", (e) ->
|
|
e.preventDefault()
|
|
submit()
|
|
|
|
if attrs.autoSubmit
|
|
submit()
|
|
}
|
|
|
|
App.directive "formMessages", () ->
|
|
return {
|
|
restrict: "E"
|
|
template: """
|
|
<div class="alert" ng-class="{
|
|
'alert-danger': form.response.message.type == 'error',
|
|
'alert-success': form.response.message.type != 'error'
|
|
}" ng-show="!!form.response.message">
|
|
{{form.response.message.text}}
|
|
</div>
|
|
<div ng-transclude></div>
|
|
"""
|
|
transclude: true
|
|
scope: {
|
|
form: "=for"
|
|
}
|
|
|
|
}
|
|
|
|
|
|
App.directive 'complexPassword', ->
|
|
require: ['^asyncForm', 'ngModel']
|
|
|
|
link: (scope, element, attrs, ctrl) ->
|
|
|
|
PassField.Config.blackList = []
|
|
defaultPasswordOpts =
|
|
pattern: ""
|
|
length:
|
|
min: 6
|
|
max: 128
|
|
allowEmpty: false
|
|
allowAnyChars: false
|
|
isMasked: true
|
|
showToggle: false
|
|
showGenerate: false
|
|
showTip:false
|
|
showWarn:false
|
|
checkMode : PassField.CheckModes.STRICT
|
|
chars:
|
|
digits: "1234567890"
|
|
letters: "abcdefghijklmnopqrstuvwxyz"
|
|
letters_up: "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
|
|
symbols: "@#$%^&*()-_=+[]{};:<>/?!£€.,"
|
|
|
|
opts = _.defaults(window.passwordStrengthOptions || {}, defaultPasswordOpts)
|
|
if opts.length.min == 1
|
|
opts.acceptRate = 0 #this allows basically anything to be a valid password
|
|
passField = new PassField.Field("passwordField", opts);
|
|
|
|
[asyncFormCtrl, ngModelCtrl] = ctrl
|
|
|
|
ngModelCtrl.$parsers.unshift (modelValue) ->
|
|
isValid = passField.validatePass()
|
|
email = asyncFormCtrl.getEmail() || window.usersEmail
|
|
if !isValid
|
|
scope.complexPasswordErrorMessage = passField.getPassValidationMessage()
|
|
else if (email? and email != "")
|
|
startOfEmail = email?.split("@")?[0]
|
|
if modelValue.indexOf(email) != -1 or modelValue.indexOf(startOfEmail) != -1
|
|
isValid = false
|
|
scope.complexPasswordErrorMessage = "Password can not contain email address"
|
|
if opts.length.max? and modelValue.length == opts.length.max
|
|
isValid = false
|
|
scope.complexPasswordErrorMessage = "Maximum password length #{opts.length.max} reached"
|
|
if opts.length.min? and modelValue.length < opts.length.min
|
|
isValid = false
|
|
scope.complexPasswordErrorMessage = "Password too short, minimum #{opts.length.min}"
|
|
ngModelCtrl.$setValidity('complexPassword', isValid)
|
|
return modelValue
|