build method
- BuildContext context
Describes the part of the user interface represented by this widget.
The framework calls this method when this widget is inserted into the tree in a given BuildContext and when the dependencies of this widget change (e.g., an InheritedWidget referenced by this widget changes). This method can potentially be called in every frame and should not have any side effects beyond building a widget.
The framework replaces the subtree below this widget with the widget returned by this method, either by updating the existing subtree or by removing the subtree and inflating a new subtree, depending on whether the widget returned by this method can update the root of the existing subtree, as determined by calling Widget.canUpdate.
Typically implementations return a newly created constellation of widgets that are configured with information from this widget's constructor and from the given BuildContext.
The given BuildContext contains information about the location in the tree at which this widget is being built. For example, the context provides the set of inherited widgets for this location in the tree. A given widget might be built with multiple different BuildContext arguments over time if the widget is moved around the tree or if the widget is inserted into the tree in multiple places at once.
The implementation of this method must only depend on:
- the fields of the widget, which themselves must not change over time, and
- any ambient state obtained from the
context
using BuildContext.dependOnInheritedWidgetOfExactType.
If a widget's build method is to depend on anything else, use a StatefulWidget instead.
See also:
- StatelessWidget, which contains the discussion on performance considerations.
Implementation
@override
Widget build(BuildContext context) {
imageCache.clear();
imageCache.clearLiveImages();
final ScrollController _scrollController = ScrollController();
return Scaffold(
appBar: AppBar(
centerTitle: false,
title: buildHyperLink == null
? Text(appBarImage)
: MarkdownBody(
data: wordWrap('''
**$appBarImage**
$buildHyperLink
'''),
styleSheet: MarkdownStyleSheet(
// Force left alignment for paragraph text.
p: Theme.of(context).textTheme.bodyMedium ??
const TextStyle(),
textAlign: WrapAlignment.start,
strong: const TextStyle(
fontSize: 20,
fontWeight: FontWeight.w500,
),
),
onTapLink: (text, href, title) {
if (href != null && href.isNotEmpty) {
launchUrl(Uri.parse(href));
}
},
),
),
body: Column(
children: [
Expanded(
child: Scrollbar(
controller: _scrollController,
thumbVisibility: true,
thickness: 8,
radius: const Radius.circular(
4,
), // Optional: customize scrollbar appearance
child: ListView.builder(
controller: _scrollController,
scrollDirection: Axis.horizontal,
itemCount: paths.length,
itemBuilder: (context, index) {
return FutureBuilder<Uint8List?>(
future: _loadImageBytes(paths[index]),
builder: (context, snapshot) {
if (snapshot.hasError) {
return Center(
child: Text('Error: ${snapshot.error}'),
);
} else if (snapshot.connectionState ==
ConnectionState.waiting) {
return const Center(child: CircularProgressIndicator());
} else if (snapshot.data == null ||
snapshot.data!.isEmpty) {
return const Center(
child: Text(
'Image not available',
style: TextStyle(
fontSize: 20,
fontWeight: FontWeight.bold,
color: Colors.redAccent,
),
),
);
} else {
final bytes = snapshot.data!;
return Padding(
padding: const EdgeInsets.all(8.0),
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
Row(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.center,
children: [
Flexible(
fit: FlexFit.loose,
child: MarkdownBody(
data: wordWrap(titles[index]),
selectable: true,
onTapLink: (text, href, title) {
final Uri url = Uri.parse(href ?? '');
launchUrl(url);
},
),
),
const SizedBox(
width: 8,
),
MarkdownTooltip(
message: '''
**Enlarge.** Tap here to view the plot enlarged to the
maximum size within the app.
''',
child: IconButton(
icon: const Icon(
Icons.zoom_out_map,
color: Colors.blue,
),
onPressed: () {
showImageDialog(context, bytes);
},
),
),
const SizedBox(
width: 8,
),
MarkdownTooltip(
message: '''
**Open.** Tap here to open the plot in a separate window
to the Rattle app itself. This allows you to retain a
view of the plot while you navigate through other plots
and analyses. If you choose the external app to be
**Inkscape**, for example, then you can edit the plot,
including the text, colours, etc. Note that Inkscape can
be a little slow to startup. The choice of app depends
on your operating system settings and can be overridden
in the Rattle **Settings**.
''',
child: IconButton(
icon: const Icon(
Icons.open_in_new,
color: Colors.blue,
),
onPressed: () async {
String fileName =
'plot_${Random().nextInt(10000)}.svg';
File tempFile =
File('$tempDir/$fileName');
await File(paths[index])
.copy(tempFile.path);
final prefs = await SharedPreferences
.getInstance();
final imageViewerApp =
prefs.getString('imageViewerApp');
Platform.isWindows
? Process.run(
imageViewerApp!,
[tempFile.path],
runInShell: true,
)
: Process.run(
imageViewerApp!,
[tempFile.path],
);
},
),
),
const SizedBox(
width: 8,
),
MarkdownTooltip(
message: '''
**Save.** Tap here to save the plot in your preferred
format (**svg**, **pdf**, or **png**). You can directly
choose your desired format by replacing the default
*svg* filename extension with either *pdf* or *png*. The
file is saved to your local storage. Perfect for
including in reports or keeping for future
reference. The **svg** format is particularly convenient
as you can edit all details of the plot with an
application like **Inkscape**.
''',
child: IconButton(
icon: const Icon(
Icons.save,
color: Colors.blue,
),
onPressed: () async {
String fileName =
paths[index].split('/').last;
String? pathToSave = await selectFile(
defaultFileName: fileName,
allowedExtensions: [
'svg',
'pdf',
'png',
],
);
if (pathToSave != null) {
String extension = pathToSave
.split('.')
.last
.toLowerCase();
if (extension == 'svg') {
await File(paths[index])
.copy(pathToSave);
} else if (extension == 'pdf') {
await _exportToPdf(
paths[index],
pathToSave,
);
} else if (extension == 'png') {
await _exportToPng(
paths[index],
pathToSave,
);
} else {
showOk(
title: 'Error',
context: context,
content: '''
An unsupported filename extension was
provided: .$extension. Please try again
and select a filename with one of the
supported extensions: .svg, .pdf, or .png.
''',
);
}
}
},
),
),
const SizedBox(width: 5),
],
),
Expanded(
child: InteractiveViewer(
maxScale: 5,
alignment: Alignment.topCenter,
child: svgImage
? SvgPicture.memory(
bytes,
fit: BoxFit.scaleDown,
)
: Image.memory(bytes),
),
),
],
),
);
}
},
);
},
),
),
),
],
),
);
}