DEV Community

Cover image for Integrate the Gemini REST API in Flutter: Unlock Powerful Generative Language Models for Your Next App
Sayed Ali Alkamel
Sayed Ali Alkamel

Posted on

Integrate the Gemini REST API in Flutter: Unlock Powerful Generative Language Models for Your Next App

Imagine integrating powerful AI capabilities directly into your Flutter app. With the Gemini REST API and Flutter’s intuitive framework, this is no longer a dream! This guide empowers Flutter developers with the knowledge to harness AI's potential by seamlessly connecting their apps to Gemini’s advanced functionalities.

The Gemini Advantage:

**The Gemini REST API serves as the gateway to Gemini, Google's next-generation AI model. This API unlocks a wide array of capabilities, including text-based analysis, question answering, and potentially even image recognition (with the Gemini Pro Vision model).

Target Audience:

This guide is tailored for Flutter developers eager to leverage AI in their applications. Whether you’re building a chatbot, a creative writing assistant, or an intelligent search tool, Gemini and Flutter can be your winning combination.

Screenshots:

Image description


Jumpstart Generative AI in Your Flutter Apps with Google AI Studio:

Google AI Studio is your launchpad for integrating Gemini, Google’s next-generation generative AI model, into your Flutter projects. This intuitive browser-based IDE empowers you to rapidly experiment with prompts and tailor Gemini’s responses to match your app’s functionality perfectly. Craft chatbots, generate creative text formats, or translate languages — all within a user-friendly interface designed specifically for developers. Google AI Studio eliminates the need for intricate setup processes, allowing you to focus on building innovative Flutter applications that harness the true potential of generative AI.


Gemini REST API: A Deep Dive:

The Gemini REST API offers a programmatic way to interact with Google’s powerful generative language model from your Flutter app. This API utilizes a POST request to send structured data in JSON format. Here’s a breakdown:

  • Content-Type Header: This header sets the content type of the request body to application/json, indicating JSON-formatted data is following.
  • Request Body: The body contains the prompt you want Gemini to respond to. It’s wrapped in a JSON object with nested structures:
  • "contents": An array containing a single object for the prompt.
  • "parts": Another array containing a single object for the specific prompt text.
  • "text": This key holds the actual prompt string, in this case, "Explain how AI works".
  • API Endpoint: The URL points to the specific Gemini model (gemini-pro) and the desired action (generateContent). Remember to replace YOUR_API_KEY it with your actual Google Cloud Platform API key for authentication. For more information, click here

Testing with Postman (Simple Steps):

  1. Install Postman: Download and install the Postman application for your operating system.
  2. Create a POST Request: In Postman, create a new request and set the method to POST.
  3. Set the URL: Paste the following URL, replacing YOUR_API_KEY with your actual key:
https://generativelanguage.googleapis.com/v1beta/models/gemini-pro:generateContent?key=YOUR_API_KEY
Enter fullscreen mode Exit fullscreen mode
  1. Set the Content-Type Header: In the Headers tab, add a header named “Content-Type” and set its value to “application/json”.

  2. Create the Request Body: In the Body tab, select the “raw” option and set the format to “JSON”. Paste the following code, replacing the prompt text if desired:

{
  "contents": [
    {
      "parts": [
        {
          "text": "Explain how AI works"
        }
      ]
    }
  ]
}
Enter fullscreen mode Exit fullscreen mode
  1. Send the Request: Click the “Send” button. If successful, you’ll receive a response from Gemini containing its generated text in JSON format.

Image description


Flutter UI

In this step-by-step guide, we’ll delve into the code behind a basic chat interface that interacts with the Gemini REST API using Flutter. This interface allows you to send text messages and receive responses generated by Gemini’s powerful language models. As a Flutter tech influencer, understanding how to build such an interaction will empower you to create innovative applications that leverage the potential of generative AI.

1. Scaffolding the Foundation (Scaffold Widget)

The Scaffold widget serves as the fundamental building block of our application's user interface (UI). It provides a pre-built structure for common UI elements like an app bar, body content, and a floating action button (FAB) if needed.

  • We customize the appBar using Theme.of(context).colorScheme.primary to set the background color based on the app's primary theme.
  • The title is dynamically set using widget.title.
  • We leverage Theme.of(context).textTheme.titleLarge to ensure consistent styling based on the app's theme.

2. Creating a Safe Space (SafeArea Widget)

The SafeArea widget ensures that our UI elements are displayed correctly within the viewable area, even on devices with notches or rounded corners.

3. Adding Padding (Padding Widget)

The Padding widget adds a margin around the child widgets, creating a visual separation and enhancing readability. Here, we apply padding of 16.0 on the left, top, and right sides, and 20.0 on the bottom.

4. Organizing Content (Column Widget)

The Column widget arranges its child widgets vertically one below the other. We use mainAxisAlignment: MainAxisAlignment.spaceBetween to distribute the child elements evenly within the available space, creating a balanced layout.

5. Handling Loading States (_buildLoadingIndicator())

This part, indicated by _isLoading ? _buildLoadingIndicator() : Container(), conditionally displays a loading indicator while the application fetches data from the Gemini API. The _buildLoadingIndicator() function would create a visual representation (like a progress bar or spinner), informing the user that something is happening in the background.


6. Displaying Response Messages (Expanded Widget and Text Widget)

  • The Expanded widget ensures that the response message text takes up all available space within the Column.
  • The SingleChildScrollView widget allows the text to scroll if it overflows the allocated space, preventing content from being cut off.
  • The Text widget displays the actual response message received from the Gemini API. The font size is set to 18.0 for better readability.

7. User Input Field (TextField Widget)

  • The TextField widget is where the user types their messages to be sent to Gemini.
  • We create a _chatController to manage the user's input.
  • The decoration property is used to customize the appearance of the text field:
    • hintText: "Type a message" provides a placeholder text displayed when the field is empty.
    • hintStyle: const TextStyle(color: Colors.grey) sets the color of the placeholder text.
    • OutlineInputBorder defines the border style, using a circular radius of 20.0 for a rounded look.
    • borderSide configures the border's color (deep purple) and thickness (2.0).
    • contentPadding sets the padding inside the text field for better user experience (16.0 on all sides).
  • The maxLines property is set to 5 to allow users to enter multi-line messages if needed.

8. Sending Messages (ElevatedButton.icon Widget)

  • The ElevatedButton.icon widget creates a visually appealing button with an icon for sending messages.
  • onPressed: () => _callGemini() defines the action that occurs when the button is pressed. This calls a function _callGemini() that handles sending the user's message to the Gemini API and fetching the response.
  • iconAlignment: IconAlignment.end positions the icon at the right end of the button.
  • We customize the button’s appearance using ElevatedButton.styleFrom:
    • backgroundColor sets the button's background color (deep purple).
    • shape defines a rounded rectangle shape with a borderRadius of 50.0 for a more modern look.

Widget Tree

Scaffold
  -> AppBar (backgroundColor, title)
  -> SafeArea
    -> Padding
      -> Column (mainAxisAlignment)
        -> (Conditional) If _isLoading is true:
          -> _buildLoadingIndicator() (CircularProgressIndicator)
        -> Else:
          -> Expanded
            -> SingleChildScrollView
              -> Text (responseMessage, style)
        -> TextField (controller, decoration, maxLines)
        -> Padding
          -> ElevatedButton.icon (onPressed, iconAlignment, label, icon, style)
Enter fullscreen mode Exit fullscreen mode

Understanding the Code: The _callGemini Function

The _callGemini function serves as the heart of our chat interface's interaction with the Gemini REST API. Let's break down its functionality step-by-step:

1. Updating the Loading State

  1. The function begins by setting the _isLoading state to true using setState.
  2. This triggers a UI rebuild, displaying a loading indicator while fetching data from the API.

2. Constructing the API Request URL

  • A constant string (url) holds the base URL for interacting with the Gemini API, including:
    • The model identifier (gemini-pro)
    • The API endpoint (generateContent)
    • A placeholder for your API key ($GEMINI_API_KEY)
  • The Uri.parse(url) method converts the string into a Uri object, which represents a structured web address.

3. Creating the Request Object

  • An instance of the Request class (defined in a separate model file) is created.
  • This object will hold the data structure required by the Gemini API for your request.

4. Building Request Content (Parts and Contents)

  1. A list named partsList is created, containing a single Parts object.
    • The Parts object has a text field that stores the user’s message, retrieved from the _chatController.
  2. Another list named contentsList is created, containing a single Contents object.
    • The Contents object has a parts field that references the previously created partsList.
  3. The contents field of the Request object is assigned the contentsList.

This nested structure organizes the request data as expected by the Gemini API.

5. Sending the Request

  • The http.post function (from the http package) is used to send a POST request to the uri (the API endpoint).
  • The body argument is set to the JSON-encoded representation of the request object using jsonEncode(request.toJson()).
    • This method converts the Dart object into a JSON string format for sending over the network.

6. Processing the Response

  1. The await keyword ensures the code waits for the API's response before proceeding.
  2. The rawResponse variable holds the raw HTTP response object.
  3. jsonDecode(rawResponse.body) parses the JSON response body into a Dart map structure.
  4. The Response.fromJson constructor (likely defined in a separate model file) takes the decoded JSON map and uses it to create a Response object.
    • This object has properties that map to the response structure defined by the Gemini API.

7. Updating UI and Displaying Response

  1. Another setState call updates the UI state:
    • _isLoading is set to false to hide the loading indicator.
    • responseMessage is updated with the first candidate's text extracted from the Response object.
    • This assumes the Response object has a nested structure with candidates, content, parts, and finally the text field for the actual response message.

Full main.dart code:

import 'dart:convert';
import 'package:flutter/cupertino.dart';
import 'package:flutter/material.dart';
import 'package:http/http.dart' as http;
import 'Model/contents.dart';
import 'Model/parts.dart';
import 'Model/request.dart';
import 'Model/response.dart';
void main() {
runApp(const MyApp());
}
class MyApp extends StatelessWidget {
const MyApp({super.key});
@override
Widget build(BuildContext context) {
return MaterialApp(
debugShowCheckedModeBanner: false,
title: 'Flutter Gemini REST API Demo',
theme: ThemeData(
colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple),
useMaterial3: true,
textTheme: const TextTheme(
bodyMedium: TextStyle(color: Colors.black87),
titleLarge: TextStyle(color: Colors.white),
),
elevatedButtonTheme: ElevatedButtonThemeData(
style: ElevatedButton.styleFrom(
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(50.0),
),
minimumSize: const Size(88, 36),
padding: EdgeInsets.zero,
),
),
),
home: const MyHomePage(title: 'Flutter Gemini REST API Demo'),
);
}
}
class MyHomePage extends StatefulWidget {
const MyHomePage({super.key, required this.title});
final String title;
@override
State<MyHomePage> createState() => _MyHomePageState();
}
class _MyHomePageState extends State<MyHomePage> {
final String GEMINI_API_KEY = "YOUR_API_KEY_HERE"; //TODO: Replace with your actual API key
final TextEditingController _chatController = TextEditingController();
String responseMessage = "Hello 👋, No response yet!";
bool _isLoading = false;
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
backgroundColor: Theme.of(context).colorScheme.primary,
title: Text(widget.title, style: Theme.of(context).textTheme.titleLarge),
),
body: SafeArea(
child: Padding(
padding: const EdgeInsets.fromLTRB(16.0, 16.0, 16.0, 20.0),
child: Column(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
_isLoading ? _buildLoadingIndicator() : Container(),
Expanded(
child: SingleChildScrollView(
child: Text(
responseMessage,
style: const TextStyle(fontSize: 18.0),
),
),
),
TextField(
controller: _chatController,
decoration: InputDecoration(
hintText: "Type a message",
hintStyle: const TextStyle(color: Colors.grey),
border: OutlineInputBorder(
borderRadius: BorderRadius.circular(20.0),
borderSide: const BorderSide(
color: Colors.deepPurple,
width: 2.0,
),
),
contentPadding: const EdgeInsets.all(16.0),
),
maxLines: 5,
),
Padding(
padding: const EdgeInsets.only(top: 16.0),
child: ElevatedButton.icon(
onPressed: () => _callGemini(),
iconAlignment: IconAlignment.end,
label: const Text('Send', style: TextStyle(color: Colors.white)),
icon: const Icon(Icons.send, color: Colors.white),
style: ElevatedButton.styleFrom(
backgroundColor: Colors.deepPurple,
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(50.0),
),
padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 20.0),
),
),
),
],
),
),
),
);
}
Widget _buildLoadingIndicator() {
return const CircularProgressIndicator(color: Colors.deepPurple);
}
_callGemini() async {
setState(() {
_isLoading = true;
});
final url = "https://generativelanguage.googleapis.com/v1/models/gemini-pro:generateContent?key=$GEMINI_API_KEY";
final uri = Uri.parse(url);
Request request = Request();
List<Parts> partsList = [Parts(text: _chatController.text)];
List<Contents> contentsList = [Contents(parts: partsList)];
request.contents = contentsList;
final rawResponse = await http.post(uri, body: jsonEncode(request.toJson()));
Response response = Response.fromJson(jsonDecode(rawResponse.body));
setState(() {
_isLoading = false;
responseMessage = response.candidates!.first.content!.parts!.first.text!;
});
}
}

Further Enhancements

  1. Error Handling for API Calls

    • Implement robust error handling strategies to gracefully manage network issues or unexpected responses.
  2. More User-Friendly Chat Interface

    • Refine the UI/UX design for improved aesthetics and user interaction.
    • Possibly add features like message timestamps and conversation history.
  3. Display Additional Response Information

    • Integrate UI elements to show extra data (e.g., confidence scores) to help users evaluate the quality of responses.

Source Code

The complete source code for this project is available on GitHub.

https://github.com/sayed3li97/Gemini-REST-API-Flutter

Top comments (0)