在先前的文章中,介紹了從網路上下載資料,並且進行解析,顯示在 App 上,以及如何在 App 中播放 YouTube 影片,在此製作一個小範例 [滾石唱片],抓取 [滾石唱片] YouTube 頻道資料,點選影片時可進行播放。
前言
在先前的文章中,介紹了從網路上下載資料,並且進行解析,顯示在 App 上,以及如何在 App 中播放 YouTube 影片,在此製作一個小範例 [滾石唱片],抓取 [滾石唱片] YouTube 頻道資料,點選影片時可進行播放。
實作
滾石唱片在 YouTube 的頻道網址
http://www.youtube.com/user/RockRecordsTaipei
其 gdata 資料網址
http://gdata.youtube.com/feeds/api/users/RockRecordsTaipei/uploads?orderby=updated
新增專案
首先修改 ItemViewModel.cs
using System;
using System.ComponentModel;
using System.Diagnostics;
using System.Net;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Documents;
using System.Windows.Ink;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Animation;
using System.Windows.Shapes;
namespace VideoApp
{
public class ItemViewModel : INotifyPropertyChanged
{
private string _link;
public string Link
{
get
{
return _link;
}
set
{
if (value != _link)
{
_link = value;
NotifyPropertyChanged("Link");
}
}
}
private string _image;
public string Image
{
get
{
return _image;
}
set
{
if (value != _image)
{
_image = value;
NotifyPropertyChanged("Image");
}
}
}
private string _player;
public string Player
{
get
{
return _player;
}
set
{
if (value != _player)
{
_player = value;
NotifyPropertyChanged("Player");
}
}
}
public event PropertyChangedEventHandler PropertyChanged;
private void NotifyPropertyChanged(String propertyName)
{
PropertyChangedEventHandler handler = PropertyChanged;
if (null != handler)
{
handler(this, new PropertyChangedEventArgs(propertyName));
}
}
}
}
開啟 MainViewModel.cs,撰寫程式碼,將 [滾石唱片] YouTube 頻道的 gdata 資料下載,透過 Linq 分析後,將資料加入 Items 中,由於使用了 Xml.Linq,因此請先加入參考。
完整程式碼 :
using System;
using System.ComponentModel;
using System.Collections.Generic;
using System.Diagnostics;
using System.Text;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Data;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Imaging;
using System.Windows.Shapes;
using System.Collections.ObjectModel;
using System.Net;
using System.IO;
using System.Linq;
using System.Xml.Linq;
namespace VideoApp
{
public class MainViewModel : INotifyPropertyChanged
{
public MainViewModel()
{
Items = new ObservableCollection();
}
///
/// ItemViewModel 物件的集合。
///
public static ObservableCollection Items { get; private set; }
private string _sampleProperty = "Sample Runtime Property Value";
///
/// 範例 ViewModel 屬性; 這個屬性是用在檢視中,以使用繫結來顯示其值
///
///
public string SampleProperty
{
get
{
return _sampleProperty;
}
set
{
if (value != _sampleProperty)
{
_sampleProperty = value;
NotifyPropertyChanged("SampleProperty");
}
}
}
public bool IsDataLoaded
{
get;
private set;
}
///
/// 建立並加入一些 ItemViewModel 物件到 Items 集合。
///
public void LoadData()
{
// 下載 [滾石唱片] YouTube 頻道 gdata 資料
HttpWebRequest request =
(HttpWebRequest)WebRequest.Create("http://gdata.youtube.com/feeds/api/users/RockRecordsTaipei/uploads?orderby=updated");
request.BeginGetResponse(new AsyncCallback(GetHttpDocumentCallback), request);
this.IsDataLoaded = true;
}
private static void GetHttpDocumentCallback(IAsyncResult res)
{
HttpWebRequest request = (HttpWebRequest)res.AsyncState;
HttpWebResponse response = (HttpWebResponse)request.EndGetResponse(res);
using (var reader = new StreamReader(response.GetResponseStream()))
{
var resultStream = reader.ReadToEnd();
XDocument resdoc = XDocument.Parse(resultStream.Trim());
XNamespace ns_media = "http://search.yahoo.com/mrss/";
XNamespace ns_default = "http://www.w3.org/2005/Atom";
var newsitems = from newsitem in resdoc.Descendants(ns_media + "group")
select new ItemViewModel
{
Link = newsitem.Element(ns_media + "title").Value,
Image = (newsitem.Elements(ns_media + "thumbnail").Last()).Attribute("url").Value,
Player = (newsitem.Elements(ns_media + "player").Last()).Attribute("url").Value,
};
Deployment.Current.Dispatcher.BeginInvoke(new Action(() =>
{
foreach (var i in newsitems)
{
Items.Add(i);
}
}));
}
}
public event PropertyChangedEventHandler PropertyChanged;
private void NotifyPropertyChanged(String propertyName)
{
PropertyChangedEventHandler handler = PropertyChanged;
if (null != handler)
{
handler(this, new PropertyChangedEventArgs(propertyName));
}
}
}
}
以設計模式開啟 MainPage.xaml,修改 Panorama 控制項配置,綁定相關資料,並且加入 Image 的 Tap 事件,當使用者點選 Image 時播放該 YouTube 影片。
<phone:PhoneApplicationPage
x:Class="VideoApp.MainPage"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:phone="clr-namespace:Microsoft.Phone.Controls;assembly=Microsoft.Phone"
xmlns:shell="clr-namespace:Microsoft.Phone.Shell;assembly=Microsoft.Phone"
xmlns:controls="clr-namespace:Microsoft.Phone.Controls;assembly=Microsoft.Phone.Controls"
xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
mc:Ignorable="d" d:DesignWidth="480" d:DesignHeight="800"
d:DataContext="{d:DesignData SampleData/MainViewModelSampleData.xaml}"
FontFamily="{StaticResource PhoneFontFamilyNormal}"
FontSize="{StaticResource PhoneFontSizeNormal}"
Foreground="{StaticResource PhoneForegroundBrush}"
SupportedOrientations="Portrait" Orientation="Portrait"
shell:SystemTray.IsVisible="False">
<!--LayoutRoot 是放置所有頁面的根資料格-->
<Grid x:Name="LayoutRoot" Background="Transparent">
<!--Panorama 控制項-->
<controls:Panorama Title="我的應用程式">
<controls:Panorama.Background>
<ImageBrush ImageSource="PanoramaBackground.png"/>
</controls:Panorama.Background>
<!--Panorama 項目 1-->
<controls:PanoramaItem Header="滾石唱片">
<!--文字會換行的雙行清單-->
<ListBox x:Name="lb1" Margin="0,0,-12,0" ItemsSource="{Binding Items}">
<ListBox.ItemTemplate>
<DataTemplate>
<Grid Margin="0,0,0,12">
<Grid.ColumnDefinitions>
<ColumnDefinition Width="auto" />
<ColumnDefinition Width="*" />
</Grid.ColumnDefinitions>
<Image x:Name="imgData" Grid.Column="0" Width="200" Height="150" Source="{Binding Image}"
Margin="6,0,0,0" VerticalAlignment="Top" Tap="imgData_Tap" Tag="{Binding Player}" />
<TextBlock Grid.Column="1" Text="{Binding Link}" TextWrapping="Wrap"
Margin="4,-4,0,0" Style="{StaticResource PhoneTextTitle2Style}"
VerticalAlignment="Top" FontSize="22"/>
</Grid>
</DataTemplate>
</ListBox.ItemTemplate>
</ListBox>
</controls:PanoramaItem>
</controls:Panorama>
</Grid>
</phone:PhoneApplicationPage>
畫面配置如下圖所示
開啟 MainPage.xaml 程式碼,撰寫 Image_Tap 事件程式碼,將選擇的 Image 的 YouTube 網址,透過 MyToolkit 的 YouTube 類別進行播放,請先透過 NuGet 加入 MyToolkit.Extended。
完整程式碼 :
using System;
using System.Collections.Generic;
using System.Linq;
using System.Net;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Media.Animation;
using System.Windows.Shapes;
using Microsoft.Phone.Controls;
using MyToolkit.Multimedia;
namespace VideoApp
{
public partial class MainPage : PhoneApplicationPage
{
// 建構函式
public MainPage()
{
InitializeComponent();
// 將清單方塊控制項的資料內容設為範例資料
DataContext = App.ViewModel;
this.Loaded += new RoutedEventHandler(MainPage_Loaded);
}
// 載入 ViewModel 項目的資料
private void MainPage_Loaded(object sender, RoutedEventArgs e)
{
if (!App.ViewModel.IsDataLoaded)
{
App.ViewModel.LoadData();
}
}
private void imgData_Tap(object sender, System.Windows.Input.GestureEventArgs e)
{
var imgDataTapd = (Image)sender;
string strYoutube = imgDataTapd.Tag.ToString();
if (strYoutube.Length == 0)
{
MessageBox.Show("沒有取得網址,無法播放");
}
else
{
Uri uriYoutube = new Uri(strYoutube);
foreach (string strQuery in uriYoutube.Query.Split(new char[] { '&' }))
{
if (strQuery.IndexOf("v=") >= 0)
{
Deployment.Current.Dispatcher.BeginInvoke(() =>
{
YouTube.Play(strQuery.Replace("?", "").Replace(@"v=", ""), YouTubeQuality.Quality480P, null);
});
}
}
}
}
}
}
執行程式,一開始顯示滾石唱片 YouTube 頻道。
點選 Image,播放該 YouTube 影片。
範例下載