Finilization v1.1.0.0
This commit is contained in:
parent
2b3da349b0
commit
e10ba53c1a
|
@ -3,7 +3,7 @@
|
|||
"solution": {
|
||||
"name": "SPFx app dev Simple Password Vault",
|
||||
"id": "d091b369-f63d-459c-846e-5a4323e31745",
|
||||
"version": "1.0.0.0",
|
||||
"version": "1.1.0.0",
|
||||
"includeClientSideAssets": true,
|
||||
"skipFeatureDeployment": true,
|
||||
"isDomainIsolated": false,
|
||||
|
|
|
@ -1,6 +1,6 @@
|
|||
{
|
||||
"name": "sp-fx-app-dev-simple-password-vault",
|
||||
"version": "1.0.0",
|
||||
"version": "1.1.0",
|
||||
"private": true,
|
||||
"main": "lib/index.js",
|
||||
"engines": {
|
||||
|
|
|
@ -1,6 +1,3 @@
|
|||
export interface IVaultData {
|
||||
masterPW: string;
|
||||
// username: string;
|
||||
// password: string;
|
||||
// note: string;
|
||||
}
|
|
@ -1,7 +1,6 @@
|
|||
import * as CryptoJS from 'crypto-js';
|
||||
import { isNullOrEmpty, toBoolean } from '@spfxappdev/utility';
|
||||
import { SessionStorage } from '@spfxappdev/storage';
|
||||
import { IVaultData } from '@src/models/IVaultData';
|
||||
import { ModuleType } from '@src/models';
|
||||
|
||||
|
||||
|
@ -71,7 +70,7 @@ export class PasswordVaultService implements IPasswordVaultService {
|
|||
|
||||
this.cache = new SessionStorage(cacheSettings);
|
||||
this.isVaultOpen = toBoolean(this.cache.get(this.encryptedInstanceId));
|
||||
let pwFromCache = this.cache.get(this.encryptedMasterPWInstanceId);
|
||||
const pwFromCache = this.cache.get(this.encryptedMasterPWInstanceId);
|
||||
this.encryptedMasterPw = isNullOrEmpty(pwFromCache) ? "" : pwFromCache;
|
||||
|
||||
if(!isNullOrEmpty(this.encryptedMasterPw)) {
|
||||
|
@ -103,7 +102,7 @@ export class PasswordVaultService implements IPasswordVaultService {
|
|||
public open(masterPW: string, encryptedMasterPW: string): boolean {
|
||||
const masterPWEncrypted = CryptoJS.HmacSHA256(masterPW, this.masterSecretKey);
|
||||
|
||||
if(masterPWEncrypted.toString() == encryptedMasterPW) {
|
||||
if(masterPWEncrypted.toString() === encryptedMasterPW) {
|
||||
this.isVaultOpen = true;
|
||||
this.cache.set(this.encryptedInstanceId, true);
|
||||
this.encryptedMasterPw = this.encrypt(masterPW, this.masterSecretKey);
|
||||
|
@ -115,7 +114,6 @@ export class PasswordVaultService implements IPasswordVaultService {
|
|||
return false;
|
||||
}
|
||||
|
||||
|
||||
public isOpen(): boolean {
|
||||
return this.isVaultOpen;
|
||||
}
|
||||
|
|
|
@ -22,9 +22,18 @@
|
|||
"officeFabricIconFontName": "ProtectRestrict",
|
||||
"properties": {
|
||||
"masterPW": "",
|
||||
"username": "",
|
||||
"password": "",
|
||||
"note": ""
|
||||
"modules": [
|
||||
{
|
||||
"id": "718def44-3e0c-48bf-8843-ccabefece27e",
|
||||
"data": "",
|
||||
"type": "UserField"
|
||||
},
|
||||
{
|
||||
"id": "65025036-e345-43f7-9561-6d3026294b2e",
|
||||
"data": "",
|
||||
"type": "PasswordField"
|
||||
},
|
||||
]
|
||||
}
|
||||
}]
|
||||
}
|
||||
|
|
|
@ -39,15 +39,15 @@ export default class PasswordVaultWebPart extends SPFxAppDevClientSideWebPart<IP
|
|||
Title: this.properties.Title,
|
||||
passwordVaultService: this.passwordVaultService,
|
||||
masterPW: this.properties.masterPW,
|
||||
// username: this.properties.username,
|
||||
// password: this.properties.password,
|
||||
// note: this.properties.note,
|
||||
modules: this.properties.modules||[],
|
||||
onTitleChanged: (title: string): void => {
|
||||
this.onTitleChanged(title);
|
||||
},
|
||||
onVaultDataChanged: (encryptedMasterPw: string, modules: IModule[]): void => {
|
||||
this.onVaultDataChanged(encryptedMasterPw, modules);
|
||||
},
|
||||
onVaultPasswordChanged: (encryptedMasterPw: string): void => {
|
||||
this.onMasterPasswordChanged(encryptedMasterPw);
|
||||
}
|
||||
}
|
||||
);
|
||||
|
@ -66,7 +66,7 @@ export default class PasswordVaultWebPart extends SPFxAppDevClientSideWebPart<IP
|
|||
});
|
||||
}
|
||||
|
||||
if (!this.helper.functions.isNullOrEmpty(oldProps.username)) {
|
||||
if (!this.helper.functions.isNullOrEmpty(oldProps.password)) {
|
||||
this.properties.modules.push({
|
||||
id: Guid.newGuid().toString(),
|
||||
type: ModuleType.PasswordField,
|
||||
|
@ -81,6 +81,23 @@ export default class PasswordVaultWebPart extends SPFxAppDevClientSideWebPart<IP
|
|||
data: oldProps.note
|
||||
});
|
||||
}
|
||||
|
||||
this.removePropertiesFromOldVersion();
|
||||
}
|
||||
|
||||
private removePropertiesFromOldVersion(): void {
|
||||
const oldProps: any = this.properties as any;
|
||||
if (this.helper.functions.isset(oldProps.username)) {
|
||||
delete oldProps.username;
|
||||
}
|
||||
|
||||
if (this.helper.functions.isset(oldProps.password)) {
|
||||
delete oldProps.password;
|
||||
}
|
||||
|
||||
if (this.helper.functions.isset(oldProps.note)) {
|
||||
delete oldProps.note;
|
||||
}
|
||||
}
|
||||
|
||||
public getLogCategory(): string {
|
||||
|
@ -94,9 +111,11 @@ export default class PasswordVaultWebPart extends SPFxAppDevClientSideWebPart<IP
|
|||
public onVaultDataChanged(encryptedMasterPW: string, modules: IModule[]): void {
|
||||
this.properties.masterPW = encryptedMasterPW;
|
||||
this.properties.modules = modules;
|
||||
// this.properties.note = vaultData.note;
|
||||
// this.properties.password = vaultData.password;
|
||||
// this.properties.username = vaultData.username;
|
||||
this.removePropertiesFromOldVersion();
|
||||
}
|
||||
|
||||
private onMasterPasswordChanged(encryptedMasterPW: string): void {
|
||||
this.properties.masterPW = encryptedMasterPW;
|
||||
}
|
||||
|
||||
protected onDispose(): void {
|
||||
|
|
|
@ -1,5 +1,4 @@
|
|||
import * as React from 'react';
|
||||
import styles from './PasswordVault.module.scss';
|
||||
import * as strings from 'PasswordVaultWebPartStrings';
|
||||
import { Label } from 'office-ui-fabric-react';
|
||||
import { RichText } from "@pnp/spfx-controls-react/lib/RichText";
|
||||
|
|
|
@ -1,6 +1,5 @@
|
|||
import * as React from 'react';
|
||||
import styles from './PasswordVault.module.scss';
|
||||
import { ActionButton, Callout, Icon, TooltipHost, DirectionalHint, TextField, ITextField } from 'office-ui-fabric-react';
|
||||
import { Callout, Icon, DirectionalHint, TextField, ITextField } from 'office-ui-fabric-react';
|
||||
import * as strings from 'PasswordVaultWebPartStrings';
|
||||
import { getDeepOrDefault, issetDeep, isset } from '@spfxappdev/utility';
|
||||
|
||||
|
@ -30,8 +29,6 @@ export default class PasswordField extends React.Component<IPasswordFieldProps,
|
|||
}
|
||||
|
||||
return this.renderEditMode();
|
||||
|
||||
|
||||
}
|
||||
|
||||
public renderDisplayMode(): JSX.Element {
|
||||
|
|
|
@ -34,6 +34,7 @@
|
|||
&::before {
|
||||
border-top-style: solid;
|
||||
border-top-width: 2px;
|
||||
border-top-color: $ms-color-neutralTertiary;
|
||||
content: "";
|
||||
height: 0;
|
||||
position: absolute;
|
||||
|
@ -49,6 +50,7 @@
|
|||
border-radius: 50%;
|
||||
border-style: solid;
|
||||
border-width: 2px;
|
||||
border-color: $ms-color-neutralTertiary;
|
||||
box-sizing: border-box;
|
||||
height: 24px;
|
||||
line-height: 12px;
|
||||
|
@ -57,7 +59,7 @@
|
|||
position: relative;
|
||||
transition: opacity .3s ease-in-out;
|
||||
width: 24px;
|
||||
background-color: #000;
|
||||
background-color: $ms-color-neutralTertiary;
|
||||
display: block;
|
||||
|
||||
|
||||
|
@ -104,13 +106,19 @@
|
|||
padding: 10px;
|
||||
}
|
||||
}
|
||||
|
||||
:global {
|
||||
.ms-CommandBar {
|
||||
padding: 0 0 10px;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
:global {
|
||||
|
||||
.ql-editor
|
||||
{
|
||||
padding: 0px !important;
|
||||
padding: 8px 0 0 !important;
|
||||
}
|
||||
|
||||
.ql-editor[contenteditable='true'] {
|
||||
|
|
|
@ -3,7 +3,7 @@ import styles from './PasswordVault.module.scss';
|
|||
import { SPFxAppDevWebPartComponent, ISPFxAppDevWebPartComponentProps } from '@spfxappdev/framework';
|
||||
import PasswordVaultWebPart from '../PasswordVaultWebPart';
|
||||
import { IPasswordVaultService } from '@src/services/PasswordVaultService';
|
||||
import { DefaultButton, IconButton, MessageBar, MessageBarType, PrimaryButton, TextField } from 'office-ui-fabric-react';
|
||||
import { Dialog, DefaultButton, IconButton, MessageBar, MessageBarType, PrimaryButton, TextField, CommandBar, ICommandBarItemProps, DialogFooter, DialogType } from 'office-ui-fabric-react';
|
||||
import { WebPartTitle } from "@pnp/spfx-controls-react/lib/WebPartTitle";
|
||||
import * as strings from 'PasswordVaultWebPartStrings';
|
||||
import AddNewModule from './AddNewModule';
|
||||
|
@ -13,12 +13,14 @@ import UserField from './UserField';
|
|||
import PasswordField from './PasswordField';
|
||||
import NoteField from './NoteField';
|
||||
import { cloneDeep } from '@microsoft/sp-lodash-subset';
|
||||
import '@spfxappdev/utility/lib/extensions/ArrayExtensions';
|
||||
|
||||
interface IPasswordVaultState {
|
||||
isVaultOpen: boolean;
|
||||
showWrongMasterInfo: boolean;
|
||||
isSaveButtonDisabled: boolean;
|
||||
modules: IModule[];
|
||||
modules: IModule[];
|
||||
showChangePasswordDialog: boolean;
|
||||
}
|
||||
|
||||
export interface IPasswordVaultProps extends ISPFxAppDevWebPartComponentProps<PasswordVaultWebPart> {
|
||||
|
@ -26,10 +28,10 @@ export interface IPasswordVaultProps extends ISPFxAppDevWebPartComponentProps<Pa
|
|||
masterPW?: string;
|
||||
onTitleChanged?(value: string): void;
|
||||
onVaultDataChanged?(encryptedMaster: string, modules: IModule[]): void;
|
||||
onVaultPasswordChanged?(encryptedMaster: string): void;
|
||||
modules?: IModule[];
|
||||
}
|
||||
//TODO:
|
||||
// "Change Password" as command button, only if vault already have set a master password
|
||||
|
||||
|
||||
export default class PasswordVault extends SPFxAppDevWebPartComponent<PasswordVaultWebPart, IPasswordVaultProps, IPasswordVaultState> {
|
||||
|
||||
|
@ -43,7 +45,8 @@ export default class PasswordVault extends SPFxAppDevWebPartComponent<PasswordVa
|
|||
isVaultOpen: this.isVaultOpen,
|
||||
showWrongMasterInfo: false,
|
||||
isSaveButtonDisabled: this.helper.functions.isNullOrEmpty(this.props.masterPW),
|
||||
modules: cloneDeep(this.props.modules)
|
||||
modules: cloneDeep(this.props.modules),
|
||||
showChangePasswordDialog: false
|
||||
};
|
||||
|
||||
private get isVaultOpen(): boolean {
|
||||
|
@ -61,6 +64,8 @@ export default class PasswordVault extends SPFxAppDevWebPartComponent<PasswordVa
|
|||
|
||||
private enteredMasterPW: string = "";
|
||||
|
||||
private repeatedEnteredMasterPW: string = "";
|
||||
|
||||
private encryptedModuleData: Record<string, string> = {};
|
||||
|
||||
private decryptedModuleData: Record<string, string> = {};
|
||||
|
@ -152,26 +157,17 @@ export default class PasswordVault extends SPFxAppDevWebPartComponent<PasswordVa
|
|||
|
||||
{showForm &&
|
||||
<>
|
||||
<MessageBar messageBarType={MessageBarType.warning}>
|
||||
{strings.DontLoseMasterpasswordLabel}
|
||||
</MessageBar>
|
||||
|
||||
{this.renderCommandBarButtons()}
|
||||
{this.renderChangePasswordDialog()}
|
||||
|
||||
{this.isNewVault &&
|
||||
<MessageBar messageBarType={MessageBarType.warning}>
|
||||
{strings.DontLoseMasterpasswordLabel}
|
||||
</MessageBar>
|
||||
}
|
||||
<div className='spfxappdev-grid'>
|
||||
<div className="spfxappdev-grid-row">
|
||||
<div className="spfxappdev-grid-col spfxappdev-sm12">
|
||||
<TextField
|
||||
label={this.isNewVault ? strings.SetMasterPasswordLabel : strings.ChangeMasterPasswordLabel}
|
||||
type="password"
|
||||
required={this.isNewVault}
|
||||
canRevealPassword={true}
|
||||
onChange={(ev: any, newValue: string) => {
|
||||
this.enteredMasterPW = newValue;
|
||||
this.setState({
|
||||
isSaveButtonDisabled: this.isSaveButtonDisabled()
|
||||
});
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
{this.isNewVault && this.renderMasterPasswordControls()}
|
||||
|
||||
{this.state.modules.map((module: IModule, index: number): JSX.Element => {
|
||||
return this.renderModuleEditMode(module, index);
|
||||
|
@ -187,31 +183,7 @@ export default class PasswordVault extends SPFxAppDevWebPartComponent<PasswordVa
|
|||
<div className="spfxappdev-grid-row grid-footer">
|
||||
<div className="spfxappdev-grid-col spfxappdev-sm12">
|
||||
<PrimaryButton disabled={this.state.isSaveButtonDisabled} onClick={() => {
|
||||
this.isNewVault = false;
|
||||
let encryptedMaster = this.currentMasterPW;
|
||||
|
||||
if(!this.enteredMasterPW.IsEmpty()) {
|
||||
encryptedMaster = this.props.passwordVaultService.setMasterPassword(this.enteredMasterPW);
|
||||
this.currentMasterPW = encryptedMaster;
|
||||
}
|
||||
|
||||
const encryptedModules: IModule[] = [];
|
||||
this.state.modules.forEach((module: IModule) => {
|
||||
const encryptedValue: string = this.props.passwordVaultService.encryptModuleData(module.type, this.decryptedModuleData[module.id]);
|
||||
|
||||
encryptedModules.push({
|
||||
id: module.id,
|
||||
data: encryptedValue,
|
||||
type: module.type
|
||||
});
|
||||
});
|
||||
|
||||
this.props.onVaultDataChanged(encryptedMaster, encryptedModules);
|
||||
|
||||
this.setState({
|
||||
modules: encryptedModules
|
||||
});
|
||||
|
||||
this.onSaveButtonClick();
|
||||
}}>
|
||||
{strings.SaveLabel}
|
||||
</PrimaryButton>
|
||||
|
@ -274,19 +246,19 @@ export default class PasswordVault extends SPFxAppDevWebPartComponent<PasswordVa
|
|||
|
||||
<div className={styles["edit-container--content"]}>
|
||||
|
||||
{module.type == ModuleType.UserField &&
|
||||
{module.type === ModuleType.UserField &&
|
||||
<UserField defaultValue={this.decryptedModuleData[module.id]} tabIndex={index} onChange={(newVal: string) => {
|
||||
this.decryptedModuleData[module.id] = newVal;
|
||||
}} isDisplayMode={false} />
|
||||
}
|
||||
|
||||
{module.type == ModuleType.PasswordField &&
|
||||
{module.type === ModuleType.PasswordField &&
|
||||
<PasswordField defaultValue={this.decryptedModuleData[module.id]} tabIndex={index} onChange={(newVal: string) => {
|
||||
this.decryptedModuleData[module.id] = newVal;
|
||||
}} isDisplayMode={false} />
|
||||
}
|
||||
|
||||
{module.type == ModuleType.NoteField &&
|
||||
{module.type === ModuleType.NoteField &&
|
||||
<NoteField defaultValue={this.decryptedModuleData[module.id]} onChange={(newVal: string) => {
|
||||
|
||||
this.decryptedModuleData[module.id] = newVal;
|
||||
|
@ -307,15 +279,15 @@ export default class PasswordVault extends SPFxAppDevWebPartComponent<PasswordVa
|
|||
<>
|
||||
<div className="spfxappdev-grid-row" key={module.id}>
|
||||
<div className="spfxappdev-grid-col spfxappdev-sm12">
|
||||
{module.type == ModuleType.UserField &&
|
||||
{module.type === ModuleType.UserField &&
|
||||
<UserField defaultValue={this.decryptedModuleData[module.id]} isDisplayMode={true} tabIndex={index} />
|
||||
}
|
||||
|
||||
{module.type == ModuleType.PasswordField &&
|
||||
{module.type === ModuleType.PasswordField &&
|
||||
<PasswordField defaultValue={this.decryptedModuleData[module.id]} isDisplayMode={true} tabIndex={index} />
|
||||
}
|
||||
|
||||
{module.type == ModuleType.NoteField &&
|
||||
{module.type === ModuleType.NoteField &&
|
||||
<NoteField defaultValue={this.decryptedModuleData[module.id]} isDisplayMode={true} />
|
||||
}
|
||||
</div>
|
||||
|
@ -350,7 +322,7 @@ export default class PasswordVault extends SPFxAppDevWebPartComponent<PasswordVa
|
|||
this.enteredMasterPW = newValue;
|
||||
}}
|
||||
onKeyUp={(ev: React.KeyboardEvent<HTMLInputElement>) => {
|
||||
if(ev.keyCode == 13) {
|
||||
if(ev.keyCode === 13) {
|
||||
this.onOpenVault();
|
||||
}
|
||||
}}
|
||||
|
@ -375,9 +347,123 @@ export default class PasswordVault extends SPFxAppDevWebPartComponent<PasswordVa
|
|||
</>);
|
||||
}
|
||||
|
||||
private renderMasterPasswordControls(): JSX.Element {
|
||||
return (
|
||||
<div className="spfxappdev-grid-row">
|
||||
<div className="spfxappdev-grid-col spfxappdev-sm12">
|
||||
<TextField
|
||||
label={this.isNewVault ? strings.SetMasterPasswordLabel : strings.ChangeMasterPasswordLabel}
|
||||
type="password"
|
||||
required={this.isNewVault}
|
||||
canRevealPassword={true}
|
||||
onChange={(ev: any, newValue: string) => {
|
||||
this.enteredMasterPW = newValue;
|
||||
this.setState({
|
||||
isSaveButtonDisabled: this.isSaveButtonDisabled()
|
||||
});
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div className="spfxappdev-grid-col spfxappdev-sm12">
|
||||
<TextField
|
||||
label={strings.RepeatMasterPasswordLabel}
|
||||
type="password"
|
||||
required={this.isNewVault}
|
||||
canRevealPassword={true}
|
||||
onChange={(ev: any, newValue: string) => {
|
||||
this.repeatedEnteredMasterPW = newValue;
|
||||
this.setState({
|
||||
isSaveButtonDisabled: this.isSaveButtonDisabled()
|
||||
});
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
</div>);
|
||||
}
|
||||
|
||||
private renderChangePasswordDialog(): JSX.Element {
|
||||
return (
|
||||
<Dialog
|
||||
hidden={!this.state.showChangePasswordDialog}
|
||||
dialogContentProps={{
|
||||
title: strings.ChangeMasterPasswordDialogTitle,
|
||||
type: DialogType.normal,
|
||||
}}
|
||||
onDismiss={() => { this.toggleChangePasswordDialogVisibility(); }}
|
||||
>
|
||||
|
||||
<MessageBar messageBarType={MessageBarType.warning}>
|
||||
{strings.DontLoseMasterpasswordLabel}
|
||||
</MessageBar>
|
||||
|
||||
{this.renderMasterPasswordControls()}
|
||||
|
||||
<DialogFooter>
|
||||
<PrimaryButton
|
||||
disabled={this.state.isSaveButtonDisabled}
|
||||
onClick={() => { this.onChangePasswordClick(); }}
|
||||
text={strings.SaveLabel} />
|
||||
<DefaultButton onClick={() => { this.toggleChangePasswordDialogVisibility(); }} text={strings.CancelLabel} />
|
||||
</DialogFooter>
|
||||
</Dialog>
|
||||
);
|
||||
}
|
||||
|
||||
private renderCommandBarButtons(): JSX.Element {
|
||||
const buttons: ICommandBarItemProps[] = [];
|
||||
|
||||
const saveButton: ICommandBarItemProps = {
|
||||
key: 'saveSettings',
|
||||
text: strings.SaveLabel,
|
||||
disabled: this.isSaveButtonDisabled(),
|
||||
iconProps: { iconName: 'Save' },
|
||||
onClick: () => {
|
||||
this.onSaveButtonClick();
|
||||
}
|
||||
}
|
||||
|
||||
buttons.push(saveButton);
|
||||
|
||||
if(!this.isNewVault) {
|
||||
const changeMasterPwButton: ICommandBarItemProps = {
|
||||
key: 'changeMasterPassword',
|
||||
text: strings.ChangeMasterPasswordButtonText,
|
||||
iconProps: { iconName: 'PasswordField' },
|
||||
onClick: () => {
|
||||
this.toggleChangePasswordDialogVisibility();
|
||||
},
|
||||
}
|
||||
|
||||
buttons.push(changeMasterPwButton);
|
||||
}
|
||||
|
||||
if(!this.helper.functions.isNullOrEmpty(this.currentMasterPW)) {
|
||||
const closeButton: ICommandBarItemProps = {
|
||||
key: 'closeVault',
|
||||
text: strings.CloseVaultLabel,
|
||||
iconProps: { iconName: 'Lock' },
|
||||
onClick: () => {
|
||||
this.closeVault();
|
||||
}
|
||||
}
|
||||
|
||||
buttons.push(closeButton);
|
||||
}
|
||||
|
||||
return (
|
||||
<CommandBar
|
||||
items={buttons}
|
||||
/>
|
||||
);
|
||||
}
|
||||
|
||||
private isSaveButtonDisabled(): boolean {
|
||||
if(this.isNewVault && this.helper.functions.isNullOrEmpty(this.enteredMasterPW)) {
|
||||
if((this.isNewVault || this.state.showChangePasswordDialog) && this.helper.functions.isNullOrEmpty(this.enteredMasterPW)) {
|
||||
return true;
|
||||
}
|
||||
|
||||
if(!this.enteredMasterPW.Equals(this.repeatedEnteredMasterPW, false)) {
|
||||
return true;
|
||||
}
|
||||
|
||||
|
@ -404,8 +490,6 @@ export default class PasswordVault extends SPFxAppDevWebPartComponent<PasswordVa
|
|||
});
|
||||
}
|
||||
|
||||
console.log("SSC onOpen", this.decryptedModuleData);
|
||||
|
||||
this.setState({
|
||||
isVaultOpen: isCorrectPW,
|
||||
showWrongMasterInfo: !isCorrectPW
|
||||
|
@ -461,4 +545,55 @@ export default class PasswordVault extends SPFxAppDevWebPartComponent<PasswordVa
|
|||
modules: this.state.modules
|
||||
});
|
||||
}
|
||||
|
||||
private onSaveButtonClick(): void {
|
||||
this.isNewVault = false;
|
||||
let encryptedMaster = this.currentMasterPW;
|
||||
|
||||
if(!this.enteredMasterPW.IsEmpty()) {
|
||||
encryptedMaster = this.props.passwordVaultService.setMasterPassword(this.enteredMasterPW);
|
||||
this.currentMasterPW = encryptedMaster;
|
||||
}
|
||||
|
||||
const encryptedModules: IModule[] = [];
|
||||
this.state.modules.forEach((module: IModule) => {
|
||||
const encryptedValue: string = this.props.passwordVaultService.encryptModuleData(module.type, this.decryptedModuleData[module.id]);
|
||||
|
||||
encryptedModules.push({
|
||||
id: module.id,
|
||||
data: encryptedValue,
|
||||
type: module.type
|
||||
});
|
||||
});
|
||||
|
||||
this.props.onVaultDataChanged(encryptedMaster, encryptedModules);
|
||||
|
||||
this.setState({
|
||||
modules: encryptedModules
|
||||
});
|
||||
|
||||
this.enteredMasterPW = '';
|
||||
this.repeatedEnteredMasterPW = '';
|
||||
}
|
||||
|
||||
private onChangePasswordClick(): void {
|
||||
let encryptedMaster = this.currentMasterPW;
|
||||
|
||||
if(!this.enteredMasterPW.IsEmpty()) {
|
||||
encryptedMaster = this.props.passwordVaultService.setMasterPassword(this.enteredMasterPW);
|
||||
this.currentMasterPW = encryptedMaster;
|
||||
this.props.onVaultPasswordChanged(encryptedMaster);
|
||||
}
|
||||
|
||||
this.enteredMasterPW = '';
|
||||
this.repeatedEnteredMasterPW = '';
|
||||
this.toggleChangePasswordDialogVisibility();
|
||||
}
|
||||
|
||||
private toggleChangePasswordDialogVisibility(): void {
|
||||
this.setState({
|
||||
showChangePasswordDialog: !this.state.showChangePasswordDialog,
|
||||
isSaveButtonDisabled: !this.state.showChangePasswordDialog
|
||||
});
|
||||
}
|
||||
}
|
|
@ -1,6 +1,5 @@
|
|||
import * as React from 'react';
|
||||
import styles from './PasswordVault.module.scss';
|
||||
import { ActionButton, Callout, Icon, TooltipHost, DirectionalHint, TextField, ITextField } from 'office-ui-fabric-react';
|
||||
import { Callout, Icon, DirectionalHint, TextField, ITextField } from 'office-ui-fabric-react';
|
||||
import * as strings from 'PasswordVaultWebPartStrings';
|
||||
import { getDeepOrDefault, issetDeep, isset } from '@spfxappdev/utility';
|
||||
|
||||
|
|
|
@ -5,6 +5,7 @@ define([], function() {
|
|||
"PasswordLabel": "Passwort",
|
||||
"MasterPasswordLabel": "Master Passwort",
|
||||
"SetMasterPasswordLabel": "Master Passwort",
|
||||
"RepeatMasterPasswordLabel": "Master Passwort wiederholen",
|
||||
"ChangeMasterPasswordLabel": "Master Passwort ändern",
|
||||
"UsernameLabel": "Benutezrname",
|
||||
"NoteLabel": "Notiz",
|
||||
|
@ -22,5 +23,8 @@ define([], function() {
|
|||
"PasswordModuleLabel": "Passwort",
|
||||
"UsernameModuleLabel": "Benutzername",
|
||||
"NoteModuleLabel": "Notiz",
|
||||
"ChangeMasterPasswordButtonText": "Master Passwort ändern",
|
||||
"ChangeMasterPasswordDialogTitle": "Master Passwort ändern",
|
||||
"CancelLabel": "Abbrechen"
|
||||
}
|
||||
});
|
|
@ -5,6 +5,7 @@ define([], function() {
|
|||
"PasswordLabel": "Password",
|
||||
"MasterPasswordLabel": "Master Password",
|
||||
"SetMasterPasswordLabel": "Master Password",
|
||||
"RepeatMasterPasswordLabel": "Repeat Master Passwort",
|
||||
"ChangeMasterPasswordLabel": "Change Master Password",
|
||||
"UsernameLabel": "Username",
|
||||
"NoteLabel": "Note",
|
||||
|
@ -22,5 +23,8 @@ define([], function() {
|
|||
"PasswordModuleLabel": "Password",
|
||||
"UsernameModuleLabel": "Username",
|
||||
"NoteModuleLabel": "Note",
|
||||
"ChangeMasterPasswordButtonText": "Change Master Passwort",
|
||||
"ChangeMasterPasswordDialogTitle": "Change Master Passwort",
|
||||
"CancelLabel": "Cancel"
|
||||
}
|
||||
});
|
|
@ -4,6 +4,7 @@ declare interface IPasswordVaultWebPartStrings {
|
|||
PasswordLabel: string;
|
||||
MasterPasswordLabel: string;
|
||||
SetMasterPasswordLabel: string;
|
||||
RepeatMasterPasswordLabel: string;
|
||||
ChangeMasterPasswordLabel: string;
|
||||
UsernameLabel: string;
|
||||
NoteLabel: string;
|
||||
|
@ -21,6 +22,9 @@ declare interface IPasswordVaultWebPartStrings {
|
|||
PasswordModuleLabel: string;
|
||||
UsernameModuleLabel: string;
|
||||
NoteModuleLabel: string;
|
||||
ChangeMasterPasswordButtonText: string;
|
||||
ChangeMasterPasswordDialogTitle: string;
|
||||
CancelLabel: string;
|
||||
}
|
||||
|
||||
declare module 'PasswordVaultWebPartStrings' {
|
||||
|
|
Loading…
Reference in New Issue