カウンターアプリから新しいアプリを作る(初心者向け)続き

image

この記事は、【 可茂IT塾 Advent Calendar 2021 】の20日目の記事です。

Version

  • Flutter 2.6.0
  • Dart 2.14.4

カウンターアプリをもとに新しいアプリを作る(続き)

前回、カウンターアプリを使って組み込み型を操作するアプリを作りました。 それをもとに少し修正してみたいと思います。内容はFloatingActionButtonを押すたびにListとMapで表示されていた要素とkey、valueが一つずつ順番に色とフォントを変えて表示されます。

コード修正

修正前

// List部分
Text(_string,
     style: TextStyle(color: Colors.blue),
    ),
// Map部分
Text(_japaneseMlbPlayers.toString(),
     style: TextStyle(color: Colors.blue),
    ),

これをRichTextを使ってフォントの色とサイズを変えようと思い、以下のようなコードに修正しました。

修正(List部分)

// 途中省略
showListElement(_splitString, _fiveCounter),
// 途中省略
Widget showListElement(List list, int counter) {
  return RichText(
    text: TextSpan(
      style: TextStyle(color: Colors.black),
      children: [
        listElementTextSpan(list, 0, counter),
        listElementTextSpan(list, 1, counter),
        listElementTextSpan(list, 2, counter),
        listElementTextSpan(list, 3, counter),
        listElementTextSpan(list, 4, counter),
      ],
    ),
  );
}

listElementTextSpan(List list, int element, int counter) {
  return TextSpan(
    text: '${list[element]}',
    style:
        counter == element ? TextStyle(color: Colors.blue, fontSize: 24) : null,
        // カウンターの数字とelementの数字が同じときにTextStyleを変更する
  );
}

修正(Map部分)

// 途中省略
showMapElement(_japaneseMlbPlayers, _fiveCounter),
// 途中省略
Widget showMapElement(Map map, int counter) {
  return RichText(
    text: TextSpan(
      style: TextStyle(color: Colors.black),
      children: [
        mapKeyElementTextSpan(map, 0, counter),
        mapValueElementTextSpan(map, 0, counter),
        mapKeyElementTextSpan(map, 1, counter),
        mapValueElementTextSpan(map, 1, counter),
        mapKeyElementTextSpan(map, 2, counter),
        mapValueElementTextSpan(map, 2, counter),
        mapKeyElementTextSpan(map, 3, counter),
        mapValueElementTextSpan(map, 3, counter),
        mapKeyElementTextSpan(map, 4, counter),
        mapValueElementTextSpan(map, 4, counter),
      ],
    ),
  );
}

mapKeyElementTextSpan(Map map, int element, int counter) {
  return TextSpan(
    text: '${map.keys.elementAt(element)} ',
    style: counter == element
        ? TextStyle(color: Colors.orange, fontSize: 24)
        : null,
    // keyのelementとカウンターの数字が一致したときにTextStyleを変更する
  );
}

mapValueElementTextSpan(Map map, int element, int counter) {
  return TextSpan(
    text: '${map.values.elementAt(element)} ',
    style: counter == element
        ? TextStyle(color: Colors.green, fontSize: 24)
        : null,
    // valueのelementとカウンターの数字が一致したときにTextStyleを変更する
  );
}


これで、カウンターの数字が変化するごとにListとMapの文字の色と大きさが変化するアプリが完成しました。

コード全文

コード全文は以下の通りです。

import 'package:flutter/material.dart';

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

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

class MyHomePage extends StatefulWidget {
  MyHomePage({Key? key, required this.title}) : super(key: key);

  final String title;

  
  State<StatefulWidget> createState() => _MyHomePageState();
}

class _MyHomePageState extends State<MyHomePage> {
  int _counter = 0;
  String _string = 'イヌ サル キジ タヌキ キツネ';
  Map<String, String> _japaneseMlbPlayers = {
    'Shohei Ohtani': 'Los Angeles Angels',
    'Yu Darvish': 'San Diego Padres',
    'Kenta Maeda': 'Minnesota Twins',
    'Shogo Akiyama': 'Cincinnati Reds',
    'Yoshi Tsutsugo': 'Pittsburgh Pirates',
  };
  bool _isChecked = false;

  void _incrementCounter() {
    setState(() {
      _counter++;
      _isChecked = !_isChecked;
    });
  }

  
  Widget build(BuildContext context) {
    int _fiveCounter = (_counter + 5) % 5;
    List _splitString = _string.split(' ');

    return Scaffold(
      appBar: AppBar(
        title: Text(widget.title),
      ),
      body: Center(
        child: Column(
          mainAxisAlignment: MainAxisAlignment.start,
          crossAxisAlignment: CrossAxisAlignment.center,
          children: <Widget>[
            SizedBox(
              height: 20,
            ),
            showListElement(_splitString, _fiveCounter),
            SizedBox(
              height: 20,
            ),
            showMapElement(_japaneseMlbPlayers, _fiveCounter),
          ],
        ),
      ),
      floatingActionButton: FloatingActionButton(
        onPressed: _incrementCounter,
        tooltip: 'Increment',
        child: Icon(Icons.add),
      ),
    );
  }
}

Widget showListElement(List list, int counter) {
  return RichText(
    text: TextSpan(
      style: TextStyle(color: Colors.black),
      children: [
        listElementTextSpan(list, 0, counter),
        listElementTextSpan(list, 1, counter),
        listElementTextSpan(list, 2, counter),
        listElementTextSpan(list, 3, counter),
        listElementTextSpan(list, 4, counter),
      ],
    ),
  );
}

listElementTextSpan(List list, int element, int counter) {
  return TextSpan(
    text: '${list[element]}',
    style:
        counter == element ? TextStyle(color: Colors.blue, fontSize: 24) : null,
  );
}

Widget showMapElement(Map map, int counter) {
  return RichText(
    text: TextSpan(
      style: TextStyle(color: Colors.black),
      children: [
        mapKeyElementTextSpan(map, 0, counter),
        mapValueElementTextSpan(map, 0, counter),
        mapKeyElementTextSpan(map, 1, counter),
        mapValueElementTextSpan(map, 1, counter),
        mapKeyElementTextSpan(map, 2, counter),
        mapValueElementTextSpan(map, 2, counter),
        mapKeyElementTextSpan(map, 3, counter),
        mapValueElementTextSpan(map, 3, counter),
        mapKeyElementTextSpan(map, 4, counter),
        mapValueElementTextSpan(map, 4, counter),
      ],
    ),
  );
}

mapKeyElementTextSpan(Map map, int element, int counter) {
  return TextSpan(
    text: '${map.keys.elementAt(element)} ',
    style: counter == element
        ? TextStyle(color: Colors.orange, fontSize: 24)
        : null,
  );
}

mapValueElementTextSpan(Map map, int element, int counter) {
  return TextSpan(
    text: '${map.values.elementAt(element)} ',
    style: counter == element
        ? TextStyle(color: Colors.green, fontSize: 24)
        : null,
  );
}

まとめ

最初は、RichText内のTextSpanを一つ一つ定義していたので、長いコードになってしまいました。ちょっと外に出すとコードが簡潔になりますね。 もっと効率のよい書き方もあるような気がしますが、少しずつ覚えていきたいと思います。

お知らせ

可茂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