キャンバスを利用して好きな図形を描画できる CustomPaint の利用

ここでは Flutter のウィジェットで、キャンバス (canvas) を使うための CustomPaint ウィジェットについて説明します。 CustomPaint を使うと canvas のドローコマンドが使えます。

ここでは半径を渡して円を描くだけのカスタムペイント・ウィジェットを作成します。

そのウィジェットに対して、Tween アニメーションを利用して異なる半径を渡し、再描画することで円のアニメーションを実現します。

import 'package:flutter/material.dart';
import 'package:flutter/animation.dart';

void main() => runApp(
      MaterialApp(
        debugShowCheckedModeBanner: false,
        home: MyApp(),
      ),
    );

class MyApp extends StatefulWidget {
  @override
  State<StatefulWidget> createState() {
    return _State();
  }
}

class _State extends State<MyApp> with SingleTickerProviderStateMixin {
  Animation<double> _animation;
  AnimationController _animationController;

  @override
  void initState() {
    _animationController =
        AnimationController(duration: const Duration(seconds: 1), vsync: this);
    _animation = Tween(begin: 10.0, end: 100.0).animate(_animationController)
      ..addListener(() {
        setState(() {});
      });

    super.initState();
  }

  @override
  void dispose() {
    _animationController.dispose();
    super.dispose();
  }
  
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text('CustomPaint Test'),
      ),
      body: Center(
        child: Column(
          children: <Widget>[
            Container(
              padding: EdgeInsets.all(20.0),
              child: RaisedButton(
                child: Text('Start/Stop'),
                onPressed: () {
                  if (_animationController.isAnimating) {
                    _animationController.reset();
                  } else {
                    _animationController.repeat();
                  }
                },
              ),
            ),
            Opacity(
              opacity: _animationController.isAnimating ? 1.0 : 0.00,
              child: CustomPaint(
                painter: _CirclePainter(_animation.value),
              ),
            ),
          ],
        ),
      ),
    );
  }
}

class _CirclePainter extends CustomPainter {
  double radius;

  _CirclePainter(this.radius);

  @override
  void paint(Canvas canvas, Size size) {
    var c = Offset(0, 150.0);
    var paint = Paint()
      ..isAntiAlias = true
      ..color = Colors.blue
      ..strokeWidth = 5.0
      ..style = PaintingStyle.stroke;
    canvas.drawCircle(
      c,
      radius,
      paint,
    );
  }

  @override
  bool shouldRepaint(CustomPainter oldDelegate) {
    return true;
  }
}

CustomPaint ウィジェットは、キャンバスに描画する操作を行う CustomPainter と組み合わせて使います。

CustomPainter は2つのメソッドを実装します。ひとつは paint メソッド。もうひとつは shouldRepaint メソッドです。

paint メソッドでは実際にキャンバスに描画します。Canvas と Size オブジェクトが渡されます。shouldRepaint メソッドは再描画が必要かどうかを返します。一度描画したら状態が変わらないものを描画する場合は、再描画する必要がありませんので、常に false を返します。

ここまでお読みいただき、誠にありがとうございます。SNS 等でこの記事をシェアしていただけますと、大変励みになります。どうぞよろしくお願いします。

© 2025 Flutter 入門