top of page
  • AutorenbildFlutter Akademie

Taschenrechner in Flutter



Wer Flutter Anfänger ist und erst einmal mit den Basics von Flutter anfangen will, kann mit einem Flutter Taschenrechner beginnen. Dieses Flutter Tutorial zeigt euch ganz einfach Schritt für Schritt wie ihr eure eigenen Flutter Taschenrechner bauen könnt.


Wenn wir fertig sind, wird der Taschenrechner so aussehen:


1. Das Projekt erstellen


Wie ihr ein Flutter Projekt erstellt, erfahrt ihr in diesem Artikel: Flutter First Steps.


2. Das UI erstellen


Zuerst löschen wir alles, was in body enthalten ist und ersetzen es durch das Layout unseres Taschenrechners.



 @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text(widget.title),
      ),
      body: Container(
        child: Column(
          children: <Widget>[
            Container(
              width: double.infinity,
              alignment: Alignment.centerRight,
              color: Colors.white,
              padding: EdgeInsets.symmetric(vertical: 24.0,     
                horizontal: 12.0),
              child: Text(
                0,
                overflow: TextOverflow.clip,
                maxLines: 1,
                style: TextStyle(
                  fontSize: 40.0,
                  fontWeight: FontWeight.bold,
                ),
              ),
            ),
            Column(
              children: <Widget>[
                Row(
                  children: <Widget>[
                    counterButton("7"),
                    counterButton("8"),
                    counterButton("9"),
                    counterButton(operands["div"]),
                  ],
                ),
                Row(
                  children: <Widget>[
                    counterButton("4"),
                    counterButton("5"),
                    counterButton("6"),
                    counterButton(operands["mul"]),
                  ],
                ),
                Row(
                  children: <Widget>[
                    counterButton("1"),
                    counterButton("2"),
                    counterButton("3"),
                    counterButton(operands["sub"]),
                  ],
                ),
                Row(
                  children: <Widget>[
                    counterButton("0"),
                    counterButton("Clear"),
                    counterButton(operands["equals"]),
                    counterButton(operands["add"]),
                  ],
                ),
              ],
            ),
          ],
        ),
      ),
    );
  }

In der ersten Column ist das Feld für die Anzeige unseres Ergebnisses. Danach folgen die Rows für die Zahlen und Befehle. Ich habe das Styling für die einzelnen Buttons, für eine bessere Übersicht, ausgelagert. Hierfür habe ich einfach ein Widget counterButton deklariert und es in die Rows eingefügt.



Widget counterButton(var buttonText) {
    return Expanded(
      child: OutlinedButton(
        onPressed: () => {},
        child: Text(buttonText,
            style: TextStyle(
                fontSize: 20.0,
                fontWeight: FontWeight.bold,
                color: Colors.black)),
        style: ButtonStyle(
          padding: MaterialStateProperty.all(EdgeInsets.all(24.0)),
          backgroundColor: MaterialStateProperty.all<Color> .        (Colors.white),
          textStyle: MaterialStateProperty.all(
            TextStyle(
              //backgroundColor: Colors.yellow,
              color: Colors.black,
            ),
          ),
        ),
      ),
    );
  }
    

3. Funktionen zu Buttons hinzufügen


Jetzt sind die Buttons zwar da, wenn man sie klickt, passiert jedoch nichts. Das werden wir jetzt ändern.


Variable Output


Erstellt zuerst einmal die Variable output in der _MyHomePageState State-Klasse (die sich einen State merkt) und ersetzt im Container der Ergebnisanzeige Text("0", ...) durch Text(output, ...). Nun wird anstatt einem festgeschriebenen Text der Inhalt einer Variable im Text-Widget ausgegeben.



class _MyHomePageState extends State<MyHomePage> {
  var output = "0";
  ...
}
  

Container(
  width: double.infinity,
  alignment: Alignment.centerRight,
  color: Colors.white,
  padding: EdgeInsets.symmetric(vertical: 24.0, horizontal: 12.0),
  child: Text(
    output,
    overflow: TextOverflow.clip,
    maxLines: 1,
    style: TextStyle(
      fontSize: 40.0,
      fontWeight: FontWeight.bold,
    ),
  ),
),
  


onPressed Event


Jetzt ist es Zeit dem onPressed Event der Buttons eine Funktion anzufügen, die jedes Mal ausgeführt wird, wenn der Button gedrückt wird. Wir nennen die Funktion einfach buttonPressed, welche die Value buttonText hat.



Widget counterButton(String buttonText) {
    return Expanded(
      child: new OutlineButton(
          onPressed: () => buttonPressed(buttonText),
          child: new Text(buttonText,
              style: TextStyle(fontSize: 20.0, 
                fontWeight: FontWeight.bold)),
          color: Colors.white,
          textColor: Colors.black,
          padding: EdgeInsets.all(24.0)
       ),
    );
  }


Dann deklarieren wir die Funktion in der _MyHomePageState State-Klasse wie folgt:



buttonPressed(String buttonText) {}

Bevor wir die Funktion mit ifs und elses füllen, müssen wir noch ein paar Variablen deklarieren.


  valueOfOperation = 0.0;
  firstValue = 0.0;
  secondValue = 0.0;
  operand = "";

valueOfOperation entspricht der Anzeige. firstValue und secondValue sind die beiden eingegebenen Werte. Der operand ist der mathematische Operator, also +,-,* oder /.


Kalkulation


Jetzt können wir der buttonPressed Funktion sagen, was sie mit den eingegebenen Werten tun soll.



buttonPressed(String buttonText) {
    if (buttonText == "Clear") {
      valueOfOperation = 0.0;
      firstValue = 0.0;
      secondValue = 0.0;
      operand = "";
      //Wenn gleich gedrückt wird
    } else if (buttonText == operands["equals"]) {
      secondValue = double.parse(output);
      if (operand == operands["add"]) {
        valueOfOperation = firstValue + secondValue;
      } else if (operand == operands["sub"]) {
        valueOfOperation = firstValue - secondValue;
      } else if (operand == operands["mul"]) {
        valueOfOperation = firstValue * secondValue;
      } else if (operand == operands["div"]) {
        valueOfOperation = firstValue / secondValue;
      }
      firstValue = 0.0;
      secondValue = 0.0;
      operand = "";
      //Wenn ein anderer Operand gedrückt wird
    } else if (operands.containsValue(buttonText)) {
      firstValue = double.parse(output);
      operand = buttonText;
      valueOfOperation = 0.0;
      //Wenn eine Zahl gedrückt wird
    } else {
      if (valueOfOperation == 0.0) {
        valueOfOperation = double.parse(buttonText);
      } else {
        var zahlensplit = valueOfOperation.toString().split(".");
        var ganzzahl = zahlensplit[0];
        var kommazahl = zahlensplit[1];
        valueOfOperation =            double.parse(ganzzahl + 
          buttonText + "." + kommazahl);
      }
    }
      

Bei der Eingabe von Clear wird keine Berechnung ausgeführt, sondern die Werte einfach auf 0 zurückgesetzt.


Wenn der buttonText +,-,* oder / ist, legen wir fest, dass der erste Output, der ja ein String ist, zu einem Double geparset wird und in firstValue gespeichert soll. Das gleiche soll für den zweiten Output geschehen.


Wenn der buttonText “=” entspricht, dann wird die Berechnung für das jeweilige Zeichen ausgeführt und das Ergebnis wieder in ein String umgewandelt.


Dann resetten wir noch den Wert von firstValue, secondValue und valueOfOperation zu 0.


Dann fügen wir unserem valueOfOperation noch den buttonText hinzu, damit dieser angezeigt wird, wenn keine Kalkulation durchgeführt wird.


Ganz zum Schluss geben wir, über setState, noch den Wert von valueOfOperation an output, damit er ausgegeben wird.



setState(
  () {
    var valueOfOperationString = valueOfOperation.toString();
    var valueOfOperationSplit = valueOfOperationString.split(".");
    if (valueOfOperationSplit[1] == "0") {
      output = valueOfOperationSplit[0];
    } else {
      output = valueOfOperationString;
    }
  },
);
  


Der komplette Code in der Main.dart



import 'package:flutter/material.dart';
void main() {
  runApp(MyApp());
}
class MyApp extends StatelessWidget {
  @override
  Widget build(BuildContext context) {
    return MaterialApp(
      title: 'Flutter Demo',
      theme: ThemeData(
        primarySwatch: Colors.blue,
      ),
      home: const MyHomePage(title: 'Taschenrechner'),
    );
  }
}
class MyHomePage extends StatefulWidget {
  const MyHomePage({Key? key, required this.title}) : super(key: key);
  final String title;
  @override
  State<MyHomePage> createState() => _MyHomePageState();
}
class _MyHomePageState extends State<MyHomePage> {
  Map<String, String> operands = {
    "add": "+",
    "sub": "-",
    "mul": "*",
    "div": "/",
    "equals": "=",
  };
  var output = "0";
  var valueOfOperation = 0.0;
  var firstValue = 0.0;
  var secondValue = 0.0;
  var operand = "";
  buttonPressed(String buttonText) {
    if (buttonText == "Clear") {
      valueOfOperation = 0.0;
      firstValue = 0.0;
      secondValue = 0.0;
      operand = "";
      //Wenn gleich gedrückt wird
    } else if (buttonText == operands["equals"]) {
      secondValue = double.parse(output);
      if (operand == operands["add"]) {
        valueOfOperation = firstValue + secondValue;
      } else if (operand == operands["sub"]) {
        valueOfOperation = firstValue - secondValue;
      } else if (operand == operands["mul"]) {
        valueOfOperation = firstValue * secondValue;
      } else if (operand == operands["div"]) {
        valueOfOperation = firstValue / secondValue;
      }
      firstValue = 0.0;
      secondValue = 0.0;
      operand = "";
      //Wenn ein anderer Operand gedrückt wird
    } else if (operands.containsValue(buttonText)) {
      firstValue = double.parse(output);
      operand = buttonText;
      valueOfOperation = 0.0;
      //Wenn eine Zahl gedrückt wird
    } else {
      if (valueOfOperation == 0.0) {
        valueOfOperation = double.parse(buttonText);
      } else {
        var zahlensplit = valueOfOperation.toString().split(".");
        var ganzzahl = zahlensplit[0];
        var kommazahl = zahlensplit[1];
        valueOfOperation =            double.parse(ganzzahl + buttonText + "." + kommazahl);
      }
    }
    setState(
      () {
        var valueOfOperationString = valueOfOperation.toString();
        var valueOfOperationSplit = valueOfOperationString.split(".");
        if (valueOfOperationSplit[1] == "0") {
          output = valueOfOperationSplit[0];
        } else {
          output = valueOfOperationString;
        }
      },
    );
  }
  Widget counterButton(var buttonText) {
    return Expanded(
      child: OutlinedButton(
        onPressed: () => buttonPressed(buttonText),
        child: Text(buttonText,
            style: const TextStyle(
                fontSize: 20.0,
                fontWeight: FontWeight.bold,
                color: Colors.black)),
        style: ButtonStyle(
          padding: MaterialStateProperty.all(
        const EdgeInsets.all(24.0)),
          backgroundColor: MaterialStateProperty.all<Color>(Colors.white),
          textStyle: MaterialStateProperty.all(
            const TextStyle(
              //backgroundColor: Colors.yellow,
              color: Colors.black,
            ),
          ),
        ),
      ),
    );
  }
  @override
  Widget build(BuildContext context) {
    return Scaffold(
      appBar: AppBar(
        title: Text(widget.title),
      ),
      body: Container(
        child: Column(
          children: <Widget>[
            Container(
              width: double.infinity,
              alignment: Alignment.centerRight,
              color: Colors.white,
              padding: const EdgeInsets.symmetric(vertical: 24.0, horizontal: 12.0),
              child: Text(
                output,
                overflow: TextOverflow.clip,
                maxLines: 1,
                style: const TextStyle(
                  fontSize: 40.0,
                  fontWeight: FontWeight.bold,
                ),
              ),
            ),
            Column(
              children: <Widget>[
                Row(
                  children: <Widget>[
                    counterButton("7"),
                    counterButton("8"),
                    counterButton("9"),
                    counterButton(operands["div"]),
                  ],
                ),
                Row(
                  children: <Widget>[
                    counterButton("4"),
                    counterButton("5"),
                    counterButton("6"),
                    counterButton(operands["mul"]),
                  ],
                ),
                Row(
                  children: <Widget>[
                    counterButton("1"),
                    counterButton("2"),
                    counterButton("3"),
                    counterButton(operands["sub"]),
                  ],
                ),
                Row(
                  children: <Widget>[
                    counterButton("0"),
                    counterButton("Clear"),
                    counterButton(operands["equals"]),
                    counterButton(operands["add"]),
                  ],
                ),
              ],
            ),
          ],
        ),
      ),
    );
  }
}


Was haben wir gelernt?


Es ist nicht schwer leichte Anwendungen in Flutter auszuführen. Der nächste Schritt in der Lernkurve ist Werte in einem Formular einzugeben und in einer lokalen Datenbank zu speichern. Probiert es aus!

bottom of page