ส่งผ่านค่าไปยังวิดเจ็ตก่อนหน้า

Aug 16 2020

ฉันมีรูปแบบง่ายๆข้างในมี CircularAvatar เมื่อกดนี้แสดง ModalBottomSheet เพื่อเลือกระหว่างถ่ายภาพจากแกลเลอรี่หรือกล้อง เพื่อให้วิดเจ็ตของฉันมีขนาดกะทัดรัดมากขึ้นฉันจึงแยกมันออกเป็นบางไฟล์

  1. FormDosenScreen (เป็นหน้าจอหลัก)
  2. DosenImagePicker (เป็นเพียง CircularAvatar)
  3. ModalBottomSheetPickImage (เพื่อแสดง ModalBottomSheet)

ปัญหาคือผมไม่ทราบวิธีการที่จะผ่านค่าจากModalBottomSheetPickImageเพื่อFormDosenScreen เนื่องจากค่าจากModalBottomSheetPickImageฉันจะใช้เพื่อแทรกการทำงาน

ฉันประสบความสำเร็จในการส่งต่อจากวิดเจ็ตที่สามไปยังวิดเจ็ตที่สอง แต่เมื่อฉันส่งอีกครั้งจากวิดเจ็ตที่สองไปยังวิดเจ็ตแรกค่าเป็นโมฆะและฉันคิดว่าปัญหากำลังส่งผ่านจากวิดเจ็ตที่สองไปยังวิดเจ็ตแรก

ฉันจะส่งต่อจากวิดเจ็ตที่สามไปยังวิดเจ็ตแรกได้อย่างไร

วิดเจ็ตแรก

class FormDosenScreen extends StatefulWidget {
  static const routeNamed = '/formdosen-screen';

  @override
  _FormDosenScreenState createState() => _FormDosenScreenState();
}

class _FormDosenScreenState extends State<FormDosenScreen> {
  String selectedFile;
  @override
  Widget build(BuildContext context) {
    final detectKeyboardOpen = MediaQuery.of(context).viewInsets.bottom;
    print('trigger');
    return Scaffold(
      appBar: AppBar(
        centerTitle: true,
        title: Text('Tambah Dosen'),
        actions: <Widget>[
          PopupMenuButton(
            itemBuilder: (_) => [
              PopupMenuItem(
                child: Text('Tambah Pelajaran'),
                value: 'add_pelajaran',
              ),
            ],
            onSelected: (String value) {
              switch (value) {
                case 'add_pelajaran':
                  Navigator.of(context).pushNamed(FormPelajaranScreen.routeNamed);
                  break;
                default:
              }
            },
          )
        ],
      ),
      body: Stack(
        fit: StackFit.expand,
        children: <Widget>[
          SingleChildScrollView(
            child: Column(
              mainAxisSize: MainAxisSize.min,
              crossAxisAlignment: CrossAxisAlignment.stretch,
              children: <Widget>[
                SizedBox(height: 20),
                DosenImagePicker(onPickedImage: (file) => selectedFile = file),
                SizedBox(height: 20),
                Card(
                  margin: const EdgeInsets.symmetric(horizontal: 15, vertical: 10),
                  child: Padding(
                    padding: const EdgeInsets.all(20),
                    child: Column(
                      crossAxisAlignment: CrossAxisAlignment.stretch,
                      children: <Widget>[
                        TextFormFieldCustom(
                          onSaved: (value) {},
                          labelText: 'Nama Dosen',
                        ),
                        SizedBox(height: 20),
                        TextFormFieldCustom(
                          onSaved: (value) {},
                          prefixIcon: Icon(Icons.email),
                          labelText: 'Email Dosen',
                          keyboardType: TextInputType.emailAddress,
                        ),
                        SizedBox(height: 20),
                        TextFormFieldCustom(
                          onSaved: (value) {},
                          keyboardType: TextInputType.number,
                          inputFormatter: [
                            // InputNumberFormat(),
                            WhitelistingTextInputFormatter.digitsOnly
                          ],
                          prefixIcon: Icon(Icons.local_phone),
                          labelText: 'Telepon Dosen',
                        ),
                      ],
                    ),
                  ),
                ),
                SizedBox(height: kToolbarHeight),
              ],
            ),
          ),
          Positioned(
            child: Visibility(
              visible: detectKeyboardOpen > 0 ? false : true,
              child: RaisedButton(
                onPressed: () {
                  print(selectedFile);
                },
                materialTapTargetSize: MaterialTapTargetSize.shrinkWrap,
                color: colorPallete.primaryColor,
                child: Text(
                  'SIMPAN',
                  style: TextStyle(fontWeight: FontWeight.bold, fontFamily: AppConfig.headerFont),
                ),
                shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(20)),
                textTheme: ButtonTextTheme.primary,
              ),
            ),
            bottom: kToolbarHeight / 2,
            left: sizes.width(context) / 15,
            right: sizes.width(context) / 15,
          )
        ],
      ),
    );
  }
}

วิดเจ็ตที่สอง


class DosenImagePicker extends StatefulWidget {
  final Function(String file) onPickedImage;
  DosenImagePicker({@required this.onPickedImage});
  @override
  DosenImagePickerState createState() => DosenImagePickerState();
}

class DosenImagePickerState extends State<DosenImagePicker> {
  String selectedImage;
  @override
  Widget build(BuildContext context) {
    return Align(
      alignment: Alignment.center,
      child: InkWell(
        onTap: () async {
          await showModalBottomSheet(
            context: context,
            builder: (context) => ModalBottomSheetPickImage(
              onPickedImage: (file) {
                setState(() {
                  selectedImage = file;
                  widget.onPickedImage(selectedImage);
                  print('Hellooo dosen image picker $selectedImage');
                });
              },
            ),
          );
        },
        child: CircleAvatar(
          foregroundColor: colorPallete.black,
          backgroundImage: selectedImage == null ? null : MemoryImage(base64.decode(selectedImage)),
          radius: sizes.width(context) / 6,
          backgroundColor: colorPallete.accentColor,
          child: selectedImage == null ? Text('Pilih Gambar') : SizedBox(),
        ),
      ),
    );
  }
}

วิดเจ็ตที่สาม


class ModalBottomSheetPickImage extends StatelessWidget {
  final Function(String file) onPickedImage;

  ModalBottomSheetPickImage({@required this.onPickedImage});

  @override
  Widget build(BuildContext context) {
    return SizedBox(
      child: Padding(
        padding: const EdgeInsets.all(15.0),
        child: Wrap(
          alignment: WrapAlignment.spaceEvenly,
          children: <Widget>[
            InkWell(
              onTap: () async {
                final String resultBase64 =
                    await commonFunction.pickImage(quality: 80, returnFile: ReturnFile.BASE64);
                onPickedImage(resultBase64);
              },
              child: CircleAvatar(
                foregroundColor: colorPallete.white,
                backgroundColor: colorPallete.green,
                child: Icon(Icons.camera_alt),
              ),
            ),
            InkWell(
              onTap: () async {
                final String resultBase64 =
                    await commonFunction.pickImage(returnFile: ReturnFile.BASE64, isCamera: false);
                onPickedImage(resultBase64);
              },
              child: CircleAvatar(
                foregroundColor: colorPallete.white,
                backgroundColor: colorPallete.blue,
                child: Icon(Icons.photo_library),
              ),
            ),
          ],
        ),
      ),
    );
  }
}

คำตอบ

2 Uni Aug 17 2020 at 04:30

วิธีที่สะอาดและง่ายที่สุดคือผ่านผู้ให้บริการ เป็นหนึ่งในโซลูชันการจัดการสถานะที่คุณสามารถใช้เพื่อส่งผ่านค่ารอบ ๆ แอปและสร้างเฉพาะวิดเจ็ตที่เปลี่ยนแปลงใหม่ (เช่นเมื่อค่าของวิดเจ็ตข้อความเปลี่ยนไป) นี่คือวิธีที่คุณสามารถใช้ Provider ในสถานการณ์ของคุณ:

นี่คือลักษณะของโมเดลของคุณ:

class ImageModel extends ChangeNotifier {
  String _base64Image;
  get base64Image => _base64Image;
  set base64Image(String base64Image) {
    _base64Image = base64Image;
    notifyListeners();
  }
}

อย่าลืมเพิ่ม getters และ setters เพื่อให้คุณสามารถใช้ alertListeners () ได้หากคุณมี UI ที่ขึ้นอยู่กับมัน

นี่คือวิธีที่คุณสามารถเข้าถึงค่าของ ImageModel ใน UI ของคุณ:

final model=Provider.of<ImageModel>(context,listen:false);
String image=model.base64Image; //get data
model.base64Image=resultBase64; //set your image data after you used ImagePicker

นี่คือวิธีที่คุณสามารถแสดงข้อมูลของคุณในวิดเจ็ตข้อความ (ตามหลักการแล้วคุณควรใช้ Selector แทน Consumer เพื่อให้วิดเจ็ตสร้างใหม่ก็ต่อเมื่อค่าที่รับฟังมีการเปลี่ยนแปลง):

@override
Widget build(BuildContext context) {
 //other widgets
 Selector<ImageModel, String>(
  selector: (_, model) => model.base64Image,
  builder: (_, image, __) {
   return Text(image);
     },
   );
  }
 )
}
VasilKanev Aug 17 2020 at 09:45

คุณสามารถบรรลุสิ่งนี้ได้อย่างง่ายดาย หากคุณใช้งาน Blocs