Local weather station with BME280 & Android integration

  • Student: Andreea Miu
  • Master: SAS

Introduction

This project describes a local weather station, which is included in a larger Android application.

The purpose of this project is to provide a proof-of-concept for adding local weather stations to standard weather applications, in order to have a more granular approach on weather based on customizable location and measured data.

I have also uploaded this project on Github, for easier access to the full code and to provide the possibility of testing and reproducing the application.

Project demo

A demo video of the application running can be found at the following link.

Hardware Setup

Components

  • ESP32-DevKitC-V4 board
  • BME280 environmental sensor
  • Jumper wires

Schematics

ESP32 and BME280 schematics

In the above schematics, the following pins are connected:

  • VCC of BME280 - 3v3 of ESP32
  • GND of BME280 - GND of ESP32
  • SCL/SCK of BME280 - IO22 / GPIO pin 22 of ESP32 (default GPIO pin for I2C SCL)
  • SDI/SDA of BME280 - IO21 / GPIO pin 21 of ESP32 (default GPIO ppin for I2C SDA)

Physical setup

The physical setup is easily reproducible. This allows for scalability if creating multiple local weather stations is desired. The BME280 sensor used comes with female-to-male jumper wires that form a connection port suitable for the sensor. The female end of each wire can be connected to the ESP32 board pins, according to the schematics described above.

Software Setup

The local weather station will send data to a database, in order to facilitate access to retrieved data. For this project, a Firebase real-time database is used.

Arduino code

The Arduino code consists of a few key elements used, from both a Firebase connection perspective, and a BME280 sensor perspective. The full code can be found on this Github repository.

In order to measure altitude correctly, BME280 needs a reference pressure for sea level:

#define SEALEVELPRESSURE_HPA (1013.25)

BME280 can transmit data using either I2C or SPI. For this project, I2C was used. By default, the sensor uses I2C, but if the user would like to use SPI, this is possible by editing the if structure and calling bme using the SPI pins as parameters.

#define USEIIC 1

#if(USEIIC)
  Adafruit_BME280 bme;
#else
  // define SPI pins
#endif

In order to use Firebase and connect to WiFi for data transmission, multiple defines are used.

In the setup function, the BME280 sensor is initialized and the network connection is established.

void setup(){
    Serial.begin(115200);

    // Init the BME280 sensor
    bool rslt;
    rslt = bme.begin();  
    if (!rslt) {
      Serial.println("Could not initialize sensor, make sure it is properly connected!");
      while(1);
    }
    Serial.println("BME280 successfully initialized");
    
    .......

    // Start connecting to WiFi network
    WiFi.begin(WIFI_SSID, WIFI_PASSWORD);
    Serial.print("Connecting to Wi-Fi");
    while (WiFi.status() != WL_CONNECTED){
      Serial.print(".");
      delay(300);
    }
    Serial.println();
    Serial.print("Connected with IP: ");
    Serial.println(WiFi.localIP());
    Serial.println();
    
    /* Assign the api key (required) */
    config.api_key = API_KEY;
    
    /* Assign the RTDB URL (required) */
    config.database_url = DATABASE_URL;
    
    /* Sign up */
    if (Firebase.signUp(&config, &auth, "", "")){
      Serial.println("ok");
      signupOK = true;
    }
    
    .......
  }

In the loop() function, I call the sendValues() function, which sends values for temperature, humidity, pressure and altitude respectively. For example, temperature can be fetched and sent to the Firebase database in the following way:

if (Firebase.RTDB.setFloat(&fbdo, "temperature", bme.readTemperature())){
      Serial.println("PASSED");
      Serial.println("PATH: " + fbdo.dataPath());
      Serial.println("TYPE: " + fbdo.dataType());
    }
    else {
      Serial.println("FAILED");
      Serial.println("REASON: " + fbdo.errorReason());
    }

Firebase

The Firebase database is a real-time database, which stores the values captured by the sensor in respective nodes: temperature, humidity, pressure and altitude respectively. The below picture gives a look into the database from an admin perspective:

Android application

The Android application used is part of a larger project, which also provides access to global weather forecast using an API. By using the data collected by the BME280 sensor and stored in the Firebase database, the granularity of data (across a city, for example) is increased. The screenshots below illustrate how the app can be navigated in order to access the local weather station in Bucharest. A different forecast can also be retrieved for Bucharest using the API, thus providing two points of data collection in Bucharest.

For developing the application, I used Android Studio, and Java as the programming language.

After configuring the connection to Firebase using Android Studio's Firebase plugin, the nodes and values of the nodes can be retrieved in real-time using a ValueEventListener. For example, getting the temperature value and showing it in a TextView can be done in the below manner:

databaseReference = FirebaseDatabase.getInstance().getReference();

// Event listener
databaseReference.addValueEventListener(
    new ValueEventListener() {
    @Override
    public void onDataChange(@NonNull DataSnapshot snapshot) {
        temp_val =
            snapshot.child("temperature").getValue(Float.class);
        temperatureTxt.setText(temp_val.toString() + "°C");
    }

The full application can be viewed using this link. Below are some screenshots that illustrate how to navigate the app for viewing information from the local weather station.

References

iothings/proiecte/2022sric/weatherly-local-station.txt · Last modified: 2023/06/01 21:45 by andreea.miu
CC Attribution-Share Alike 3.0 Unported
www.chimeric.de Valid CSS Driven by DokuWiki do yourself a favour and use a real browser - get firefox!! Recent changes RSS feed Valid XHTML 1.0