0 purchases
tutorial stage
TutorialStage #
A Flutter package for creating highly customized in-app tutorials.
Getting Started #
Add dependency to pubspec.yaml
dependencies:
tutorial_stage: <latest-version>
copied to clipboard
Import the package
import 'package:tutorial_stage/tutorial_stage.dart';
copied to clipboard
Add TutorialStage to your widget tree
TutorialStage(
child: SomeWidget(),
)
copied to clipboard
Adding TutorialContents
enum TutorialIdentifier {
button,
title,
counter,
}
class TutorialContentExample extends StatelessWidget {
const TutorialContentExample({
super.key,
required this.key,
required this.text,
});
final GlobalKey key;
final String text;
@override
Widget build(BuildContext context) {
final Rect rect = key.boxPosition!.rect.withPadding(const EdgeInsets.all(4));
return SpotlightStage(
rect: rect,
borderRadius: const BorderRadius.all(Radius.circular(4)),
children: <Widget>[
AlignRect(
rect: rect,
alignment: const Alignment(0.0, 2.0),
child: Padding(
padding: const EdgeInsets.only(top: 8),
child: ElevatedButton(
onPressed: () => TutorialStage.of(context).next(),
child: Text(text),
),
),
),
],
);
}
}
class ButtonTutorialContent extends AnimatedTutorialContent {
ButtonTutorialContent(this._buttonKey)
: super(identifier: TutorialIdentifier.button);
final GlobalKey _buttonKey;
@override
Widget buildContent(BuildContext context) {
return TutorialContentExample(
key: _buttonKey,
text: 'Button',
);
}
}
class TitleTutorialContent extends AnimatedTutorialContent {
TitleTutorialContent(this._titleKey)
: super(identifier: TutorialIdentifier.title);
final GlobalKey _titleKey;
@override
Widget buildContent(BuildContext context) {
return TutorialContentExample(
key: _titleKey,
text: 'Title',
);
}
}
class CounterTutorialContent extends AnimatedTutorialContent {
CounterTutorialContent(this._counterKey)
: super(identifier: TutorialIdentifier.counter);
final GlobalKey _counterKey;
@override
Future<void> start() async {
await Scrollable.ensureVisible(
_counterKey.currentContext!,
duration: const Duration(milliseconds: 300),
);
}
@override
Widget buildContent(BuildContext context) {
return TutorialContentExample(
key: _counterKey,
text: 'Counter',
);
}
}
copied to clipboard
Starting the tutorial
final GlobalKey _buttonKey = GlobalKey();
final GlobalKey _titleKey = GlobalKey();
final GlobalKey _counterKey = GlobalKey();
@override
Widget build(BuildContext context) {
return TutorialStage(
child: Scaffold(
appBar: AppBar(
title: Text(
widget.title,
key: _titleKey,
),
),
body: SingleChildScrollView(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
const Text('This is text'),
Padding(
padding: EdgeInsets.only(
top: MediaQuery.of(context).size.height,
),
),
Text(
'This the target text',
key: _counterKey,
style: Theme.of(context).textTheme.headline4,
),
const SizedBox(height: 200),
],
),
),
floatingActionButton: FloatingActionButton(
key: _buttonKey,
onPressed: _startTutorial,
child: const Text('Start'),
),
),
);
}
void _startTutorial() {
TutorialStage.build(
context: context,
contents: <TutorialContent>[
_ButtonTutorialContent(_buttonKey),
_TitleTutorialContent(_titleKey),
_CounterTutorialContent(_counterKey),
],
).start();
}
copied to clipboard
For personal and professional use. You cannot resell or redistribute these repositories in their original state.
There are no reviews.