Flutterでconfettiパッケージを使って紙吹雪を出す方法

image

Fultterで紙吹雪を出す方法を紹介します。

confettiDemo confettiDemo

Version

・Flutter 2.2.0
・Dart 2.13.0
・confetti 0.6.0-nullsafety

紙吹雪の出し方

confettiパッケージのインストール

まずconfettiをインストールします。pubspec.yamlに以下を追記し、次いでPub getをクリックします。

confetti: ^0.6.0-nullsafetys

そして紙吹雪を出したいファイルで以下を宣言します。

import 'package:confetti/confetti.dart';

コントローラを宣言

以下のようにコントローラを宣言します。

final _controller =
      ConfettiController(duration: const Duration(milliseconds: 500));

durationは紙吹雪を何秒出すかの設定です。例えばここをseconds: 10にしたら紙吹雪が10秒間出続けます。

ConfettiWidget()にコントローラを設定

ConfettiWidget()が紙吹雪を出すウィジェットになります。これをbuild関数の中に追加し、confettiControllerプロパティに先ほど宣言した_controllerを設定します。

ConfettiWidget(
              confettiController: _controller,
            )

紙吹雪を発動させるタイミングを設定

この1行が走れば紙吹雪が舞います。

_controller.play();

紙吹雪のカスタマイズ

ConfettiWidgetにはさまざまなプロパティがあります。お好みで設定して使いましょう。

噴出の仕方

blastDirectionalityプロパティにexplosivedirectionalのどちらかを設定します。
explosiveは爆発するように中央からランダムな方向に紙吹雪が噴出します。なので方向を指定する必要がありません。
directionalは一定方向に噴出します。こちらはblastDirectionで方向を指定する必要があります。

ConfettiWidget(
      confettiController: _controller,
      blastDirectionality: BlastDirectionality.explosive,
    );

噴出方向

blastDirectionプロパティに方向を設定します。
piはπのことです。0(2π)で右、π/2で下、πで左、3π/2で上。なぜか数学でやった単位円と逆まわりですね。

import 'dart:math';

ConfettiWidget(
      confettiController: _controller,
      blastDirectionality: BlastDirectionality.directional,
      blastDirection: pi / 2,
    );

放出頻度

emissionFrequencyプロパティで紙吹雪が放出される頻度を設定します。
0〜1の値にする必要があり、値が大きいほど紙吹雪が放出される頻度が増えます。

ConfettiWidget(
      confettiController: _controller,
      emissionFrequency: 1,
    );

一度に発射する紙の数

numberOfParticlesプロパティで一度に発射する紙の数を設定します。
デフォルトは10です。

ConfettiWidget(
      confettiController: _controller,
      numberOfParticles: 30,
    );

繰り返し

shouldLoopプロパティで繰り返すかどうかの設定をします。
デフォルトはfalseです。

ConfettiWidget(
      confettiController: _controller,
      shouldLoop: false,
    );

噴出初速の最大最小

maxBlastForceminBlastForceに噴出初速の最大値と最小値を設定します。
デフォルトは最大20、最小5です。

ConfettiWidget(
      confettiController: _controller,
      maxBlastForce: 5,
      minBlastForce: 2,
    );

発射位置の印の表示非表示

displayTargetをtrueにすることで紙吹雪の発射位置に印がついてわかりやすくなります。
デフォルトはfalseです。

ConfettiWidget(
      confettiController: _controller,
      displayTarget: true,
    );

colorsプロパティに色を設定できます。設定しなかった場合色はランダムになります。

ConfettiWidget(
      confettiController: _controller,
      colors: [Colors.green, Colors.pink,Colors.orange],
    );

サイズ

maximumSizeプロパティで紙吹雪のサイズの最大値を、minimumSizeプロパティで紙吹雪のサイズの最小値を設定します。
たとえば、minimumSizeをSize(10,10)、maximumSizeをSize(50,50)に設定すると、最小サイズ(10、10)と最大サイズ(50、50)[幅、高さ]の間のサイズの紙吹雪が作成されます。

ConfettiWidget(
      confettiController: _controller,
      maximumSize: Size(40, 40),
      minimumSize: Size(20, 20),
    );

紙の落ちる速さ

gravityプロパティに紙の落ちる速さを設定します。
0〜1の値にする必要があり、値が大きいほど速く落ちます。デフォルトは0.1です。

ConfettiWidget(
      confettiController: _controller,
      gravity: 0.3,
    );

抗力

particleDragプロパティに抗力を設定します。
抗力は空気抵抗みたいなイメージです。0〜1の値にする必要があります。値が大きいほど、散らばらずゆっくりと落ちていきます。デフォルトは0.05です。

ConfettiWidget(
      confettiController: _controller,
      particleDrag: 0.001,
    );

描画領域

canvasプロパティで紙吹雪が表示される領域のサイズを設定します。
デフォルトでは、フルスクリーンサイズに設定されています。

ConfettiWidget(
      confettiController: _controller,
      canvas: Size(100, 100),
    );

紙吹雪の形

createParticlePathプロパティに紙吹雪の形を設定できます。
デフォルトは長方形です。

ConfettiWidget(
      confettiController: _controller,
      createParticlePath: drawStar,
    );

カスタムcreateParticlePathの例

Path drawStar(Size size) {
    // Method to convert degree to radians
    double degToRad(double deg) => deg * (pi / 180.0);

    const numberOfPoints = 5;
    final halfWidth = size.width / 2;
    final externalRadius = halfWidth;
    final internalRadius = halfWidth / 2.5;
    final degreesPerStep = degToRad(360 / numberOfPoints);
    final halfDegreesPerStep = degreesPerStep / 2;
    final path = Path();
    final fullAngle = degToRad(360);
    path.moveTo(size.width, halfWidth);

    for (double step = 0; step < fullAngle; step += degreesPerStep) {
      path.lineTo(halfWidth + externalRadius * cos(step),
          halfWidth + externalRadius * sin(step));
      path.lineTo(halfWidth + internalRadius * cos(step + halfDegreesPerStep),
          halfWidth + internalRadius * sin(step + halfDegreesPerStep));
    }
    path.close();
    return path;
  }

コード全文

参考までにサンプルアプリのコードを載せておきます。

import 'package:flutter/material.dart';
import 'package:confetti/confetti.dart';
import 'dart:math';

void main() {
  runApp(MyApp());
}

class MyApp extends StatelessWidget {
  
  Widget build(BuildContext context) {
    return MaterialApp(
      theme: ThemeData(
        primarySwatch: Colors.blue,
      ),
      home: MyHomePage(),
    );
  }
}

class MyHomePage extends StatefulWidget {
  
  _MyHomePageState createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {
  final _controller =
      ConfettiController(duration: const Duration(milliseconds: 500));

  
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text('紙吹雪サンプル'),
      ),
      body: Stack(
        children: [
          Center(
            child: ElevatedButton(
              child: Text('紙吹雪舞わせる'),
              onPressed: () {
                _controller.play();
              },
            ),
          ),
          Container(
            padding: EdgeInsets.all(30),
            alignment: Alignment.topCenter,
            child: _buildConfetti(),
          ),
        ],
      ),
    );
  }

  Widget _buildConfetti() {
    return ConfettiWidget(
      confettiController: _controller,
      blastDirectionality: BlastDirectionality.explosive,
      // blastDirection: pi / 2,
      emissionFrequency: 1,
      numberOfParticles: 30,
      shouldLoop: false,
      maxBlastForce: 5,
      minBlastForce: 2,
      displayTarget: true,
      colors: [Colors.green, Colors.pink, Colors.orange],
      maximumSize: Size(40, 40),
      minimumSize: Size(20, 20),
      gravity: 0.3,
      particleDrag: 0.001,
      // canvas: Size(100, 100),
      createParticlePath: drawStar,
    );
  }

  Path drawStar(Size size) {
    // Method to convert degree to radians
    double degToRad(double deg) => deg * (pi / 180.0);

    const numberOfPoints = 5;
    final halfWidth = size.width / 2;
    final externalRadius = halfWidth;
    final internalRadius = halfWidth / 2.5;
    final degreesPerStep = degToRad(360 / numberOfPoints);
    final halfDegreesPerStep = degreesPerStep / 2;
    final path = Path();
    final fullAngle = degToRad(360);
    path.moveTo(size.width, halfWidth);

    for (double step = 0; step < fullAngle; step += degreesPerStep) {
      path.lineTo(halfWidth + externalRadius * cos(step),
          halfWidth + externalRadius * sin(step));
      path.lineTo(halfWidth + internalRadius * cos(step + halfDegreesPerStep),
          halfWidth + internalRadius * sin(step + halfDegreesPerStep));
    }
    path.close();
    return path;
  }
}

参考

https://pub.dev/packages/confetti/versions/0.6.0-nullsafety

お知らせ

可茂IT塾ではFlutter/Reactのインターンを募集しています!

可茂IT塾ではFlutter/Reactのインターンを募集しています!

可茂IT塾ではFlutter/Reactのインターンを募集しています!可茂IT塾のエンジニアの判断で、一定以上のスキルをを習得した方には有給でのインターンも受け入れています。

Read More
U30可茂ITインターンハッカソン

U30可茂ITインターンハッカソン

12月28,29日開催。2日間でアプリ開発の企画から完成までを目指す!U30可茂ITインターンハッカソンを開催します。

Read More

タグ

Flutter (118)初心者向け (29)イベント (18)Google Apps Script (16)Nextjs (12)可茂IT塾 (10)React (8)Firebase (7)riverpod (6)ChatGPT (5)vscode (5)デザイン (5)新卒 (4)就活 (4)Figma (4)Dart (4)JavaScript (4)お知らせ (4)FlutterWeb (3)Prisma (3)NestJS (3)Slack (3)TypeScript (3)ワーケーション (3)インターン (3)設計 (2)線型計画法 (2)事例 (2)Git (2)Image (2)File (2)Material Design (2)経験談 (2)画像 (2)iOS (2)アプリ開発 (2)React Hooks (2)tailwindcss (2)社会人 (2)大学生 (2)RSS (1)Google (1)Web (1)CodeRunner (1)個人開発 (1)Android (1)Unity (1)WebView (1)Twitter (1)フルリモート (1)TextScaler (1)textScaleFactor (1)学生向け (1)supabase (1)Java (1)Spring Boot (1)shell script (1)正規表現 (1)table (1)テーブル (1)hooks (1)react (1)パワーポイント (1)趣味 (1)モンスターボール (1)CSS (1)SCSS (1)Cupertino (1)ListView (1)就活浪人 (1)既卒 (1)保守性 (1)iPad (1)シェアハウス (1)スクレイピング (1)PageView (1)画面遷移 (1)flutter_hooks (1)Gmail (1)GoogleWorkspace (1)ShaderMask (1)google map (1)Google Places API (1)GCPコンソール (1)Google_ML_Kit (1)Vercel (1)Google Domains (1)DeepLeaning (1)深層学習 (1)Google Colab (1)コード生成 (1)GitHub Copilot (1)オンラインオフィス (1)javascript (1)css (1)html (1)オブジェクト指向 (1)クラスの継承 (1)ポリモーフィズム (1)LINE (1)Bitcoin (1)bitFlyer (1)コミュニティー (1)文系エンジニア (1)build_runner (1)freezed (1)Freezed (1)ヒーター (1)作業効率 (1) (1)Flutter実践開発 (1) (1)permission_handler (1)flutter_local_notifications (1)markdown (1)GlobalKey (1)ValueKey (1)Key (1)アイコン (1)go_router (1)FireStorage (1)debug (1)datetime_picker (1)Apple Store Connect (1)FlutterGen (1)デバッグ (1)Widget Inspector (1)VRChat (1)API (1)検索機能 (1)Shader (1)Navigator (1)メール送信 (1)FlutterFlow (1)Firebase App Distribution (1)Fastlane (1)Dio (1)CustomClipper (1)ClipPath (1)カスタム認証 (1)アニメーション (1)Arduino (1)ESP32 (1)フリーランス (1)会社員 (1)mac (1)csv (1)docker (1)GithubActions (1)Dialog (1)BI (1)LifeHack (1)ショートカット (1)Chrome (1)高校生 (1)キャリア教育 (1)非同期処理 (1)生体認証 (1)BackdropFilter (1)レビュー (1)getAuth (1)クローズドテスト (1)PlayConsole (1)Algolia (1)コンサルティング (1)Symbol (1)

お知らせ

可茂IT塾ではFlutter/Reactのインターンを募集しています!

可茂IT塾ではFlutter/Reactのインターンを募集しています!

可茂IT塾ではFlutter/Reactのインターンを募集しています!可茂IT塾のエンジニアの判断で、一定以上のスキルをを習得した方には有給でのインターンも受け入れています。

Read More
U30可茂ITインターンハッカソン

U30可茂ITインターンハッカソン

12月28,29日開催。2日間でアプリ開発の企画から完成までを目指す!U30可茂ITインターンハッカソンを開催します。

Read More