flutter_story 1.0.1 copy "flutter_story: ^1.0.1" to clipboard
flutter_story: ^1.0.1 copied to clipboard

A flutter plugin for creating story (like Instagram, WhatsApp, Telegram, Facebook, etc) for all platforms.

example/lib/main.dart

import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';
import 'package:flutter_story/flutter_story.dart';

void main() {
  runApp(const StoryExampleApp());
}

class StoryExampleApp extends StatelessWidget {
  const StoryExampleApp({super.key});

  @override
  Widget build(BuildContext context) {
    return const MaterialApp(
      debugShowCheckedModeBanner: false,
      title: "Flutter Story - Example App",
      home: StoryExamplePage(),
    );
  }
}

class StoryExamplePage extends StatefulWidget {
  const StoryExamplePage({super.key});

  @override
  State<StoryExamplePage> createState() => _StoryExamplePageState();
}

class _StoryExamplePageState extends State<StoryExamplePage> {
  StoryController storyController = StoryController();
  late List<StoryModel> stories;

  @override
  void initState() {
    super.initState();
    stories = getStories();
  }

  @override
  void dispose() {
    super.dispose();
    storyController.dispose();
  }

  @override
  Widget build(BuildContext context) {
    return Scaffold(
      body: SafeArea(
        child: SingleChildScrollView(
          physics: const BouncingScrollPhysics(),
          child: Column(
            crossAxisAlignment: CrossAxisAlignment.start,
            children: [
              Story.builder(
                  controller: storyController,
                  itemCount: stories.length,
                  itemBuilder: (context, index) {
                    StoryModel s = stories[index];
                    return StoryUser(
                      avatar: s.avatar,
                      label: s.label,
                      children: s.cards == null
                          ? []
                          : s.cards!
                          .map((card) => StoryCard(
                        onVisited: (cardIndex) {
                          setState(() {
                            card.visited = true;
                          });
                        },
                        footer: StoryCardFooter(
                          messageBox: StoryCardMessageBox(
                            child: Center(
                              child: SizedBox(
                                width: MediaQuery.of(context)
                                    .size
                                    .width /
                                    1.5,
                                height: MediaQuery.of(context)
                                    .size
                                    .width /
                                    1.5,
                                child: Column(
                                  mainAxisAlignment:
                                  MainAxisAlignment.center,
                                  children: [
                                    Row(
                                      mainAxisAlignment:
                                      MainAxisAlignment
                                          .spaceAround,
                                      children: [
                                        MaterialButton(
                                          minWidth: 0,
                                          padding: EdgeInsets.zero,
                                          shape:
                                          const CircleBorder(),
                                          child: const Text(
                                            "😂",
                                            style: TextStyle(
                                                fontSize: 32),
                                          ),
                                          onPressed: () {},
                                        ),
                                        MaterialButton(
                                          minWidth: 0,
                                          padding: EdgeInsets.zero,
                                          shape:
                                          const CircleBorder(),
                                          child: const Text(
                                            "😮",
                                            style: TextStyle(
                                                fontSize: 32),
                                          ),
                                          onPressed: () {},
                                        ),
                                        MaterialButton(
                                          minWidth: 0,
                                          padding: EdgeInsets.zero,
                                          shape:
                                          const CircleBorder(),
                                          child: const Text(
                                            "😍",
                                            style: TextStyle(
                                                fontSize: 32),
                                          ),
                                          onPressed: () {},
                                        ),
                                      ],
                                    ),
                                    const SizedBox(height: 30),
                                    Row(
                                      mainAxisAlignment:
                                      MainAxisAlignment
                                          .spaceAround,
                                      children: [
                                        MaterialButton(
                                          minWidth: 0,
                                          padding: EdgeInsets.zero,
                                          shape:
                                          const CircleBorder(),
                                          child: const Text(
                                            "😢",
                                            style: TextStyle(
                                                fontSize: 32),
                                          ),
                                          onPressed: () {},
                                        ),
                                        MaterialButton(
                                          minWidth: 0,
                                          padding: EdgeInsets.zero,
                                          shape:
                                          const CircleBorder(),
                                          child: const Text(
                                            "👏",
                                            style: TextStyle(
                                                fontSize: 32),
                                          ),
                                          onPressed: () {},
                                        ),
                                        MaterialButton(
                                          minWidth: 0,
                                          padding: EdgeInsets.zero,
                                          shape:
                                          const CircleBorder(),
                                          child: const Text(
                                            "🔥",
                                            style: TextStyle(
                                                fontSize: 32),
                                          ),
                                          onPressed: () {},
                                        ),
                                      ],
                                    ),
                                  ],
                                ),
                              ),
                            ),
                          ),
                          likeButton: StoryCardLikeButton(
                            onLike: (cardLike) {},
                          ),
                          forwardButton: StoryCardForwardButton(
                            onForward: (cardIndex) {},
                          ),
                        ),
                        color: card.color,
                        visited: card.visited,
                        cardDuration: card.duration,
                        childOverlay: card.childOverlay,
                        child: card.child,
                      ))
                          .toList(),
                    );
                  }),
              Container(
                margin: const EdgeInsets.fromLTRB(0, 5, 0, 20),
                decoration: BoxDecoration(
                  border: Border(
                    top: BorderSide(width: 1, color: Colors.grey.withAlpha(50)),
                  ),
                ),
                child: Column(
                  children: [
                    getPost(
                        name: "Benjamin",
                        avatar: "assets/avatar1.jpg",
                        image: "assets/post1.jpg"),
                    getPost(
                        name: "Liam",
                        avatar: "assets/avatar2.jpg",
                        image: "assets/post2.jpg"),
                    getPost(
                        name: "Oliver",
                        avatar: "assets/avatar3.jpg",
                        image: "assets/post3.jpg"),
                  ],
                ),
              ),
            ],
          ),
        ),
      ),
    );
  }

  List<StoryModel> getStories() {
    List<StoryModel> storyList = [];
    for (int i = 1; i <= 10; i++) {
      storyList.add(StoryModel(
          id: i + 2,
          avatar: Image.asset("assets/avatar$i.jpg"),
          label: Text(
            userLabel(i),
            style: const TextStyle(color: Colors.black),
          ),
          cards: [
            storyCard1(i),
            storyCard2(i),
          ]));
    }
    return storyList;
  }

  StoryCardModel storyCard1(int i) => StoryCardModel(
    color: Colors.purple,
    childOverlay: Container(
      margin: const EdgeInsets.all(50),
      child: const Center(
        child: Text(
          "Lorem Ipsum is simply dummy text of the printing "
              "and typesetting industry. Lorem Ipsum has been the industry's "
              "standard dummy text ever since the 1500s, when an unknown "
              "printer took a galley of type and scrambled it to "
              "make a type specimen book. It has survived not only five centuries,"
              " but also the leap into electronic typesetting, "
              "remaining essentially unchanged. It was popularised in the 1960s "
              "with the release of Letraset sheets containing "
              "Lorem Ipsum passages, and more recently with "
              "desktop publishing software like Aldus PageMaker "
              "including versions of Lorem Ipsum.",
          style: TextStyle(color: Colors.white, fontSize: 20, height: 1.5),
        ),
      ),
    ),
  );

  StoryCardModel storyCard2(int i) => StoryCardModel(
    child: Container(
      color: Colors.purple,
      child: Column(
        mainAxisAlignment: MainAxisAlignment.center,
        children: [
          Container(
            width: 100,
            color: Colors.white,
            padding: const EdgeInsets.all(10),
            child: Center(
              child: Text(
                "user $i",
                style: const TextStyle(color: Colors.black, fontSize: 20),
              ),
            ),
          ),
          const SizedBox(height: 10),
          Image.asset("assets/avatar$i.jpg"),
          const SizedBox(height: 10),
          Container(
            width: 250,
            padding: const EdgeInsets.all(5),
            decoration: BoxDecoration(
              color: Colors.red.withAlpha(150),
              borderRadius: const BorderRadius.all(Radius.circular(10)),
            ),
            child: const Center(
              child: Text(
                "This is a container widget",
                style: TextStyle(color: Colors.black, fontSize: 20),
              ),
            ),
          ),
          const SizedBox(height: 10),
          Container(
            width: 350,
            padding: const EdgeInsets.all(5),
            decoration: BoxDecoration(
                color: Colors.black.withAlpha(50),
                border: Border.all(color: Colors.white, width: 1)),
            child: Center(
              child: Text(
                "This is a container widget",
                style:
                TextStyle(color: Colors.white, fontSize: 20, shadows: [
                  Shadow(
                      color: Colors.black.withAlpha(150),
                      blurRadius: 20,
                      offset: const Offset(0, 0))
                ]),
              ),
            ),
          ),
          const SizedBox(height: 10),
          Center(
            child: Text(
              "Story $i",
              style: const TextStyle(
                color: Colors.white,
                fontSize: 20,
                shadows: [
                  Shadow(
                      color: Colors.red,
                      blurRadius: 20,
                      offset: Offset(0, 0))
                ],
              ),
            ),
          ),
        ],
      ),
    ),
  );

  String userLabel(int storyIndex) {
    String label = "";
    switch (storyIndex) {
      case 1:
        return "Oliver";
      case 2:
        return "Liam";
      case 3:
        return "Benjamin";
      case 4:
        return "James";
      case 5:
        return "Alexander";
      case 6:
        return "John";
      case 7:
        return "Ava";
      case 8:
        return "Emma";
      case 9:
        return "Ava";
      case 10:
        return "Lili";
    }
    return label;
  }

  Widget getPost(
      {required String name, required String avatar, required String image}) {
    return Container(
      margin: const EdgeInsets.only(top: 5),
      height: 296,
      child: Column(
        crossAxisAlignment: CrossAxisAlignment.start,
        children: [
          Row(
            mainAxisAlignment: MainAxisAlignment.spaceBetween,
            children: [
              Row(
                children: [
                  Container(
                    margin: const EdgeInsets.fromLTRB(10, 0, 10, 0),
                    height: 35,
                    width: 35,
                    clipBehavior: Clip.antiAlias,
                    decoration: const BoxDecoration(
                        borderRadius: BorderRadius.all(Radius.circular(35))),
                    child: Image.asset(avatar),
                  ),
                  Text(
                    name,
                    style: const TextStyle(
                      fontWeight: FontWeight.bold,
                    ),
                  ),
                ],
              ),
              IconButton(
                splashRadius: 20,
                iconSize: 20,
                onPressed: () {},
                icon: const Icon(CupertinoIcons.ellipsis_vertical),
              ),
            ],
          ),
          Container(
            margin: const EdgeInsets.only(top: 0),
            height: 200,
            decoration: BoxDecoration(
              color: Colors.black.withAlpha(40),
              image: DecorationImage(
                  image: AssetImage(image),
                  fit: BoxFit.cover,
                  alignment: Alignment.topLeft),
            ),
          ),
          Row(
            mainAxisAlignment: MainAxisAlignment.spaceBetween,
            children: [
              Container(
                margin: const EdgeInsets.fromLTRB(10, 10, 10, 0),
                decoration: const BoxDecoration(
                    borderRadius: BorderRadius.all(Radius.circular(7))),
                child: const Text(
                  "Description...",
                  style: TextStyle(
                    color: Color(0xff777777),
                  ),
                ),
              ),
              Row(
                children: [
                  IconButton(
                    splashRadius: 20,
                    onPressed: () {},
                    icon: const Icon(CupertinoIcons.heart),
                  ),
                  IconButton(
                    splashRadius: 20,
                    onPressed: () {},
                    iconSize: 23,
                    icon: const Icon(CupertinoIcons.chat_bubble),
                  ),
                  IconButton(
                    splashRadius: 20,
                    onPressed: () {},
                    iconSize: 21,
                    icon: const Icon(Icons.share_outlined),
                  ),
                ],
              ),
            ],
          ),
        ],
      ),
    );
  }
}

class StoryModel {
  StoryModel({
    this.id,
    this.avatar,
    this.label,
    this.cards,
  });

  int? id;
  Widget? avatar;
  Text? label;
  List<StoryCardModel>? cards;
}

class StoryCardModel {
  StoryCardModel({
    this.visited = false,
    this.duration = const Duration(seconds: 2),
    this.color = const Color(0xff333333),
    this.childOverlay,
    this.child,
  });

  bool visited;
  Duration duration;
  Color color;
  Widget? childOverlay;
  Widget? child;
}
23
likes
150
points
297
downloads
screenshot

Publisher

verified publishershirvanie.ir

Weekly Downloads

A flutter plugin for creating story (like Instagram, WhatsApp, Telegram, Facebook, etc) for all platforms.

Homepage
Repository (GitHub)
View/report issues

Documentation

API reference

License

MIT (license)

Dependencies

flutter

More

Packages that depend on flutter_story