Kindle Launcher を作るときに、Amazon から商品検索をしたかったので、これに嵌りました。サンプルコード自体は Amazon で提供されているものの、WinRT(Windows ストアアプリ)では動かなかったんですよね。
■アフリエイトキーを取得する
http://affiliate.amazon.co.jp/
上記からアフリエイトキーを作成します。AWS とは違うので注意が必要(暫く使っていなかったので探した探した)。まあ、商品検索 API を使いたいのにクレジットカード登録とかは変な話なので。
API キーの取得等は以下を参照してください。アプリでは「アクセスキー」と「シークレットキー」の2つを使います。
Access Key IDとSecret Access Keyの取得 – Amazon Web サービス
http://www.ajaxtower.jp/ecs/pre/index1.html
■SignedRequestHelper.cs を修正する
C# では、アフリエイト API を使うときにヘルパークラスを使うと便利です。WPF や Windows Phone の場合は、提供されている SignedRequestHelper のままでいいのですが、WinRT の場合はちょっと違います。
[C#] Amazon Product Advertising API の利用(REST サンプルコードの修正) | プログラミング生放送
http://pronama.azurewebsites.net/2014/05/23/csharp-amazon-product-advertising-api-rest-sample-code-modified/
↓こんな風に修正してください。
SignedRequestHelper.cs for WinRT
https://gist.github.com/moonmile/16c950174d0c276cc2c3
System.Security.Cryptography じゃなくて、Windows.Security.Cryptography.Core を使うってのがミソですね。下に元ネタがあります。
c# – Working with hmacsha256 in windows store app – Stack Overflow
http://stackoverflow.com/questions/13293420/working-with-hmacsha256-in-windows-store-app
SHA で MAC キーを使うところが少し違うだけで、ヘルパークラスの使い方は変わりません。
おそらく、WinRT での違いがあって、ストア版の Amazon アプリが増えないのかもしれません。セキュリティ回りの名前空間とクラスが再整理(改悪?)されているので、いままでのコードが使えないのが面倒です。互換クラスを作ってしまえばよいような気もするのですが。
■指定したキーワードで商品情報を検索する
SignedRequestHelper を使って Amazon の商品を検索しているところがこんな感じです。
private const string MY_AWS_ACCESS_KEY_ID = "YOUR_AWS_ACCESS_KEY_ID";
private const string MY_AWS_SECRET_KEY = "YOUR_AWS_SECRET_KEY";
private const string DESTINATION = "ecs.amazonaws.jp";
/// <summary>
/// 検索ボタンをクリック
/// </summary>
/// <param name="sender"></param>
/// <param name="e"></param>
private async void OnButtonItemSearch(object sender, RoutedEventArgs e)
{
var helper = new SignedRequestHelper(MY_AWS_ACCESS_KEY_ID, MY_AWS_SECRET_KEY, DESTINATION, "mycompany-22");
var param = new Dictionary<string, String>();
param["Service"] = "AWSECommerceService";
param["Version"] = "2011-08-01";
param["Operation"] = "ItemSearch";
param["Keywords"] = this.vm.Title + " Kindle";
param["Author"] = this.vm.Author;
param["SearchIndex"] = "All"; // "Books";
param["ResponseGroup"] = "Large";
try
{
var requestUrl = helper.Sign(param);
var request = new System.Net.Http.HttpClient();
var res = await request.GetStringAsync(new Uri(requestUrl));
// Debug.WriteLine(res);
var doc = ExDocument.LoadXml(res);
int TotalPages = doc * "TotalPages";
int TotalResults = doc * "TotalResults";
vm.Total = TotalResults;
int count = 0;
vm.Books.Clear();
for (int i = 1; i <= TotalPages; i++)
{
param["ItemPage"] = i.ToString();
await Task.Delay(1000); // ちょっとだけ待つ
var reqUrl = helper.Sign(param);
res = await request.GetStringAsync(new Uri(reqUrl));
doc = ExDocument.LoadXml(res);
var ASINs = doc * "Item" * "ASIN";
foreach (var it in ASINs)
{
count++;
var asin = it.Value;
var item = it.Parent;
var format = item * "ItemAttributes" * "Format";
if (format == "Kindle本")
{
string title = item * "ItemAttributes" * "Title";
string iconurl = item * "LargeImage" * "URL";
Debug.WriteLine("{0} ASIN:{1} Title:{2} img:{3} ", count, asin, title, iconurl);
vm.Books.Add(new Book { ASIN = asin, Title = title, IconUrl = iconurl });
vm.TotalKindle = vm.Books.Count();
vm.TotalStr = string.Format("{0}/{1}", vm.TotalKindle, vm.Total);
}
}
}
}
catch (Exception ex)
{
// MessageBox.Show("API制限に達しました。暫く経ってから実行してくださいn" + ex.Message);
Debug.WriteLine(ex.Message);
}
}
ItemSearch で戻される検索はページ単位で返ってくるのと、検索数が制限されています。サイトでは結構な数で検索できるのですが、API の場合は、4000件(400ページ)程度のようです。
この検索では、商品画像(本の表紙)を取るために Large にして商品データを取っています。まあ、Kindle Launcher の書籍検索がちょっと遅いのはそれが原因でもあります。ResponseGroup を変更するともうちょっとデータ量を絞れるでしょう。
SearchIndex を Books だけにすると「和書」だけになるので、All にしています。洋書も検索結果に入れたかったのです。Keywords に Kindle を含めているのは「Kindle本」に絞るためです。こうすると、おそらく Kindle 対象の本だけが検索できるはずです。そして、もういちど取得したデータを改めて」「Kindle本」で検索しています。
データは XML 形式で取得できるので適当にパースします。ここでは自前の ExDoc を使ってパースをしています。
https://github.com/moonmile/ExDoc/
にあるのは、Windows デスクトップ版なので、近いうちに PCL 版をアップしておきます。
XML データからクラスを作成する VS のアドイン(だっけ?)もあるので、それを使うのもよいでしょう。
■検索した商品を表示する
表示自体は GridView でやっています。
そのままのテンプレートでは使いづらいので「GridView.ItemTemplate」で自前テンプレートを作ります。
<!-- 水平スクロール グリッド -->
<GridView
Grid.Column="1" Grid.Row="1"
x:Name="itemGridView"
ItemsSource="{Binding Books}"
TabIndex="1"
Padding="10"
SelectionMode="{Binding SelectMode}"
IsSwipeEnabled="false"
Tapped="itemGridView_Tapped">
<GridView.ItemTemplate>
<DataTemplate>
<Grid HorizontalAlignment="Left" Width="250" Height="250">
<Border Background="{StaticResource ListViewItemPlaceholderBackgroundThemeBrush}">
<Image Source="{Binding IconUrl}" Stretch="UniformToFill" />
</Border>
<StackPanel VerticalAlignment="Bottom" Orientation="Vertical"
Background="{StaticResource ListViewItemOverlayBackgroundThemeBrush}">
<TextBlock Text="{Binding ASIN}"
Height="18"
Foreground="{StaticResource ListViewItemOverlaySecondaryForegroundThemeBrush}"
TextWrapping="NoWrap" Margin="3"/>
<TextBlock Text="{Binding Title}"
Foreground="{StaticResource ListViewItemOverlayForegroundThemeBrush}"
Height="30" Margin="3"
TextWrapping="NoWrap"/>
</StackPanel>
</Grid>
</DataTemplate>
</GridView.ItemTemplate>
</GridView>
IconUrl とか ASIN とかをバインドしておけば、自動的に表示されますね。
こんな風に表示されます。

