|  | // Copyright 2013 The Flutter Authors. All rights reserved. | 
|  | // Use of this source code is governed by a BSD-style license that can be | 
|  | // found in the LICENSE file. | 
|  |  | 
|  | // ignore_for_file: public_member_api_docs | 
|  |  | 
|  | import 'package:flutter/material.dart'; | 
|  | import 'package:image_picker/image_picker.dart'; | 
|  | import 'package:share/share.dart'; | 
|  |  | 
|  | import 'image_previews.dart'; | 
|  |  | 
|  | void main() { | 
|  | runApp(DemoApp()); | 
|  | } | 
|  |  | 
|  | class DemoApp extends StatefulWidget { | 
|  | @override | 
|  | DemoAppState createState() => DemoAppState(); | 
|  | } | 
|  |  | 
|  | class DemoAppState extends State<DemoApp> { | 
|  | String text = ''; | 
|  | String subject = ''; | 
|  | List<String> imagePaths = []; | 
|  |  | 
|  | @override | 
|  | Widget build(BuildContext context) { | 
|  | return MaterialApp( | 
|  | title: 'Share Plugin Demo', | 
|  | home: Scaffold( | 
|  | appBar: AppBar( | 
|  | title: const Text('Share Plugin Demo'), | 
|  | ), | 
|  | body: SingleChildScrollView( | 
|  | child: Padding( | 
|  | padding: const EdgeInsets.all(24.0), | 
|  | child: Column( | 
|  | crossAxisAlignment: CrossAxisAlignment.start, | 
|  | children: <Widget>[ | 
|  | TextField( | 
|  | decoration: const InputDecoration( | 
|  | labelText: 'Share text:', | 
|  | hintText: 'Enter some text and/or link to share', | 
|  | ), | 
|  | maxLines: 2, | 
|  | onChanged: (String value) => setState(() { | 
|  | text = value; | 
|  | }), | 
|  | ), | 
|  | TextField( | 
|  | decoration: const InputDecoration( | 
|  | labelText: 'Share subject:', | 
|  | hintText: 'Enter subject to share (optional)', | 
|  | ), | 
|  | maxLines: 2, | 
|  | onChanged: (String value) => setState(() { | 
|  | subject = value; | 
|  | }), | 
|  | ), | 
|  | const Padding(padding: EdgeInsets.only(top: 12.0)), | 
|  | ImagePreviews(imagePaths, onDelete: _onDeleteImage), | 
|  | ListTile( | 
|  | leading: Icon(Icons.add), | 
|  | title: Text("Add image"), | 
|  | onTap: () async { | 
|  | final imagePicker = ImagePicker(); | 
|  | final pickedFile = await imagePicker.getImage( | 
|  | source: ImageSource.gallery, | 
|  | ); | 
|  | if (pickedFile != null) { | 
|  | setState(() { | 
|  | imagePaths.add(pickedFile.path); | 
|  | }); | 
|  | } | 
|  | }, | 
|  | ), | 
|  | const Padding(padding: EdgeInsets.only(top: 12.0)), | 
|  | Builder( | 
|  | builder: (BuildContext context) { | 
|  | return ElevatedButton( | 
|  | child: const Text('Share'), | 
|  | onPressed: text.isEmpty && imagePaths.isEmpty | 
|  | ? null | 
|  | : () => _onShare(context), | 
|  | ); | 
|  | }, | 
|  | ), | 
|  | const Padding(padding: EdgeInsets.only(top: 12.0)), | 
|  | Builder( | 
|  | builder: (BuildContext context) { | 
|  | return ElevatedButton( | 
|  | child: const Text('Share With Empty Origin'), | 
|  | onPressed: () => _onShareWithEmptyOrigin(context), | 
|  | ); | 
|  | }, | 
|  | ), | 
|  | ], | 
|  | ), | 
|  | ), | 
|  | )), | 
|  | ); | 
|  | } | 
|  |  | 
|  | _onDeleteImage(int position) { | 
|  | setState(() { | 
|  | imagePaths.removeAt(position); | 
|  | }); | 
|  | } | 
|  |  | 
|  | _onShare(BuildContext context) async { | 
|  | // A builder is used to retrieve the context immediately | 
|  | // surrounding the ElevatedButton. | 
|  | // | 
|  | // The context's `findRenderObject` returns the first | 
|  | // RenderObject in its descendent tree when it's not | 
|  | // a RenderObjectWidget. The ElevatedButton's RenderObject | 
|  | // has its position and size after it's built. | 
|  | final RenderBox box = context.findRenderObject() as RenderBox; | 
|  |  | 
|  | if (imagePaths.isNotEmpty) { | 
|  | await Share.shareFiles(imagePaths, | 
|  | text: text, | 
|  | subject: subject, | 
|  | sharePositionOrigin: box.localToGlobal(Offset.zero) & box.size); | 
|  | } else { | 
|  | await Share.share(text, | 
|  | subject: subject, | 
|  | sharePositionOrigin: box.localToGlobal(Offset.zero) & box.size); | 
|  | } | 
|  | } | 
|  |  | 
|  | _onShareWithEmptyOrigin(BuildContext context) async { | 
|  | await Share.share("text"); | 
|  | } | 
|  | } |