data:image/s3,"s3://crabby-images/48983/4898318718a1f503d88de95c219a30ceec199762" alt="windows-typescript-0004-03"
こんにちは!Popoです。
この記事を読むと「Angularでの画面遷移」が解決できます。
前回、Angularについて勉強しました。
data:image/s3,"s3://crabby-images/00a0f/00a0f776a911e78a3dea73411db576b320df41e3" alt=""
スマフォアプリの場合、色々な画面に遷移をして各画面に実装されている機能を使用したりします。
data:image/s3,"s3://crabby-images/0d601/0d60179b063001788e1d955899822fdbc7a2568d" alt=""
Web画面での画面遷移てどうやってやるんだろう?
画面遷移というのは、アプリケーションを開発する上で避けては通れません🤔
これは必ずマスターしておきたい機能です!
data:image/s3,"s3://crabby-images/00f5b/00f5b1a5cfc3ef7af58448cce1aee4cfd1df4b0f" alt=""
data:image/s3,"s3://crabby-images/00f5b/00f5b1a5cfc3ef7af58448cce1aee4cfd1df4b0f" alt=""
data:image/s3,"s3://crabby-images/00f5b/00f5b1a5cfc3ef7af58448cce1aee4cfd1df4b0f" alt=""
画面遷移!抑えておかねば!!
早速、画面遷移を勉強してみたいと思います。
今までの続きから始めますので、環境構築等がまだの方は下記記事を参考にしてください。
data:image/s3,"s3://crabby-images/2c121/2c121f8895e8c8ea65b5b3084c7937dc63c671f3" alt=""
data:image/s3,"s3://crabby-images/2c121/2c121f8895e8c8ea65b5b3084c7937dc63c671f3" alt=""
data:image/s3,"s3://crabby-images/00a0f/00a0f776a911e78a3dea73411db576b320df41e3" alt=""
data:image/s3,"s3://crabby-images/00a0f/00a0f776a911e78a3dea73411db576b320df41e3" alt=""
data:image/s3,"s3://crabby-images/239ec/239ecac7e531a241f97144c1707969d31a55be07" alt=""
data:image/s3,"s3://crabby-images/239ec/239ecac7e531a241f97144c1707969d31a55be07" alt=""
Angular、TypeScriptの動作環境
data:image/s3,"s3://crabby-images/16660/166600647d389fa60c406b1908e7e085807026b2" alt="Angular、TypeScriptの動作環境"
data:image/s3,"s3://crabby-images/16660/166600647d389fa60c406b1908e7e085807026b2" alt="Angular、TypeScriptの動作環境"
今回開発時の環境ソフトのバージョンです。
項目 | バージョン |
Node | v18.17.0 |
TypeScript | Version 5.1.6 |
OS | Windows 11 |
VSCode | version1.81.0 |
Angular CLI | 16.2.0 |
Angularのルータて何?
data:image/s3,"s3://crabby-images/46008/46008c7f3832e24713771bb7d8fb0b9fe8aed8e4" alt="Angularのルータて何?"
data:image/s3,"s3://crabby-images/46008/46008c7f3832e24713771bb7d8fb0b9fe8aed8e4" alt="Angularのルータて何?"
Swiftの場合、UINavigationControllerを利用してViewContorollerをプッシュさせたり、backでポップさせたりして画画面遷移を行っています。(他にも方法はあります。)
Angularの場合は、コンポーネントを組み合わせて画面を構成しているため、このコンポーネントを入れ替えることで画面遷移を表現しています。
この機能はルーティングと呼ばれ、RouterModuleによって提供されます。
ルート設定
プロジェクトを作成してRouterModuleを利用する場合、地図となるルートを記述する必要があります。(下記は参考ロジックです)
import { NgModule } from '@angular/core';
import { RouterModule, Routes } from '@angular/router';
// RouterModuleとRoutes型をインポート
import { LoginComponent } from './screen1/screen1.component';
// サンプル用コンポーネント screen1
import { MenuComponent } from './screen2/screen2.component';
// サンプル用コンポーネント screen2
const routes: Routes = [
{ path: 'screen2', component: screen2Component },
// http://localhost:3000/screen2 にアクセスされたらscreen2Component を表示する
{ path: '', redirectTo: 'screen1', pathMatch: 'full' },
// 以外のURLにアクセスされたら screen1Component を表示する
{ path: 'screen1', component: screen1Component }
// http://localhost:3000/screen1 にアクセスされたらscreen1Component を表示する
];
@NgModule({
imports: [RouterModule.forRoot(routes)],
exports: [RouterModule]
})
export class AppRoutingModule { }
ルート型(「const routes: Routes」)は配列になっています。この配列に指定された順番通りに優先順位が決まります。
スクリプトで何かしらの処理を行った場合に遷移させたい場合は、「navigate」メソッドを使用します。
<div class="container">
<h1>screen1</h1>
<form (ngSubmit)="onSubmit()" #screen1Form="ngForm">
<table>
<tr>
<td><label for="id">入力欄</label></td>
<td><input type="text" id="id" required [(ngModel)]="model.input1" name="input1" #idValid="ngModel">
<span [hidden]="idValid.valid || idValid.pristine">
入力して下さい。
</span>
</td>
</tr>
</table>
<button type="submit" [disabled]="!screen1Form.form.valid">入力</button>
<br /> {{model.message}}
</form>
</div>
import { Component } from '@angular/core';
import { Router } from '@angular/router'
import { Screen1Model } from './Screen1Model';
@Component({
selector: 'app-screen1',
templateUrl: './screen1.component.html',
styleUrls: ['./screen1.component.css']
})
export class Screen1Component {
model = new Screen1Model('', '');
constructor(private router: Router) { }
ngOnInit(): void {
}
onSubmit(): void {
if (this.model.input1 == "test") {
this.router.navigate(['screen2']);
} else {
this.model.message = "入力エラー";
}
}
}
Angularのプロジェクトを作成してみよう
data:image/s3,"s3://crabby-images/0f671/0f671e59c38654d3b3e0fbed5c751902516738e1" alt="Angularのプロジェクトを作成してみよう"
data:image/s3,"s3://crabby-images/0f671/0f671e59c38654d3b3e0fbed5c751902516738e1" alt="Angularのプロジェクトを作成してみよう"
はじめにプロジェクトを作成します。
前回Angularのプロジェクトを作成した時は、下記のコマンドでした。
ng new <プロジェクト名>
今回は下記になります。「–routing」のオプションを追加します。
ng new プロジェクト名 --routing
結構時間がかかります。
data:image/s3,"s3://crabby-images/4cc15/4cc15e49765950802220d1599960655414ce9086" alt=""
data:image/s3,"s3://crabby-images/4cc15/4cc15e49765950802220d1599960655414ce9086" alt=""
「git」はインストールしていないのでエラーになっています。
data:image/s3,"s3://crabby-images/226bf/226bf862fd6171731a1fb48cf04d950fa495634a" alt=""
data:image/s3,"s3://crabby-images/226bf/226bf862fd6171731a1fb48cf04d950fa495634a" alt=""
自動的に必要なファイルが作成されます。
data:image/s3,"s3://crabby-images/e30f9/e30f91039e9bbb9ad0a0c094c2ed5662ba74f9f1" alt="プロジェクト作成3"
data:image/s3,"s3://crabby-images/e30f9/e30f91039e9bbb9ad0a0c094c2ed5662ba74f9f1" alt="プロジェクト作成3"
各画面のコンポーネントを作成してみましょう。
ng generate component screen1
ng generate component screen2
data:image/s3,"s3://crabby-images/405fa/405fa251160347bfd721095a4639d999ab2522c2" alt="Angularのプロジェクトを作成3"
data:image/s3,"s3://crabby-images/405fa/405fa251160347bfd721095a4639d999ab2522c2" alt="Angularのプロジェクトを作成3"
サクッと作成できました。
data:image/s3,"s3://crabby-images/a0678/a0678c6bb7a59956cdd3971bec36fc128432f01f" alt="プロジェクト作成5"
data:image/s3,"s3://crabby-images/a0678/a0678c6bb7a59956cdd3971bec36fc128432f01f" alt="プロジェクト作成5"
下記エラーが発生場合は、前回記事の「サーバー起動」を参照してください。
data:image/s3,"s3://crabby-images/00a0f/00a0f776a911e78a3dea73411db576b320df41e3" alt=""
data:image/s3,"s3://crabby-images/00a0f/00a0f776a911e78a3dea73411db576b320df41e3" alt=""
Xcodeの場合、UIでプロジェクトを作成できますが、TypeScriptの場合はコマンド入力で作成になります。
data:image/s3,"s3://crabby-images/00f5b/00f5b1a5cfc3ef7af58448cce1aee4cfd1df4b0f" alt=""
data:image/s3,"s3://crabby-images/00f5b/00f5b1a5cfc3ef7af58448cce1aee4cfd1df4b0f" alt=""
data:image/s3,"s3://crabby-images/00f5b/00f5b1a5cfc3ef7af58448cce1aee4cfd1df4b0f" alt=""
少し面倒ですね!
実際にアプリケーションを開発してみましょう!
data:image/s3,"s3://crabby-images/9da6c/9da6c3b92af9f6c500a101e60c73b21870597c8f" alt="実際にアプリケーションを開発してみましょう!"
data:image/s3,"s3://crabby-images/9da6c/9da6c3b92af9f6c500a101e60c73b21870597c8f" alt="実際にアプリケーションを開発してみましょう!"
それでは、画面遷移するアプリケーションを作成してみましょう。
簡単な画面ですが、Screen1画面で「test」という文字列を入力して「入力」ボタンを押下すると、Screen2画面に遷移する遷移になります。
data:image/s3,"s3://crabby-images/7d458/7d45846575900d50abffb4518b94a355ee3c5ebe" alt=""
data:image/s3,"s3://crabby-images/7d458/7d45846575900d50abffb4518b94a355ee3c5ebe" alt=""
Screen1画面関連
はじめにメインとなるScreen1画面を作成していきましょう。
src\app\screen1\Screen1Model.ts
入力画面のモデルクラス「Screen1Model」を新規に作成します。
data:image/s3,"s3://crabby-images/11a2f/11a2f65053ddfe8a7501c97bf7566be8b61f97d2" alt="入力画面1"
data:image/s3,"s3://crabby-images/11a2f/11a2f65053ddfe8a7501c97bf7566be8b61f97d2" alt="入力画面1"
中身は下記の通り。入力テキストとメッセージ項目を定義します。
export class Screen1Model {
constructor(
public input1: string,
public message?: string
) { }
}
src\app\screen1\screen1.component.ts
入力画面コンポーネントに処理と画面遷移の処理を追加します。
import { Component } from '@angular/core';
import { Router } from '@angular/router'
import { Screen1Model } from './Screen1Model';
@Component({
selector: 'app-screen1',
templateUrl: './screen1.component.html',
styleUrls: ['./screen1.component.css']
})
export class Screen1Component {
model = new Screen1Model('', '');
constructor(private router: Router) { }
ngOnInit(): void {
}
onSubmit(): void {
if (this.model.input1 == "test") {
this.router.navigate(['screen2']);
} else {
this.model.message = "入力エラー";
}
}
}
src\app\login\login.component.html
入力画面は、自動で生成されたhtmlファイルを修正します。
<div class="container">
<h1>screen1</h1>
<form (ngSubmit)="onSubmit()" #screen1Form="ngForm">
<table>
<tr>
<td><label for="id">入力欄</label></td>
<td><input type="text" id="id" required [(ngModel)]="model.input1" name="input1" #idValid="ngModel">
<span [hidden]="idValid.valid || idValid.pristine">
入力して下さい。
</span>
</td>
</tr>
</table>
<button type="submit" [disabled]="!screen1Form.form.valid">入力</button>
<br /> {{model.message}}
</form>
</div>
src\app\app.module.ts
テンプレート駆動フォームを使用するので@NgModuleにFormsModuleを追加します。
Angular には2つのフォームスタイルがありますが、今回はテンプレート駆動フォームを使用してみます。
import { NgModule } from '@angular/core';
import { BrowserModule } from '@angular/platform-browser';
import { FormsModule } from '@angular/forms';
import { AppRoutingModule } from './app-routing.module';
import { AppComponent } from './app.component';
import { Screen1Component } from './screen1/screen1.component';
import { Screen2Component } from './screen2/screen2.component';
@NgModule({
declarations: [
AppComponent,
Screen1Component,
Screen2Component
],
imports: [
BrowserModule,
AppRoutingModule,
FormsModule
],
providers: [],
bootstrap: [AppComponent]
})
export class AppModule { }
Screen2画面関連
次にScreen2画面を作成してみます。
戻るボタン押下でScreen1画面に遷移します。
src\app\screen2\screen2.component.ts
「入力」ボタン押下後は、「戻る」ボタン表示だけの画面にしました。
import { Component, OnInit } from '@angular/core';
@Component({
selector: 'app-screen2',
template:'<h1>screen2</h1><a routerLink="/screen1">戻る</a>'
})
export class Screen2Component implements OnInit {
constructor() { }
ngOnInit(): void {
}
}
ルーター機能関連
Angularでの画面遷移機能で重要なルータ機能部分のロジックを実装してみましょう。
src\app\app.component.ts
「template」に「router-outlet」を設定します。
import { Component } from '@angular/core';
@Component({
selector: 'app-root',
template: '<router-outlet></router-outlet>'
})
export class AppComponent {
}
src\app\app-routing.module.ts
ルータ情報を設定します。
import { NgModule } from '@angular/core';
import { RouterModule, Routes } from '@angular/router';
import { Screen1Component } from './screen1/screen1.component';
import { Screen2Component } from './screen2/screen2.component';
const routes: Routes = [
{ path: 'screen2', component: Screen2Component },
{ path: '', redirectTo: 'screen1', pathMatch: 'full' },
{ path: 'screen1', component: Screen1Component }
];
@NgModule({
imports: [RouterModule.forRoot(routes)],
exports: [RouterModule]
})
export class AppRoutingModule { }
実行
それでは、web画面を表示してみましょう!
ターミナルから下記コマンドを入力します。
ng serve --open
コンパイルが正常終了すると、「Compiled successfully.」が表示されweb画面が表示されます。
data:image/s3,"s3://crabby-images/e0401/e040145680d449f7bdf514f3739c2c750f9c5632" alt="コンパイル"
data:image/s3,"s3://crabby-images/e0401/e040145680d449f7bdf514f3739c2c750f9c5632" alt="コンパイル"
data:image/s3,"s3://crabby-images/74136/74136c0cd841eaee65504567fb65d82aabdcbd7d" alt="実行1"
data:image/s3,"s3://crabby-images/74136/74136c0cd841eaee65504567fb65d82aabdcbd7d" alt="実行1"
「test」を入力して入力ボタンを押下すると次画面に遷移します。
data:image/s3,"s3://crabby-images/5f458/5f458718bc3a94822cd44c03103cc2bc247011e6" alt="実行2"
data:image/s3,"s3://crabby-images/5f458/5f458718bc3a94822cd44c03103cc2bc247011e6" alt="実行2"
data:image/s3,"s3://crabby-images/16fb3/16fb356ec648a78c60342a49fe2bdc6488928fe4" alt="実行3"
data:image/s3,"s3://crabby-images/16fb3/16fb356ec648a78c60342a49fe2bdc6488928fe4" alt="実行3"
「戻る」ボタン押下で前画面に戻ります。
data:image/s3,"s3://crabby-images/7ba0b/7ba0b895e325a28d366f95c2a3e6380e94700e45" alt="実行4"
data:image/s3,"s3://crabby-images/7ba0b/7ba0b895e325a28d366f95c2a3e6380e94700e45" alt="実行4"
「test」以外の入力はエラーになります。
data:image/s3,"s3://crabby-images/994f9/994f9502700089c897a7efe6a98257909fc341c6" alt="実行5"
data:image/s3,"s3://crabby-images/994f9/994f9502700089c897a7efe6a98257909fc341c6" alt="実行5"
data:image/s3,"s3://crabby-images/00f5b/00f5b1a5cfc3ef7af58448cce1aee4cfd1df4b0f" alt=""
data:image/s3,"s3://crabby-images/00f5b/00f5b1a5cfc3ef7af58448cce1aee4cfd1df4b0f" alt=""
data:image/s3,"s3://crabby-images/00f5b/00f5b1a5cfc3ef7af58448cce1aee4cfd1df4b0f" alt=""
できました!
まとめ
data:image/s3,"s3://crabby-images/414c2/414c249179163cbb53f55e3bd9b9f1012d6a97f8" alt="まとめ"
data:image/s3,"s3://crabby-images/414c2/414c249179163cbb53f55e3bd9b9f1012d6a97f8" alt="まとめ"
今回はAngularで画面遷移を実装してみました。肝となるのはルータ機能です。
Angularでの画面遷移はルータ機能を利用!
値等も引き継ぐことができるようなので、一度実装してみたいと思っています。
ついついSwiftと比べてしまうのですが、Angularはファイル数が多いですね。😓
Objective-Cといった感じですね!
Objective-CからSwiftに移行して、ヘッダーファイルが無くなりすっきりしました。
TypeScriptをやっていると、Objective-Cに戻った感覚があります!😱
でもそれはそれ!TypeScriptはTypeScriptでしっかり勉強していきたいと思っています。
data:image/s3,"s3://crabby-images/00f5b/00f5b1a5cfc3ef7af58448cce1aee4cfd1df4b0f" alt=""
data:image/s3,"s3://crabby-images/00f5b/00f5b1a5cfc3ef7af58448cce1aee4cfd1df4b0f" alt=""
data:image/s3,"s3://crabby-images/00f5b/00f5b1a5cfc3ef7af58448cce1aee4cfd1df4b0f" alt=""
それではまた!