2016-08-08 18:48:45 -04:00
|
|
|
extends ../../../ts/_cache/tutorial/toh-pt6.jade
|
2016-06-28 16:12:49 -04:00
|
|
|
|
|
|
|
block includes
|
|
|
|
include ../_util-fns
|
|
|
|
- var _Http = 'BrowserClient';
|
|
|
|
- var _Angular_Http = 'Dart <code>BrowserClient</code>'
|
|
|
|
- var _httpUrl = 'https://pub.dartlang.org/packages/http'
|
|
|
|
- var _Angular_http_library = 'Dart <a href="' + _httpUrl + '"><b>http</b></a> package'
|
2016-08-12 14:21:16 -04:00
|
|
|
- var _HttpModule = 'BrowserClient'
|
2016-06-28 16:12:49 -04:00
|
|
|
- var _JSON_stringify = 'JSON.encode'
|
|
|
|
|
|
|
|
block start-server-and-watch
|
|
|
|
:marked
|
|
|
|
### Keep the app compiling and running
|
|
|
|
Open a terminal/console window.
|
|
|
|
Start the Dart compiler, watch for changes, and start our server by entering the command:
|
|
|
|
|
|
|
|
code-example(language="bash").
|
|
|
|
pub serve
|
2016-08-12 14:21:16 -04:00
|
|
|
|
2016-06-28 16:12:49 -04:00
|
|
|
block http-library
|
|
|
|
:marked
|
2016-08-12 14:21:16 -04:00
|
|
|
We'll be using the !{_Angular_http_library}'s
|
2016-06-28 16:12:49 -04:00
|
|
|
`BrowserClient` class to communicate with a server.
|
|
|
|
|
|
|
|
### Pubspec updates
|
2016-08-12 14:21:16 -04:00
|
|
|
|
|
|
|
We need to add package dependencies for the
|
2016-08-02 12:59:35 -04:00
|
|
|
`stream_transformers` and !{_Angular_http_library}s.
|
2016-06-28 16:12:49 -04:00
|
|
|
|
|
|
|
We also need to add a `resolved_identifiers` entry, to inform the [angular2
|
|
|
|
transformer][ng2x] that we'll be using `BrowserClient`. (For an explanation of why
|
|
|
|
this extra configuration is needed, see the [HTTP client chapter][guide-http].) We'll
|
|
|
|
also need to use `Client` from http, so let's add that now as well.
|
|
|
|
|
|
|
|
Update `pubspec.yaml` to look like this (additions are highlighted):
|
|
|
|
|
|
|
|
[guide-http]: ../guide/server-communication.html#!#http-providers
|
|
|
|
[ng2x]: https://github.com/angular/angular/wiki/Angular-2-Dart-Transformer
|
|
|
|
|
2016-08-02 12:59:35 -04:00
|
|
|
- var stylePattern = { pnk: /(http.*|stream.*|resolved_identifiers:|Browser.*|Client.*)/gm };
|
2016-06-28 16:12:49 -04:00
|
|
|
+makeExcerpt('pubspec.yaml', 'additions', null, stylePattern)
|
|
|
|
|
|
|
|
block http-providers
|
|
|
|
:marked
|
|
|
|
Before our app can use `#{_Http}`, we have to register it as a service provider.
|
|
|
|
|
2016-08-12 14:21:16 -04:00
|
|
|
We should be able to access `!{_Http}` services from anywhere in the application.
|
|
|
|
So we register it in the `bootstrap` call where we
|
|
|
|
launch the application and its root `AppComponent`.
|
|
|
|
|
|
|
|
+makeExcerpt('app/main.ts','v1')
|
|
|
|
|
|
|
|
:marked
|
|
|
|
Notice that we supply `!{_HttpModule}` in a list, as the second parameter to
|
|
|
|
the `bootstrap` method. This has the same effect as the `providers` list in
|
|
|
|
`@Component` annotation.
|
|
|
|
|
2016-06-28 16:12:49 -04:00
|
|
|
block backend
|
|
|
|
:marked
|
|
|
|
We want to replace `BrowserClient`, the service that talks to the remote server,
|
2016-08-12 14:21:16 -04:00
|
|
|
with the in-memory web API service.
|
2016-06-28 16:12:49 -04:00
|
|
|
Our in-memory web API service, shown below, is implemented using the
|
|
|
|
`http` library `MockClient` class.
|
|
|
|
All `http` client implementations share a common `Client` interface, so
|
|
|
|
we'll have our app use the `Client` type so that we can freely switch between
|
|
|
|
implementations.
|
|
|
|
|
|
|
|
block dont-be-distracted-by-backend-subst
|
2016-08-02 12:59:35 -04:00
|
|
|
//- No backend substitution but we do need to comment on the changes to Hero.
|
|
|
|
:marked
|
|
|
|
As is common for web API services, our mock in-memory service will be
|
|
|
|
encoding and decoding heroes in JSON format, so we enhance the `Hero`
|
|
|
|
class with these capabilities:
|
|
|
|
|
|
|
|
+makeExample('lib/hero.dart')
|
2016-06-28 16:12:49 -04:00
|
|
|
|
|
|
|
block get-heroes-details
|
|
|
|
:marked
|
|
|
|
To get the list of heroes, we first make an asynchronous call to
|
|
|
|
`http.get()`. Then we use the `_extractData` helper method to decode the
|
|
|
|
response payload (`body`).
|
|
|
|
|
|
|
|
block hero-detail-comp-extra-imports-and-vars
|
|
|
|
//- N/A
|
|
|
|
|
|
|
|
block hero-detail-comp-updates
|
|
|
|
:marked
|
|
|
|
### Edit in the *HeroDetailComponent*
|
|
|
|
|
2016-08-12 14:21:16 -04:00
|
|
|
We already have `HeroDetailComponent` for viewing details about a specific hero.
|
2016-06-28 16:12:49 -04:00
|
|
|
Supporting edit functionality is a natural extension of the detail view,
|
|
|
|
so we are able to reuse `HeroDetailComponent` with a few tweaks.
|
|
|
|
|
|
|
|
block hero-detail-comp-save-and-goback
|
|
|
|
//- N/A
|
|
|
|
|
|
|
|
block add-new-hero-via-detail-comp
|
|
|
|
//- N/A
|
|
|
|
|
|
|
|
block heroes-comp-add
|
|
|
|
//- N/A
|
|
|
|
|
|
|
|
block review
|
|
|
|
//- Not showing animated gif due to differences between TS and Dart implementations.
|
|
|
|
|
2016-08-02 12:59:35 -04:00
|
|
|
block observables-section-intro
|
|
|
|
:marked
|
|
|
|
Recall that `HeroService.getHeroes()` awaits for an `http.get()`
|
|
|
|
response and yields a _Future_ `List<Hero>`, which is fine when we are only
|
|
|
|
interested in a single result.
|
|
|
|
|
|
|
|
block search-criteria-intro
|
|
|
|
:marked
|
|
|
|
A [StreamController][], as its name implies, is a controller for a [Stream][] that allows us to
|
|
|
|
manipulate the underlying stream by adding data to it, for example.
|
|
|
|
|
|
|
|
In our sample, the underlying stream of strings (`_searchTerms.stream`) represents the hero
|
|
|
|
name search patterns entered by the user. Each call to `search` puts a new string into
|
|
|
|
the stream by calling `add` over the controller.
|
|
|
|
|
|
|
|
[Stream]: https://api.dartlang.org/stable/dart-async/Stream-class.html
|
|
|
|
[StreamController]: https://api.dartlang.org/stable/dart-async/StreamController-class.html
|
|
|
|
|
|
|
|
block observable-transformers
|
|
|
|
:marked
|
|
|
|
Fortunately, there are stream transformers that will help us reduce the request flow.
|
|
|
|
We'll make fewer calls to the `HeroSearchService` and still get timely results. Here's how:
|
|
|
|
|
|
|
|
* `transform(new Debounce(... 300)))` waits until the flow of search terms pauses for 300
|
|
|
|
milliseconds before passing along the latest string. We'll never make requests more frequently
|
|
|
|
than 300ms.
|
|
|
|
|
|
|
|
* `distinct()` ensures that we only send a request if a search term has changed.
|
|
|
|
There's no point in repeating a request for the same search term.
|
|
|
|
|
|
|
|
* `transform(new FlatMapLatest(...))` applies a map-like transformer that (1) calls our search
|
|
|
|
service for each search term that makes it through the debounce and distinct gauntlet and (2)
|
|
|
|
returns only the most recent search service result, discarding any previous results.
|
|
|
|
|
|
|
|
* `handleError()` handles errors. Our simple example prints the error to the console; a real
|
|
|
|
life application should do better.
|
2016-07-19 18:30:42 -04:00
|
|
|
|
2016-06-28 16:12:49 -04:00
|
|
|
block filetree
|
|
|
|
.filetree
|
2016-07-01 14:05:45 -04:00
|
|
|
.file angular2_tour_of_heroes
|
2016-06-28 16:12:49 -04:00
|
|
|
.children
|
|
|
|
.file lib
|
|
|
|
.children
|
|
|
|
.file app_component.css
|
2016-08-02 12:59:35 -04:00
|
|
|
.file app_component.dart
|
2016-06-28 16:12:49 -04:00
|
|
|
.file dashboard_component.css
|
|
|
|
.file dashboard_component.dart
|
2016-08-02 12:59:35 -04:00
|
|
|
.file dashboard_component.html
|
2016-06-28 16:12:49 -04:00
|
|
|
.file hero.dart
|
|
|
|
.file hero_detail_component.css
|
|
|
|
.file hero_detail_component.dart
|
2016-08-02 12:59:35 -04:00
|
|
|
.file hero_detail_component.html
|
|
|
|
.file hero_search_component.css (new)
|
|
|
|
.file hero_search_component.dart (new)
|
|
|
|
.file hero_search_component.html (new)
|
|
|
|
.file hero_search_service.dart (new)
|
2016-06-28 16:12:49 -04:00
|
|
|
.file hero_service.dart
|
|
|
|
.file heroes_component.css
|
|
|
|
.file heroes_component.dart
|
2016-08-02 12:59:35 -04:00
|
|
|
.file heroes_component.html
|
2016-06-28 16:12:49 -04:00
|
|
|
.file in_memory_data_service.dart (new)
|
|
|
|
.file web
|
|
|
|
.children
|
|
|
|
.file main.dart
|
|
|
|
.file index.html
|
|
|
|
.file styles.css
|
|
|
|
.file pubspec.yaml
|
|
|
|
|
|
|
|
block file-summary
|
|
|
|
+makeTabs(
|
2016-08-02 12:59:35 -04:00
|
|
|
`toh-6/dart/lib/dashboard_component.dart,
|
|
|
|
toh-6/dart/lib/dashboard_component.html,
|
|
|
|
toh-6/dart/lib/hero.dart,
|
2016-06-28 16:12:49 -04:00
|
|
|
toh-6/dart/lib/hero_detail_component.dart,
|
|
|
|
toh-6/dart/lib/hero_detail_component.html,
|
|
|
|
toh-6/dart/lib/hero_service.dart,
|
2016-08-02 12:59:35 -04:00
|
|
|
toh-6/dart/lib/heroes_component.css,
|
|
|
|
toh-6/dart/lib/heroes_component.dart`,
|
|
|
|
null,
|
|
|
|
`lib/dashboard_component.dart,
|
|
|
|
lib/dashboard_component.html,
|
|
|
|
lib/hero.dart,
|
2016-06-28 16:12:49 -04:00
|
|
|
lib/hero_detail_component.dart,
|
|
|
|
lib/hero_detail_component.html,
|
|
|
|
lib/hero_service.dart,
|
2016-08-02 12:59:35 -04:00
|
|
|
lib/heroes_component.css,
|
|
|
|
lib/heroes_component.dart`)
|
2016-06-28 16:12:49 -04:00
|
|
|
|
2016-08-02 12:59:35 -04:00
|
|
|
+makeTabs(
|
|
|
|
`toh-6/dart/lib/hero_search_component.css,
|
|
|
|
toh-6/dart/lib/hero_search_component.dart,
|
|
|
|
toh-6/dart/lib/hero_search_component.html,
|
|
|
|
toh-6/dart/lib/hero_search_service.dart`,
|
|
|
|
null,
|
|
|
|
`lib/hero_search_component.css,
|
|
|
|
lib/hero_search_component.dart,
|
|
|
|
lib/hero_search_component.html,
|
|
|
|
lib/hero_search_service.dart`)
|
2016-08-12 14:21:16 -04:00
|
|
|
|
2016-08-02 12:59:35 -04:00
|
|
|
+makeTabs(
|
|
|
|
`toh-6/dart/pubspec.yaml,
|
|
|
|
toh-6/dart/web/main.dart`,
|
2016-08-12 14:21:16 -04:00
|
|
|
null,
|
2016-08-02 12:59:35 -04:00
|
|
|
`pubspec.yaml,
|
|
|
|
web/main.dart`)
|