webmaster@1
|
1 // $Id: user.js,v 1.6 2007/09/12 18:29:32 goba Exp $ |
webmaster@1
|
2 |
webmaster@1
|
3 /** |
webmaster@1
|
4 * Attach handlers to evaluate the strength of any password fields and to check |
webmaster@1
|
5 * that its confirmation is correct. |
webmaster@1
|
6 */ |
webmaster@1
|
7 Drupal.behaviors.password = function(context) { |
webmaster@1
|
8 var translate = Drupal.settings.password; |
webmaster@1
|
9 $("input.password-field:not(.password-processed)", context).each(function() { |
webmaster@1
|
10 var passwordInput = $(this).addClass('password-processed'); |
webmaster@1
|
11 var parent = $(this).parent(); |
webmaster@1
|
12 // Wait this number of milliseconds before checking password. |
webmaster@1
|
13 var monitorDelay = 700; |
webmaster@1
|
14 |
webmaster@1
|
15 // Add the password strength layers. |
webmaster@1
|
16 $(this).after('<span class="password-strength"><span class="password-title">'+ translate.strengthTitle +'</span> <span class="password-result"></span></span>').parent(); |
webmaster@1
|
17 var passwordStrength = $("span.password-strength", parent); |
webmaster@1
|
18 var passwordResult = $("span.password-result", passwordStrength); |
webmaster@1
|
19 parent.addClass("password-parent"); |
webmaster@1
|
20 |
webmaster@1
|
21 // Add the password confirmation layer. |
webmaster@1
|
22 var outerItem = $(this).parent().parent(); |
webmaster@1
|
23 $("input.password-confirm", outerItem).after('<span class="password-confirm">'+ translate["confirmTitle"] +' <span></span></span>').parent().addClass("confirm-parent"); |
webmaster@1
|
24 var confirmInput = $("input.password-confirm", outerItem); |
webmaster@1
|
25 var confirmResult = $("span.password-confirm", outerItem); |
webmaster@1
|
26 var confirmChild = $("span", confirmResult); |
webmaster@1
|
27 |
webmaster@1
|
28 // Add the description box at the end. |
webmaster@1
|
29 $(confirmInput).parent().after('<div class="password-description"></div>'); |
webmaster@1
|
30 var passwordDescription = $("div.password-description", $(this).parent().parent()).hide(); |
webmaster@1
|
31 |
webmaster@1
|
32 // Check the password fields. |
webmaster@1
|
33 var passwordCheck = function () { |
webmaster@1
|
34 // Remove timers for a delayed check if they exist. |
webmaster@1
|
35 if (this.timer) { |
webmaster@1
|
36 clearTimeout(this.timer); |
webmaster@1
|
37 } |
webmaster@1
|
38 |
webmaster@1
|
39 // Verify that there is a password to check. |
webmaster@1
|
40 if (!passwordInput.val()) { |
webmaster@1
|
41 passwordStrength.css({ visibility: "hidden" }); |
webmaster@1
|
42 passwordDescription.hide(); |
webmaster@1
|
43 return; |
webmaster@1
|
44 } |
webmaster@1
|
45 |
webmaster@1
|
46 // Evaluate password strength. |
webmaster@1
|
47 |
webmaster@1
|
48 var result = Drupal.evaluatePasswordStrength(passwordInput.val()); |
webmaster@1
|
49 passwordResult.html(result.strength == "" ? "" : translate[result.strength +"Strength"]); |
webmaster@1
|
50 |
webmaster@1
|
51 // Map the password strength to the relevant drupal CSS class. |
webmaster@1
|
52 var classMap = { low: "error", medium: "warning", high: "ok" }; |
webmaster@1
|
53 var newClass = classMap[result.strength] || ""; |
webmaster@1
|
54 |
webmaster@1
|
55 // Remove the previous styling if any exists; add the new class. |
webmaster@1
|
56 if (this.passwordClass) { |
webmaster@1
|
57 passwordResult.removeClass(this.passwordClass); |
webmaster@1
|
58 passwordDescription.removeClass(this.passwordClass); |
webmaster@1
|
59 } |
webmaster@1
|
60 passwordDescription.html(result.message); |
webmaster@1
|
61 passwordResult.addClass(newClass); |
webmaster@1
|
62 if (result.strength == "high") { |
webmaster@1
|
63 passwordDescription.hide(); |
webmaster@1
|
64 } |
webmaster@1
|
65 else { |
webmaster@1
|
66 passwordDescription.addClass(newClass); |
webmaster@1
|
67 } |
webmaster@1
|
68 this.passwordClass = newClass; |
webmaster@1
|
69 |
webmaster@1
|
70 // Check that password and confirmation match. |
webmaster@1
|
71 |
webmaster@1
|
72 // Hide the result layer if confirmation is empty, otherwise show the layer. |
webmaster@1
|
73 confirmResult.css({ visibility: (confirmInput.val() == "" ? "hidden" : "visible") }); |
webmaster@1
|
74 |
webmaster@1
|
75 var success = passwordInput.val() == confirmInput.val(); |
webmaster@1
|
76 |
webmaster@1
|
77 // Remove the previous styling if any exists. |
webmaster@1
|
78 if (this.confirmClass) { |
webmaster@1
|
79 confirmChild.removeClass(this.confirmClass); |
webmaster@1
|
80 } |
webmaster@1
|
81 |
webmaster@1
|
82 // Fill in the correct message and set the class accordingly. |
webmaster@1
|
83 var confirmClass = success ? "ok" : "error"; |
webmaster@1
|
84 confirmChild.html(translate["confirm"+ (success ? "Success" : "Failure")]).addClass(confirmClass); |
webmaster@1
|
85 this.confirmClass = confirmClass; |
webmaster@1
|
86 |
webmaster@1
|
87 // Show the indicator and tips. |
webmaster@1
|
88 passwordStrength.css({ visibility: "visible" }); |
webmaster@1
|
89 passwordDescription.show(); |
webmaster@1
|
90 }; |
webmaster@1
|
91 |
webmaster@1
|
92 // Do a delayed check on the password fields. |
webmaster@1
|
93 var passwordDelayedCheck = function() { |
webmaster@1
|
94 // Postpone the check since the user is most likely still typing. |
webmaster@1
|
95 if (this.timer) { |
webmaster@1
|
96 clearTimeout(this.timer); |
webmaster@1
|
97 } |
webmaster@1
|
98 |
webmaster@1
|
99 // When the user clears the field, hide the tips immediately. |
webmaster@1
|
100 if (!passwordInput.val()) { |
webmaster@1
|
101 passwordStrength.css({ visibility: "hidden" }); |
webmaster@1
|
102 passwordDescription.hide(); |
webmaster@1
|
103 return; |
webmaster@1
|
104 } |
webmaster@1
|
105 |
webmaster@1
|
106 // Schedule the actual check. |
webmaster@1
|
107 this.timer = setTimeout(passwordCheck, monitorDelay); |
webmaster@1
|
108 }; |
webmaster@1
|
109 // Monitor keyup and blur events. |
webmaster@1
|
110 // Blur must be used because a mouse paste does not trigger keyup. |
webmaster@1
|
111 passwordInput.keyup(passwordDelayedCheck).blur(passwordCheck); |
webmaster@1
|
112 confirmInput.keyup(passwordDelayedCheck).blur(passwordCheck); |
webmaster@1
|
113 }); |
webmaster@1
|
114 }; |
webmaster@1
|
115 |
webmaster@1
|
116 /** |
webmaster@1
|
117 * Evaluate the strength of a user's password. |
webmaster@1
|
118 * |
webmaster@1
|
119 * Returns the estimated strength and the relevant output message. |
webmaster@1
|
120 */ |
webmaster@1
|
121 Drupal.evaluatePasswordStrength = function(value) { |
webmaster@1
|
122 var strength = "", msg = "", translate = Drupal.settings.password; |
webmaster@1
|
123 |
webmaster@1
|
124 var hasLetters = value.match(/[a-zA-Z]+/); |
webmaster@1
|
125 var hasNumbers = value.match(/[0-9]+/); |
webmaster@1
|
126 var hasPunctuation = value.match(/[^a-zA-Z0-9]+/); |
webmaster@1
|
127 var hasCasing = value.match(/[a-z]+.*[A-Z]+|[A-Z]+.*[a-z]+/); |
webmaster@1
|
128 |
webmaster@1
|
129 // Check if the password is blank. |
webmaster@1
|
130 if (!value.length) { |
webmaster@1
|
131 strength = ""; |
webmaster@1
|
132 msg = ""; |
webmaster@1
|
133 } |
webmaster@1
|
134 // Check if length is less than 6 characters. |
webmaster@1
|
135 else if (value.length < 6) { |
webmaster@1
|
136 strength = "low"; |
webmaster@1
|
137 msg = translate.tooShort; |
webmaster@1
|
138 } |
webmaster@1
|
139 // Check if password is the same as the username (convert both to lowercase). |
webmaster@1
|
140 else if (value.toLowerCase() == translate.username.toLowerCase()) { |
webmaster@1
|
141 strength = "low"; |
webmaster@1
|
142 msg = translate.sameAsUsername; |
webmaster@1
|
143 } |
webmaster@1
|
144 // Check if it contains letters, numbers, punctuation, and upper/lower case. |
webmaster@1
|
145 else if (hasLetters && hasNumbers && hasPunctuation && hasCasing) { |
webmaster@1
|
146 strength = "high"; |
webmaster@1
|
147 } |
webmaster@1
|
148 // Password is not secure enough so construct the medium-strength message. |
webmaster@1
|
149 else { |
webmaster@1
|
150 // Extremely bad passwords still count as low. |
webmaster@1
|
151 var count = (hasLetters ? 1 : 0) + (hasNumbers ? 1 : 0) + (hasPunctuation ? 1 : 0) + (hasCasing ? 1 : 0); |
webmaster@1
|
152 strength = count > 1 ? "medium" : "low"; |
webmaster@1
|
153 |
webmaster@1
|
154 msg = []; |
webmaster@1
|
155 if (!hasLetters || !hasCasing) { |
webmaster@1
|
156 msg.push(translate.addLetters); |
webmaster@1
|
157 } |
webmaster@1
|
158 if (!hasNumbers) { |
webmaster@1
|
159 msg.push(translate.addNumbers); |
webmaster@1
|
160 } |
webmaster@1
|
161 if (!hasPunctuation) { |
webmaster@1
|
162 msg.push(translate.addPunctuation); |
webmaster@1
|
163 } |
webmaster@1
|
164 msg = translate.needsMoreVariation +"<ul><li>"+ msg.join("</li><li>") +"</li></ul>"; |
webmaster@1
|
165 } |
webmaster@1
|
166 |
webmaster@1
|
167 return { strength: strength, message: msg }; |
webmaster@1
|
168 }; |
webmaster@1
|
169 |
webmaster@1
|
170 /** |
webmaster@1
|
171 * Set the client's system timezone as default values of form fields. |
webmaster@1
|
172 */ |
webmaster@1
|
173 Drupal.setDefaultTimezone = function() { |
webmaster@1
|
174 var offset = new Date().getTimezoneOffset() * -60; |
webmaster@1
|
175 $("#edit-date-default-timezone, #edit-user-register-timezone").val(offset); |
webmaster@1
|
176 }; |
webmaster@1
|
177 |
webmaster@1
|
178 /** |
webmaster@1
|
179 * On the admin/user/settings page, conditionally show all of the |
webmaster@1
|
180 * picture-related form elements depending on the current value of the |
webmaster@1
|
181 * "Picture support" radio buttons. |
webmaster@1
|
182 */ |
webmaster@1
|
183 Drupal.behaviors.userSettings = function (context) { |
webmaster@1
|
184 $('div.user-admin-picture-radios input[type=radio]:not(.userSettings-processed)', context).addClass('userSettings-processed').click(function () { |
webmaster@1
|
185 $('div.user-admin-picture-settings', context)[['hide', 'show'][this.value]](); |
webmaster@1
|
186 }); |
webmaster@1
|
187 }; |
webmaster@1
|
188 |