LoginSignup
5
4

More than 5 years have passed since last update.

第2回 Angular勉強会 〜楽曲検索アプリ作成〜

Last updated at Posted at 2018-07-30

前回に続き、今回は 第2回「Webフロントエンド開発勉強会」 で作成した楽曲検索アプリを Angular で実装することを目的として、コンポーネント、サービス、HTTP通信について説明します。

第1回 Angular勉強会を実施していない方は事前に実施しておいてください。

前回の復習

検索ボタンを押したら固定の楽曲情報を表示するページを作成してください。

$ ng generate component study2/music-search
app-routing.module.ts
import { EcSiteComponent } from './study2/ec-site/ec-site.component';

const routes: Routes = [

  :

  { path: 'study2/music-search', component: MusicSearchComponent },
];
index.component.html
<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>
music-search.component.html
<div>
  <button>検索</button>
  <div id="resultContent">
  </div>
</div>
music-search.component.ts
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' }
    ];
  }
}

image.png

答え
music-search.component.ts
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' }
    ];
  }
}
music-search.component.html
<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>
music-search.component.css
.track-name {
  font-weight: bold;
}

コンポーネント

HTMLの要素をカプセル化して再利用可能なパーツを提供するための仕組みをコンポーネントといいます。

これまで作ってきた各ページもコンポーネントですが、コンポーネントを複数組み合わせて利用することで、それぞれが持つ機能や役割が明確になったり、パーツの再利用が可能になったりします。

以下のコンポーネントを作成してページに追加してください。

$ ng generate component study2/ec-site
$ ng generate component study2/product
app-routing.module.ts
import { EcSiteComponent } from './study2/ec-site/ec-site.component';

const routes: Routes = [

  :

  { path: 'study2/ec-site', component: EcSiteComponent },
];
index.component.html
<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.tsselector: 'app-product' の部分が、ProductComponent を利用するためのセレクタ名(タグ名)となり、@Input('product') product: ProductInfo; が、このコンポーネントへ渡すプロパティとなります。
コンポーネントを利用する場合は、 <app-product [product]="商品オブジェクト"></app-product> のように指定します。

product.component.ts
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() {
  }

}
product.component.html
<div class="product">
  <img [src]="product?.image">
  <p>{{product?.name}}</p>
</div>
product.component.css
.product {
    width: 130px;
    margin: 8px;
    padding: 8px;
    border: 1px solid gray;
    background-color: #f7f7f7;
    text-align: center;
}
ec-site.component.ts
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}」を買い物かごへ追加`);
  }
}
ec-site.component.html
<div class="products">
  <app-product *ngFor="let product of products" [product]="product"></app-product>
</div>
ec-site.component.css
.products {
    display: flex;
    flex-wrap: wrap;
}

image.png

コンポーネントにコンテンツを表示

コンポーネント内に <ng-content> を利用することで、呼び出し元で指定したコンテンツを表示することができます。

product.component.html
<div class="product">
  <img [src]="product?.image">
  <p>{{product?.name}}</p>
  <ng-content></ng-content>
</div>
ec-site.component.html
<div class="products">
  <app-product *ngFor="let product of products" [product]="product">
    <p>セール中!</p>
  </app-product>
</div>

image.png

<ng-content>select="セレクタ" を付けることで、複数のコンテンツを表示することもできます。

product.component.html
<div class="product">
  <img [src]="product?.image">
  <p>{{product?.name}}</p>
  <ng-content select=".sale"></ng-content>
  <ng-content select="div"></ng-content>
</div>
ec-site.component.html
<div class="products">
  <app-product *ngFor="let product of products" [product]="product">
    <p class="sale">セール中!</p>
    <div>お知らせ</div>
  </app-product>
</div>

image.png

コンポーネントのイベントを処理する

コンポーネント内で発生したイベントを呼び出し元のコンポーネントへ伝えるには OutputEventEmitter を使用します。

@Output() プロパティ名 = new EventEmitter<型>(); で定義して、emit(オブジェクト) でイベントを伝えます。

product.component.ts
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);
  }
}
product.component.html
<div class="product">
  <img [src]="product?.image">
  <p>{{product?.name}}</p>
  <button (click)="onAddClick()">追加</button>
</div>
ec-site.component.html
<div class="products">
  <app-product *ngFor="let product of products" [product]="product" (onAddProduct)="onAddProduct($event)">
  </app-product>
</div>
ec-site.component.ts
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
product.service.ts
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;
  }
}
ec-site.component.ts
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}」を買い物かごへ追加`);
  }
}
ec-site.component.html
<div class="products">
  <app-product *ngFor="let product of productService.products()" [product]="product" (onAddProduct)="onAddProduct($event)">
  </app-product>
</div>

インスタンス生成方法

サービスは、モジュールやコンポーネントの providers プロパティでインスタンスの生成方法を指定することができます。

デフォルトで useClass が指定され、providedInroot を指定した場合、または app.module で指定した場合は、アプリケーション全体で1つのインスタンス(シングルトン)となります。

指定方法 説明
useClass クラスを指定して、Injectされるたびにそのクラスのインスタンスを生成。
useValue オブジェクトを指定して、Injectされるたびにそのオブジェクトを参照。
useExisting DIトークン(サービス)を指定して、エイリアス(別名)で生成。
useFactory ファクトリー関数を指定して、Injectの際にオブジェクトを生成。

HTTP通信

Angular の HTTP通信は @angular/common/httpHttpClientModule で提供されています。
app.module.tsHttpClientModule のインポートを追加してください。

Angular 4.2 までのHTTP通信は @angular/httpHttpModule を利用していましたが、Angular 4.3 からは改良された @angular/common/httpHttpClientModule が追加され、これまでの HttpModule は非推奨となりました。

app.module.ts
import { HttpClientModule } from '@angular/common/http';

:

@NgModule({
  declarations: [
      :
  ],
  imports: [
      :
    HttpClientModule,
  ],
  providers: [],
  bootstrap: [AppComponent]
})
export class AppModule { }

新にコンポーネントを生成し、http.component.tsHttpClient をインポートし、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
http.component.ts
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
    );
  }
}
http.component.html
<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 だけですが、オプションの observeresponse を指定することで、ヘッダーやステータスコード等を含むレスポンスを取得できます。

http.component.ts
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

答え
music-item.component.html
<div>
  <img [src]="item?.artworkUrl100">
  <div class="track-name">{{item?.trackName}}</div>
  <div class="artist-name">{{item?.artistName}}</div>
</div>
music-item.component.css
.track-name {
    font-weight: bold;
}
music-item.component.ts
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() {
  }

}
music-search.component.html
<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

答え
music-search.service.ts
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' }
    ];
  }
}
music-search.component.ts
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 を呼び出し、テキストボックスに入力したキーワードのアーティストの楽曲を検索し取得結果を表示してください。

step3

答え
music-search.service.ts
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);
  }
}
music-search.component.html
<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>
music-search.component.ts
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. 検索対象の切り替え

アーティスト名、曲名のラジオボタンを追加して、選択した値を対象として検索するようにしてください。

step4

答え
music-search.component.html
<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>
music-search.component.ts
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;
    });
  }
}
music-search.service.ts
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 の楽曲ページを表示してください。

step5

答え
music-search.component.ts
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. 横並びレイアウト

検索結果を横に並べて表示してください。

step6

答え
music-search.component.css
#resultContent {
    display: flex;
    flex-wrap: wrap;
}

app-music-item {
    width: 200px;
    margin-bottom: 16px;
}

Extra1. ページング

ページング機能を追加してください。

ex1.gif

答え
music-search.component.html
<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>
music-search.component.css
#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;
}
music-search.component.ts
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);
  }
}
music-search.service.ts
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);
  }
}

5
4
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
5
4