前回に続き、今回は 第2回「Webフロントエンド開発勉強会」 で作成した楽曲検索アプリを Angular で実装することを目的として、コンポーネント、サービス、HTTP通信について説明します。
第1回 Angular勉強会を実施していない方は事前に実施しておいてください。
前回の復習
検索ボタンを押したら固定の楽曲情報を表示するページを作成してください。
$ ng generate component study2/music-search
import { EcSiteComponent } from './study2/ec-site/ec-site.component';
const routes: Routes = [
:
{ path: 'study2/music-search', component: MusicSearchComponent },
];
<h1>第1回 Angular勉強会</h1>
<ul>
<li><a routerLink="/type-script">TypeScript</a></li>
<li><a routerLink="/data-binding">data-binding</a></li>
<li><a routerLink="/pipe">pipe</a></li>
<li><a routerLink="/directive">directive</a></li>
</ul>
<h1>第2回 Angular勉強会</h1>
<ul>
<li><a routerLink="/study2/music-search">music-search</a></li>
</ul>
<div>
<button>検索</button>
<div id="resultContent">
</div>
</div>
import { Component, OnInit } from '@angular/core';
class MusicItem {
trackName: string;
artistName: string;
artworkUrl100: string;
}
@Component({
selector: 'app-music-search',
templateUrl: './music-search.component.html',
styleUrls: ['./music-search.component.css']
})
export class MusicSearchComponent implements OnInit {
constructor() { }
ngOnInit() {
}
/**
* 検索実行
*/
onSearch() {
const results = [
{ artistName: 'アーティスト名1', trackName: '楽曲名1', artworkUrl100: 'https://is5-ssl.mzstatic.com/image/thumb/Music6/v4/a5/df/97/a5df97ec-b7e4-7f78-625a-f331603b0756/source/100x100bb.jpg' },
{ artistName: 'アーティスト名2', trackName: '楽曲名2', artworkUrl100: 'https://is4-ssl.mzstatic.com/image/thumb/Music/v4/fb/d8/04/fbd8047c-293e-1f90-4b06-929bbeca20b1/source/100x100bb.jpg' },
{ artistName: 'アーティスト名3', trackName: '楽曲名3', artworkUrl100: 'https://is2-ssl.mzstatic.com/image/thumb/Music/v4/4d/2e/39/4d2e39ef-fd24-a902-e704-cb7dc2ae6c5b/source/100x100bb.jpg' },
{ artistName: 'アーティスト名4', trackName: '楽曲名4', artworkUrl100: 'https://is5-ssl.mzstatic.com/image/thumb/Music/v4/54/ff/d6/54ffd6b2-3f3f-ce92-189d-20c3143b2380/source/100x100bb.jpg' },
{ artistName: 'アーティスト名5', trackName: '楽曲名5', artworkUrl100: 'https://is2-ssl.mzstatic.com/image/thumb/Music6/v4/ae/4a/fd/ae4afdd5-0dc7-31a4-ab96-eacc806a32d5/source/100x100bb.jpg' }
];
}
}
答え
import { Component, OnInit } from '@angular/core';
class MusicItem {
trackName: string;
artistName: string;
artworkUrl100: string;
}
@Component({
selector: 'app-music-search',
templateUrl: './music-search.component.html',
styleUrls: ['./music-search.component.css']
})
export class MusicSearchComponent implements OnInit {
results: Array<MusicItem>;
constructor() { }
ngOnInit() {
}
/**
* 検索実行
*/
onSearch() {
this.results = [
{ artistName: 'アーティスト名1', trackName: '楽曲名1', artworkUrl100: 'https://is5-ssl.mzstatic.com/image/thumb/Music6/v4/a5/df/97/a5df97ec-b7e4-7f78-625a-f331603b0756/source/100x100bb.jpg' },
{ artistName: 'アーティスト名2', trackName: '楽曲名2', artworkUrl100: 'https://is4-ssl.mzstatic.com/image/thumb/Music/v4/fb/d8/04/fbd8047c-293e-1f90-4b06-929bbeca20b1/source/100x100bb.jpg' },
{ artistName: 'アーティスト名3', trackName: '楽曲名3', artworkUrl100: 'https://is2-ssl.mzstatic.com/image/thumb/Music/v4/4d/2e/39/4d2e39ef-fd24-a902-e704-cb7dc2ae6c5b/source/100x100bb.jpg' },
{ artistName: 'アーティスト名4', trackName: '楽曲名4', artworkUrl100: 'https://is5-ssl.mzstatic.com/image/thumb/Music/v4/54/ff/d6/54ffd6b2-3f3f-ce92-189d-20c3143b2380/source/100x100bb.jpg' },
{ artistName: 'アーティスト名5', trackName: '楽曲名5', artworkUrl100: 'https://is2-ssl.mzstatic.com/image/thumb/Music6/v4/ae/4a/fd/ae4afdd5-0dc7-31a4-ab96-eacc806a32d5/source/100x100bb.jpg' }
];
}
}
<div>
<button (click)="onSearch()">検索</button>
<div id="resultContent">
<div *ngFor="let item of results">
<img [src]="item?.artworkUrl100">
<div class="track-name">{{item?.trackName}}</div>
<div class="artist-name">{{item?.artistName}}</div>
</div>
</div>
</div>
.track-name {
font-weight: bold;
}
コンポーネント
HTMLの要素をカプセル化して再利用可能なパーツを提供するための仕組みをコンポーネントといいます。
これまで作ってきた各ページもコンポーネントですが、コンポーネントを複数組み合わせて利用することで、それぞれが持つ機能や役割が明確になったり、パーツの再利用が可能になったりします。
以下のコンポーネントを作成してページに追加してください。
$ ng generate component study2/ec-site
$ ng generate component study2/product
import { EcSiteComponent } from './study2/ec-site/ec-site.component';
const routes: Routes = [
:
{ path: 'study2/ec-site', component: EcSiteComponent },
];
<h1>第1回 Angular勉強会</h1>
<ul>
<li><a routerLink="/type-script">TypeScript</a></li>
<li><a routerLink="/data-binding">data-binding</a></li>
<li><a routerLink="/pipe">pipe</a></li>
<li><a routerLink="/directive">directive</a></li>
</ul>
<h1>第2回 Angular勉強会</h1>
<ul>
<li><a routerLink="/study2/ec-site">ec-site</a></li>
</ul>
コンポーネントを作成する
ECサイトの商品を想定した商品コンポーネント ProductComponent
を作成してみましょう。
product.component.ts
の selector: 'app-product'
の部分が、ProductComponent
を利用するためのセレクタ名(タグ名)となり、@Input('product') product: ProductInfo;
が、このコンポーネントへ渡すプロパティとなります。
コンポーネントを利用する場合は、 <app-product [product]="商品オブジェクト"></app-product>
のように指定します。
import { Component, OnInit, Input } from '@angular/core';
export class ProductInfo {
id: number;
name: string;
image?: string;
}
@Component({
selector: 'app-product',
templateUrl: './product.component.html',
styleUrls: ['./product.component.css']
})
export class ProductComponent implements OnInit {
@Input('product') product: ProductInfo;
constructor() { }
ngOnInit() {
}
}
<div class="product">
<img [src]="product?.image">
<p>{{product?.name}}</p>
</div>
.product {
width: 130px;
margin: 8px;
padding: 8px;
border: 1px solid gray;
background-color: #f7f7f7;
text-align: center;
}
import { Component } from '@angular/core';
import { ProductInfo } from './../product/product.component'
@Component({
selector: 'app-ec-site',
templateUrl: './ec-site.component.html',
styleUrls: ['./ec-site.component.css']
})
export class EcSiteComponent {
products: ProductInfo[] = [
{ id: 0, name: 'TシャツA', image: '' },
{ id: 1, name: 'TシャツB', image: '' },
{ id: 2, name: 'TシャツC', image: '' },
{ id: 3, name: 'ボトムスA', image: '' },
{ id: 4, name: 'ボトムスB', image: '' },
];
constructor() { }
onAddProduct(product: ProductInfo) {
alert(`「${product.name}」を買い物かごへ追加`);
}
}
<div class="products">
<app-product *ngFor="let product of products" [product]="product"></app-product>
</div>
.products {
display: flex;
flex-wrap: wrap;
}
コンポーネントにコンテンツを表示
コンポーネント内に <ng-content>
を利用することで、呼び出し元で指定したコンテンツを表示することができます。
<div class="product">
<img [src]="product?.image">
<p>{{product?.name}}</p>
<ng-content></ng-content>
</div>
<div class="products">
<app-product *ngFor="let product of products" [product]="product">
<p>セール中!</p>
</app-product>
</div>
<ng-content>
に select="セレクタ"
を付けることで、複数のコンテンツを表示することもできます。
<div class="product">
<img [src]="product?.image">
<p>{{product?.name}}</p>
<ng-content select=".sale"></ng-content>
<ng-content select="div"></ng-content>
</div>
<div class="products">
<app-product *ngFor="let product of products" [product]="product">
<p class="sale">セール中!</p>
<div>お知らせ</div>
</app-product>
</div>
コンポーネントのイベントを処理する
コンポーネント内で発生したイベントを呼び出し元のコンポーネントへ伝えるには Output
と EventEmitter
を使用します。
@Output() プロパティ名 = new EventEmitter<型>();
で定義して、emit(オブジェクト)
でイベントを伝えます。
import { Component, OnInit, Input, Output, EventEmitter } from '@angular/core';
export class ProductInfo {
id: number;
name: string;
image?: string;
}
@Component({
selector: 'app-product',
templateUrl: './product.component.html',
styleUrls: ['./product.component.css']
})
export class ProductComponent implements OnInit {
@Input('product') product: ProductInfo;
@Output() onAddProduct = new EventEmitter<ProductInfo>();
constructor() { }
ngOnInit() {
}
onAddClick() {
this.onAddProduct.emit(this.product);
}
}
<div class="product">
<img [src]="product?.image">
<p>{{product?.name}}</p>
<button (click)="onAddClick()">追加</button>
</div>
<div class="products">
<app-product *ngFor="let product of products" [product]="product" (onAddProduct)="onAddProduct($event)">
</app-product>
</div>
import { Component, OnInit } from '@angular/core';
import { ProductInfo } from './../product/product.component'
@Component({
selector: 'app-component',
templateUrl: './ec-site.component.html',
styleUrls: ['./ec-site.component.css']
})
export class EcSiteComponent implements OnInit {
products: ProductInfo[] = [
{ id: 0, name: 'TシャツA', image: '' },
{ id: 1, name: 'TシャツB', image: '' },
{ id: 2, name: 'TシャツC', image: '' },
{ id: 3, name: 'ボトムスA', image: '' },
{ id: 4, name: 'ボトムスB', image: '' },
];
constructor() { }
ngOnInit() {
}
onAddProduct(product: ProductInfo) {
alert(`「${product.name}」を買い物かごへ追加`);
}
}
ライフサイクル
Angularには、コンポーネントの生成時や、状態の変更時など、特定のタイミングで呼ばれるライフサイクルメソッドが用意されています。
メソッド名 | 実行タイミング |
---|---|
ngOnChanges | コンポーネントの入力プロパティ(@Input )変更時 |
ngOnInit | コンポーネントの初期化時 |
ngDoCheck | 変更を検知したとき |
ngAfterContentInit | 外部コンテンツ初期化時 |
ngAfterContentChecked | 外部コンテンツ変更時 |
ngAfterViewInit | 自分自身と子コンポーネントのビュー初期化時 |
ngAfterViewChecked | 自分自身と子コンポーネントのビュー変更時 |
ngOnDestroy | コンポーネントが破棄されるとき |
詳細は以下で説明されています。
https://qiita.com/ksh-fthr/items/ccd9861f919c4aa30ae8
サービス
共通のビジネスロジックを提供する仕組みをサービスといいます。
コンポーネントやディレクティブの constructor
にサービス型の引数を指定することで、依存性注入(Dependency Injection、DI とも呼ばれます)され、使うことができます。
アクセス修飾子を指定しない場合は constructor内のみ、private
を指定した場合はクラス内、public
を指定した場合はクラス外へ公開されます。
$ ng generate service study2/product/product
import { Injectable } from '@angular/core';
import { ProductInfo } from './product.component'
@Injectable({
providedIn: 'root'
})
export class ProductService {
private _products: ProductInfo[] = [
{ id: 0, name: 'TシャツA', image: '' },
{ id: 1, name: 'TシャツB', image: '' },
{ id: 2, name: 'TシャツC', image: '' },
{ id: 3, name: 'ボトムスA', image: '' },
{ id: 4, name: 'ボトムスB', image: '' },
];
constructor() { }
products() {
return this._products;
}
}
import { Component } from '@angular/core';
import { ProductInfo } from './../product/product.component'
import { ProductService } from '../product/product.service';
@Component({
selector: 'app-ec-site',
templateUrl: './ec-site.component.html',
styleUrls: ['./ec-site.component.css']
})
export class EcSiteComponent {
constructor(public productService: ProductService) { }
onAddProduct(product: ProductInfo) {
alert(`「${product.name}」を買い物かごへ追加`);
}
}
<div class="products">
<app-product *ngFor="let product of productService.products()" [product]="product" (onAddProduct)="onAddProduct($event)">
</app-product>
</div>
インスタンス生成方法
サービスは、モジュールやコンポーネントの providers
プロパティでインスタンスの生成方法を指定することができます。
デフォルトで useClass
が指定され、providedIn
で root
を指定した場合、または app.module で指定した場合は、アプリケーション全体で1つのインスタンス(シングルトン)となります。
指定方法 | 説明 |
---|---|
useClass | クラスを指定して、Injectされるたびにそのクラスのインスタンスを生成。 |
useValue | オブジェクトを指定して、Injectされるたびにそのオブジェクトを参照。 |
useExisting | DIトークン(サービス)を指定して、エイリアス(別名)で生成。 |
useFactory | ファクトリー関数を指定して、Injectの際にオブジェクトを生成。 |
HTTP通信
Angular の HTTP通信は @angular/common/http
の HttpClientModule
で提供されています。
app.module.ts
に HttpClientModule
のインポートを追加してください。
Angular 4.2 までのHTTP通信は
@angular/http
のHttpModule
を利用していましたが、Angular 4.3 からは改良された@angular/common/http
のHttpClientModule
が追加され、これまでのHttpModule
は非推奨となりました。
import { HttpClientModule } from '@angular/common/http';
:
@NgModule({
declarations: [
:
],
imports: [
:
HttpClientModule,
],
providers: [],
bootstrap: [AppComponent]
})
export class AppModule { }
新にコンポーネントを生成し、http.component.ts
で HttpClient
をインポートし、constructor で DI (Dependency Injection)してください。
this.http.get(url, option)
で HTTP の GETリクエストが送信でき、.subscribe()
で結果を処理できます。
get<型>()
で、レスポンスの型を指定できます。(指定しない場合は any となる)
HttpClient
には get
の他に post
, put
, delete
などのメソッドが用意されています。
https://angular.io/api/common/http/HttpClient
$ ng generate component study2/http
import { Component, OnInit } from '@angular/core';
import { HttpClient } from '@angular/common/http';
class ResponseBody {
resultCount: number;
results: Array<any>;
}
@Component({
selector: 'app-http',
templateUrl: './http.component.html',
styleUrls: ['./http.component.css']
})
export class HttpComponent implements OnInit {
url = 'https://itunes.apple.com/lookup?id=292706922';
response: ResponseBody;
error: any;
// HttpClient を DI する
constructor(private http: HttpClient) { }
ngOnInit() {
// HTTP GETリクエスト
this.http.get<ResponseBody>(this.url).subscribe(
// 成功時のコールバック
response => this.response = response,
// 失敗時のコールバック
error => this.error = error
);
}
}
<p>{{url}}</p>
<pre *ngIf="response">
{{response | json}}
</pre>
<pre *ngIf="error">
{{error | json}}
</pre>
https://itunes.apple.com/lookup?id=292706922
{
"resultCount": 1,
"results": [
{
"wrapperType": "artist",
"artistType": "Artist",
"artistName": "AKB48",
"artistLinkUrl": "https://itunes.apple.com/us/artist/akb48/292706922?uo=4",
"artistId": 292706922,
"amgArtistId": 989429,
"primaryGenreName": "J-Pop",
"primaryGenreId": 27
}
]
}
レスポンスのフォーマットはデフォルトでは JSON ですが、オプションの responseType
で指定することもできます。
this.http.get(this.url, { responseType: 'text' })
レスポンスの内容はデフォルトでは body だけですが、オプションの observe
に response
を指定することで、ヘッダーやステータスコード等を含むレスポンスを取得できます。
import { Component, OnInit } from '@angular/core';
import { HttpClient, HttpResponse } from '@angular/common/http';
class ResponseBody {
resultCount: number;
results: Array<any>;
}
@Component({
selector: 'app-http',
templateUrl: './http.component.html',
styleUrls: ['./http.component.css']
})
export class HttpComponent implements OnInit {
url = 'https://itunes.apple.com/lookup?id=292706922';
response: HttpResponse<ResponseBody>;
error: any;
// HttpClient を DI する
constructor(private http: HttpClient) { }
ngOnInit() {
// HTTP GETリクエスト
this.http.get<ResponseBody>(this.url, { observe: 'response' }).subscribe(
// 成功時のコールバック
response => this.response = response,
// 失敗時のコールバック
error => this.error = error
);
}
}
https://itunes.apple.com/lookup?id=292706922
{
"headers": {
"normalizedNames": {},
"lazyUpdate": null
},
"status": 200,
"statusText": "OK",
"url": "https://itunes.apple.com/lookup?id=292706922",
"ok": true,
"type": 4,
"body": {
"resultCount": 1,
"results": [
{
"wrapperType": "artist",
"artistType": "Artist",
"artistName": "AKB48",
"artistLinkUrl": "https://itunes.apple.com/us/artist/akb48/292706922?uo=4",
"artistId": 292706922,
"amgArtistId": 989429,
"primaryGenreName": "J-Pop",
"primaryGenreId": 27
}
]
}
}
楽曲検索アプリ作成
Step1. 楽曲情報を Component化
前回の復習で作成したページの楽曲情報を Component で作成してください。
$ ng generate component study2/music-search/music-item
答え
<div>
<img [src]="item?.artworkUrl100">
<div class="track-name">{{item?.trackName}}</div>
<div class="artist-name">{{item?.artistName}}</div>
</div>
.track-name {
font-weight: bold;
}
import { Component, OnInit, Input } from '@angular/core';
import { MusicItem } from '../music-search.service';
@Component({
selector: 'app-music-item',
templateUrl: './music-item.component.html',
styleUrls: ['./music-item.component.css']
})
export class MusicItemComponent implements OnInit {
@Input() item: MusicItem;
constructor() { }
ngOnInit() {
}
}
<div>
<input type="text">
<button (click)="onSearch()">検索</button>
<div id="resultContent">
<app-music-item *ngFor="let item of results" [item]="item"></app-music-item>
</div>
</div>
Step2. データ取得処理を Service化
データ取得処理を Service で作成してください。
$ ng generate service study2/music-search/music-search
答え
import { Injectable } from '@angular/core';
export class MusicItem {
trackName: string;
artistName: string;
artworkUrl100: string;
}
@Injectable({
providedIn: 'root'
})
export class MusicSearchService {
constructor() { }
search() {
return [
{ artistName: 'アーティスト名1', trackName: '楽曲名1', artworkUrl100: 'https://is5-ssl.mzstatic.com/image/thumb/Music6/v4/a5/df/97/a5df97ec-b7e4-7f78-625a-f331603b0756/source/100x100bb.jpg' },
{ artistName: 'アーティスト名2', trackName: '楽曲名2', artworkUrl100: 'https://is4-ssl.mzstatic.com/image/thumb/Music/v4/fb/d8/04/fbd8047c-293e-1f90-4b06-929bbeca20b1/source/100x100bb.jpg' },
{ artistName: 'アーティスト名3', trackName: '楽曲名3', artworkUrl100: 'https://is2-ssl.mzstatic.com/image/thumb/Music/v4/4d/2e/39/4d2e39ef-fd24-a902-e704-cb7dc2ae6c5b/source/100x100bb.jpg' },
{ artistName: 'アーティスト名4', trackName: '楽曲名4', artworkUrl100: 'https://is5-ssl.mzstatic.com/image/thumb/Music/v4/54/ff/d6/54ffd6b2-3f3f-ce92-189d-20c3143b2380/source/100x100bb.jpg' },
{ artistName: 'アーティスト名5', trackName: '楽曲名5', artworkUrl100: 'https://is2-ssl.mzstatic.com/image/thumb/Music6/v4/ae/4a/fd/ae4afdd5-0dc7-31a4-ab96-eacc806a32d5/source/100x100bb.jpg' }
];
}
}
import { Component, OnInit } from '@angular/core';
import { MusicSearchService, MusicItem } from './music-search.service';
@Component({
selector: 'app-music-search',
templateUrl: './music-search.component.html',
styleUrls: ['./music-search.component.css']
})
export class MusicSearchComponent implements OnInit {
constructor(private musicSearch: MusicSearchService) { }
/** 検索結果 */
results: MusicItem[];
ngOnInit() {
}
/**
* 検索実行
*/
onSearch() {
this.results = this.musicSearch.search();
}
}
Step3. APIで取得した結果を表示
iTunes Search API を呼び出し、テキストボックスに入力したキーワードのアーティストの楽曲を検索し取得結果を表示してください。
答え
import { Injectable } from '@angular/core';
import { HttpClient } from '@angular/common/http';
export class ResponseBody {
resultCount: number;
results: MusicItem[];
}
export class MusicItem {
trackName: string;
artistName: string;
artworkUrl100: string;
}
@Injectable({
providedIn: 'root'
})
export class MusicSearchService {
constructor(private http: HttpClient) { }
search(keyword: string) {
var url = `https://itunes.apple.com/search?term=${keyword}&country=jp&media=music&attribute=artistTerm`;
return this.http.get<ResponseBody>(url);
}
}
<div>
<input type="text" #keyword>
<button (click)="onSearch(keyword.value)">検索</button>
<div id="resultContent">
<app-music-item *ngFor="let item of results" [item]="item"></app-music-item>
</div>
</div>
import { Component, OnInit } from '@angular/core';
import { MusicSearchService, MusicItem } from './music-search.service';
@Component({
selector: 'app-music-search',
templateUrl: './music-search.component.html',
styleUrls: ['./music-search.component.css']
})
export class MusicSearchComponent implements OnInit {
constructor(private musicSearch: MusicSearchService) { }
/** 検索結果 */
results: MusicItem[];
ngOnInit() {
}
/**
* 検索実行
* @param keyword 検索キーワード
*/
onSearch(keyword: string) {
this.musicSearch.search(keyword).subscribe(response => {
this.results = response.results;
});
}
}
Step4. 検索対象の切り替え
アーティスト名、曲名のラジオボタンを追加して、選択した値を対象として検索するようにしてください。
答え
<div>
<input type="text" #keyword>
<button (click)="onSearch(keyword.value)">検索</button>
<div>
<input type="radio" name="attribute" value="artistTerm" [(ngModel)]="attribute">アーティスト名
<input type="radio" name="attribute" value="songTerm" [(ngModel)]="attribute">曲名
</div>
<div id="resultContent">
<app-music-item *ngFor="let item of results" [item]="item"></app-music-item>
</div>
</div>
import { Component, OnInit } from '@angular/core';
import { MusicSearchService, MusicItem } from './music-search.service';
@Component({
selector: 'app-music-search',
templateUrl: './music-search.component.html',
styleUrls: ['./music-search.component.css']
})
export class MusicSearchComponent implements OnInit {
attribute = 'artistTerm';
constructor(private musicSearch: MusicSearchService) { }
/** 検索結果 */
results: MusicItem[];
ngOnInit() {
}
/**
* 検索実行
* @param keyword 検索キーワード
*/
onSearch(keyword: string) {
this.musicSearch.search(keyword, this.attribute).subscribe(response => {
this.results = response.results;
});
}
}
import { Injectable } from '@angular/core';
import { HttpClient } from '@angular/common/http';
export class ResponseBody {
resultCount: number;
results: MusicItem[];
}
export class MusicItem {
trackName: string;
artistName: string;
artworkUrl100: string;
}
@Injectable({
providedIn: 'root'
})
export class MusicSearchService {
constructor(private http: HttpClient) { }
search(keyword: string, attribute: string) {
var url = `https://itunes.apple.com/search?term=${keyword}&country=jp&media=music&attribute=${attribute}`;
return this.http.get<ResponseBody>(url);
}
}
Step5. iTunes ページ表示
検索結果の楽曲をクリックしたときに、新しいウィンドウで iTunes の楽曲ページを表示してください。
答え
import { Component, OnInit } from '@angular/core';
import { MusicSearchService, MusicItem } from './music-search.service';
@Component({
selector: 'app-music-search',
templateUrl: './music-search.component.html',
styleUrls: ['./music-search.component.css']
})
export class MusicSearchComponent implements OnInit {
attribute = 'artistTerm';
constructor(private musicSearch: MusicSearchService) { }
/** 検索結果 */
results: MusicItem[];
ngOnInit() {
}
/**
* 検索実行
* @param keyword 検索キーワード
*/
onSearch(keyword: string) {
this.musicSearch.search(keyword, this.attribute).subscribe(response => {
this.results = response.results;
});
}
/**
* 楽曲クリック時
* @param item 楽曲情報
*/
onItemClick(item: MusicItem) {
// iTunesのページを新しいウィンドウで開く
window.open(item.trackViewUrl, 'trackView', 'width=400, height=600');
}
}
Step6. 横並びレイアウト
検索結果を横に並べて表示してください。
答え
#resultContent {
display: flex;
flex-wrap: wrap;
}
app-music-item {
width: 200px;
margin-bottom: 16px;
}
Extra1. ページング
ページング機能を追加してください。
答え
<div>
<input type="text" [(ngModel)]="keyword">
<button (click)="onSearch(0)">検索</button>
<div>
<input type="radio" name="attribute" value="artistTerm" [(ngModel)]="attribute">アーティスト名
<input type="radio" name="attribute" value="songTerm" [(ngModel)]="attribute">曲名
</div>
<div id="resultContent">
<app-music-item *ngFor="let item of results" [item]="item" (click)="onItemClick(item)"></app-music-item>
</div>
<div id="pager">
<button (click)="onPrev()">←</button>
<span id="pageIndex">{{currentPage + 1}}</span>
<button (click)="onNext()">→</button>
</div>
</div>
#resultContent {
display: flex;
flex-wrap: wrap;
}
app-music-item {
width: 200px;
margin-bottom: 16px;
}
#resultContent {
display: flex;
flex-wrap: wrap;
}
#pager {
display: flex;
justify-content: space-around;
}
import { Component, OnInit } from '@angular/core';
import { MusicSearchService, MusicItem } from './music-search.service';
@Component({
selector: 'app-music-search',
templateUrl: './music-search.component.html',
styleUrls: ['./music-search.component.css']
})
export class MusicSearchComponent implements OnInit {
keyword: string;
attribute = 'artistTerm';
currentPage = 0;
constructor(private musicSearch: MusicSearchService) { }
/** 検索結果 */
results: MusicItem[];
ngOnInit() {
}
/**
* 検索実行
*/
onSearch(page = 0) {
this.currentPage = page;
this.musicSearch.search(this.keyword, this.attribute, page).subscribe(response => {
this.results = response.results;
});
}
/**
* 楽曲クリック時
* @param item 楽曲情報
*/
onItemClick(item: MusicItem) {
// iTunesのページを新しいウィンドウで開く
window.open(item.trackViewUrl, 'trackView', 'width=400, height=600');
}
onPrev() {
if (0 < this.currentPage) {
this.onSearch(this.currentPage - 1);
}
}
onNext() {
this.onSearch(this.currentPage + 1);
}
}
import { Injectable } from '@angular/core';
import { HttpClient } from '@angular/common/http';
export class ResponseBody {
resultCount: number;
results: MusicItem[];
}
export class MusicItem {
trackName: string;
artistName: string;
artworkUrl100: string;
trackViewUrl: string;
}
@Injectable({
providedIn: 'root'
})
export class MusicSearchService {
limit = 10;
constructor(private http: HttpClient) { }
search(keyword: string, attribute: string, page: number) {
var url = `https://itunes.apple.com/search?term=${keyword}&country=jp&media=music&attribute=${attribute}&limit=${this.limit}&offset=${this.limit * page}`;
return this.http.get<ResponseBody>(url);
}
}