更新注册服务

change
panqihua 5 years ago
parent 2d788991b3
commit 94ad8edfa2
  1. 11
      angular.json
  2. 13
      src/app/account/account/account.module.ts
  3. 4
      src/app/account/login/login.component.html
  4. 9
      src/app/account/login/login.service.ts
  5. 51
      src/app/account/register/register.component.html
  6. 96
      src/app/account/register/register.component.ts
  7. 58
      src/app/account/register/register.service.ts
  8. 7
      src/app/app.module.ts
  9. 19
      src/app/commons.ts
  10. 7
      src/app/forum/forum/forum.module.ts
  11. 5
      src/app/interface/Http.ts
  12. 20
      src/app/interface/HttpInterface.ts
  13. 30
      src/app/interface/JSONRequest.ts
  14. 38
      src/app/interface/JSONResponse.ts
  15. 4
      src/app/interface/Result.ts
  16. 13
      src/assets/i18n/en-US.json
  17. 13
      src/assets/i18n/zh-CN.json
  18. 21
      src/environments/environment.rap2.ts

@ -64,6 +64,14 @@
"maximumError": "10kb" "maximumError": "10kb"
} }
] ]
},
"rap2": {
"fileReplacements": [
{
"replace": "src/environments/environment.ts",
"with": "src/environments/environment.rap2.ts"
}
]
} }
} }
}, },
@ -75,6 +83,9 @@
"configurations": { "configurations": {
"production": { "production": {
"browserTarget": "PocketCommunityWeb:build:production" "browserTarget": "PocketCommunityWeb:build:production"
},
"rap2": {
"browserTarget": "PocketCommunityWeb:build:rap2"
} }
} }
}, },

@ -6,15 +6,24 @@ import {LoginComponent} from '../login/login.component';
import {RegisterComponent} from '../register/register.component'; import {RegisterComponent} from '../register/register.component';
// 重置密码组件 // 重置密码组件
import {ResetpwdComponent} from '../resetpwd/resetpwd.component'; import {ResetpwdComponent} from '../resetpwd/resetpwd.component';
import {TranslateModule} from '@ngx-translate/core';
import {RouterModule} from '@angular/router';
import {ReactiveFormsModule} from '@angular/forms';
// 提示框组件
import {MessageComponent} from '../../message/message.component';
import {InputComponent} from '../../input/input.component';
/** /**
* *
*/ */
@NgModule({ @NgModule({
declarations: [LoginComponent, RegisterComponent, ResetpwdComponent], declarations: [LoginComponent, RegisterComponent, ResetpwdComponent, MessageComponent, InputComponent],
imports: [ imports: [
CommonModule CommonModule,
TranslateModule,
RouterModule,
ReactiveFormsModule,
] ]
}) })
export class AccountModule { } export class AccountModule { }

@ -8,7 +8,7 @@
<span class="input-group-text">{{ 'login.manager_name' | translate }}</span> <span class="input-group-text">{{ 'login.manager_name' | translate }}</span>
</div> </div>
<input autocomplete="managerName" type="text" class="form-control text-center" <input autocomplete="managerName" type="text" class="form-control text-center"
[placeholder]="('tip.input' | translate)+('login.manager_name' | translate)" formControlName="managerName"> [placeholder]="'tip.input' | translate:{value:'login.manager_name'|translate}" formControlName="managerName">
</div> </div>
<!-- 管理员密码--> <!-- 管理员密码-->
@ -17,7 +17,7 @@
<span class="input-group-text">{{ 'login.password' | translate }}</span> <span class="input-group-text">{{ 'login.password' | translate }}</span>
</div> </div>
<input autocomplete="new-password" type="password" class="form-control text-center" <input autocomplete="new-password" type="password" class="form-control text-center"
[placeholder]="('tip.input' | translate)+('login.password' | translate)" formControlName="password"> [placeholder]="'tip.input' | translate:{value:'login.manager_name'|translate}" formControlName="password">
<div class="input-group-append"> <div class="input-group-append">
<button class="btn btn-outline-secondary" type="button" <button class="btn btn-outline-secondary" type="button"
[routerLink]="['/reset_pwd']">{{ 'button.forget_pwd' | translate }}</button> [routerLink]="['/reset_pwd']">{{ 'button.forget_pwd' | translate }}</button>

@ -1,19 +1,20 @@
import {Injectable} from '@angular/core'; import {Injectable} from '@angular/core';
import {CookieService} from 'ngx-cookie-service'; import {CookieService} from 'ngx-cookie-service';
import {Router} from '@angular/router'; import {Router} from '@angular/router';
import {HttpInterface} from '../../interface/Http';
import {MessageService} from '../../message/message.service'; import {MessageService} from '../../message/message.service';
import {JSONRequest} from '../../interface/JSONRequest';
@Injectable({ @Injectable({
providedIn: 'root' providedIn: 'root'
}) })
export class LoginService implements HttpInterface { export class LoginService extends JSONRequest {
constructor( constructor(
private cookieService: CookieService, private cookieService: CookieService,
private router: Router, private router: Router,
private messageService: MessageService private messageService: MessageService
) { ) {
super();
} }
/** /**
@ -52,8 +53,4 @@ export class LoginService implements HttpInterface {
this.messageService.danger('登陆失败'); this.messageService.danger('登陆失败');
return false; return false;
} }
url(): string {
return '/api/manager/login';
}
} }

@ -7,8 +7,12 @@
<div class="input-group-prepend"> <div class="input-group-prepend">
<span class="input-group-text">{{ 'login.manager_name' | translate }}</span> <span class="input-group-text">{{ 'login.manager_name' | translate }}</span>
</div> </div>
<input type="text" class="form-control text-center" <input type="text" class="form-control text-center" [class.is-invalid]="checkValue('managerName')"
[placeholder]="('tip.input' | translate)+('login.manager_name' | translate)" formControlName="managerName"> [placeholder]="'tip.input' | translate:{value:'login.manager_name'|translate}"
formControlName="managerName"/>
<div class="invalid-feedback font-weight-bold text-center">
{{'tip.notnull' | translate:{value:'login.manager_name'|translate} }}
</div>
</div> </div>
<!-- 管理员密码--> <!-- 管理员密码-->
@ -17,7 +21,12 @@
<span class="input-group-text">{{ 'login.password' | translate }}</span> <span class="input-group-text">{{ 'login.password' | translate }}</span>
</div> </div>
<input autocomplete="password" type="password" class="form-control text-center" <input autocomplete="password" type="password" class="form-control text-center"
[placeholder]="('tip.input' | translate)+('login.password' | translate)" formControlName="password"> [class.is-invalid]="checkValue('password')"
[placeholder]="'tip.input' | translate:{value:'login.password'|translate}" formControlName="password"/>
<div class="invalid-feedback font-weight-bold text-center">
{{'tip.notnull' | translate:{value:'login.password'|translate} }}
</div>
</div> </div>
@ -27,8 +36,14 @@
<span class="input-group-text">{{ 'register.confirm_pwd' | translate }}</span> <span class="input-group-text">{{ 'register.confirm_pwd' | translate }}</span>
</div> </div>
<input autocomplete="confirmPassword" type="password" class="form-control text-center" <input autocomplete="confirmPassword" type="password" class="form-control text-center"
[placeholder]="('tip.input' | translate)+('register.confirm_pwd' | translate)" [class.is-invalid]="(checkPwd()|async) !==''"
formControlName="confirmPassword"> [placeholder]="'tip.input' | translate:{value:'register.confirm_pwd'|translate}"
formControlName="confirmPassword"/>
<div class="invalid-feedback font-weight-bold text-center">
{{ checkPwd()|async }}
</div>
</div> </div>
<!-- 手机号--> <!-- 手机号-->
@ -36,29 +51,41 @@
<div class="input-group-prepend"> <div class="input-group-prepend">
<span class="input-group-text">{{ 'register.mobile' | translate }}</span> <span class="input-group-text">{{ 'register.mobile' | translate }}</span>
</div> </div>
<input type="number" class="form-control text-center" <input type="number" class="form-control text-center" [class.is-invalid]="(checkMobile()|async) !==''"
[placeholder]="('tip.input' | translate)+('register.mobile' | translate)" formControlName="mobile"> [placeholder]="'tip.input' | translate:{value:'register.mobile'|translate}" formControlName="mobile"/>
<div class="invalid-feedback font-weight-bold text-center">
{{ checkMobile()|async }}
</div>
</div> </div>
<!-- 邮箱--> <!-- 邮箱-->
<div class="input-group mb-3 mx-auto col-7"> <div class="input-group mb-3 mx-auto col-8">
<div class="input-group-prepend"> <div class="input-group-prepend">
<span class="input-group-text">{{ 'register.email' | translate }}</span> <span class="input-group-text">{{ 'register.email' | translate }}</span>
</div> </div>
<input type="text" class="form-control text-center" <input type="text" class="form-control text-center"
[placeholder]="('tip.input' | translate)+('register.email' | translate)" formControlName="email"> [placeholder]="'tip.input' | translate:{value:'register.email'|translate}" formControlName="email" [class.is-invalid]="checkValue('email')" />
<select class="custom-select" formControlName="emailType">
<option selected value="-1">{{'tip.select'|translate:{value:'register.email_type'|translate} }}</option>
<option *ngFor="let item of (emailType$|async)?.body" [value]="item.suffix">{{item.name}}</option>
</select>
<div class="invalid-feedback font-weight-bold text-center">
{{'tip.notnull' | translate:{value:'register.email'|translate} }}
</div>
</div> </div>
<div class="col-7 mx-auto text-center mb-3"> <div class="col-7 mx-auto text-center mb-3">
<!-- 注册按钮--> <!-- 注册按钮-->
<button type="button" (click)="register()" <button type="button" (click)="register()" [class.disabled]="checkForm()" [disabled]="checkForm()"
class="btn btn-primary btn-lg">{{ 'button.register' | translate }}</button> class="btn btn-primary btn-lg disabled">{{ 'button.register' | translate }}</button>
<!-- 返回登陆按钮--> <!-- 返回登陆按钮-->
<button type="button" [routerLink]="['/login']" <button type="button" [routerLink]="['/login']"
class="btn btn-info btn-lg ml-3">{{ 'button.backLogin' | translate }}</button> class="btn btn-info btn-lg ml-3">{{ 'button.backLogin' | translate }}</button>
</div> </div>
<app-message></app-message> <app-message></app-message>
</form> </form>
</div> </div>

@ -1,37 +1,123 @@
import {Component, OnInit} from '@angular/core'; import {Component, OnInit} from '@angular/core';
import {FormBuilder} from '@angular/forms'; import {FormBuilder} from '@angular/forms';
import {Commons} from '../../commons'; import {Commons} from '../../commons';
import {Router} from '@angular/router';
import {RegisterService} from './register.service'; import {RegisterService} from './register.service';
import {TranslateService} from '@ngx-translate/core';
import {Observable, of} from 'rxjs';
@Component({ @Component({
selector: 'app-register', selector: 'app-register',
templateUrl: './register.component.html', templateUrl: './register.component.html',
styleUrls: ['./register.component.scss'] styleUrls: ['./register.component.scss']
}) })
// 注册模块 // 注册模块
export class RegisterComponent extends Commons implements OnInit { export class RegisterComponent extends Commons implements OnInit {
// 注册表单 // 注册表单
registerForm = this.fb.group({ registerForm = this.fb.group({
// 管理员名
managerName: [], managerName: [],
// 密码
password: [], password: [],
// 确认密码
confirmPassword: [], confirmPassword: [],
// 手机号
mobile: [], mobile: [],
email: [] // 邮箱
email: [],
// 邮箱类型
emailType: this.fb.control('-1')
}); });
/**
*
*/
emailType$ = this.registerService.getEmailType();
// 表单验证
validForm = {
confirmPassword: {
flag: false
},
mobile: {
flag: false
}
};
constructor( constructor(
private fb: FormBuilder, private fb: FormBuilder,
private router: Router, private registerService: RegisterService,
private registerService: RegisterService private translateService: TranslateService
) { ) {
super(); super();
} }
/**
*
* @param name formControlName
*/
checkValue(name): boolean {
if (!this.validForm.hasOwnProperty(name)) {
this.validForm[name] = {};
}
this.validForm[name].flag = this.registerForm.value[name];
return !this.validForm[name].flag;
}
/**
*
*/
checkForm(): boolean {
for (const key in this.validForm) {
if (!this.validForm[key].flag) {
return true;
}
}
return this.registerForm.value.emailType === '-1';
}
/**
*
*/
checkPwd(): Observable<string> {
if (!this.registerForm.value.confirmPassword) {
this.validForm.confirmPassword.flag = false;
return this.translateService.get('tip.password_null');
} else if (this.registerForm.value.password !== this.registerForm.value.confirmPassword) {
this.validForm.confirmPassword.flag = false;
return this.translateService.get('tip.password_diff');
} else {
this.validForm.confirmPassword.flag = true;
return of('');
}
}
/**
*
*/
checkMobile(): Observable<string> {
if (!this.registerForm.value.mobile) {
this.validForm.mobile.flag = false;
return this.translateService.get('tip.mobile_null');
} else if (!/^1[3456789]\d{9}$/.test(this.registerForm.value.mobile)) {
this.validForm.mobile.flag = false;
return this.translateService.get('tip.mobile_error');
} else {
this.validForm.mobile.flag = true;
return of('');
}
}
ngOnInit(): void { ngOnInit(): void {
} }
register() { register() {
this.registerService.register(JSON.stringify(this.registerForm.value)); this.registerService.register(this.registerForm.value);
} }
} }

@ -1,22 +1,66 @@
import { Injectable } from '@angular/core'; import {Injectable} from '@angular/core';
import {HttpInterface} from '../../interface/Http'; import {JSONRequest} from '../../interface/JSONRequest';
import {HttpClient} from '@angular/common/http';
import {catchError, tap} from 'rxjs/operators';
import {JSONResponse} from '../../interface/JSONResponse';
import {HttpInterface} from '../../interface/HttpInterface';
import {Observable} from 'rxjs';
import {Result} from '../../interface/Result';
import {Router} from '@angular/router';
import {MessageService} from '../../message/message.service';
@Injectable({ @Injectable({
providedIn: 'root' providedIn: 'root'
}) })
export class RegisterService implements HttpInterface { export class RegisterService extends JSONRequest {
constructor() { } constructor(
private http: HttpClient,
private router: Router,
private messageService: MessageService
) {
super();
this.httpError.result = Result.FAIL;
this.httpError.message = '注册失败';
}
/** /**
* *
* @param body * @param body
*/ */
register(body: string) { register(body) {
this.http.post<JSONResponse<any>>(HttpInterface.register, body, this.httpOptions)
.pipe(
catchError(this.handleError<any>('注册', this.httpError))
).subscribe(r => {
if (r.result === Result.OK) {
this.messageService.info('注册成功');
this.router.navigateByUrl('/login');
} else {
this.messageService.danger('注册失败');
}
});
}
/**
*
*/
getEmailType(): Observable<any> {
return this.http.get<JSONResponse<any>>(HttpInterface.getEmailType)
.pipe(
catchError(this.handleError<any>('获取邮箱类型'))
);
} }
url(): string { /**
return '/api/manager/register'; *
*/
sendEmail(sender): Observable<any> {
return this.http.post<JSONResponse<any>>(HttpInterface.sendCode, {email: 'sender'})
.pipe(
catchError(this.handleError<any>('发送邮件验证码'))
);
} }
} }

@ -7,18 +7,18 @@ import {AppComponent} from './app.component';
// 响应式表单 // 响应式表单
import {ReactiveFormsModule} from '@angular/forms'; import {ReactiveFormsModule} from '@angular/forms';
// 国际化支持 // 国际化支持
import {HttpClient, HttpClientModule} from '@angular/common/http';
import {TranslateLoader, TranslateModule} from '@ngx-translate/core'; import {TranslateLoader, TranslateModule} from '@ngx-translate/core';
import {TranslateHttpLoader} from '@ngx-translate/http-loader'; import {TranslateHttpLoader} from '@ngx-translate/http-loader';
// cookie // cookie
import {CookieService} from 'ngx-cookie-service'; import {CookieService} from 'ngx-cookie-service';
// 提示框组件
import {MessageComponent} from './message/message.component'
// 异常处理组件 // 异常处理组件
import {ErrorComponent} from './error/error.component'; import {ErrorComponent} from './error/error.component';
// 账号管理模块 // 账号管理模块
import {AccountModule} from './account/account/account.module'; import {AccountModule} from './account/account/account.module';
// 论坛模块
import {ForumModule} from './forum/forum/forum.module'; import {ForumModule} from './forum/forum/forum.module';
// Http请求模块
import {HttpClient, HttpClientModule} from '@angular/common/http';
export function HttpLoaderFactory(http: HttpClient) { export function HttpLoaderFactory(http: HttpClient) {
return new TranslateHttpLoader(http); return new TranslateHttpLoader(http);
@ -30,7 +30,6 @@ export function HttpLoaderFactory(http: HttpClient) {
@NgModule({ @NgModule({
declarations: [ declarations: [
AppComponent, AppComponent,
MessageComponent,
ErrorComponent ErrorComponent
], ],
imports: [ imports: [

@ -1,24 +1,5 @@
import {environment} from './../environments/environment';
// 组件通用配置 // 组件通用配置
export class Commons { export class Commons {
// 页面高度=屏幕的高度/2 // 页面高度=屏幕的高度/2
height = 'height:' + screen.height / 2 + 'px'; height = 'height:' + screen.height / 2 + 'px';
// post请求
request(url, b, res) {
const header = new Headers();
header.append('Content-Type', 'application/json');
const req = new Request(environment.apiServer + url,
{method: 'POST', body: b, headers: header});
fetch(req).then(r => {
if (r.status === 200) {
return r.json();
} else {
throw new Error('666');
}
}).then(res).catch(err => {
console.error(err);
});
}
} }

@ -2,6 +2,7 @@ import { NgModule } from '@angular/core';
import { CommonModule } from '@angular/common'; import { CommonModule } from '@angular/common';
// 论坛组件 // 论坛组件
import {ForumComponent} from '../forum.component'; import {ForumComponent} from '../forum.component';
import {TranslateModule} from '@ngx-translate/core';
/** /**
@ -9,8 +10,12 @@ import {ForumComponent} from '../forum.component';
*/ */
@NgModule({ @NgModule({
declarations: [ForumComponent], declarations: [ForumComponent],
exports: [
ForumComponent
],
imports: [ imports: [
CommonModule CommonModule,
TranslateModule
] ]
}) })
export class ForumModule { } export class ForumModule { }

@ -1,5 +0,0 @@
// 请求接口
export interface HttpInterface {
// 接口地址
url(): string;
}

@ -0,0 +1,20 @@
// 环境变量
import {environment} from '../../environments/environment';
// 请求接口
const HttpInterface = {
// 注册接口
register: '/api/manager/register',
// 获取邮箱类型接口
getEmailType: '/api/manager/emailType',
// 登陆接口
login: '/api/manager/login',
// 发送验证码接口
sendCode: '/api/manager/sendcode'
};
// tslint:disable-next-line:forin
for (const key in HttpInterface) {
HttpInterface[key] = environment.apiServer + HttpInterface[key];
}
console.debug(HttpInterface);
export {HttpInterface};

@ -0,0 +1,30 @@
import {HttpInterface} from './HttpInterface';
import {Observable, of} from 'rxjs';
import {HttpHeaders} from '@angular/common/http';
import {JSONResponse} from './JSONResponse';
export abstract class JSONRequest {
protected httpOptions = {
headers: new HttpHeaders({ 'Content-Type': 'application/json' })
};
protected httpError = new JSONResponse<any>();
/**
* Handle Http operation that failed.
* Let the app continue.
* @param operation - name of the operation that failed
* @param result - optional value to return as the observable result
*/
protected handleError<T>(operation = 'operation', result?: T) {
return (error: any): Observable<T> => {
// TODO: send the error to remote logging infrastructure
console.error(error); // log to console instead
// Let the app keep running by returning an empty result.
return of(result as T);
};
}
}

@ -0,0 +1,38 @@
import {Result} from './Result';
export class JSONResponse<T> {
// 响应结果
// tslint:disable-next-line:variable-name
private _result: Result;
// 响应详细结果
// tslint:disable-next-line:variable-name
private _message: string;
// 自定义其他响应信息
// tslint:disable-next-line:variable-name
private _body: T;
get result(): Result {
return this._result;
}
set result(value: Result) {
this._result = value;
}
get message(): string {
return this._message;
}
set message(value: string) {
this._message = value;
}
get body(): T {
return this._body;
}
set body(value: T) {
this._body = value;
}
}

@ -0,0 +1,4 @@
export enum Result {
OK = 'OK',
FAIL = 'FAIL'
}

@ -10,7 +10,8 @@
"register": { "register": {
"confirm_pwd": "confrim password", "confirm_pwd": "confrim password",
"mobile": "mobile", "mobile": "mobile",
"email": "email" "email": "email",
"emailType": "email type"
}, },
"reset_pwd": { "reset_pwd": {
"code": "verification code" "code": "verification code"
@ -28,8 +29,14 @@
"complaint": "time:{{time}}" "complaint": "time:{{time}}"
}, },
"tip":{ "tip":{
"input": "please input ", "input": "please input {{value}}",
"HELLO": "Hello {{value}} welcome to you" "HELLO": "Hello {{value}} welcome to you",
"notnull": "The {{value}} cannot be empty",
"password_null": "the password cannot be empty",
"password_diff": "The passwords do not match",
"mobile_null": "The phone number cannot be empty",
"mobile_error": "The phone number is illegal",
"select": "please choose an {{value}}"
}, },
"button": { "button": {
"login": "login", "login": "login",

@ -10,7 +10,8 @@
"register": { "register": {
"confirm_pwd": "确认密码", "confirm_pwd": "确认密码",
"mobile": "手机", "mobile": "手机",
"email": "邮箱" "email": "邮箱",
"email_type": "邮箱类型"
}, },
"reset_pwd": { "reset_pwd": {
"code": "验证码" "code": "验证码"
@ -28,8 +29,14 @@
"complaint_time": "投诉时间:{{time}}" "complaint_time": "投诉时间:{{time}}"
}, },
"tip":{ "tip":{
"input": "请输入", "input": "请输入{{value}}",
"HELLO": "欢迎管理员{{value}}登陆" "HELLO": "欢迎管理员{{value}}登陆",
"notnull": "{{value}}不能为空",
"password_null": "确认密码不能为空",
"password_diff": "密码和确认密码不一致",
"mobile_null": "手机号不能为空",
"mobile_error": "手机号不合法",
"select": "请选择{{value}}"
}, },
"button": { "button": {
"login": "登陆", "login": "登陆",

@ -0,0 +1,21 @@
// This file can be replaced during build by using the `fileReplacements` array.
// `ng build --prod` replaces `environment.ts` with `environment.prod.ts`.
// The list of file replacements can be found in `angular.json`.
export const environment = {
production: false,
// 服务端地址
apiServer: 'http://rap2.taobao.org:38080/app/mock/245830',
// token key
tokenKey: 'token'
};
/*
* For easier debugging in development mode, you can import the following file
* to ignore zone related error stack frames such as `zone.run`, `zoneDelegate.invokeTask`.
*
* This import should be commented out in production mode because it will have a negative impact
* on performance if an error is thrown.
*/
// import 'zone.js/dist/zone-error'; // Included with Angular CLI.
Loading…
Cancel
Save