คุณยังรู้สึกว่า ตอนนี้เขียนโปรแกรมทำอะไรที่ต้องแสดงข้อมูลที เป็นเรื่องยุ่งยากน่าเบื่อหรือเปล่า? แล้วทำไม คุณ Pandula ถึงสามารถเขียนโปรแกรม Twitter ได้ง่ายมากๆ บน WP7? ลองมาดูกันว่า WP7/Silverlight มีอะไรที่ช่วยให้ชีวิตเราง่ายขึ้นไม่น่าเบื่อขึ้นบ้างกันดีกว่า
*** ขออภัยท่านผู้อ่านครับ ผมกำลังพยายามหาทางใส่ Source Code ที่อ่านง่ายกว่านี้อยู่ครับ ***
*** Tool สำหรับบทความนี้ โหลดได้จาก ตาม Link ในโพสนี้
ผมว่า ทุกท่านคงจะต้องเคยเขียนโค๊ดกันแบบนี้
this.txtName.Text = MyDataObject.Name;
หรือไม่ก็.
MyDataObject.Name = this.txtName.Text;
หรือไม่ก็
foreach( var item in DataLayer.GetCustomer() )
{
this.ListView1.Items.Add( );
}
หรือแม้แต่การใช้ Object Data Source ก็ตาม
แต่ในการพัฒนาโปรแกรมด้วย Silverlight รวมไปถึง Windows phone 7 นั้น เราจะไม่ได้ใช้วิธีการแบบเดิมๆ ในการติดต่อกับข้อมูลแบบนั้นอีกแล้ว เพราะ WP7/Silverlight นั้น มีความสามารถด้านการใช้งานข้อมูลที่แจ่มกว่านั้นแบบ Built-in ที่เรียกว่า Binding ครับ
Binding?
แปลกันตรงๆ ก็คือการผูกครับ ถ้าลองกลับไปดู Code ใน WP7 Tutorial #1 จะเห็นว่า คุณ pandula แทบไม่ได้เขียนโค๊ดเลย แต่สามารถแสดงผล Tweet ออกมาเป็นรายการสวยงามได้ พร้อมรูปประกอบ (ดังรูป) เขาทำได้อย่างไรกัน?
ลองดู Hilight ของ Code XAML ซึ่งหน้าตาคล้ายกับ จากบทความของคุณ pandula ครับ
สังเกตที่คำว่า Text=Binding UserName และ Text=Binding Message มันเป็นการบอกว่า เราจะผูก Property Text ของ TextBlock (คือ Label แบบหนึ่ง) เข้ากับ UserName และ Message ครับ แล้ว Username กับ Message มาจากไหนละ? ก็คลาสที่คุณ pandula สร้างไว้นั่นเองครับ
แล้วเราเอา TwitterItem หลายๆ อัน ใส่เข้า ListBox ตอนไหน??? ก็คือตอนที่ WebClient ทำการ Download ข้อมูลจาก Twitter เสร็จแล้ว คุณ Pandula ก็ทำการสร้างรายการของ TwitterItem ด้วย LINQ (ที่เห็นเป็น LINQ-to-XML)
NOTE: ถ้าสงสัยว่า อะไรคือ LINQ มันคือภาษาในการเลือกข้อมูลออกมาจาก Object ครับ ลักษณะคล้าย SQL ผมได้เขียนแนะนำไว้ มีด้วยกัน สองตอน ซึ่งคุณ chaowman ก็ได้เขียน Tutorial แปลจากภาษา SQL เป็น LINQ ไว้แล้วด้วย สำหรับผู้ที่คล่อง SQL ตามอ่านได้จากบทความของคุณ chaowman ตอนที่ 1,ตอนที่ 2, ตอนที่ 3 ครับ
เดี๋ยวนะ ทำไมทำโปรแกรม แล้วมี Xml ด้วยละ
นั่นก็เพราะว่า ต่อไปนี้ โปรแกรม จะไม่ได้มีแค่ Code อย่างเดียวแล้วครับ เพราะหลักฐานทางวิทยาศาสตร์ ซึ่งก็ได้แก่โปรแกรมทั้งหลายที่เราเขียนกันอยู่ ได้พิสูจน์แล้วว่า โปรแกรมเมอร์ ไม่อาจสร้างโปรแกรมที่ดูดีได้ ในขณะที่ยังสามารถส่งได้ทันเวลา และมี Feature ครบได้ ด้วย Code เพียงอย่างเดียว มันจะต้องให้มือดีอย่าง Designer เขาเข้ามาช่วย
จะเห็นว่า เมื่อเราสร้าง Project ขึ้นมาใหม่ด้วย Windows phone Tool หน้า Main ของเรา มันจะไม่ได้มีแค่ .cs แล้ว แต่จะมีไฟล์ .xaml คู่กันด้วย โดยใน .xaml ก็จะมีแค่ Markup (xml) ที่กำหนดว่า หน้าตาโปรแกรมจะออกมาเป็นอย่างไร และ .cs ก็จะมีแต่ Code ที่จะกำหนดว่า โปรแกรมจะทำงานอย่างไร เห็นภาพแล้วใช่ไหมครับ? Designer ก็แก้ไฟล์ .xaml ไป เราก็แก้ไฟล์ .cs ของเราไป ไม่ต้องยุ่งเกี่ยวกัน แต่ว่าทำงานร่วมกันได้ คนนึงออกแบบหน้าจอ คนนึงกำหนดว่า หน้าจอนั้น จะทำงานได้อย่างไร
ทดลองเล่นกับ xaml ดูครับ ลองลาก Listbox ออกมาวางไว้ จะเห็นว่า ใน Xaml ก็จะมี Tag Listbox ปรากฏขึ้น
และถ้าเราทำการแก้ไข Property ใดๆ ก็ตาม เช่น การเพิ่ม Items เข้าใน Listbox เราก็จะเห็นว่า มี Tag เกิดขึ้นตามมาด้วย
เตรียมพร้อม Project เราให้รับ DataBinding
การที่จะใช้ DataBinding ได้อย่างสะดวกเต็มที่ และทำให้เกิดการแยกระหว่าง UI และ Code ได้จริงๆ เราจะต้องมีการเตรียมการเล็กน้อยครับ ซึ่งเทคนิคหนึ่งที่ผมชอบใช้ ก็คือสิ่งที่เรียกว่า Data Island เป็น การสร้างคลาสที่เป็นศูนย์รวมของข้อมูลที่เราจะใช้ในโปรแกรมนั่นเองครับ
เช่น ถ้าโปรแกรม Twitter ของเรา ที่ใช้ Username และ List ของ Twitter เราก็อาจสร้างคลาสที่เป็น Data Island ได้ ดังนี้
ProgramData.cs (Data Island)
- using System;
- using System.Net;
- using System.Collections.Generic;
- using System.ComponentModel;
- namespace WindowsPhoneApplication2
- {
- public class ProgramData : INotifyPropertyChanged
- {
- #region (Relaxed) Singleton Pattern
- private static ProgramData _Default;
- ///
- /// Get the default instance of ProgramData
- ///
- public static ProgramData Default
- {
- get
- {
- if ( _Default == null )
- {
- _Default = new ProgramData();
- }
- return _Default;
- }
- }
- #endregion
- private string _Username;
- private List _Tweets;
- ///
- ///Get or set the value of Username
- ///
- public string Username
- {
- get
- {
- return _Username;
- }
- set
- {
- _Username = value;
- this.OnPropertyChanged("Username");
- }
- }
- ///
- ///Get or set the value of Tweets
- ///
- public List Tweets
- {
- get
- {
- return _Tweets;
- }
- set
- {
- _Tweets = value;
- this.OnPropertyChanged("Tweets");
- }
- }
- #region INotifyPropertyChanged Members
- public event PropertyChangedEventHandlerPropertyChanged;
- protected void OnPropertyChanged(string propertyName)
- {
- if (PropertyChanged != null)
- {
- PropertyChanged(this,
- newPropertyChangedEventArgs(propertyName));
- }
- }
- #endregion
- }
- }
การสร้างคลาสที่เป็น INotifyPropertyChanged จะเป็นการสร้างคลาส ที่รองรับการ Binding โดยสมบูรณ์ เพราะว่า คลาสนี้ สามารถส่ง Event ออกมาบอกชาวโลกได้ว่า มีอะไรเปลี่ยนแปลง (Event PropertyChanged) โดยคอนโทรลต่างๆ ที่เป็น UI เมื่อ Bind เข้ากับ Property ใดๆ แล้ว มันก็จะรอฟัง Event นี้ นั่นเอง
ดังนั้น แทนที่เราจะดาวน์โหลด Twitter แล้ว Return ตรงๆ เราก็จะทำแบบนี้แทน
ProgramService.cs
- using System;
- using System.Windows;
- using System.Net;
- using System.Linq;
- using System.Xml.Linq;
- namespace WindowsPhoneApplication2
- {
- public static class ProgramService
- {
- public static void DownloadTweet()
- {
- WebClient client = new WebClient();
- client.DownloadStringCompleted += delegate(object sender, DownloadStringCompletedEventArgs e)
- {
- if ( e.Error != null )
- {
- MessageBox.Show( e.Error.Message );
- return;
- }
- ProgramData.Default.Tweets = (from tweet in XElement.Parse(e.Result).Descendants("status")
- select new TwitterItem
- {
- AvatarUrl = tweet.Element("user").Element("profile_image_url").Value,
- Message = tweet.Element("text").Value,
- UserName = tweet.Element("user").Element("name").Value
- }).ToList(); // Read data now, not later
- };
- client.DownloadStringAsync(new Uri("http://api.twitter.com/1/statuses/user_timeline.xml?screen_name=" + ProgramData.Default.Username));
- }
- }
- }
ข้อสังเกต
- ผมได้ทำการเซ็ท Tweets ลงไปใน ProgramData.Tweets โดยตรง ถึงการเซ็ทนี้ ก็จะทำให้เกิด Event PropertyChanged ปริยาย
- การที่ผมต้อง .ToList และที่ต้องให้ ProgramData.Tweets เป็น List แทนที่จะเป็น IEnumberable เนื่องจาก LINQ เป็น Lazy Evaluation ครับ ผมอยากให้มีข้อมูลพร้อมเลยทันที สำหรับการแสดงผล ไม่ใช่ว่าต้องมาอ่านอีกที
- Code ตั้งแต่บรรทัดที่ 14 ถึง 29 รันคนละ Thread กับ Thread UI ครับ ถ้าเราเขียนแบบปกติ ใช้คอนโทรลตรงๆ เราจะต้องระวัง แต่แบบนี้ ไม่ต้องเลย ตัว Framework มัน Handle เองอยู่แล้ว
เปิด Blend ขึ้นมาซะ!
หนึ่งในอีกโปรแกรมที่เราต้องใช้ก็คือ Blend ครับ เราจะสวมรอยเป็น Designer และใช้ Blend ในการออกหน้าจอเพิ่มเติม และผูกหน้าจอ เข้ากับข้อมูลที่เราเตรียมไว้ให้
ซึ่งการจะเริ่มผูกทุกอย่างได้ เราจะต้องมี DataContext ซึ่งก็คือ แหล่งข้อมูล ที่เราจะผูกด้วย และบังเอิญเราก็มีแหล่งข้อมูลที่พร้อมใช้แล้วพอดี ก็ ProgramData นั่นเอง การใช้ DataContext ทำได้โดยการเลือกที่คอนโทรลตัวบนสุด แล้วกด New ที่ Property DataContext ทางขวามือ
แล้วเลือก ProgramData จากรายการที่มี (ถ้าไม่มี ลอง ไปที่เมนู Project แล้วสั่ง Build 1 ครั้ง)
เจ้า DataContext นี้ มันจะ Propagate ลงไปทุกคอนโทรล ที่อยู่ภายใต้ตัวที่เรา Set DataContext ดังนั้น การ Set ที่ตัวใหญ่สุด ก็คือการ Set DataContext สำหรับทั้งหน้านั่นเอง
จากนั้น ก็ถึงเวลาที่เราจะเริ่ม Bind กันละครับ สำหรับโปรแกรมของเรา ก็ควรจะมี Textbox หนึ่งตัว สำหรับใส่ชื่อ และ Listbox อีกตัว เพื่อแสดงรายการของ Tweet ที่โหลดมาได้ เริ่มจากการ Bind เจ้า Textbox ก่อนเลย กดเครื่องหมายสี่เหลี่ยม ด้านหลัง Property Text ครับ จะมีเมนูขึ้นมา ให้เราเลือก Data Binding
แล้วเลือกว่า จะ Bind จาก Data Context และใช้ Property Username
ก็จะเห็นข้อความใน Textbox เปลี่ยนเป็นค่าที่เราตั้งไว้ให้กับ Property ของ Username ทันที ซึ่งผมแอบไปเพิ่ม Code ให้มันเป็น default ส่วนของ Listbox นั้นทำแบบเดียวกัน แต่เป็นการ Bind Property ที่ชื่อ ItemsSource เข้ากับ Tweets ครับ ถ้าทดลองตั้งข้อมูลใน List ไว้ ข้อมูลจะปรากฏขึ้นใน Listbox ด้วย
Data Template
แต่ว่า การแสดงผล Tweet มันยังไม่สวยเหมือนของคุณ Pandula เลยใช่ไหมละ? ข้อดีอีกอย่างของ WP7/Silverlight ก็คือ เราสามารถกำหนดการแสดงผลของข้อมูลใน Listbox ได้ด้วย เราเรียกมันว่า Data Template ครับ การสร้าง Data Template นั้น มีหลายวิธี แต่หนึ่งในวิธีที่ผมพบว่าสะดวกก็คือ ใช้มันตรงๆ ในหน้านั้นเลย เริ่มจากการคลิ๊กขวา แล้วเข้าเมนู
แล้วก็ตั้งชื่อให้มัน
จากนั้น Blend ก็จะเข้าสู่โหมดการแก้ไข Template ซึ่งเราสามารถนำคอนโทรล วางลงไปใน Template ได้ตามปกติ แล้ว Blend ก็จะ Preview ให้ดูด้วยว่า เมื่อวางลงไปแล้ว หน้าตามันจะออกมาเป็นอย่างไร แน่นอนว่า คอนโทรลที่เราวางลงไปใน Template ก็ยังสามารถ Binding ได้ครับ
โดยตอนที่ Bind จะเห็นว่า จะเป็นการ Bind กับ TweetItem ซึ่งก็คือ Item แต่ละชิ้นภายที่ถูกส่งเข้ามายัง Listbox นั่นเอง
หลังจากแก้ไขจนหนำใจแล้ว ก็กด Breadcrumb ด้านบน เพื่อกลับสู่หน้าจอปกติ
Finishing Touch
คราวนี้ เราก็พร้อมที่จะให้โปรแกรมทำงานแล้ว เราอาจจะสร้างปุ่มขึ้นมา แล้วผูก Event Click เพื่อให้โปรแกรมโหลด Tweet ใหม่ ก็ได้
หลังจากเราตั้งชื่อ Event แล้วกด Enter Blend ก็จะเปิดหน้า Source มาให้แก้ไขได้ทันที เราจะพิมพ์จากใน Blend เลยก็ได้ครับ ไม่ว่ากัน แต่ถ้าลองรันดู จะเห็นว่า กดปุ่มไป ก็ไม่มีอะไรเกิดขึ้น แม้ว่าจะเขียนถูกแล้วก็ตาม ทำไมละ???
ข้อควรระวัง
อย่าลืมว่า ProgramService จะทำการ Assign ค่า ใส่ ProgramData.Default.Tweet ซึ่งเป็นคนละ Instance กันกับตัวที่ Blend สร้างให้ เราจะต้องทำการตั้งค่า DataContext ของหน้าจอนี้ เป็น ProgramData.Default ก่อนครับ ซึ่ง Code ที่สมบูรณ์ จะมีดังนี้
MainPage.cs
- 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;
- namespace WindowsPhoneApplication2
- {
- public partial class MainPage : PhoneApplicationPage
- {
- public MainPage()
- {
- InitializeComponent();
- SupportedOrientations = SupportedPageOrientation.Portrait | SupportedPageOrientation.Landscape;
- this.DataContext = ProgramData.Default;
- }
- private void LoadButton_Click(object sender, System.Windows.RoutedEventArgs e)
- {
- // TODO: Add event handler implementation here.
- ProgramService.DownloadTweet();
- }
- }
- }
ยังไม่จบ
ตอนหน้า เราจะมาทำให้โปรแกรมนี้สมบูรณ์ขึ้น ด้วย Visual State Manager ครับ
WP7 Tutorial : #2 Silverlight, XAML, DataContext!?!?
คุณยังรู้สึกว่า ตอนนี้เขียนโปรแกรมทำอะไรที่ต้องแสดงข้อมูลที เป็นเรื่องยุ่งยากน่าเบื่อหรือเปล่า? แล้วทำไม คุณ Pandula ถึงสามารถเขียนโปรแกรม Twitter ได้ง่ายมากๆ บน WP7? ลองมาดูกันว่า WP7/Silverlight มีอะไรที่ช่วยให้ชีวิตเราง่ายขึ้นไม่น่าเบื่อขึ้นบ้างกันดีกว่า
*** ขออภัยท่านผู้อ่านครับ ผมกำลังพยายามหาทางใส่ Source Code ที่อ่านง่ายกว่านี้อยู่ครับ ***
*** Tool สำหรับบทความนี้ โหลดได้จาก ตาม Link ในโพสนี้
ผมว่า ทุกท่านคงจะต้องเคยเขียนโค๊ดกันแบบนี้
หรือไม่ก็.
หรือไม่ก็
หรือแม้แต่การใช้ Object Data Source ก็ตาม
แต่ในการพัฒนาโปรแกรมด้วย Silverlight รวมไปถึง Windows phone 7 นั้น เราจะไม่ได้ใช้วิธีการแบบเดิมๆ ในการติดต่อกับข้อมูลแบบนั้นอีกแล้ว เพราะ WP7/Silverlight นั้น มีความสามารถด้านการใช้งานข้อมูลที่แจ่มกว่านั้นแบบ Built-in ที่เรียกว่า Binding ครับ
Binding?
แปลกันตรงๆ ก็คือการผูกครับ ถ้าลองกลับไปดู Code ใน WP7 Tutorial #1 จะเห็นว่า คุณ pandula แทบไม่ได้เขียนโค๊ดเลย แต่สามารถแสดงผล Tweet ออกมาเป็นรายการสวยงามได้ พร้อมรูปประกอบ (ดังรูป) เขาทำได้อย่างไรกัน?
ลองดู Hilight ของ Code XAML ซึ่งหน้าตาคล้ายกับ จากบทความของคุณ pandula ครับ
สังเกตที่คำว่า Text=Binding UserName และ Text=Binding Message มันเป็นการบอกว่า เราจะผูก Property Text ของ TextBlock (คือ Label แบบหนึ่ง) เข้ากับ UserName และ Message ครับ แล้ว Username กับ Message มาจากไหนละ? ก็คลาสที่คุณ pandula สร้างไว้นั่นเองครับ
แล้วเราเอา TwitterItem หลายๆ อัน ใส่เข้า ListBox ตอนไหน??? ก็คือตอนที่ WebClient ทำการ Download ข้อมูลจาก Twitter เสร็จแล้ว คุณ Pandula ก็ทำการสร้างรายการของ TwitterItem ด้วย LINQ (ที่เห็นเป็น LINQ-to-XML)
NOTE: ถ้าสงสัยว่า อะไรคือ LINQ มันคือภาษาในการเลือกข้อมูลออกมาจาก Object ครับ ลักษณะคล้าย SQL ผมได้เขียนแนะนำไว้ มีด้วยกัน สองตอน ซึ่งคุณ chaowman ก็ได้เขียน Tutorial แปลจากภาษา SQL เป็น LINQ ไว้แล้วด้วย สำหรับผู้ที่คล่อง SQL ตามอ่านได้จากบทความของคุณ chaowman ตอนที่ 1,ตอนที่ 2, ตอนที่ 3 ครับ
เดี๋ยวนะ ทำไมทำโปรแกรม แล้วมี Xml ด้วยละ
นั่นก็เพราะว่า ต่อไปนี้ โปรแกรม จะไม่ได้มีแค่ Code อย่างเดียวแล้วครับ เพราะหลักฐานทางวิทยาศาสตร์ ซึ่งก็ได้แก่โปรแกรมทั้งหลายที่เราเขียนกันอยู่ ได้พิสูจน์แล้วว่า โปรแกรมเมอร์ ไม่อาจสร้างโปรแกรมที่ดูดีได้ ในขณะที่ยังสามารถส่งได้ทันเวลา และมี Feature ครบได้ ด้วย Code เพียงอย่างเดียว มันจะต้องให้มือดีอย่าง Designer เขาเข้ามาช่วย
จะเห็นว่า เมื่อเราสร้าง Project ขึ้นมาใหม่ด้วย Windows phone Tool หน้า Main ของเรา มันจะไม่ได้มีแค่ .cs แล้ว แต่จะมีไฟล์ .xaml คู่กันด้วย โดยใน .xaml ก็จะมีแค่ Markup (xml) ที่กำหนดว่า หน้าตาโปรแกรมจะออกมาเป็นอย่างไร และ .cs ก็จะมีแต่ Code ที่จะกำหนดว่า โปรแกรมจะทำงานอย่างไร เห็นภาพแล้วใช่ไหมครับ? Designer ก็แก้ไฟล์ .xaml ไป เราก็แก้ไฟล์ .cs ของเราไป ไม่ต้องยุ่งเกี่ยวกัน แต่ว่าทำงานร่วมกันได้ คนนึงออกแบบหน้าจอ คนนึงกำหนดว่า หน้าจอนั้น จะทำงานได้อย่างไร
ทดลองเล่นกับ xaml ดูครับ ลองลาก Listbox ออกมาวางไว้ จะเห็นว่า ใน Xaml ก็จะมี Tag Listbox ปรากฏขึ้น
และถ้าเราทำการแก้ไข Property ใดๆ ก็ตาม เช่น การเพิ่ม Items เข้าใน Listbox เราก็จะเห็นว่า มี Tag เกิดขึ้นตามมาด้วย
เตรียมพร้อม Project เราให้รับ DataBinding
การที่จะใช้ DataBinding ได้อย่างสะดวกเต็มที่ และทำให้เกิดการแยกระหว่าง UI และ Code ได้จริงๆ เราจะต้องมีการเตรียมการเล็กน้อยครับ ซึ่งเทคนิคหนึ่งที่ผมชอบใช้ ก็คือสิ่งที่เรียกว่า Data Island เป็น การสร้างคลาสที่เป็นศูนย์รวมของข้อมูลที่เราจะใช้ในโปรแกรมนั่นเองครับ
เช่น ถ้าโปรแกรม Twitter ของเรา ที่ใช้ Username และ List ของ Twitter เราก็อาจสร้างคลาสที่เป็น Data Island ได้ ดังนี้
ProgramData.cs (Data Island)
การสร้างคลาสที่เป็น INotifyPropertyChanged จะเป็นการสร้างคลาส ที่รองรับการ Binding โดยสมบูรณ์ เพราะว่า คลาสนี้ สามารถส่ง Event ออกมาบอกชาวโลกได้ว่า มีอะไรเปลี่ยนแปลง (Event PropertyChanged) โดยคอนโทรลต่างๆ ที่เป็น UI เมื่อ Bind เข้ากับ Property ใดๆ แล้ว มันก็จะรอฟัง Event นี้ นั่นเอง
ดังนั้น แทนที่เราจะดาวน์โหลด Twitter แล้ว Return ตรงๆ เราก็จะทำแบบนี้แทน
ProgramService.cs
ข้อสังเกต
เปิด Blend ขึ้นมาซะ!
หนึ่งในอีกโปรแกรมที่เราต้องใช้ก็คือ Blend ครับ เราจะสวมรอยเป็น Designer และใช้ Blend ในการออกหน้าจอเพิ่มเติม และผูกหน้าจอ เข้ากับข้อมูลที่เราเตรียมไว้ให้
ซึ่งการจะเริ่มผูกทุกอย่างได้ เราจะต้องมี DataContext ซึ่งก็คือ แหล่งข้อมูล ที่เราจะผูกด้วย และบังเอิญเราก็มีแหล่งข้อมูลที่พร้อมใช้แล้วพอดี ก็ ProgramData นั่นเอง การใช้ DataContext ทำได้โดยการเลือกที่คอนโทรลตัวบนสุด แล้วกด New ที่ Property DataContext ทางขวามือ
แล้วเลือก ProgramData จากรายการที่มี (ถ้าไม่มี ลอง ไปที่เมนู Project แล้วสั่ง Build 1 ครั้ง)
เจ้า DataContext นี้ มันจะ Propagate ลงไปทุกคอนโทรล ที่อยู่ภายใต้ตัวที่เรา Set DataContext ดังนั้น การ Set ที่ตัวใหญ่สุด ก็คือการ Set DataContext สำหรับทั้งหน้านั่นเอง
จากนั้น ก็ถึงเวลาที่เราจะเริ่ม Bind กันละครับ สำหรับโปรแกรมของเรา ก็ควรจะมี Textbox หนึ่งตัว สำหรับใส่ชื่อ และ Listbox อีกตัว เพื่อแสดงรายการของ Tweet ที่โหลดมาได้ เริ่มจากการ Bind เจ้า Textbox ก่อนเลย กดเครื่องหมายสี่เหลี่ยม ด้านหลัง Property Text ครับ จะมีเมนูขึ้นมา ให้เราเลือก Data Binding
แล้วเลือกว่า จะ Bind จาก Data Context และใช้ Property Username
ก็จะเห็นข้อความใน Textbox เปลี่ยนเป็นค่าที่เราตั้งไว้ให้กับ Property ของ Username ทันที ซึ่งผมแอบไปเพิ่ม Code ให้มันเป็น default ส่วนของ Listbox นั้นทำแบบเดียวกัน แต่เป็นการ Bind Property ที่ชื่อ ItemsSource เข้ากับ Tweets ครับ ถ้าทดลองตั้งข้อมูลใน List ไว้ ข้อมูลจะปรากฏขึ้นใน Listbox ด้วย
Data Template
แต่ว่า การแสดงผล Tweet มันยังไม่สวยเหมือนของคุณ Pandula เลยใช่ไหมละ? ข้อดีอีกอย่างของ WP7/Silverlight ก็คือ เราสามารถกำหนดการแสดงผลของข้อมูลใน Listbox ได้ด้วย เราเรียกมันว่า Data Template ครับ การสร้าง Data Template นั้น มีหลายวิธี แต่หนึ่งในวิธีที่ผมพบว่าสะดวกก็คือ ใช้มันตรงๆ ในหน้านั้นเลย เริ่มจากการคลิ๊กขวา แล้วเข้าเมนู
แล้วก็ตั้งชื่อให้มัน
จากนั้น Blend ก็จะเข้าสู่โหมดการแก้ไข Template ซึ่งเราสามารถนำคอนโทรล วางลงไปใน Template ได้ตามปกติ แล้ว Blend ก็จะ Preview ให้ดูด้วยว่า เมื่อวางลงไปแล้ว หน้าตามันจะออกมาเป็นอย่างไร แน่นอนว่า คอนโทรลที่เราวางลงไปใน Template ก็ยังสามารถ Binding ได้ครับ
โดยตอนที่ Bind จะเห็นว่า จะเป็นการ Bind กับ TweetItem ซึ่งก็คือ Item แต่ละชิ้นภายที่ถูกส่งเข้ามายัง Listbox นั่นเอง
หลังจากแก้ไขจนหนำใจแล้ว ก็กด Breadcrumb ด้านบน เพื่อกลับสู่หน้าจอปกติ
Finishing Touch
คราวนี้ เราก็พร้อมที่จะให้โปรแกรมทำงานแล้ว เราอาจจะสร้างปุ่มขึ้นมา แล้วผูก Event Click เพื่อให้โปรแกรมโหลด Tweet ใหม่ ก็ได้
หลังจากเราตั้งชื่อ Event แล้วกด Enter Blend ก็จะเปิดหน้า Source มาให้แก้ไขได้ทันที เราจะพิมพ์จากใน Blend เลยก็ได้ครับ ไม่ว่ากัน แต่ถ้าลองรันดู จะเห็นว่า กดปุ่มไป ก็ไม่มีอะไรเกิดขึ้น แม้ว่าจะเขียนถูกแล้วก็ตาม ทำไมละ???
ข้อควรระวัง
อย่าลืมว่า ProgramService จะทำการ Assign ค่า ใส่ ProgramData.Default.Tweet ซึ่งเป็นคนละ Instance กันกับตัวที่ Blend สร้างให้ เราจะต้องทำการตั้งค่า DataContext ของหน้าจอนี้ เป็น ProgramData.Default ก่อนครับ ซึ่ง Code ที่สมบูรณ์ จะมีดังนี้
MainPage.cs
ยังไม่จบ
ตอนหน้า เราจะมาทำให้โปรแกรมนี้สมบูรณ์ขึ้น ด้วย Visual State Manager ครับ