@gocredible/angular-password-strength-meter
Version:
[](https://github.com/GoCredible/password-strength-meter/actions/workflows/ci-workflow.yml) [ {
this.minPasswordLength = 8;
this.enableFeedback = false;
this.enableAsync = false;
this.colors = [];
this.numberOfProgressBarItems = 5;
this.strengthChange = new EventEmitter();
this.baseClass = 'psm';
this.passwordStrengthMeterService = inject(IPasswordStrengthMeterService);
this.passwordStrength = null;
this.feedback = null;
this.prevPasswordStrength = null;
this.passwordChangeObservable$ = new Subject();
this.init();
}
ngOnChanges(changes) {
if (changes['password']) {
this.passwordChangeObservable$.next(this.password);
}
}
init() {
this.passwordChangeObservable$
.pipe(distinctUntilChanged(), debounceTime(100), switchMap((value) => {
if (!value) {
return of({ score: null, feedback: null });
}
if (value && value.length < this.minPasswordLength) {
return of({ score: 0, feedback: null });
}
if (this.enableAsync) {
return this.calculateScoreAsync(value);
}
const result = this.calculateScore(value);
return of(result);
}), takeUntilDestroyed())
.subscribe((result) => {
this.passwordStrength = result.score;
this.feedback = result.feedback;
// Only emit the passwordStrength if it changed
if (this.prevPasswordStrength !== this.passwordStrength) {
this.strengthChange.emit(this.passwordStrength);
this.prevPasswordStrength = this.passwordStrength;
}
});
}
calculateScore(value) {
if (this.enableFeedback) {
return this.passwordStrengthMeterService.scoreWithFeedback(value);
}
const feedbackResult = {
score: this.passwordStrengthMeterService.score(value),
feedback: null,
};
return feedbackResult;
}
calculateScoreAsync(value) {
if (this.enableFeedback) {
return this.passwordStrengthMeterService.scoreWithFeedbackAsync(value);
}
return this.passwordStrengthMeterService
.scoreAsync(value)
.then((result) => ({
score: result,
feedback: null,
}));
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: PasswordStrengthMeterComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.12", type: PasswordStrengthMeterComponent, isStandalone: true, selector: "password-strength-meter", inputs: { password: "password", minPasswordLength: "minPasswordLength", enableFeedback: ["enableFeedback", "enableFeedback", booleanAttribute], enableAsync: ["enableAsync", "enableAsync", booleanAttribute], colors: "colors", numberOfProgressBarItems: "numberOfProgressBarItems" }, outputs: { strengthChange: "strengthChange" }, host: { properties: { "class": "this.baseClass" } }, usesOnChanges: true, ngImport: i0, template: "<div\n class=\"psm__progress-bar\"\n role=\"progressbar\"\n [passwordStrength]=\"passwordStrength\"\n [numberOfProgressBarItems]=\"numberOfProgressBarItems\"\n [colors]=\"colors\"\n>\n <div class=\"psm__progress-bar-items\"></div>\n <div class=\"psm__progress-bar-overlay\"></div>\n</div>\n\n@if(enableFeedback && feedback) {\n\n @if(feedback.warning) {\n <small class=\"psm__feedback\">\n {{ feedback.warning }}\n </small>\n }\n\n @if(feedback.suggestions && feedback.suggestions.length) {\n <small class=\"psm__suggestion\">\n @for (suggestion of feedback.suggestions; track suggestion; let isLast = $last) {\n {{ suggestion }}{{ isLast ? '' : ' ' }}\n }\n </small>\n }\n}\n\n", styles: ["*,*:before,*:after{box-sizing:border-box}.psm__progress-bar{position:relative;height:3px;margin:10px auto;border-radius:3px;background:#fff}.psm__progress-bar-items{display:flex;justify-content:space-evenly;width:100%;height:100%;background:#ddd}.psm__progress-bar-item{height:100%;z-index:5;border-right:4px solid #fff}.psm__progress-bar-item:last-child{border-right:0px}.psm__progress-bar-overlay{position:absolute;background:red;border-radius:3px;height:3px;top:0;transition:width .5s ease-in-out,background .25s}.psm__feedback,.psm__suggestion{font-size:70%;font-weight:400;color:#6c757d!important;display:block;margin-top:.25rem}\n"], dependencies: [{ kind: "directive", type: PSMProgressBarDirective, selector: ".psm__progress-bar", inputs: ["numberOfProgressBarItems", "passwordStrength", "colors"] }] }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: PasswordStrengthMeterComponent, decorators: [{
type: Component,
args: [{ standalone: true, selector: 'password-strength-meter', imports: [PSMProgressBarDirective], template: "<div\n class=\"psm__progress-bar\"\n role=\"progressbar\"\n [passwordStrength]=\"passwordStrength\"\n [numberOfProgressBarItems]=\"numberOfProgressBarItems\"\n [colors]=\"colors\"\n>\n <div class=\"psm__progress-bar-items\"></div>\n <div class=\"psm__progress-bar-overlay\"></div>\n</div>\n\n@if(enableFeedback && feedback) {\n\n @if(feedback.warning) {\n <small class=\"psm__feedback\">\n {{ feedback.warning }}\n </small>\n }\n\n @if(feedback.suggestions && feedback.suggestions.length) {\n <small class=\"psm__suggestion\">\n @for (suggestion of feedback.suggestions; track suggestion; let isLast = $last) {\n {{ suggestion }}{{ isLast ? '' : ' ' }}\n }\n </small>\n }\n}\n\n", styles: ["*,*:before,*:after{box-sizing:border-box}.psm__progress-bar{position:relative;height:3px;margin:10px auto;border-radius:3px;background:#fff}.psm__progress-bar-items{display:flex;justify-content:space-evenly;width:100%;height:100%;background:#ddd}.psm__progress-bar-item{height:100%;z-index:5;border-right:4px solid #fff}.psm__progress-bar-item:last-child{border-right:0px}.psm__progress-bar-overlay{position:absolute;background:red;border-radius:3px;height:3px;top:0;transition:width .5s ease-in-out,background .25s}.psm__feedback,.psm__suggestion{font-size:70%;font-weight:400;color:#6c757d!important;display:block;margin-top:.25rem}\n"] }]
}], ctorParameters: () => [], propDecorators: { password: [{
type: Input,
args: [{ required: true }]
}], minPasswordLength: [{
type: Input
}], enableFeedback: [{
type: Input,
args: [{ transform: booleanAttribute }]
}], enableAsync: [{
type: Input,
args: [{ transform: booleanAttribute }]
}], colors: [{
type: Input
}], numberOfProgressBarItems: [{
type: Input
}], strengthChange: [{
type: Output
}], baseClass: [{
type: HostBinding,
args: ['class']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"password-strength-meter.component.js","sourceRoot":"","sources":["../../../../projects/password-strength-meter/src/lib/password-strength-meter.component.ts","../../../../projects/password-strength-meter/src/lib/password-strength-meter.component.html"],"names":[],"mappings":"AAAA,uDAAuD;AACvD,OAAO,EACL,SAAS,EACT,KAAK,EAGL,MAAM,EACN,YAAY,EACZ,WAAW,EACX,MAAM,EACN,gBAAgB,GACjB,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,kBAAkB,EAAE,MAAM,4BAA4B,CAAC;AAChE,OAAO,EACL,OAAO,EACP,YAAY,EACZ,oBAAoB,EACpB,EAAE,EACF,SAAS,GACV,MAAM,MAAM,CAAC;AAEd,OAAO,EAGL,6BAA6B,GAC9B,MAAM,mCAAmC,CAAC;AAC3C,OAAO,EAAE,uBAAuB,EAAE,MAAM,8BAA8B,CAAC;;AASvE,MAAM,OAAO,8BAA8B;IA2BzC;QAxBS,sBAAiB,GAAG,CAAC,CAAC;QAES,mBAAc,GAAG,KAAK,CAAC;QAEvB,gBAAW,GAAG,KAAK,CAAC;QAEnD,WAAM,GAAa,EAAE,CAAC;QAEtB,6BAAwB,GAAG,CAAC,CAAC;QAE5B,mBAAc,GAAG,IAAI,YAAY,EAAiB,CAAC;QAEvC,cAAS,GAAG,KAAK,CAAC;QAEhC,iCAA4B,GAAkC,MAAM,CAC1E,6BAA6B,CAC9B,CAAC;QAEF,qBAAgB,GAAkB,IAAI,CAAC;QACvC,aAAQ,GAAoB,IAAI,CAAC;QAEzB,yBAAoB,GAAkB,IAAI,CAAC;QAC3C,8BAAyB,GAAG,IAAI,OAAO,EAAU,CAAC;QAGxD,IAAI,CAAC,IAAI,EAAE,CAAC;IACd,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,UAAU,CAAC,EAAE,CAAC;YACxB,IAAI,CAAC,yBAAyB,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QACrD,CAAC;IACH,CAAC;IAEO,IAAI;QACV,IAAI,CAAC,yBAAyB;aAC3B,IAAI,CACH,oBAAoB,EAAE,EACtB,YAAY,CAAC,GAAG,CAAC,EACjB,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE;YAClB,IAAI,CAAC,KAAK,EAAE,CAAC;gBACX,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;YAC7C,CAAC;YAED,IAAI,KAAK,IAAI,KAAK,CAAC,MAAM,GAAG,IAAI,CAAC,iBAAiB,EAAE,CAAC;gBACnD,OAAO,EAAE,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC;YAC1C,CAAC;YAED,IAAI,IAAI,CAAC,WAAW,EAAE,CAAC;gBACrB,OAAO,IAAI,CAAC,mBAAmB,CAAC,KAAK,CAAC,CAAC;YACzC,CAAC;YAED,MAAM,MAAM,GAAG,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;YAC1C,OAAO,EAAE,CAAC,MAAM,CAAC,CAAC;QACpB,CAAC,CAAC,EACF,kBAAkB,EAAE,CACrB;aACA,SAAS,CAAC,CAAC,MAAsB,EAAE,EAAE;YACpC,IAAI,CAAC,gBAAgB,GAAG,MAAM,CAAC,KAAK,CAAC;YACrC,IAAI,CAAC,QAAQ,GAAG,MAAM,CAAC,QAAQ,CAAC;YAEhC,+CAA+C;YAC/C,IAAI,IAAI,CAAC,oBAAoB,KAAK,IAAI,CAAC,gBAAgB,EAAE,CAAC;gBACxD,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;gBAChD,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC,gBAAgB,CAAC;YACpD,CAAC;QACH,CAAC,CAAC,CAAC;IACP,CAAC;IAEO,cAAc,CAAC,KAAa;QAClC,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;YACxB,OAAO,IAAI,CAAC,4BAA4B,CAAC,iBAAiB,CAAC,KAAK,CAAC,CAAC;QACpE,CAAC;QAED,MAAM,cAAc,GAAG;YACrB,KAAK,EAAE,IAAI,CAAC,4BAA4B,CAAC,KAAK,CAAC,KAAK,CAAC;YACrD,QAAQ,EAAE,IAAI;SACf,CAAC;QAEF,OAAO,cAAc,CAAC;IACxB,CAAC;IAEO,mBAAmB,CAAC,KAAa;QACvC,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;YACxB,OAAO,IAAI,CAAC,4BAA4B,CAAC,sBAAsB,CAAC,KAAK,CAAC,CAAC;QACzE,CAAC;QAED,OAAO,IAAI,CAAC,4BAA4B;aACrC,UAAU,CAAC,KAAK,CAAC;aACjB,IAAI,CAAC,CAAC,MAAM,EAAE,EAAE,CAAC,CAAC;YACjB,KAAK,EAAE,MAAM;YACb,QAAQ,EAAE,IAAI;SACf,CAAC,CAAC,CAAC;IACR,CAAC;+GAhGU,8BAA8B;mGAA9B,8BAA8B,wLAKrB,gBAAgB,+CAEhB,gBAAgB,8NC1CtC,qtBA4BA,srBDKY,uBAAuB;;4FAEtB,8BAA8B;kBAP1C,SAAS;iCACI,IAAI,YACN,yBAAyB,WAG1B,CAAC,uBAAuB,CAAC;wDAGP,QAAQ;sBAAlC,KAAK;uBAAC,EAAE,QAAQ,EAAE,IAAI,EAAE;gBAEhB,iBAAiB;sBAAzB,KAAK;gBAEkC,cAAc;sBAArD,KAAK;uBAAC,EAAE,SAAS,EAAE,gBAAgB,EAAE;gBAEE,WAAW;sBAAlD,KAAK;uBAAC,EAAE,SAAS,EAAE,gBAAgB,EAAE;gBAE7B,MAAM;sBAAd,KAAK;gBAEG,wBAAwB;sBAAhC,KAAK;gBAEI,cAAc;sBAAvB,MAAM;gBAEe,SAAS;sBAA9B,WAAW;uBAAC,OAAO","sourcesContent":["/* eslint-disable @angular-eslint/component-selector */\nimport {\n  Component,\n  Input,\n  OnChanges,\n  SimpleChanges,\n  Output,\n  EventEmitter,\n  HostBinding,\n  inject,\n  booleanAttribute,\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport {\n  Subject,\n  debounceTime,\n  distinctUntilChanged,\n  of,\n  switchMap,\n} from 'rxjs';\n\nimport {\n  Feedback,\n  FeedbackResult,\n  IPasswordStrengthMeterService,\n} from './password-strength-meter-service';\nimport { PSMProgressBarDirective } from './psm-progress-bar.directive';\n\n@Component({\n  standalone: true,\n  selector: 'password-strength-meter',\n  templateUrl: './password-strength-meter.component.html',\n  styleUrls: ['./password-strength-meter.component.scss'],\n  imports: [PSMProgressBarDirective],\n})\nexport class PasswordStrengthMeterComponent implements OnChanges {\n  @Input({ required: true }) password!: string;\n\n  @Input() minPasswordLength = 8;\n\n  @Input({ transform: booleanAttribute }) enableFeedback = false;\n\n  @Input({ transform: booleanAttribute }) enableAsync = false;\n\n  @Input() colors: string[] = [];\n\n  @Input() numberOfProgressBarItems = 5;\n\n  @Output() strengthChange = new EventEmitter<number | null>();\n\n  @HostBinding('class') baseClass = 'psm';\n\n  private passwordStrengthMeterService: IPasswordStrengthMeterService = inject(\n    IPasswordStrengthMeterService\n  );\n\n  passwordStrength: number | null = null;\n  feedback: Feedback | null = null;\n\n  private prevPasswordStrength: number | null = null;\n  private passwordChangeObservable$ = new Subject<string>();\n\n  constructor() {\n    this.init();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['password']) {\n      this.passwordChangeObservable$.next(this.password);\n    }\n  }\n\n  private init(): void {\n    this.passwordChangeObservable$\n      .pipe(\n        distinctUntilChanged(),\n        debounceTime(100),\n        switchMap((value) => {\n          if (!value) {\n            return of({ score: null, feedback: null });\n          }\n\n          if (value && value.length < this.minPasswordLength) {\n            return of({ score: 0, feedback: null });\n          }\n\n          if (this.enableAsync) {\n            return this.calculateScoreAsync(value);\n          }\n\n          const result = this.calculateScore(value);\n          return of(result);\n        }),\n        takeUntilDestroyed()\n      )\n      .subscribe((result: FeedbackResult) => {\n        this.passwordStrength = result.score;\n        this.feedback = result.feedback;\n\n        // Only emit the passwordStrength if it changed\n        if (this.prevPasswordStrength !== this.passwordStrength) {\n          this.strengthChange.emit(this.passwordStrength);\n          this.prevPasswordStrength = this.passwordStrength;\n        }\n      });\n  }\n\n  private calculateScore(value: string): FeedbackResult {\n    if (this.enableFeedback) {\n      return this.passwordStrengthMeterService.scoreWithFeedback(value);\n    }\n\n    const feedbackResult = {\n      score: this.passwordStrengthMeterService.score(value),\n      feedback: null,\n    };\n\n    return feedbackResult;\n  }\n\n  private calculateScoreAsync(value: string): Promise<FeedbackResult> {\n    if (this.enableFeedback) {\n      return this.passwordStrengthMeterService.scoreWithFeedbackAsync(value);\n    }\n\n    return this.passwordStrengthMeterService\n      .scoreAsync(value)\n      .then((result) => ({\n        score: result,\n        feedback: null,\n      }));\n  }\n}\n","<div\n  class=\"psm__progress-bar\"\n  role=\"progressbar\"\n  [passwordStrength]=\"passwordStrength\"\n  [numberOfProgressBarItems]=\"numberOfProgressBarItems\"\n  [colors]=\"colors\"\n>\n  <div class=\"psm__progress-bar-items\"></div>\n  <div class=\"psm__progress-bar-overlay\"></div>\n</div>\n\n@if(enableFeedback && feedback) {\n\n  @if(feedback.warning) {\n    <small class=\"psm__feedback\">\n      {{ feedback.warning }}\n    </small>\n  }\n\n  @if(feedback.suggestions && feedback.suggestions.length) {\n    <small class=\"psm__suggestion\">\n      @for (suggestion of feedback.suggestions; track suggestion; let isLast = $last) {\n        {{ suggestion }}{{ isLast ? '' : ' ' }}\n      }\n    </small>\n  }\n}\n\n"]}