Skip to content

Commit

Permalink
Add BiColorBargraph using Ht16k33 (dotnet#1915)
Browse files Browse the repository at this point in the history
* Add BiColorBargraph using Ht16k33

* Add images

* Update per feedback

* Update notes

* Add Fill method

* Update src/devices/Display/BiColorBarGraph.cs
  • Loading branch information
richlander authored Sep 19, 2022
1 parent 55a4ec0 commit fa1960c
Show file tree
Hide file tree
Showing 9 changed files with 361 additions and 19 deletions.
31 changes: 31 additions & 0 deletions src/devices/Display/BarColor.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.

namespace Iot.Device.Display
{
/// <summary>
/// Describes LED colors in a Bargraph.
/// </summary>
public enum BarColor
{
/// <summary>
/// Disable LED.
/// </summary>
Off = 0,

/// <summary>
/// Enable red LED.
/// </summary>
Red = 1,

/// <summary>
/// Enable green LED.
/// </summary>
Green = 2,

/// <summary>
/// Enable both green and red LEDs, producing a yellow color.
/// </summary>
Yellow = 3
}
}
189 changes: 189 additions & 0 deletions src/devices/Display/BiColorBarGraph.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,189 @@
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.

using System;
using System.Device.I2c;

namespace Iot.Device.Display
{
/// <summary>
/// Represents a 24-Segment bargraph that can display multiple colors.
/// </summary>
// Product: https://www.adafruit.com/product/1721
public class BiColorBarGraph : Ht16k33
{
private readonly byte[] _displayBuffer = new byte[7];
private readonly BarColor[] _biColorSegment = new BarColor[24];

/// <summary>
/// Initialize BarGraph display
/// </summary>
/// <param name="i2cDevice">The <see cref="I2cDevice"/> to create with.</param>
public BiColorBarGraph(I2cDevice i2cDevice)
: base(i2cDevice)
{
}

/// <summary>
/// Indexer for <see cref="BiColorBarGraph"/>.
/// </summary>
public BarColor this[int index]
{
get => _biColorSegment[index];
set
{
_biColorSegment[index] = value;
UpdateBuffer(index);
if (BufferingEnabled)
{
Flush();
}
}
}

/// <summary>
/// Enable all LEDs.
/// </summary>
public void Fill(BarColor color)
{
byte fill = 0xFF;
switch (color)
{
case BarColor.Red:
_displayBuffer[1] = fill;
_displayBuffer[3] = fill;
_displayBuffer[5] = fill;
break;
case BarColor.Green:
_displayBuffer[2] = fill;
_displayBuffer[4] = fill;
_displayBuffer[6] = fill;
break;
case BarColor.Yellow:
Span<byte> displayBuffer = _displayBuffer;
displayBuffer.Fill(fill);
displayBuffer[0] = 0x00;
break;
default:
break;
}

if (BufferingEnabled)
{
_i2cDevice.Write(_displayBuffer);
}
}

/// <inheritdoc/>
public override void Clear()
{
_displayBuffer.AsSpan().Clear();
if (BufferingEnabled)
{
_i2cDevice.Write(_displayBuffer);
}
}

/// <inheritdoc/>
public override void Flush() => _i2cDevice.Write(_displayBuffer);

/// <inheritdoc/>
public override void Write(ReadOnlySpan<byte> data, int startAddress = 0)
{
// Note: first byte is command data; does not affect display
foreach (byte b in data)
{
_displayBuffer[startAddress++] = b;
}

if (BufferingEnabled)
{
Flush();
}
}

/*
Task: Update data for the bargraph
The following diagram shows the intended orientation of the bargraph.
pins x x x x x x x x x x x x x x x x x x x x x x x x
23 22 21 20 19 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0
Each bargraph has three segments, four LEDs each.
For the 24-segment bargraph, that's six segments of four LEDs.
Each segment is addressed separately, with 4 bits (of a byte).
The first and last 4 bits of each byte represent different segments.
Each bit represents an LED. If the bits are on, the LEDs are on.
Each bar contains two LEDs. These are separately controlled.
That means that there are eight bits to consider for each segement,
four for each color. All of which can be separately on or off.
There are seven (7) bytes in the buffer.
The first byte of the buffer is for control/command information.
It should always be `0` unless specifically sending commands (like for blinking).
Each of the following bytes are paired, the first for red and the second for green.
Each of the bytes in the pair are split, the first half for one segment,
and the second half for the matching segement on the other bargraph.
The bytes are laid out this way:
- first segment: first four bits of bytes[2] for red; first four bits of bytes[3] for green
- second segment: first four bits of bytes[4] for red; first four bits of bytes[5] for green
- third segment: first four bits of bytes[6] for red; first four bits of bytes[7] for green
------------- // boundary of the bargraph units
- fourth segment: second four bits of bytes[2] for red; second four bits of bytes[3] for green
- fifth segment: second four bits of bytes[4] for red; second four bits of bytes[5] for green
- sixth segment: second four bits of bytes[6] for red; second four bits of bytes[7] for green
This is more obvious if you write some variation of value to the i2cdevice:
byte[] buffer =
{
0, 255, 0, 0, 255, 255, 255
};
*/
private void UpdateBuffer(int index)
{
// Tasks:
// Determine the location of the bar (for `index`).
// Produce a mask for the correct bit (within four bits).
// bitmask the correct bits dependending on the desired result
// Some basic math to use:
// x = index % 4 // which third (for example, for the 24 bar graph, there are six thirds)
// y = x % 3 // which third of the bar segment to use
// z = index / 12 // which of the bar segments to use
BarColor value = _biColorSegment[index];
int unit = index / 12;
int segment = index / 4;
int third = segment % 3;
int bit = index % 4 + (index / 12) * 4;
int mask = 1 << bit;
int bufferIndex = (third * 2) + 1;
byte red = _displayBuffer[bufferIndex];
byte green = _displayBuffer[bufferIndex + 1];

switch (value)
{
case BarColor.Off:
_displayBuffer[bufferIndex] = (byte)(red & ~mask);
_displayBuffer[bufferIndex + 1] = (byte)(green & ~mask);
break;
case BarColor.Red:
_displayBuffer[bufferIndex] = (byte)(red ^ mask);
break;
case BarColor.Green:
_displayBuffer[bufferIndex + 1] = (byte)(green ^ mask);
break;
case BarColor.Yellow:
_displayBuffer[bufferIndex] = (byte)(red ^ mask);
_displayBuffer[bufferIndex + 1] = (byte)(green ^ mask);
break;
default:
break;
}
}
}
}
4 changes: 3 additions & 1 deletion src/devices/Display/Display.csproj
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,10 @@
<Compile Include="FontHelper.cs" />
<Compile Include="ISevenSegmentDisplay.cs" />
<Compile Include="Large4Digit7SegmentDisplay.cs" />
<Compile Include="BiColorBarGraph.cs" />
<Compile Include="BarColor.cs" />
</ItemGroup>
<ItemGroup>
<None Include="README.md" />
</ItemGroup>
</Project>
</Project>
2 changes: 2 additions & 0 deletions src/devices/Display/Display.sln
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,8 @@ Project("{2150E333-8FDC-42A3-9474-1A3956D46DE8}") = "samples", "samples", "{20F2
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Large4Digit7SegmentDisplay.sample", "samples\Large4Digit7SegmentDisplay.sample.csproj", "{C4D7AED7-B340-4DCD-974C-5150C6D5F074}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "BiColorBargraph.sample", "samples\BiColorBargraph.sample.csproj", "{C4D7AED7-B340-4DCD-974C-5150C6D5F074}"
EndProject
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Display", "Display.csproj", "{4AAE8B9E-8540-4582-8083-D1F14104BAAC}"
EndProject
Global
Expand Down
49 changes: 32 additions & 17 deletions src/devices/Display/README.md
Original file line number Diff line number Diff line change
@@ -1,27 +1,24 @@
# Segment display driver (HT16K33)
# HT16K33 -- LED Matrix Display Driver

This project contains multipurpose LED display drivers and binding implementations for concrete display configurations.

## Documentation

The **HT16K33** is LED display driver that supports multiple LED configurations and I2C communication.
The [Ht16k33](https://cdn-shop.adafruit.com/datasheets/ht16K33v110.pdf) is a memory mapping and multi-function LED controller driver. It is used as a [backpack driver for several Adafruit products](https://www.adafruit.com/?q=Ht16k33). It supports multiple LED configurations and I2C communication.

Adafruit sells multiple display backpacks built upon this driver:

- [Adafruit LED / SEGMENTED category](https://www.adafruit.com/category/103)
- **Large4Digit7SegmentDisplay** is a binding that supports the **Adafruit 1.2" 4-Digit 7-Segment Display w/I2C Backpack** that comes in 3 colors:
- [Adafruit 1.2" 4-Digit 7-Segment Display w/I2C Backpack - Yellow](https://www.adafruit.com/product/1268)
- [Adafruit 1.2" 4-Digit 7-Segment Display w/I2C Backpack - Green](https://www.adafruit.com/product/1269)
- [Adafruit 1.2" 4-Digit 7-Segment Display w/I2C Backpack - Red](https://www.adafruit.com/product/1270)
- [HT16K33 datasheet](https://cdn-shop.adafruit.com/datasheets/ht16K33v110.pdf)
- [1.2" 4-Digit 7-Segment Display w/I2C Backpack - Yellow](https://www.adafruit.com/product/1268)
- [1.2" 4-Digit 7-Segment Display w/I2C Backpack - Green](https://www.adafruit.com/product/1269)
- [1.2" 4-Digit 7-Segment Display w/I2C Backpack - Red](https://www.adafruit.com/product/1270)
- [Bi-Color (Red/Green) 24-Bar Bargraph w/I2C Backpack Kit](https://www.adafruit.com/product/1721)

More information on wiring can be found on the respective product pages.
More information on wiring can be found on the respective product pages and at [adafruit/Adafruit_CircuitPython_HT16K33
](https://github.com/adafruit/Adafruit_CircuitPython_HT16K33) (Adafruit-maintained Python bindings).

## Usage
## 7-Segment Display Usage

![Adafruit 1.2" 4-Digit 7-Segment Display w/I2C Backpack - Green](https://cdn-shop.adafruit.com/970x728/1268-00.jpg)

```csharp
// Initialize display (busId = 1 for Raspberry Pi 2 & 3)
var display = new Large4Digit7SegmentDisplay(I2cDevice.Create(new I2cConnectionSettings(busId: 1, address: Ht16k33.DefaultI2cAddress));
using var display = new Large4Digit7SegmentDisplay(I2cDevice.Create(new I2cConnectionSettings(busId: 1, address: Ht16k33.DefaultI2cAddress));

// Set max brightness (automatically turns on display)
display.Brightness = display.MaxBrightness;
Expand All @@ -41,7 +38,25 @@ display.Dots = Dot.DecimalPoint;

// Send buffer to the device
display.Flush();
```

// Dispose display object (the device itself will not be turned off until powered down)
display.Dispose();
## Bi-Color Bargraph Usage

![Bi-Color (Red/Green) 24-Bar Bargraph w/I2C Backpack Kit](https://cdn-shop.adafruit.com/970x728/1721-00.jpg)
```csharp
using BiColorBarGraph bargraph = new(I2cDevice.Create(new I2cConnectionSettings(busId: 1, Ht16k33.DefaultI2cAddress)))
{
// Set max brightness
Brightness = Ht16k33.MaxBrightness,
BufferingEnabled = true
};

bargraph.Clear();

bargraph[0] = BarColor.RED;
bargraph[1] = BarColor.GREEN;
bargraph[2] = BarColor.YELLOW;
bargraph[3] = BarColor.OFF;
bargraph[4] = BarColor.RED;
```
11 changes: 11 additions & 0 deletions src/devices/Display/samples/BiColorBargraph.sample.csproj
Original file line number Diff line number Diff line change
@@ -0,0 +1,11 @@
<Project Sdk="Microsoft.NET.Sdk">
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>$(DefaultSampleTfms)</TargetFramework>
<EnableDefaultItems>false</EnableDefaultItems>
</PropertyGroup>
<ItemGroup>
<ProjectReference Include="..\Display.csproj" />
<Compile Include="Program.BiColorBargraph.cs" />
</ItemGroup>
</Project>
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,10 @@
<PropertyGroup>
<OutputType>Exe</OutputType>
<TargetFramework>$(DefaultSampleTfms)</TargetFramework>
<EnableDefaultItems>false</EnableDefaultItems>
</PropertyGroup>
<ItemGroup>
<ProjectReference Include="..\Display.csproj" />
<Compile Include="Program.cs" />
</ItemGroup>
</Project>
</Project>
Loading

0 comments on commit fa1960c

Please sign in to comment.