1. บทนำ
วิดเจ็ตคืออะไร
สำหรับนักพัฒนาซอฟต์แวร์ Flutter คำจำกัดความทั่วไปของวิดเจ็ตจะหมายถึงคอมโพเนนต์ UI ที่สร้างขึ้นโดยใช้เฟรมเวิร์ก Flutter ในบริบทของ Codelab นี้ วิดเจ็ตหมายถึงแอปเวอร์ชันขนาดเล็กที่ให้มุมมองข้อมูลของแอปโดยไม่ต้องเปิดแอป สำหรับ Android วิดเจ็ตจะแสดงบนหน้าจอหลัก บน iOS คุณสามารถเพิ่มการ์ดลงในหน้าจอหลัก หน้าจอล็อก หรือมุมมองวันนี้
 
วิดเจ็ตจะซับซ้อนเพียงใด
วิดเจ็ตหน้าจอหลักส่วนใหญ่ใช้งานง่าย ซึ่งอาจมีข้อความพื้นฐาน กราฟิกง่ายๆ หรือการควบคุมขั้นพื้นฐานใน Android ทั้ง Android และ iOS จำกัดองค์ประกอบและฟีเจอร์ของ UI ที่ใช้ได้
 
สร้าง UI สำหรับวิดเจ็ต
เนื่องจากข้อจำกัดของ UI เหล่านี้ คุณจึงวาด UI ของวิดเจ็ตหน้าจอหลักโดยตรงโดยใช้เฟรมเวิร์ก Flutter ไม่ได้ คุณเพิ่มวิดเจ็ตที่สร้างด้วยเฟรมเวิร์กของแพลตฟอร์ม เช่น Jetpack Compose หรือ SwiftUI ไปยังแอป Flutter แทนได้ Codelab นี้จะกล่าวถึงตัวอย่างสำหรับการแชร์ทรัพยากรระหว่างแอปและวิดเจ็ตเพื่อหลีกเลี่ยงการเขียน UI ใหม่ที่ซับซ้อน
สิ่งที่คุณจะสร้าง
ใน Codelab นี้ คุณจะได้สร้างวิดเจ็ตหน้าจอหลักทั้งใน Android และ iOS สำหรับแอป Flutter แบบง่ายๆ โดยใช้แพ็กเกจ home_widget ที่อนุญาตให้ผู้ใช้อ่านบทความ วิดเจ็ตของคุณจะ:
- แสดงข้อมูลจากแอป Flutter
 - แสดงข้อความโดยใช้เนื้อหาแบบอักษรที่แชร์จากแอป Flutter
 - แสดงรูปภาพของวิดเจ็ต Flutter ที่แสดงผล
 

แอป Flutter นี้มี 2 หน้าจอ (หรือเส้นทาง) ดังนี้
- แบบแรกจะแสดงรายการบทความข่าวพร้อมพาดหัวและคำอธิบาย
 - รายการที่ 2 แสดงบทความฉบับเต็มพร้อมแผนภูมิที่สร้างด้วย 
CustomPaint 
.
 
สิ่งที่คุณจะได้เรียนรู้
- วิธีสร้างวิดเจ็ตบนหน้าจอหลักใน iOS และ Android
 - วิธีใช้แพ็กเกจ home_Widget เพื่อแชร์ข้อมูลระหว่างวิดเจ็ต Home Screen กับแอป Flutter
 - วิธีลดจำนวนโค้ดที่ต้องเขียนใหม่
 - วิธีอัปเดตวิดเจ็ตหน้าจอหลักจากแอป Flutter
 
2. ตั้งค่าสภาพแวดล้อมในการพัฒนาซอฟต์แวร์
คุณต้องมี Flutter SDK และ IDE สำหรับทั้ง 2 แพลตฟอร์ม คุณใช้ IDE ที่ต้องการสำหรับการทำงานร่วมกับ Flutter ได้ ซึ่งอาจเป็นโค้ด Visual Studio ที่มีส่วนขยาย Dart Code และ Flutter หรือ Android Studio หรือ IntelliJ ที่มีปลั๊กอิน Flutter และ Dart ติดตั้งอยู่
วิธีสร้างวิดเจ็ตหน้าจอหลักของ iOS
- คุณเรียกใช้ Codelab นี้ในอุปกรณ์ iOS จริงหรือเครื่องจำลอง iOS ได้
 - คุณต้องกำหนดค่าระบบ macOS ด้วย Xcode IDE การดำเนินการนี้จะติดตั้งคอมไพเลอร์ที่ต้องใช้ในการสร้างแอปเวอร์ชัน iOS
 
วิธีสร้างวิดเจ็ตหน้าจอหลักของ Android
- คุณสามารถเรียกใช้ Codelab นี้ในอุปกรณ์ Android จริงหรือโปรแกรมจำลอง Android ได้
 - คุณต้องกำหนดค่าระบบการพัฒนาด้วย Android Studio การดำเนินการนี้จะติดตั้งคอมไพเลอร์ที่ต้องใช้ในการสร้างแอปเวอร์ชัน Android
 
รับรหัสเริ่มต้น
ดาวน์โหลดเวอร์ชันเริ่มต้นของโปรเจ็กต์จาก GitHub
โคลนที่เก็บ GitHub จากบรรทัดคำสั่งลงในไดเรกทอรี flutter-codelabs
$ git clone https://github.com/flutter/codelabs.git flutter-codelabs
หลังจากโคลนที่เก็บแล้ว คุณสามารถดูโค้ดสำหรับ Codelab นี้ได้ในไดเรกทอรี flutter-codelabs/homescreen_codelab ไดเรกทอรีนี้มีรหัสโครงการที่เสร็จสมบูรณ์แล้วสำหรับแต่ละขั้นตอนใน Codelab
เปิดแอปเริ่มต้น
เปิดไดเรกทอรี flutter-codelabs/homescreen_codelab/step_03 ใน IDE ที่ต้องการ
ติดตั้งแพ็กเกจ
เพิ่มแพ็กเกจที่จำเป็นทั้งหมดลงในไฟล์ pubspec.yaml ของโปรเจ็กต์แล้ว หากต้องการเรียกข้อมูลทรัพยากร Dependency ของโปรเจ็กต์ ให้เรียกใช้คำสั่งต่อไปนี้
$ flutter pub get
3. เพิ่มวิดเจ็ตพื้นฐานในหน้าจอหลัก
ก่อนอื่น ให้เพิ่มวิดเจ็ต "หน้าจอหลัก" โดยใช้เครื่องมือของแพลตฟอร์มดั้งเดิม
การสร้างวิดเจ็ตพื้นฐานในหน้าจอหลักของ iOS
การเพิ่มส่วนขยายแอปลงในแอป Flutter iOS จะคล้ายกับการเพิ่มส่วนขยายแอปลงในแอป SwiftUI หรือ UIKit ดังนี้
- เรียกใช้ 
open ios/Runner.xcworkspaceในหน้าต่างเทอร์มินัลจากไดเรกทอรีโปรเจ็กต์ Flutter หรือคลิกขวาที่โฟลเดอร์ ios จาก VSCode แล้วเลือก เปิดใน Xcode การดำเนินการนี้จะเปิดพื้นที่ทำงาน Xcode เริ่มต้นในโปรเจ็กต์ Flutter ของคุณ - เลือกไฟล์ → ใหม่ → เป้าหมายจากเมนู ซึ่งจะเป็นการเพิ่มเป้าหมายใหม่ในโปรเจ็กต์
 - รายการเทมเพลตจะปรากฏขึ้น เลือกส่วนขยายวิดเจ็ต
 - พิมพ์ "NewsWidgets" ลงในช่อง Product Name สำหรับวิดเจ็ตนี้ ล้างทั้งช่องทำเครื่องหมายรวมการถ่ายทอดสดและรวมความตั้งใจในการกำหนดค่า
 
ตรวจสอบโค้ดตัวอย่าง
เมื่อคุณเพิ่มเป้าหมายใหม่ Xcode จะสร้างโค้ดตัวอย่างตามเทมเพลตที่คุณเลือก สำหรับข้อมูลเพิ่มเติมเกี่ยวกับโค้ดที่สร้างขึ้นและ WidgetKit โปรดดูเอกสารส่วนขยายแอปของ Apple
แก้ไขข้อบกพร่องและทดสอบวิดเจ็ตตัวอย่าง
- ก่อนอื่นให้อัปเดตการกำหนดค่าของแอป Flutter คุณต้องทำเช่นนี้เมื่อเพิ่มแพ็กเกจใหม่ในแอป Flutter และวางแผนที่จะเรียกใช้เป้าหมายในโปรเจ็กต์จาก Xcode หากต้องการอัปเดตการกำหนดค่าของแอป ให้เรียกใช้คำสั่งต่อไปนี้ในไดเรกทอรีแอป Flutter
 
$ flutter build ios --config-only
- คลิก Runner เพื่อแสดงรายการเป้าหมาย เลือกเป้าหมายวิดเจ็ตที่คุณเพิ่งสร้าง ไปที่ NewsWidgets แล้วคลิกเรียกใช้ เรียกใช้เป้าหมายวิดเจ็ตจาก Xcode เมื่อคุณเปลี่ยนโค้ดวิดเจ็ตของ iOS
 

- เครื่องจำลองหรือหน้าจออุปกรณ์ควรแสดงวิดเจ็ตพื้นฐานในหน้าจอหลัก หากไม่เห็นไอคอนนี้ก็เพิ่มลงในหน้าจอได้ คลิกหน้าจอหลักค้างไว้ แล้วคลิก + ที่มุมซ้ายบน
 

- ค้นหาชื่อแอป สำหรับ Codelab นี้ ให้ค้นหา "วิดเจ็ตหน้าจอหลัก"
 

- เมื่อคุณเพิ่มวิดเจ็ต "หน้าจอหลัก" แล้ว วิดเจ็ตจะแสดงข้อความง่ายๆ ที่ระบุเวลา
 
การสร้างวิดเจ็ต Android พื้นฐาน
- หากต้องการเพิ่มวิดเจ็ตหน้าจอหลักใน Android ให้เปิดไฟล์บิลด์ของโปรเจ็กต์ใน Android Studio หาไฟล์นี้ได้ที่ android/build.gradle หรือคลิกขวาที่โฟลเดอร์ android จาก VSCode แล้วเลือกเปิดใน Android Studio
 - หลังจากสร้างโปรเจ็กต์แล้ว ให้ค้นหาไดเรกทอรีแอปที่มุมบนซ้าย เพิ่มวิดเจ็ตหน้าจอหลักใหม่ลงในไดเรกทอรีนี้ คลิกขวาที่ไดเรกทอรี เลือก New -> (ใหม่) วิดเจ็ต -> วิดเจ็ตแอป
 

- Android Studio แสดงแบบฟอร์มใหม่ เพิ่มข้อมูลพื้นฐานเกี่ยวกับวิดเจ็ตหน้าจอหลัก ซึ่งรวมถึงชื่อคลาส ตำแหน่ง ขนาด และภาษาต้นฉบับ
 
สำหรับ Codelab นี้ ให้ตั้งค่าต่อไปนี้
- ช่องชื่อชั้นเรียนไปยัง NewsWidget
 - เมนูแบบเลื่อนลงความกว้างขั้นต่ำ (เซลล์) เป็น 3
 - เมนูแบบเลื่อนลงความสูงขั้นต่ำ (เซลล์) เป็น 3
 
ตรวจสอบโค้ดตัวอย่าง
เมื่อคุณส่งแบบฟอร์ม Android Studio จะสร้างและอัปเดตไฟล์หลายไฟล์ การเปลี่ยนแปลงที่เกี่ยวข้องกับ Codelab จะแสดงอยู่ในตารางด้านล่าง
การดำเนินการ  | ไฟล์เป้าหมาย  | เปลี่ยน  | 
อัปเดต  | 
  | เพิ่มตัวรับใหม่ที่ลงทะเบียน NewsWidget  | 
สร้าง  | 
  | กำหนด UI ของวิดเจ็ตบนหน้าจอหลัก  | 
สร้าง  | 
  | กำหนดการกำหนดค่าวิดเจ็ตบนหน้าจอหลัก คุณสามารถปรับขนาดหรือชื่อวิดเจ็ตได้ในไฟล์นี้  | 
สร้าง  | 
  | มีโค้ด Kotlin เพื่อเพิ่มฟังก์ชันการทำงานลงในวิดเจ็ตหน้าจอหลัก  | 
ดูรายละเอียดเพิ่มเติมเกี่ยวกับไฟล์เหล่านี้ได้ใน Codelab นี้
แก้ไขข้อบกพร่องและทดสอบวิดเจ็ตตัวอย่าง
จากนั้นเรียกใช้แอปพลิเคชันแล้วดูวิดเจ็ตหน้าจอหลัก เมื่อสร้างแอปแล้ว ให้ไปที่หน้าจอการเลือกแอปพลิเคชันของอุปกรณ์ Android แล้วกดไอคอนสำหรับโปรเจ็กต์ Flutter นี้ค้างไว้ เลือกวิดเจ็ตจากเมนูป๊อปอัป

อุปกรณ์ Android หรือโปรแกรมจำลองจะแสดงวิดเจ็ตเริ่มต้นสำหรับ Android
4. ส่งข้อมูลจากแอป Flutter ไปยังวิดเจ็ตหน้าจอหลัก
คุณสามารถปรับแต่งวิดเจ็ตพื้นฐานที่คุณสร้างขึ้นได้ อัปเดตวิดเจ็ตหน้าจอหลักเพื่อแสดงพาดหัวและสรุปของบทความข่าว ภาพหน้าจอต่อไปนี้แสดงตัวอย่างวิดเจ็ตที่แสดงบรรทัดแรกและข้อมูลสรุป

หากต้องการส่งข้อมูลระหว่างแอปและวิดเจ็ตหน้าจอหลัก คุณต้องเขียนโค้ดด้วย Dart และที่มาพร้อมเครื่อง ส่วนนี้แบ่งออกเป็น 3 ส่วนดังนี้
- เขียนโค้ด Dart ในแอป Flutter ที่ทั้ง Android และ iOS ใช้ได้
 - การเพิ่มฟังก์ชันดั้งเดิมของ iOS
 - การเพิ่มฟังก์ชันดั้งเดิมของ Android
 
การใช้กลุ่มแอป iOS
หากต้องการแชร์ข้อมูลระหว่างแอประดับบนสุดบน iOS และส่วนขยายวิดเจ็ต เป้าหมายทั้ง 2 รายการต้องอยู่ในกลุ่มแอปเดียวกัน ดูข้อมูลเพิ่มเติมเกี่ยวกับกลุ่มแอปได้จากเอกสารประกอบเกี่ยวกับกลุ่มแอปของ Apple
อัปเดตตัวระบุกลุ่มโดยทำดังนี้
ไปที่การตั้งค่าของเป้าหมายใน Xcode ในส่วนการลงชื่อและ ความสามารถ ให้ตรวจสอบว่ามีการตั้งค่าตัวระบุทีมและชุดอุปกรณ์แล้ว
เพิ่มกลุ่มแอปลงในทั้งเป้าหมาย Runner และเป้าหมาย NewsWidgetExtension ใน Xcode ดังนี้
เลือก + ความสามารถ -> กลุ่มแอป แล้วเพิ่มกลุ่มแอปใหม่ ทำซ้ำสำหรับทั้งเป้าหมาย Runner (แอปหลัก) และเป้าหมายวิดเจ็ต

เพิ่มโค้ด Dart
ทั้งแอป iOS และ Android จะแชร์ข้อมูลกับแอป Flutter ได้หลายวิธี หากต้องการสื่อสารกับแอปเหล่านี้ ให้ใช้ประโยชน์จาก Store ของ key/value ในอุปกรณ์ iOS เรียกร้านนี้ว่า UserDefaults และ Android เรียกร้านนี้ว่า SharedPreferences แพ็กเกจ home_Widget จะรวม API เหล่านี้เพื่อช่วยให้การบันทึกข้อมูลไปยังแพลตฟอร์มต่างๆ ง่ายขึ้น และช่วยให้วิดเจ็ตบนหน้าจอหลักสามารถดึงข้อมูลที่อัปเดตแล้วได้

ข้อมูลบรรทัดแรกและคำอธิบายมาจากไฟล์ news_data.dart ไฟล์นี้มีข้อมูลจำลองและคลาสข้อมูล NewsArticle
lib/news_data.dart
class NewsArticle {
  final String title;
  final String description;
  final String? articleText;
  NewsArticle({
    required this.title,
    required this.description,
    this.articleText = loremIpsum,
  });
}
อัปเดตค่าบรรทัดแรกและคำอธิบาย
หากต้องการเพิ่มฟังก์ชันการทำงานเพื่ออัปเดตวิดเจ็ตหน้าจอหลักจากแอป Flutter ให้ไปที่ไฟล์ lib/home_screen.dart แทนที่เนื้อหาของไฟล์ด้วยโค้ดต่อไปนี้ จากนั้นแทนที่ <YOUR APP GROUP> ด้วยตัวระบุสำหรับกลุ่มแอป
lib/home_screen.dart
import 'package:flutter/material.dart';
import 'package:home_widget/home_widget.dart';             // Add this import
import 'article_screen.dart';
import 'news_data.dart';
// TODO: Replace with your App Group ID
const String appGroupId = '<YOUR APP GROUP>';              // Add from here
const String iOSWidgetName = 'NewsWidgets';
const String androidWidgetName = 'NewsWidget';             // To here.
class MyHomePage extends StatefulWidget {
  const MyHomePage({super.key});
  @override
  State<MyHomePage> createState() => _MyHomePageState();
}
void updateHeadline(NewsArticle newHeadline) {             // Add from here
  // Save the headline data to the widget
  HomeWidget.saveWidgetData<String>('headline_title', newHeadline.title);
  HomeWidget.saveWidgetData<String>(
      'headline_description', newHeadline.description);
  HomeWidget.updateWidget(
    iOSName: iOSWidgetName,
    androidName: androidWidgetName,
  );
}                                                          // To here.
class _MyHomePageState extends State<MyHomePage> {
  @override                                                // Add from here
  void initState() {
    super.initState();
    HomeWidget.setAppGroupId(appGroupId);
    // Mock read in some data and update the headline
    final newHeadline = getNewsStories()[0];
    updateHeadline(newHeadline);
  }                                                        // To here.
  @override
  Widget build(BuildContext context) {
    return Scaffold(
        appBar: AppBar(
            title: const Text('Top Stories'),
            centerTitle: false,
            titleTextStyle: const TextStyle(
                fontSize: 30,
                fontWeight: FontWeight.bold,
                color: Colors.black)),
        body: ListView.separated(
          separatorBuilder: (context, idx) {
            return const Divider();
          },
          itemCount: getNewsStories().length,
          itemBuilder: (context, idx) {
            final article = getNewsStories()[idx];
            return ListTile(
              key: Key('$idx ${article.hashCode}'),
              title: Text(article.title!),
              subtitle: Text(article.description!),
              onTap: () {
                Navigator.of(context).push(
                  MaterialPageRoute(
                    builder: (context) {
                      return ArticleScreen(article: article);
                    },
                  ),
                );
              },
            );
          },
        ));
  }
}
ฟังก์ชัน updateHeadline จะบันทึกคู่คีย์/ค่าลงในพื้นที่เก็บข้อมูลในเครื่องของอุปกรณ์ คีย์ headline_title มีค่า newHeadline.title คีย์ headline_description จะเก็บค่าของ newHeadline.description  ฟังก์ชันนี้ยังแจ้งแพลตฟอร์มเนทีฟว่าสามารถเรียกดูและแสดงผลข้อมูลใหม่สำหรับวิดเจ็ตหน้าจอหลักได้
แก้ไขปุ่มการทำงานแบบลอย
เรียกใช้ฟังก์ชัน updateHeadline เมื่อกด floatingActionButton ตามที่แสดง
lib/article_screen.dart
// New: import the updateHeadline function
import 'home_screen.dart';
...
floatingActionButton: FloatingActionButton.extended(
        onPressed: () {
          ScaffoldMessenger.of(context).showSnackBar(const SnackBar(
            content: Text('Updating home screen widget...'),
          ));
          // New: call updateHeadline
          updateHeadline(widget.article);
        },
        label: const Text('Update Homescreen'),
      ),
...
ด้วยการเปลี่ยนแปลงนี้ เมื่อผู้ใช้กดปุ่มอัปเดตพาดหัวจากหน้าบทความ ระบบจะอัปเดตรายละเอียดวิดเจ็ตบนหน้าจอหลัก
อัปเดตโค้ด iOS เพื่อแสดงข้อมูลบทความ
ใช้ Xcode เพื่ออัปเดตวิดเจ็ตหน้าจอหลักสำหรับ iOS
เปิดไฟล์ NewsWidgets.swift ใน Xcode:
กำหนดค่า TimelineEntry
แทนที่โครงสร้าง SimpleEntry ด้วยโค้ดต่อไปนี้
ios/NewsWidgets/NewsWidgets.swift
// The date and any data you want to pass into your app must conform to TimelineEntry
struct NewsArticleEntry: TimelineEntry {
    let date: Date
    let title: String
    let description:String
}
โครงสร้าง NewsArticleEntry นี้จะกำหนดข้อมูลที่เข้ามาซึ่งจะส่งผ่านไปยังวิดเจ็ตหน้าจอหลักเมื่อมีการอัปเดต ประเภท TimelineEntry ต้องมีพารามิเตอร์วันที่ ดูข้อมูลเพิ่มเติมเกี่ยวกับโปรโตคอล TimelineEntry ได้ที่เอกสารประกอบไทม์ไลน์ของ Apple
แก้ไข View ที่แสดงเนื้อหา
แก้ไขวิดเจ็ตบนหน้าจอหลักให้แสดงพาดหัวและคำอธิบายของบทความข่าวแทนที่การระบุวันที่ หากต้องการแสดงข้อความใน SwiftUI ให้ใช้มุมมอง Text หากต้องการเรียงมุมมองซ้อนกันใน SwiftUI ให้ใช้มุมมอง VStack
แทนที่มุมมอง NewsWidgetEntryView ที่สร้างขึ้นด้วยโค้ดต่อไปนี้
ios/NewsWidgets/NewsWidgets.swift
//View that holds the contents of the widget
struct NewsWidgetsEntryView : View {
    var entry: Provider.Entry
    var body: some View {
      VStack {
        Text(entry.title)
        Text(entry.description)
      }
    }
}
แก้ไขผู้ให้บริการว่าจะให้อัปเดตวิดเจ็ตหน้าจอหลักเมื่อใดและอย่างไร
แทนที่ Provider ที่มีอยู่ด้วยรหัสต่อไปนี้ จากนั้นแทนที่ตัวระบุกลุ่มแอปด้วย <YOUR APP GROUP>
ios/NewsWidgets/NewsWidgets.swift
struct Provider: TimelineProvider {
// Placeholder is used as a placeholder when the widget is first displayed
    func placeholder(in context: Context) -> NewsArticleEntry {
//      Add some placeholder title and description, and get the current date
      NewsArticleEntry(date: Date(), title: "Placeholder Title", description: "Placeholder description")
    }
// Snapshot entry represents the current time and state
    func getSnapshot(in context: Context, completion: @escaping (NewsArticleEntry) -> ()) {
      let entry: NewsArticleEntry
      if context.isPreview{
        entry = placeholder(in: context)
      }
      else{
        //      Get the data from the user defaults to display
        let userDefaults = UserDefaults(suiteName: <YOUR APP GROUP>)
        let title = userDefaults?.string(forKey: "headline_title") ?? "No Title Set"
        let description = userDefaults?.string(forKey: "headline_description") ?? "No Description Set"
        entry = NewsArticleEntry(date: Date(), title: title, description: description)
      }
        completion(entry)
    }
//    getTimeline is called for the current and optionally future times to update the widget
    func getTimeline(in context: Context, completion: @escaping (Timeline<Entry>) -> ()) {
//      This just uses the snapshot function you defined earlier
      getSnapshot(in: context) { (entry) in
// atEnd policy tells widgetkit to request a new entry after the date has passed
        let timeline = Timeline(entries: [entry], policy: .atEnd)
                  completion(timeline)
              }
    }
}
Provider ในโค้ดก่อนหน้านี้สอดคล้องกับ TimelineProvider Provider มี 3 วิธี ได้แก่
- เมธอด 
placeholderจะสร้างรายการตัวยึดตำแหน่งเมื่อผู้ใช้แสดงตัวอย่างวิดเจ็ตในหน้าจอหลักเป็นครั้งแรก 

- เมธอด 
getSnapshotจะอ่านข้อมูลจากค่าเริ่มต้นของผู้ใช้และสร้างรายการเวลาปัจจุบัน - เมธอด 
getTimelineจะแสดงรายการไทม์ไลน์ วิธีนี้จะเป็นประโยชน์เมื่อคุณมีจุดที่คาดการณ์ได้ล่วงหน้าสำหรับการอัปเดตเนื้อหา Codelab นี้ใช้ฟังก์ชัน getSnapshot เพื่อรับสถานะปัจจุบันของ เมธอด.atEndจะบอกให้วิดเจ็ตหน้าจอหลักรีเฟรชข้อมูลหลังจากเวลาปัจจุบันผ่านไป 
แสดงความคิดเห็นเกี่ยวกับ NewsWidgets_Previews
การใช้ตัวอย่างอยู่นอกขอบเขตของ Codelab นี้ สำหรับรายละเอียดเพิ่มเติมเกี่ยวกับการดูตัวอย่างวิดเจ็ตหน้าจอหลักของ SwiftUI โปรดดูเอกสารประกอบของ Apple เกี่ยวกับวิดเจ็ตการแก้ไขข้อบกพร่อง
บันทึกไฟล์ทั้งหมดและเรียกใช้เป้าหมายแอปและวิดเจ็ตอีกครั้ง
เรียกใช้เป้าหมายอีกครั้งเพื่อตรวจสอบว่าแอปและวิดเจ็ตหน้าจอหลักทำงานได้หรือไม่
- เลือกสคีมาของแอปใน Xcode เพื่อเรียกใช้เป้าหมายแอป
 - เลือกสคีมาของส่วนขยายใน Xcode เพื่อเรียกใช้เป้าหมายส่วนขยาย
 - ไปที่หน้าบทความในแอป
 - คลิกปุ่มเพื่ออัปเดตพาดหัว วิดเจ็ตหน้าจอหลักควรอัปเดตบรรทัดแรกด้วย
 
อัปเดตโค้ด Android
เพิ่ม XML ของวิดเจ็ตหน้าจอหลัก
ใน Android Studio ให้อัปเดตไฟล์ที่สร้างขึ้นในขั้นตอนก่อนหน้า โดยเปิดไฟล์ res/layout/news_widget.xml ซึ่งจะกำหนดโครงสร้างและเลย์เอาต์ของวิดเจ็ตหน้าจอหลัก เลือกโค้ดที่มุมขวาบนและแทนที่เนื้อหาของไฟล์นั้นด้วยโค้ดต่อไปนี้
android/app/res/layout/news_widget.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
   xmlns:app="http://schemas.android.com/apk/res-auto"
   xmlns:tools="http://schemas.android.com/tools"
   android:id="@+id/widget_container"
   style="@style/Widget.Android.AppWidget.Container"
   android:layout_width="wrap_content"
   android:layout_height="match_parent"
   android:background="@android:color/white"
   android:theme="@style/Theme.Android.AppWidgetContainer">
   
   <TextView
       android:id="@+id/headline_title"
       style="@style/Widget.Android.AppWidget.InnerView"
       android:layout_width="wrap_content"
       android:layout_height="wrap_content"
       android:layout_alignParentStart="true"
       android:layout_alignParentLeft="true"
       android:layout_marginStart="8dp"
       android:layout_marginLeft="8dp"
       android:background="@android:color/white"
       android:text="Title"
       android:textSize="20sp"
       android:textStyle="bold" />
   <TextView
       android:id="@+id/headline_description"
       style="@style/Widget.Android.AppWidget.InnerView"
       android:layout_width="wrap_content"
       android:layout_height="wrap_content"
       android:layout_below="@+id/headline_title"
       android:layout_alignParentStart="true"
       android:layout_alignParentLeft="true"
       android:layout_marginStart="8dp"
       android:layout_marginLeft="8dp"
       android:layout_marginTop="4dp"
       android:background="@android:color/white"
       android:text="Title"
       android:textSize="16sp" />
</RelativeLayout>
XML นี้กำหนดมุมมองข้อความ 2 มุมมอง คือมุมมองสำหรับพาดหัวบทความ และอีกมุมมองสำหรับคำอธิบายบทความ มุมมองข้อความเหล่านี้จะกำหนดการจัดรูปแบบด้วย คุณจะกลับมาที่ไฟล์นี้ตลอดทั้ง Codelab นี้
อัปเดตฟังก์ชันการทำงานของ NewsWidget
เปิดไฟล์ซอร์สโค้ด Kotlin ของ NewsWidget.kt ไฟล์นี้มีคลาสที่สร้างขึ้นชื่อว่า NewsWidget ที่ขยายคลาส AppWidgetProvider
คลาส NewsWidget มี 3 เมธอดจาก Superclass  คุณจะต้องแก้ไขเมธอด onUpdate Android เรียกใช้วิธีการนี้สำหรับวิดเจ็ตในช่วงเวลาคงที่
แทนที่เนื้อหาของไฟล์ NewsWidget.kt ด้วยโค้ดต่อไปนี้
android/app/java/com.mydomain.homescreen_widgets/NewsWidget.kt
// Import will depend on App ID.
package com.mydomain.homescreen_widgets
import android.appwidget.AppWidgetManager
import android.appwidget.AppWidgetProvider
import android.content.Context
import android.widget.RemoteViews
// New import.
import es.antonborri.home_widget.HomeWidgetPlugin
/**
 * Implementation of App Widget functionality.
 */
class NewsWidget : AppWidgetProvider() {
    override fun onUpdate(
            context: Context,
            appWidgetManager: AppWidgetManager,
            appWidgetIds: IntArray,
    ) {
        for (appWidgetId in appWidgetIds) {
            // Get reference to SharedPreferences
            val widgetData = HomeWidgetPlugin.getData(context)
            val views = RemoteViews(context.packageName, R.layout.news_widget).apply {
                val title = widgetData.getString("headline_title", null)
                setTextViewText(R.id.headline_title, title ?: "No title set")
                val description = widgetData.getString("headline_description", null)
                setTextViewText(R.id.headline_description, description ?: "No description set")
            }
            appWidgetManager.updateAppWidget(appWidgetId, views)
        }
    }
}
ตอนนี้เมื่อมีการเรียก onUpdate Android จะได้รับค่าล่าสุดจากพื้นที่เก็บข้อมูลในเครื่องโดยใช้เมธอด the widgetData.getString() จากนั้นจะเรียกใช้ setTextViewText เพื่อเปลี่ยนข้อความที่แสดงในวิดเจ็ตหน้าจอหลัก
ทดสอบการอัปเดต
ทดสอบแอปเพื่อให้แน่ใจว่าวิดเจ็ตหน้าจอหลักจะอัปเดตด้วยข้อมูลใหม่ หากต้องการอัปเดตข้อมูล ให้ใช้อัปเดตหน้าจอหลัก FloatingActionButton ในหน้าบทความ วิดเจ็ตบนหน้าจอหลักควรอัปเดตด้วยชื่อบทความ

5. การใช้แบบอักษรที่กำหนดเองของแอป Flutter ในวิดเจ็ตหน้าจอหลักของ iOS
จนถึงตอนนี้ คุณได้กำหนดค่าวิดเจ็ตหน้าจอหลักให้อ่านข้อมูลที่แอป Flutter มีให้ แอป Flutter มีแบบอักษรที่กำหนดเองซึ่งคุณอาจต้องใช้ในวิดเจ็ตหน้าจอหลัก คุณสามารถใช้แบบอักษรที่กำหนดเองได้ในวิดเจ็ตหน้าจอหลักของ iOS การใช้แบบอักษรที่กำหนดเองในวิดเจ็ตหน้าจอหลักยังไม่พร้อมใช้งานบน Android
อัปเดตโค้ด iOS
Flutter จะจัดเก็บเนื้อหาไว้ใน mainBundle ของแอปพลิเคชัน iOS คุณเข้าถึงเนื้อหาในแพ็กเกจนี้ได้จากรหัสวิดเจ็ตในหน้าจอหลัก
ในโครงสร้าง NewsWidgetsEntryView ในไฟล์ NewsWidgets.swift ให้ทำการเปลี่ยนแปลงต่อไปนี้
สร้างฟังก์ชันตัวช่วยเพื่อรับเส้นทางไปยังไดเรกทอรีเนื้อหา Flutter โดยทำดังนี้
ios/NewsWidgets/NewsWidgets.swift
struct NewsWidgetsEntryView : View {
   ...
   // New: Add the helper function.
   var bundle: URL {
           let bundle = Bundle.main
           if bundle.bundleURL.pathExtension == "appex" {
               // Peel off two directory levels - MY_APP.app/PlugIns/MY_APP_EXTENSION.appex
               var url = bundle.bundleURL.deletingLastPathComponent().deletingLastPathComponent()
               url.append(component: "Frameworks/App.framework/flutter_assets")
               return url
           }
           return bundle.bundleURL
       }
   ...
}
ลงทะเบียนแบบอักษรโดยใช้ URL ไปยังไฟล์แบบอักษรที่กำหนดเอง
ios/NewsWidgets/NewsWidgets.swift
struct NewsWidgetsEntryView : View {
   ...
   // New: Register the font.
   init(entry: Provider.Entry){
     self.entry = entry
     CTFontManagerRegisterFontsForURL(bundle.appending(path: "/fonts/Chewy-Regular.ttf") as CFURL, CTFontManagerScope.process, nil)
   }
   ...
}
อัปเดตมุมมองข้อความบรรทัดแรกเพื่อใช้แบบอักษรที่กำหนดเอง
ios/NewsWidgets/NewsWidgets.swift
struct NewsWidgetsEntryView : View {
   ...
   var body: some View {
    VStack {
      // Update the following line.
      Text(entry.title).font(Font.custom("Chewy", size: 13))
      Text(entry.description)
    }
   }
   ...
}
เมื่อคุณเรียกใช้วิดเจ็ตหน้าจอหลัก วิดเจ็ตจะใช้แบบอักษรที่กำหนดเองสำหรับบรรทัดแรกดังที่แสดงในรูปภาพต่อไปนี้

6. การแสดงผลวิดเจ็ต Flutter เป็นรูปภาพ
ในส่วนนี้คุณจะเห็นกราฟจากแอป Flutter เป็นวิดเจ็ตหน้าจอหลัก
วิดเจ็ตนี้สร้างความท้าทายได้มากกว่าข้อความที่คุณแสดงบนหน้าจอหลัก การแสดงแผนภูมิ Flutter เป็นรูปภาพจะง่ายกว่าการพยายามสร้างแผนภูมิโดยใช้คอมโพเนนต์ UI แบบดั้งเดิม
เขียนโค้ดวิดเจ็ตบนหน้าจอหลักเพื่อแสดงผลแผนภูมิ Flutter เป็นไฟล์ PNG วิดเจ็ตของหน้าจอหลักจะแสดงรูปภาพนั้นได้
เขียนโค้ดของ Dart
ในฝั่ง Dart ให้เพิ่มเมธอด renderFlutterWidget จากแพ็กเกจ home_Widget วิธีนี้ใช้วิดเจ็ต ชื่อไฟล์ และคีย์ ระบบจะแสดงรูปภาพของวิดเจ็ต Flutter และบันทึกไว้ในคอนเทนเนอร์ที่แชร์ ระบุชื่อรูปภาพในโค้ดและตรวจสอบว่าวิดเจ็ตหน้าจอหลักเข้าถึงคอนเทนเนอร์ได้ key จะบันทึกเส้นทางแบบเต็มของไฟล์เป็นสตริงในพื้นที่เก็บข้อมูลในเครื่องของอุปกรณ์ วิธีนี้จะช่วยให้วิดเจ็ตหน้าจอหลักค้นหาไฟล์ได้หากมีการเปลี่ยนชื่อในโค้ด Dart
สำหรับ Codelab นี้ คลาส LineChart ในไฟล์ lib/article_screen.dart จะแสดงแผนภูมินี้ เมธอดของบิลด์จะแสดง CustomPainter ที่ลงสีแผนภูมินี้บนหน้าจอ
หากต้องการใช้ฟีเจอร์นี้ ให้เปิดไฟล์ lib/article_screen.dart นำเข้าแพ็กเกจ home_widget จากนั้นแทนที่รหัสในชั้นเรียน _ArticleScreenState ด้วยรหัสต่อไปนี้
lib/article_screen.dart
import 'package:flutter/material.dart';
// New: import the home_widget package.
import 'package:home_widget/home_widget.dart';
import 'home_screen.dart';
import 'news_data.dart';
...
class _ArticleScreenState extends State<ArticleScreen> {
  // New: add this GlobalKey
  final _globalKey = GlobalKey();
  String? imagePath;
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text(widget.article.title!),
      ),
      // New: add this FloatingActionButton
      floatingActionButton: FloatingActionButton.extended(
        onPressed: () async {
          if (_globalKey.currentContext != null) {
            var path = await HomeWidget.renderFlutterWidget(
              const LineChart(),
              fileName: 'screenshot',
              key: 'filename',
              logicalSize: _globalKey.currentContext!.size,
              pixelRatio:
                  MediaQuery.of(_globalKey.currentContext!).devicePixelRatio,
            );
            setState(() {
              imagePath = path as String?;
            });
          }
          updateHeadline(widget.article);
        },
        label: const Text('Update Homescreen'),
      ),
      body: ListView(
        padding: const EdgeInsets.all(16.0),
        children: [
          Text(
            widget.article.description!,
            style: Theme.of(context).textTheme.titleMedium,
          ),
          const SizedBox(height: 20.0),
          Text(widget.article.articleText!),
          const SizedBox(height: 20.0),
          Center(
            // New: Add this key
            key: _globalKey,
            child: const LineChart(),
          ),
          const SizedBox(height: 20.0),
          Text(widget.article.articleText!),
        ],
      ),
    );
  }
}
ตัวอย่างนี้ทำการเปลี่ยนแปลง 3 รายการกับชั้นเรียน _ArticleScreenState
สร้าง GlobalKey
GlobalKey จะได้รับบริบทของวิดเจ็ตนั้นๆ ซึ่งจำเป็นต้องใช้เพื่อดูขนาดของวิดเจ็ตนั้น
lib/article_screen.dart
class _ArticleScreenState extends State<ArticleScreen> {
   // New: add this GlobalKey
   final _globalKey = GlobalKey();
   ...
}
เพิ่ม imagePath
พร็อพเพอร์ตี้ imagePath จะจัดเก็บตำแหน่งของรูปภาพที่แสดงวิดเจ็ต Flutter
lib/article_screen.dart
class _ArticleScreenState extends State<ArticleScreen> {
   ...
   // New: add this imagePath
   String? imagePath;
   ...
}
เพิ่มคีย์ลงในวิดเจ็ตเพื่อแสดงผล
_globalKey มีวิดเจ็ต Flutter ที่แสดงผลในรูปภาพ ในกรณีนี้ วิดเจ็ต Flutter คือศูนย์กลางที่มี LineChart
lib/article_screen.dart
class _ArticleScreenState extends State<ArticleScreen> {
   ...
   Center(
      // New: Add this key
 key: _globalKey,
 child: const LineChart(),
   ),
   ...
}
- บันทึกวิดเจ็ตเป็นรูปภาพ
 
ระบบจะเรียกเมธอด renderFlutterWidget เมื่อผู้ใช้คลิกที่ floatingActionButton เมธอดบันทึกไฟล์ PNG ที่ได้เป็น "ภาพหน้าจอ" ลงในไดเรกทอรีคอนเทนเนอร์ที่แชร์ วิธีนี้จะบันทึกเส้นทางแบบเต็มไปยังรูปภาพเป็นคีย์ชื่อไฟล์ในพื้นที่เก็บข้อมูลของอุปกรณ์ด้วย
lib/article_screen.dart
class _ArticleScreenState extends State<ArticleScreen> {
   ...
   floatingActionButton: FloatingActionButton.extended(
 onPressed: () async {
   if (_globalKey.currentContext != null) {
     var path = await HomeWidget.renderFlutterWidget(
       LineChart(),
       fileName: 'screenshot',
       key: 'filename',
       logicalSize: _globalKey.currentContext!.size,
       pixelRatio:
         MediaQuery.of(_globalKey.currentContext!).devicePixelRatio,
     );
     setState(() {
        imagePath = path as String?;
     });
    }
  updateHeadline(widget.article);
  },
   ...
}
อัปเดตโค้ด iOS
สำหรับ iOS ให้อัปเดตโค้ดเพื่อรับเส้นทางของไฟล์จากพื้นที่เก็บข้อมูล และแสดงไฟล์เป็นรูปภาพโดยใช้ SwiftUI
เปิดไฟล์ NewsWidgets.swift เพื่อทำการเปลี่ยนแปลงต่อไปนี้
เพิ่ม filename และ displaySize ในโครงสร้าง NewsArticleEntry
พร็อพเพอร์ตี้ filename จะมีสตริงที่แสดงถึงเส้นทางไปยังไฟล์ภาพ พร็อพเพอร์ตี้ displaySize จะมีขนาดวิดเจ็ตบนหน้าจอหลักในอุปกรณ์ของผู้ใช้ ขนาดของวิดเจ็ตหน้าจอหลักมาจาก context
ios/NewsWidgets/NewsWidgets.swift
struct NewsArticleEntry: TimelineEntry {
   ...
   // New: add the filename and displaySize.
   let filename: String
   let displaySize: CGSize
}
อัปเดตฟังก์ชัน placeholder
รวมตัวยึดตำแหน่ง filename และ displaySize
ios/NewsWidgets/NewsWidgets.swift
func placeholder(in context: Context) -> NewsArticleEntry {
      NewsArticleEntry(date: Date(), title: "Placeholder Title", description: "Placeholder description", filename: "No screenshot available",  displaySize: context.displaySize)
    }
รับชื่อไฟล์จาก userDefaults ใน getSnapshot
ซึ่งจะตั้งค่าตัวแปร filename เป็นค่า filename ในพื้นที่เก็บข้อมูล userDefaults เมื่อวิดเจ็ตหน้าจอหลักอัปเดต
ios/NewsWidgets/NewsWidgets.swift
func getSnapshot(
   ...
   let title = userDefaults?.string(forKey: "headline_title") ?? "No Title Set"
   let description = userDefaults?.string(forKey: "headline_description") ?? "No Description Set"
   // New: get fileName from key/value store
   let filename = userDefaults?.string(forKey: "filename") ?? "No screenshot available"
   ...
)
สร้างรูปภาพแผนภูมิที่แสดงรูปภาพจากเส้นทาง
มุมมอง ChartImage จะสร้างรูปภาพจากเนื้อหาของไฟล์ที่สร้างขึ้นในฝั่ง Dart คุณตั้งขนาดเป็น 50% ของเฟรม
ios/NewsWidgets/NewsWidgets.swift
struct NewsWidgetsEntryView : View {
   ...
   // New: create the ChartImage view
   var ChartImage: some View {
        if let uiImage = UIImage(contentsOfFile: entry.filename) {
            let image = Image(uiImage: uiImage)
                .resizable()
                .frame(width: entry.displaySize.height*0.5, height: entry.displaySize.height*0.5, alignment: .center)
            return AnyView(image)
        }
        print("The image file could not be loaded")
        return AnyView(EmptyView())
    }
   ...
}
ใช้ ChartImage ในเนื้อหา NewsWidgetsEntryView
เพิ่มมุมมอง ChartImage ลงในเนื้อหาของ NewsWidgetsEntryView เพื่อแสดง ChartImage ในวิดเจ็ตหน้าจอหลัก
ios/NewsWidgets/NewsWidgets.swift
VStack {
   Text(entry.title).font(Font.custom("Chewy", size: 13))
   Text(entry.description).font(.system(size: 12)).padding(10)
   // New: add the ChartImage to the NewsWidgetEntryView
   ChartImage
}
ทดสอบการเปลี่ยนแปลง
หากต้องการทดสอบการเปลี่ยนแปลง ให้เรียกใช้ทั้งเป้าหมายแอป Flutter (Runner) และเป้าหมายส่วนขยายจาก Xcode อีกครั้ง หากต้องการดูรูปภาพ ให้ไปที่หน้าบทความหน้าใดหน้าหนึ่งในแอป และกดปุ่มเพื่ออัปเดตวิดเจ็ตหน้าจอหลัก

อัปเดตโค้ด Android
โค้ด Android ทำงานเหมือนกับโค้ด iOS
- เปิดไฟล์ 
android/app/res/layout/news_widget.xmlซึ่งประกอบด้วยองค์ประกอบ UI ของวิดเจ็ตหน้าจอหลัก แทนที่เนื้อหาด้วยโค้ดต่อไปนี้ 
android/app/res/layout/news_widget.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
   xmlns:app="http://schemas.android.com/apk/res-auto"
   xmlns:tools="http://schemas.android.com/tools"
   android:id="@+id/widget_container"
   style="@style/Widget.Android.AppWidget.Container"
   android:layout_width="wrap_content"
   android:layout_height="match_parent"
   android:background="@android:color/white"
   android:theme="@style/Theme.Android.AppWidgetContainer">
   <TextView
       android:id="@+id/headline_title"
       style="@style/Widget.Android.AppWidget.InnerView"
       android:layout_width="wrap_content"
       android:layout_height="wrap_content"
       android:layout_alignParentStart="true"
       android:layout_alignParentLeft="true"
       android:layout_marginStart="8dp"
       android:layout_marginLeft="8dp"
       android:background="@android:color/white"
       android:text="Title"
       android:textSize="20sp"
       android:textStyle="bold" />
   <TextView
       android:id="@+id/headline_description"
       style="@style/Widget.Android.AppWidget.InnerView"
       android:layout_width="wrap_content"
       android:layout_height="wrap_content"
       android:layout_below="@+id/headline_title"
       android:layout_alignParentStart="true"
       android:layout_alignParentLeft="true"
       android:layout_marginStart="8dp"
       android:layout_marginLeft="8dp"
       android:layout_marginTop="4dp"
       android:background="@android:color/white"
       android:text="Title"
       android:textSize="16sp" />
   
   <!--New: add this image view -->
   <ImageView
       android:id="@+id/widget_image"
       android:layout_width="200dp"
       android:layout_height="200dp"
       android:layout_below="@+id/headline_description"
       android:layout_alignBottom="@+id/headline_title"
       android:layout_alignParentStart="true"
       android:layout_alignParentLeft="true"
       android:layout_marginStart="8dp"
       android:layout_marginLeft="8dp"
       android:layout_marginTop="6dp"
       android:layout_marginBottom="-134dp"
       android:layout_weight="1"
       android:adjustViewBounds="true"
       android:background="@android:color/white"
       android:scaleType="fitCenter"
       android:src="@android:drawable/star_big_on"
       android:visibility="visible"
       tools:visibility="visible" />
</RelativeLayout>
รหัสใหม่นี้จะเพิ่มรูปภาพลงในวิดเจ็ตหน้าจอหลัก ซึ่ง (สำหรับตอนนี้) จะแสดงไอคอนรูปดาวทั่วไป แทนที่ไอคอนรูปดาวนี้ด้วยรูปภาพที่คุณบันทึกไว้ในโค้ดของ Dart
- เปิดไฟล์ 
NewsWidget.ktแทนที่เนื้อหาด้วยโค้ดต่อไปนี้ 
android/app/java/com.mydomain.homescreen_widgets/NewsWidget.kt
// Import will depend on App ID.
package com.mydomain.homescreen_widgets
import android.appwidget.AppWidgetManager
import android.appwidget.AppWidgetProvider
import android.content.Context
import android.graphics.Bitmap
import android.graphics.BitmapFactory
import android.widget.RemoteViews
import java.io.File
import es.antonborri.home_widget.HomeWidgetPlugin
/**
 * Implementation of App Widget functionality.
 */
class NewsWidget : AppWidgetProvider() {
    override fun onUpdate(
            context: Context,
            appWidgetManager: AppWidgetManager,
            appWidgetIds: IntArray,
    ) {
        for (appWidgetId in appWidgetIds) {
            val widgetData = HomeWidgetPlugin.getData(context)
            val views = RemoteViews(context.packageName, R.layout.news_widget).apply {
                val title = widgetData.getString("headline_title", null)
                setTextViewText(R.id.headline_title, title ?: "No title set")
                val description = widgetData.getString("headline_description", null)
                setTextViewText(R.id.headline_description, description ?: "No description set")
                // New: Add the section below
               // Get chart image and put it in the widget, if it exists
                val imageName = widgetData.getString("filename", null)
                val imageFile = File(imageName)
                val imageExists = imageFile.exists()
                if (imageExists) {
                    val myBitmap: Bitmap = BitmapFactory.decodeFile(imageFile.absolutePath)
                    setImageViewBitmap(R.id.widget_image, myBitmap)
                } else {
                    println("image not found!, looked @: ${imageName}")
                }
                // End new code
            }
            appWidgetManager.updateAppWidget(appWidgetId, views)
        }
    }
}
โค้ดของ Dart นี้จะบันทึกภาพหน้าจอลงในพื้นที่เก็บข้อมูลในเครื่องด้วยคีย์ filename นอกจากนี้ยังได้รับเส้นทางแบบเต็มของรูปภาพและสร้างออบเจ็กต์ File จากรูปภาพดังกล่าวด้วย หากมีรูปภาพอยู่ รหัสของ Dart จะแทนที่รูปภาพในวิดเจ็ตหน้าจอหลักด้วยรูปภาพใหม่
- โหลดแอปของคุณซ้ำและไปยังหน้าจอบทความ กดอัปเดตหน้าจอหลัก วิดเจ็ตหน้าจอหลักจะแสดงแผนภูมิ
 
7. ขั้นตอนถัดไป
ยินดีด้วย
ยินดีด้วย คุณสร้างวิดเจ็ตหน้าจอหลักสำหรับแอป Flutter ใน iOS และ Android สำเร็จแล้ว
การลิงก์ไปยังเนื้อหาในแอป Flutter
คุณอาจต้องการนําผู้ใช้ไปยังหน้าที่เจาะจงในแอป ทั้งนี้ขึ้นอยู่กับตําแหน่งที่ผู้ใช้คลิก ตัวอย่างเช่น ในแอปข่าวจาก Codelab นี้ คุณอาจต้องการให้ผู้ใช้เห็นบทความข่าวสำหรับพาดหัวที่แสดงอยู่
ฟีเจอร์นี้อยู่นอกขอบเขตของ Codelab นี้ คุณสามารถดูตัวอย่างการใช้สตรีมที่แพ็กเกจ home_Widget มีให้เพื่อระบุการเรียกใช้แอปจากวิดเจ็ตหน้าจอหลักและส่งข้อความจากวิดเจ็ตหน้าจอหลักผ่าน URL ดูข้อมูลเพิ่มเติมได้ในเอกสารประกอบการทำ Deep Link ใน docs.flutter.dev
การอัปเดตวิดเจ็ตในเบื้องหลัง
ใน Codelab นี้ คุณได้ทริกเกอร์การอัปเดตวิดเจ็ตบนหน้าจอหลักโดยใช้ปุ่ม แม้ว่าวิธีนี้จะสมเหตุสมผลสำหรับการทดสอบ แต่ในโค้ดเวอร์ชันที่ใช้งานจริง คุณอาจต้องการให้แอปอัปเดตวิดเจ็ตหน้าจอหลักในเบื้องหลัง คุณสามารถใช้ปลั๊กอิน Workmanager เพื่อสร้างงานเบื้องหลังเพื่ออัปเดตทรัพยากรที่วิดเจ็ตหน้าจอหลักต้องใช้ได้ หากต้องการเรียนรู้เพิ่มเติม ให้ดูที่ส่วนการอัปเดตพื้นหลังในแพ็กเกจ home_widget
สำหรับ iOS คุณสามารถใช้วิดเจ็ตบนหน้าจอหลักส่งคำขอเครือข่ายเพื่ออัปเดต UI ของตนได้ด้วย หากต้องการควบคุมเงื่อนไขหรือความถี่ของคำขอนั้น ให้ใช้ไทม์ไลน์ หากต้องการดูข้อมูลเพิ่มเติมเกี่ยวกับการใช้ไทม์ไลน์ โปรดดู "การอัปเดตวิดเจ็ตอยู่เสมอ" ของ Apple เอกสารประกอบ