SKILL.md
$27
Adhere to the following constraints when authoring previewable widgets, as the Widget Previewer runs in a web environment:
- No Native APIs: Do not use native plugins or APIs from
dart:ioordart:ffi. Widgets with transitive dependencies ondart:ioordart:ffiwill throw exceptions upon invocation. Use conditional imports to mock or bypass these in preview mode.
- Asset Paths: Use package-based paths for assets loaded via
dart:uifromAssetAPIs (e.g.,packages/my_package_name/assets/my_image.pnginstead ofassets/my_image.png).
- Public Callbacks: Ensure all callback arguments provided to preview annotations are public and constant to satisfy code generation requirements.
- Constraints: Apply explicit constraints using the
sizeparameter in the@Previewannotation if your widget is unconstrained, as the previewer defaults to constraining them to approximately half the viewport.
Workflows
Creating a Widget Preview
Copy and track this checklist when implementing a new widget preview:
- Import
package:flutter/widget_previews.dart.
- Identify a valid target (top-level function, static method, or parameter-less public constructor).
- Apply the
@Previewannotation to the target.
- Configure preview parameters (
name,group,size,theme,brightness, etc.) as needed.
- If applying the same configuration to multiple widgets, extract the configuration into a custom class extending
Preview.
Interacting with Previews
Follow the appropriate conditional workflow to launch and interact with the Widget Previewer:
If using a supported IDE (Android Studio, IntelliJ, VS Code with Flutter 3.38+):
- Launch the IDE. The Widget Previewer starts automatically.
- Open the "Flutter Widget Preview" tab in the sidebar.
- Toggle "Filter previews by selected file" at the bottom left if you want to view previews outside the currently active file.
If using the Command Line:
- Navigate to the Flutter project's root directory.
- Run
flutter widget-preview start.
- View the automatically opened Chrome environment.
Feedback Loop: Preview Iteration
- Modify the widget code or preview configuration.
- Observe the automatic update in the Widget Previewer.
- If global state (e.g., static initializers) was modified: Click the global hot restart button at the bottom right.
- If only the local widget state needs resetting: Click the individual hot restart button on the specific preview card.
- Review errors in the IDE/CLI console -> fix -> repeat.
Examples
Basic Preview
import 'package:flutter/widget_previews.dart';
import 'package:flutter/material.dart';
@Preview(name: 'My Sample Text', group: 'Typography')
Widget mySampleText() {
return const Text('Hello, World!');
}
Custom Preview with Runtime Transformation
import 'package:flutter/widget_previews.dart';
import 'package:flutter/material.dart';
final class TransformativePreview extends Preview {
const TransformativePreview({
super.name,
super.group,
});
PreviewThemeData _themeBuilder() {
return PreviewThemeData(
materialLight: ThemeData.light(),
materialDark: ThemeData.dark(),
);
}
@override
Preview transform() {
final originalPreview = super.transform();
final builder = originalPreview.toBuilder();
builder
..name = 'Transformed - ${originalPreview.name}'
..theme = _themeBuilder;
return builder.toPreview();
}
}
@TransformativePreview(name: 'Custom Themed Button')
Widget myButton() => const ElevatedButton(onPressed: null, child: Text('Click'));
MultiPreview Implementation
import 'package:flutter/widget_previews.dart';
import 'package:flutter/material.dart';
/// Creates light and dark mode previews automatically.
final class MultiBrightnessPreview extends MultiPreview {
const MultiBrightnessPreview({required this.name});
final String name;
@override
List<Preview> get previews => const [
Preview(brightness: Brightness.light),
Preview(brightness: Brightness.dark),
];
@override
List<Preview> transform() {
final previews = super.transform();
return previews.map((preview) {
final builder = preview.toBuilder()
..group = 'Brightness'
..name = '$name - ${preview.brightness!.name}';
return builder.toPreview();
}).toList();
}
}
@MultiBrightnessPreview(name: 'Primary Card')
Widget cardPreview() => const Card(child: Padding(padding: EdgeInsets.all(8.0), child: Text('Content')));