Skip to content

Commit

Permalink
Merge pull request #1029 from auth0/feature-show-password-button
Browse files Browse the repository at this point in the history
Adding "show password" option
  • Loading branch information
hzalaz authored Jun 22, 2017
2 parents 285de28 + b1647ee commit 16cbf99
Show file tree
Hide file tree
Showing 34 changed files with 250 additions and 88 deletions.
2 changes: 2 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -194,6 +194,8 @@ The appearance of the widget and the mechanics of authentication can be customiz
+ **text {String}**: The text to show.
- **allowAutocomplete {Boolean}**: Determines whether or not the the email or username inputs will allow autocomplete (`<input autocomplete />`). Defaults to `false`.
- **scrollGlobalMessagesIntoView {Boolean}**: Determines whether or not a globalMessage should be scrolled into the user's viewport. Defaults to `true`.
- **allowShowPassword {Boolean}**: Determines whether or not add a checkbox to show the password when typing it. Defaults to `false`.


#### Theming options

Expand Down
25 changes: 25 additions & 0 deletions css/index.styl
Original file line number Diff line number Diff line change
Expand Up @@ -1326,5 +1326,30 @@ tabsHeight = 40px
margin-right 5px
position relative

// show password
.auth0-lock-input-show-password
position relative

.auth0-lock-show-password
position absolute
top 14px
right 12px
width 20px
height 14px

input[type=checkbox]
display none

input[type=checkbox] + label
background-image url('')
width 20px
height 14px
display inline-block
cursor pointer
vertical-align top

input[type=checkbox]:checked + label
background-image url('')

input[type="button"]
cursor pointer
4 changes: 2 additions & 2 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -38,7 +38,7 @@
"test:jest:watch": "jest --watch --coverage",
"publish:cdn": "ccu",
"release": "scripts/release.sh",
"i18n:translate": "grunt dist; node scripts/complete-translations.js"
"i18n:translate": "grunt dist && node scripts/complete-translations.js"
},
"devDependencies": {
"babel-core": "^6.17.0",
Expand Down Expand Up @@ -145,4 +145,4 @@
"git add"
]
}
}
}
121 changes: 89 additions & 32 deletions src/__tests__/field/__snapshots__/password_pane.test.jsx.snap
Original file line number Diff line number Diff line change
@@ -1,54 +1,111 @@
// Jest Snapshot v1, https://goo.gl/fbAQLP

exports[`PasswordPane calls \`swap\` onChange 1`] = `
exports[`PasswordPane calls \`swap\` when checkbox is clicked 1`] = `
Array [
"updateEntity",
"lock",
1,
undefined,
true,
]
`;

exports[`PasswordPane calls \`swap\` when password changes 1`] = `
Array [
"updateEntity",
"lock",
1,
"setPassword",
"newUser",
"newPassword",
"policy",
]
`;

exports[`PasswordPane disables input when submitting 1`] = `
<div
data-__type="password_input"
data-disabled={true}
data-invalidHint="blankErrorHint"
data-isValid={false}
data-onChange={[Function]}
data-placeholder="placeholder"
data-policy="policy"
data-strengthMessages={Object {}}
data-value="password"
/>
className="auth0-lock-input-show-password"
>
<div
data-__type="password_input"
data-disabled={true}
data-invalidHint="blankErrorHint"
data-isValid={false}
data-onChange={[Function]}
data-placeholder="placeholder"
data-policy="policy"
data-showPassword="showPassword"
data-strengthMessages={Object {}}
data-value="password"
/>
</div>
`;

exports[`PasswordPane renders correctly 1`] = `
<div
data-__type="password_input"
data-disabled={false}
data-invalidHint="blankErrorHint"
data-isValid={false}
data-onChange={[Function]}
data-placeholder="placeholder"
data-policy="policy"
data-strengthMessages={Object {}}
data-value="password"
/>
className="auth0-lock-input-show-password"
>
<div
data-__type="password_input"
data-disabled={false}
data-invalidHint="blankErrorHint"
data-isValid={false}
data-onChange={[Function]}
data-placeholder="placeholder"
data-policy="policy"
data-showPassword="showPassword"
data-strengthMessages={Object {}}
data-value="password"
/>
</div>
`;

exports[`PasswordPane renders correctly when \`allowShowPassword\` is true 1`] = `
<div
className="auth0-lock-input-show-password"
>
<div
data-__type="password_input"
data-disabled={false}
data-invalidHint="blankErrorHint"
data-isValid={false}
data-onChange={[Function]}
data-placeholder="placeholder"
data-policy="policy"
data-showPassword="showPassword"
data-strengthMessages={Object {}}
data-value="password"
/>
<div
className="auth0-lock-show-password"
>
<input
id="slideOne"
onChange={[Function]}
type="checkbox"
/>
<label
htmlFor="slideOne"
title="showPassword"
/>
</div>
</div>
`;

exports[`PasswordPane sets isValid as true when \`isFieldVisiblyInvalid\` is false 1`] = `
<div
data-__type="password_input"
data-disabled={false}
data-invalidHint="blankErrorHint"
data-isValid={true}
data-onChange={[Function]}
data-placeholder="placeholder"
data-policy="policy"
data-strengthMessages={Object {}}
data-value="password"
/>
className="auth0-lock-input-show-password"
>
<div
data-__type="password_input"
data-disabled={false}
data-invalidHint="blankErrorHint"
data-isValid={true}
data-onChange={[Function]}
data-placeholder="placeholder"
data-policy="policy"
data-showPassword="showPassword"
data-strengthMessages={Object {}}
data-value="password"
/>
</div>
`;
30 changes: 24 additions & 6 deletions src/__tests__/field/password_pane.test.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,8 @@ const getComponent = () => require('field/password/password_pane').default;
describe('PasswordPane', () => {
const defaultProps = {
i18n: {
str: (...keys) => keys.join(',')
str: (...keys) => keys.join(','),
html: (...keys) => keys.join(',')
},
lock: {},
placeholder: 'placeholder',
Expand All @@ -22,7 +23,7 @@ describe('PasswordPane', () => {
jest.resetModules();

jest.mock('field/index', () => ({
getFieldValue: () => 'password',
getFieldValue: (m, field) => field,
isFieldVisiblyInvalid: () => true
}));

Expand All @@ -34,7 +35,7 @@ describe('PasswordPane', () => {
id: () => 1,
submitting: () => false,
ui: {
avatar: () => false
allowShowPassword: () => false
}
}));

Expand All @@ -48,6 +49,11 @@ describe('PasswordPane', () => {
const PasswordPane = getComponent();
expectComponent(<PasswordPane {...defaultProps} />).toMatchSnapshot();
});
it('renders correctly when `allowShowPassword` is true', () => {
require('core/index').ui.allowShowPassword = () => true;
const PasswordPane = getComponent();
expectComponent(<PasswordPane {...defaultProps} />).toMatchSnapshot();
});
it('disables input when submitting', () => {
require('core/index').submitting = () => true;
const PasswordPane = getComponent();
Expand All @@ -60,12 +66,24 @@ describe('PasswordPane', () => {

expectComponent(<PasswordPane {...defaultProps} />).toMatchSnapshot();
});
it('calls `swap` onChange', () => {
it('calls `swap` when password changes', () => {
let PasswordPane = getComponent();

const wrapper = mount(<PasswordPane {...defaultProps} />);
const props = extractPropsFromWrapper(wrapper, 1);
props.onChange({ target: { value: 'newPassword' } });

const { mock } = require('store/index').swap;
expect(mock.calls.length).toBe(1);
expect(mock.calls[0]).toMatchSnapshot();
});
it('calls `swap` when checkbox is clicked', () => {
require('core/index').ui.allowShowPassword = () => true;
let PasswordPane = getComponent();

const wrapper = mount(<PasswordPane {...defaultProps} />);
const props = extractPropsFromWrapper(wrapper);
props.onChange({ target: { value: 'newUser' } });
const props = wrapper.find('div input').props();
props.onChange({ target: { checked: true } });

const { mock } = require('store/index').swap;
expect(mock.calls.length).toBe(1);
Expand Down
10 changes: 7 additions & 3 deletions src/core/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -155,6 +155,7 @@ function extractUIOptions(id, options) {
rememberLastLogin: undefined === options.rememberLastLogin ? true : !!options.rememberLastLogin,
allowAutocomplete: !!options.allowAutocomplete,
authButtonsTheme: typeof authButtons === 'object' ? authButtons : {},
allowShowPassword: !!options.allowShowPassword,
scrollGlobalMessagesIntoView: undefined === options.scrollGlobalMessagesIntoView
? true
: !!options.scrollGlobalMessagesIntoView
Expand Down Expand Up @@ -189,7 +190,8 @@ export const ui = {
authButtonsTheme: lock => getUIAttribute(lock, 'authButtonsTheme'),
rememberLastLogin: m => tget(m, 'rememberLastLogin', getUIAttribute(m, 'rememberLastLogin')),
allowAutocomplete: m => tget(m, 'allowAutocomplete', getUIAttribute(m, 'allowAutocomplete')),
scrollGlobalMessagesIntoView: lock => getUIAttribute(lock, 'scrollGlobalMessagesIntoView')
scrollGlobalMessagesIntoView: lock => getUIAttribute(lock, 'scrollGlobalMessagesIntoView'),
allowShowPassword: m => tget(m, 'allowShowPassword', getUIAttribute(m, 'allowShowPassword'))
};

const { get: getAuthAttribute } = dataFns(['core', 'auth']);
Expand Down Expand Up @@ -217,8 +219,7 @@ function extractAuthOptions(options) {
sso,
state,
nonce
} =
options.auth || {};
} = options.auth || {};

let { oidcConformant } = options;

Expand Down Expand Up @@ -585,6 +586,9 @@ export function overrideOptions(m, opts) {
if (typeof opts.allowAutocomplete === 'boolean') {
m = tset(m, 'allowAutocomplete', opts.allowAutocomplete);
}
if (typeof opts.allowShowPassword === 'boolean') {
m = tset(m, 'allowShowPassword', opts.allowShowPassword);
}

return m;
}
4 changes: 4 additions & 0 deletions src/field/password.js
Original file line number Diff line number Diff line change
Expand Up @@ -8,3 +8,7 @@ export function validatePassword(password, policy) {
export function setPassword(m, password, policy) {
return setField(m, 'password', password, validatePassword, policy);
}

export function setShowPassword(m, checked) {
return setField(m, 'showPassword', checked, () => true);
}
39 changes: 25 additions & 14 deletions src/field/password/password_pane.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -4,28 +4,39 @@ import PasswordInput from '../../ui/input/password_input';
import * as c from '../index';
import { swap, updateEntity } from '../../store/index';
import * as l from '../../core/index';
import { setPassword } from '../password';
import { setPassword, setShowPassword } from '../password';

export default class PasswordPane extends React.Component {
handleChange(e) {
handleChange = e => {
const { lock, policy } = this.props;
swap(updateEntity, 'lock', l.id(lock), setPassword, e.target.value, policy);
}
};
handleShowPasswordChange = e => {
const { lock } = this.props;
swap(updateEntity, 'lock', l.id(lock), setShowPassword, e.target.checked);
};

render() {
const { i18n, lock, placeholder, policy, strengthMessages } = this.props;

return (
<PasswordInput
value={c.getFieldValue(lock, 'password')}
invalidHint={i18n.str('blankErrorHint')}
isValid={!c.isFieldVisiblyInvalid(lock, 'password')}
onChange={::this.handleChange}
placeholder={placeholder}
strengthMessages={strengthMessages}
disabled={l.submitting(lock)}
policy={policy}
/>
<div className="auth0-lock-input-show-password">
<PasswordInput
value={c.getFieldValue(lock, 'password')}
invalidHint={i18n.str('blankErrorHint')}
isValid={!c.isFieldVisiblyInvalid(lock, 'password')}
onChange={this.handleChange}
placeholder={placeholder}
strengthMessages={strengthMessages}
disabled={l.submitting(lock)}
policy={policy}
showPassword={c.getFieldValue(lock, 'showPassword', false)}
/>
{l.ui.allowShowPassword(lock) &&
<div className="auth0-lock-show-password">
<input type="checkbox" id="slideOne" onChange={this.handleShowPasswordChange} />
<label htmlFor="slideOne" title={i18n.str('showPassword')} />
</div>}
</div>
);
}
}
Expand Down
6 changes: 5 additions & 1 deletion src/i18n/ca.js
Original file line number Diff line number Diff line change
@@ -1,3 +1,6 @@
// This file was automatically translated.
// Feel free to submit a PR if you find a more accurate translation.

export default {
error: {
forgotPassword: {
Expand Down Expand Up @@ -109,5 +112,6 @@ export default {
mfaLoginTitle: 'Verificació en 2 passos',
mfaLoginInstructions: 'Indiqueu el codi de verificació generat per la seva aplicació de mòbil.',
mfaSubmitLabel: 'Inicia sessió',
mfaCodeErrorHint: 'Utilitzeu %d xifres'
mfaCodeErrorHint: 'Utilitzeu %d xifres',
showPassword: 'Ensenya la contrasenya'
};
3 changes: 2 additions & 1 deletion src/i18n/cs.js
Original file line number Diff line number Diff line change
Expand Up @@ -112,5 +112,6 @@ export default {
mfaSubmitLabel: 'Přihlásit',
mfaCodeErrorHint: 'Použijte %d čísel',
forgotPasswordTitle: 'Obnovit heslo',
signupTitle: 'Registrovat se'
signupTitle: 'Registrovat se',
showPassword: 'Zobrazit heslo'
};
3 changes: 2 additions & 1 deletion src/i18n/da.js
Original file line number Diff line number Diff line change
Expand Up @@ -112,5 +112,6 @@ export default {
mfaLoginTitle: 'Tofaktorgodkendelse',
mfaLoginInstructions: 'Indtast venligst bekræftelseskoden genereret af din mobilapplikation.',
mfaSubmitLabel: 'Log på',
mfaCodeErrorHint: 'Brug %d tal'
mfaCodeErrorHint: 'Brug %d tal',
showPassword: 'Vis adgangskode'
};
Loading

0 comments on commit 16cbf99

Please sign in to comment.