RajivCodeLabRajiv Sharma
Published © LGPL

How to Build Web Socket on Raspberry Pi Pico W and Make Live

By the end of this tutorial, you'll have a dashboard displaying the temperature readings from your Raspberry Pi Pico W using Web Socket.

IntermediateProtip2 hours124
How to Build Web Socket on Raspberry Pi Pico W and Make Live

Things used in this project

Hardware components

Raspberry Pi Pico W
Raspberry Pi Pico W
×1

Software apps and online services

Thonny IDE
Micro Dot Library

Story

Read more

Code

index.html (UI Page)

HTML
<!DOCTYPE html>
<html>
<head>
  <title>Live Temperature Dashboard</title>
  <meta charset="UTF-8">
  <!-- Include Chart.js library -->
  <script src="https://cdn.jsdelivr.net/npm/chart.js"></script>
  <style>
 
@import url(https://fonts.googleapis.com/css?family=Open+Sans:700,300);

 
.frame {
  position: absolute;
  top: 50%;
  left: 7%;
  width: 400px;
  height: 400px;
  margin-top: -200px;
  margin-left: -200px;
  border-radius: 2px;
	
	overflow: hidden;
  background: #222;
  color: #333;
	font-family: 'Open Sans', Helvetica, sans-serif;
	-webkit-font-smoothing: antialiased;
	-moz-osx-font-smoothing: grayscale;
	background: #222;
}

.center {
  position: absolute;
  top: 50%;
  left: 50%;
  transform: translate(-50%,-50%);
}

.temparature-meter__container {
	position: relative;
	height: 200px;
	width: 200px;
	border-radius: 200px;
	box-shadow: 0 1px 2px rgba(0, 0, 0, .07);
	border: solid 10px #fff;
	overflow: hidden;
	
	&:after {
		position: absolute;
		content: "";
		height: 120px;
		width: 120px;
		background-color: #fff;
		z-index: 0;
		transform: rotate(45deg);
		bottom: -40px;
		left: 40px;
		z-index: 1;
	}
}

.temparature-meter__inner {
	display: flex;
	text-align: center;
	height: 100%;
	width: 100%;
	border-radius: 100%;
	box-shadow: inset 0 0 10px rgba(0, 0, 0, .4);
	background: linear-gradient(to right, #148d04 0%,#f52c3d 100%);
	overflow: hidden;
	
	.temparature-meter__inner-ring {
		cursor: pointer;
		position: relative;
		margin: auto;
		display: flex;
        flex-direction: column;
        justify-content: center;
		height: 130px;
		width: 130px;
		background-color: #fff;
		border-radius: 100%;
		box-shadow: 1px 2px 4px 0 rgba(0, 0, 0, 0.2);
		z-index: 9;
		
		&:after {
			content: "";
			position: absolute;
			width: 1px;
			height: 50%;
			background-image: linear-gradient(#148d04 15px, transparent 0);
			top: 0;
			left: 50%;
			transform-origin: bottom;
			transform: rotatez(-35deg);
			transition: all 0.8s ease-in-out;
		}
		
		&:hover:after {
			transform: rotatez(-20deg);
		}
		
		.temparature-count {
			display: block;
			position: absolute;
			width: 100%;
			text-align: center;
			transition: all 0.8s ease-in-out;
			
			&.c2 {
				opacity: 0;
				transform: translatex(50px) scale(0.5);
			}
		}
		
		&:hover {
			.c1 {
				opacity: 0;
				transform: translatex(-50px) scale(0.5);
			}
			.c2 {
				opacity: 1;
				transform: translatex(0) scale(1);
			}
		}
	}
	
	.temparature-inside {
		position: relative;
		font-size: 50px;
    display: inline-block;
    letter-spacing: -5px;
    font-weight: 500;
    color: #148d04;
		height: 58px;
	
		.temperature-sup {
			font-size: 20px;
			position: absolute;
			position: relative;
    	right: -45px;
    	top: -30px;
		}
	}
	
	.temparature-outside {
		font-size: 13px;
	}
	
	.temparature-meter__inner-label {
		font-size: 13px;
	}
}


    body {
      background-color: #222;
      color: #fff;
      font-family: Arial, sans-serif;
     
    }
    h1 {
      text-align: center;
      color: #148d04;
      
    }
    .chart-container {
      margin-left: 300px;
      margin-top:20px;
      text-align:center;
      height: 70vh; /* Set the height to half of the viewport height */
    }
    header {
      background-color: #333;
      color: #fff;
      padding: 10px;
      text-align: center;
       /* Remove default margin */
      display: flex; /* Use flexbox to align items */
      align-items: center; /* Vertically center items */
    }
    #temperature {
      text-align: center;
      font-size: 24px;
      margin-bottom: 20px;
      margin-top:30px;
    }
    canvas {
     
      text-align:center;
      width:100%;
     
    }
    .logo {
      width: 50px; /* Adjust the width of the logo */
      height: auto; /* Maintain aspect ratio */
      margin-right: 10px; /* Add some space between logo and title */
      margin-left: 35px;
    }
  </style>
</head>
<body>
  <header>
    <img src="" class="logo">
    <h1>Live Temperature Dashboard</h1>
  </header>
  <div class="frame">
  <div class="center">
		<div class="temparature-meter__container">
			<div class="temparature-meter__inner">
				<div class="temparature-meter__inner-ring">
					 
				<div class="temparature-inside">
						<span id="temp-meter" class="temparature-count c1"></span>
						 
						<span class="temperature-sup"></span>
					</div>
					<div class="temparature-meter__inner-label">Bedroom</div>
				</div>
			</div>
		</div>
  </div>
</div>
  <div id="temperature">Temperature: --</div>
  <div class="chart-container">
    <canvas id="temperatureGraph"></canvas>
  </div>
 

  <script>
    const socket = new WebSocket('ws://' + location.host + '/temperature');
    let chart; // Reference to the chart object

    socket.addEventListener('message', ev => {
     console.log(ev);
      const temperature = parseFloat(ev.data);
      updateGraph('temperatureGraph', 'Temperature (C)', temperature);
      updateTemperatureDisplay(temperature);
    });

    function updateTemperatureDisplay(temperature) {
      document.getElementById('temperature').textContent = `Temperature: ${temperature.toFixed(2)} C`;
      document.getElementById('temp-meter').textContent = temperature.toFixed(2);
    }

    function updateGraph(chartId, label, value) {
      const ctx = document.getElementById(chartId).getContext('2d');

      if (!chart) {
        chart = new Chart(ctx, {
          type: 'line',
          data: {
            labels: Array.from({ length: 15 }, (_, i) => i + 1),
            datasets: [{
              label: label,
              data: Array(25).fill(value), // Fills array with 10 same values for initial chart setup
              fill: true,
              borderColor: '#148d04',
              tension: 0.4
            }]
          },
          options: {
            scales: {
              x: {
                title: {
                  display: true,
                  text: 'Time',
                  color: '#ccc'
                },
                grid: {
                  color: '#444'
                },
                ticks: {
                  color: '#ccc'
                }
              },
              y: {
                title: {
                  display: true,
                  text: label,
                  color: '#ccc'
                },
                grid: {
                  color: '#444'
                },
                ticks: {
                  color: '#ccc'
                }
              }
            },
            plugins: {
              legend: {
                labels: {
                  color: '#ccc'
                }
              }
            }
          }
        });
      } else {
        // Update the existing chart data
        chart.data.labels.push(new Date().toLocaleTimeString());
        chart.data.datasets[0].data.push(value);
        if (chart.data.labels.length > 10) {
          chart.data.labels.shift(); // Remove the oldest label
          chart.data.datasets[0].data.shift(); // Remove the oldest data point
        }
        chart.update();
      }
    }
  </script>
</body>
</html>

Python Code

Python
import machine
import time
import network
from microdot import Microdot, send_file
from microdot.websocket import with_websocket

def connect_to_wifi():
    sta_if = network.WLAN(network.STA_IF)
    if not sta_if.isconnected():
        print("Connecting to the network...")
        sta_if.active(True)
        sta_if.connect("Airtel_RAJEEV", "00351553")
        while not sta_if.isconnected():
            pass
    print("Connected to IP: ", sta_if.ifconfig()[0])

connect_to_wifi()

app = Microdot()

@app.route('/')
async def index(request):
    return send_file("index.html")

@app.route('/api/led', methods=['POST'])
async def index(request):
    print(request.json)
    print(request.json["ledRed"])
    return {'success': True}

@app.route('/temperature')
@with_websocket
async def index(request, ws):
    while True:
        adc = machine.ADC(4)  # Use ADC pin GP4
        conversion_factor = 3.3 / (65535)  # ADC conversion factor
        sensor_value = adc.read_u16() * conversion_factor
        temperature = 27 - (sensor_value - 0.706) / 0.001721  # Convert sensor value to temperature (formula may vary)
        await ws.send(str(temperature))
        time.sleep(1)

app.run(debug= True, port=80)

Credits

RajivCodeLab

RajivCodeLab

6 projects • 3 followers
Creates YT videos on DIY IoT Projects: Raspberry Pi Pico, Raspberry Pi Zero, Arduino, ESP32,
Rajiv Sharma

Rajiv Sharma

17 projects • 71 followers
Having more than 10 years of experience in IoT and software technology. Founded IoTBoys to share knowledge with IoT enthusiasts.

Comments