Projects Research & Design Hub

Devices Aid Speech for People with Disabilities

Written by Matthew Oppenheim

Using the BBC micro:bit

In this project article, Matthew shares how he designed two devices that enable people with cerebral palsy to more easily access communication software to create speech. The units are based on the micro:bit, a compact MCU development board.

  • How to design devices that enable people with cerebral palsy to more easily access communication software to create speech.

  • What are the details of the micro:bit board?

  • How to use MicroPython to develop the software

  • How the “Give Me a Minute”  project was implemented

  • How the “Handshake”  project was implemented

  • micro:bit board

  • MicroPython 

  • uflash command line utility

In this article I describe how I used the BBC micro:bit [1] to create two assistive technology (AT) devices. These are designed to enable people with cerebral palsy to more easily access communication software to create speech. The ideas for the projects came from the technologists who work at Beaumont College, Lancaster, UK. This college is attended by around 100 students with a broad range of physical and/or learning impairments, many resulting from cerebral palsy. Some of the students use specialist communication software to create speech.

I will describe the micro:bit and the programming tools that are available for it, and what makes this board a suitable choice for using in AT. The area of AT that these projects contribute to is called alternative and augmentative communication (AAC). The projects were tested at the college and have been presented at the Communication Matters conference. All code and manuals are freely available on my GitHub site [2], and I created videos showing how to implement the systems.

While I was writing this article in October 2020, version 2 of the micro:bit board was announced. The new board has a faster processor, more RAM, a speaker and a microphone—all in the same form factor. The projects presented in this article run on both versions of the board.

BBC MICRO:BIT BOARD

In early 2016, every 11 to 12-year-old child in the United Kingdom was given a small, embedded microcontroller (MCU) board to learn how to program. The development of this board was driven by the British Broadcasting Corporation (BBC) with 29 partners [1]. Over five million of the boards have been manufactured. The micro:bit packs a lot into a 4cm × 5cm board. Please see Table 1 for a summary of the hardware specifications for the micro:bit v1, which was used for development of the projects in this article. Figure 1 and Figure 2 show the front and back of the board, respectively, along with a AAA battery holder.

TABLE 1 – BBC micro:bit v1 specifications
FIGURE 1 – BBC micro:bit v1, front view
FIGURE 2 – BBC micro:bit v1, back view

The micro:bit lacks the 0.1″ header pins found on most development boards, since these might not last long in the hands of the intended user group. Instead, an edge connector allows access to power, ground, the I2C bus and SPI buses and the MCU general purpose input-outputs (GPIOs). A set of 4mm holes along this connector allows banana plugs to connect with some of these signals and the voltage and ground. Crocodile clips can be used with the five larger pads.

A block diagram of the board is shown in Figure 3. This also shows the signals that are connected to the edge connector. The MCU has a built-in 12-bit ADC, which can be accessed from the edge connector. Break-out connectors are available, which the micro:bit slots into, enabling easy access to all the edge connector pads.

— ADVERTISMENT—

Advertise Here

FIGURE 3 – Block diagram of micro:bit v1 hardware

The board is powered by two AAA batteries, which connect using a JST connector, or through the micro-USB connector. The boards are programmed through the same micro-USB connector. The CPU on the micro:bit v1 is Nordic Semiconductor’s nRF51822 with an Arm Cortex-M0 32-bit processor that contains a 2.4GHz radio module. The radio module can be used for Bluetooth or with a custom radio stack. Only one of these wireless protocols can be used at a time, though.

The board contains an accelerometer and a magnetometer. External temperature can be measured using the temperature sensor on the nRF41. The nRF41 runs so cool that this temperature sensor can be used to measure the environmental temperature instead. The LEDs can be used to measure external light levels. Light induces a current in the LEDs when they are not active. This current can be measured to infer the external light level.

The micro:bit has several features that make this board a suitable choice for my AT projects:

Designed to be safe: Because these boards are designed to be safe enough to give to 11 to 12-year-old students, they are safe to distribute in the “real world.” Such assurances would be more difficult if, in contrast, I handed out homemade devices.

Somebody else makes them: If a suitable product already exists, then the fastest way to deploy a system for real-world use is often to repurpose the existing product. Personally, I like hunching, Golem-like, over a homemade circuit board, enveloped by toxic solder fumes. However, the goal of these projects is to get something that is fit for purpose and can be easily distributed and implemented.

Robust board-to-board radio: In use, I found that the custom radio stack available on the micro:bit reminded me of using the XBee nodes that I wrote about in my last Circuit Cellar article a few years ago. That article is “Full-Stack Python” (Circuit Cellar 319, February 2017) [3]. The custom radio stack works without handshaking, and proved to be reliable when I used it. The alternative wireless protocol available is to use the radio module in Bluetooth mode. I find Bluetooth to be a fickle beast. Sometimes the target node needs resetting to become visible to the master node, which is not a practical thing to ask the target user group of these projects to do.

The micro:bit’s custom radio stack can be configured to send messages with both an address and a group, reducing the potential for interference from other micro:bits. The data transmission rate can be set up to 2Mbps.

Good programming tools: Several programming platforms are freely available for the micro:bit. Because the board is aimed at school use, the programming tools are designed to be easy to use.

SOFTWARE TOOLS

I used the micro:bit implementation of MicroPython [4] for the two projects presented here. Several other programming languages and platforms are available. I played around with MakeCode, a block-based, drag-and-drop programming interface. MakeCode is used at schools to start students with programming. The programming environment shows the JavaScript code that the Blocks produce. JavaScript can also be used directly to program the boards.

— ADVERTISMENT—

Advertise Here

Mbed provides an online C compiler for the micro:bit v1, but not for v2 of the board. Lancaster University also developed a C compiler for the micro:bit, on top of which the micro:bit’s MicroPython implementation and MakeCode are built. The Lancaster University C compiler can be integrated into, for example, the Eclipse IDE, but I prefer using it from the command line with a Makefile. The Lancaster University compiler is compatible with both versions of the micro:bit.

Connecting a micro:bit to your PC creates a folder called MICROBIT. If you use one of the editors aimed at the micro:bit, such as the mu editor, then your MicroPython code can be deployed with a button click. I tend to use a text editor for coding. To deploy my MicroPython code to the micro:bit I use the uflash command line utility [5]. I created a bash script to automate running this tool each time I save the file that I am working on. I put details of this script on my website [6].

Like many people who program embedded hardware, I often use C for my projects. Using MicroPython generally enables faster development, and makes the code easily accessible for others to build on. However, there are limitations with using MicroPython compared with C. The micro:bit implementation of MicroPython lacks access to event handling, which means my MicroPython code relies on manually checking for changes of state during each iteration of a while-loop. This is not a limitation with the C compiler, which implements access to events. Due to memory constraints, there is no implementation of Bluetooth in micro:bit’s MicroPython. For this project, not having access to the Bluetooth functionality was not an issue. The board-to-board custom radio stack proved to be robust and reliable.

Software limitations: I ran out of memory using MicroPython for another project, when I structured the code using classes. I got the project to run within the memory limitations by discarding the classes and just using methods. This memory constraint, lack of event handling and the loss of Bluetooth all point me to using the Lancaster University C compiler for more advanced projects. Because v2 of the micro:bit has 128KB of RAM compared to the 16KB of RAM in the v1 boards, the memory constraints for writing in classes should be removed. However, I prefer writing event-driven code. I would like to see the assert statement included in the micro:bit’s MicroPython implementation to help with writing tests.

PROJECT 1: “GIVE ME A MINUTE”

It is difficult for many high-tech AAC users to see when they are composing a message. For instance, it may not be clear when people who use eye-tracking technology are actively preparing speech. We implemented a system that visually indicates when communication software is in use. This enables a more natural interaction, and encourages good communication practice, giving adequate time for composition and respecting personal space. The visual feedback reassures others in the conversation that the AAC user is actively involved.

The system continually monitors the AAC software to show when new messages are being entered, without the users having to trigger a “hang on” type message manually and interrupt their composition. This monitoring is done with a Python script called activity_indicator.py. I used pyinstaller to generate an executable called activity_indicator.exe, which can be run without having to install Python.

The activity_indicator script finds and monitors the window that is running the communication software. When the script detects that new text has been entered or existing text deleted, a signal is sent to a micro:bit through the USB port. The micro:bit then displays an animated pattern over about 1 second to indicate that the communication software is in active use. The activation threshold is adjustable, to allow for different software and screens sizes.

Monitoring communication software: The win32gui library is used to find the window that runs the communication software. In testing, I found that one communications package I wanted to monitor spawned more than a single window, though only one was visible. Luckily, the invisible window had a slightly different title from the window that needed monitoring. I added a list of window titles to ignore, to filter out the invisible window. The Python method for finding the window that has the communication software running inside it is shown in Listing 1. This method uses the win32gui library to interact with the Windows operating system, and to get a list of windows and their titles.

LISTING 1 - Method for locating the window containing the communication software, using win32gui


import win32gui
COM_SOFTWARE = ['grid', 'communicator']
IGNORE = ['grid 3.exe', 'users']

def find_window_handle(com_software=COM_SOFTWARE, ignore=IGNORE):
     ''' Find the communication software window. '''
     toplist, winlist = [], []

     def _enum_cb(window_handle, results):
          winlist.append((window_handle, win32gui.GetWindowText(window_handle)))

     win32gui.EnumWindows(_enum_cb, toplist)
     for sware in com_software:
          # winlist is a list of tuples (window_id, window title)
          for window_handle, title in winlist:
               if sware in title.lower() and not any (x in title.lower() for x in
               ignore):
                    return window_handle
     logging.info('no communications software found for {}'.format(com_software))
     time.sleep(0.5)

Typically, only the top 20% of the window is used to display the text being prepared for speech. By monitoring only this part of the window, the amount of resources used by the software is reduced. The Pillow image processing library counts how many black pixels are in the top 20% of the communication software window, twice a second.

Making the micro:bit hot swappable: After I tested a prototype with the target user group, I realized that I had missed important use cases. What happens if the software is started without the micro:bit attached? Or what happens if the micro:bit is removed during use and then re-connected? I needed to make the micro:bit “hot swappable.”

I wrote a custom class for setting up and tearing down the serial connection with the micro:bit. I instantiate this class in a context manager. Using a context manager means that however the serial connection is terminated, the __exit__ method of the class runs. This ensures that if the micro:bit is unplugged during use, the serial connection is cleanly closed down by the __exit__ method.

To wrap the serial port in a context manager, the serial connection is instantiated using the keyword “with” in the line:

with Serial_Con(mbit_port) as mbit_serial:

The main method from the software that runs on the communication software, called activity_indicator.py, is presented as Listing 2.

LISTING 2 - Main method for monitoring communication software and communicating with the micro:bit

def main(limit, fraction):
     logging.info('*** starting find_microbit ***\n')
     check_fraction(fraction)
     logging.info('limit={} fraction={}\n'.format(limit, fraction))
     old_black = 0
     while True:
          logging.info('*** looking for a microbit')
          mbit_port = get_comport(PID_MICROBIT, VID_MICROBIT, 115200)
          logging.info('microbit found at comport: {}'.format(mbit_port))
          with Serial_Con(mbit_port) as mbit_serial:
               # occasionally mbit_serial is not created, so is None
               if not mbit_serial:
                    logging.info('failed to create mbit_serial')
                    time.sleep(0.5)
                    continue
              logging.info('microbit serial port created at: {}'.format(mbit_port))
               while True:
                    time.sleep(0.5)
                    # look for the top fraction of a window running target software
                    window_top = get_window_top(fraction)
                    if window_top is None:
                         continue
                    # count black pixels in top fraction of target window
                    new_black = num_new_black_pixels(window_top)
                    logging.debug('new_black: {}'.format(new_black))
                    if new_black is None:
                         continue
                    is_limit_exceeded = check_limit(new_black, old_black, limit)
                    if is_limit_exceeded:
                         try:
                              mbit_serial.write(b'flash')
                         except serial.SerialException as e:
                         logging.info('connection to microbit failed{}'.format(e))
                              break
                    old_black = new_black

I use the Singleton pattern to ensure that only one instantiation of the serial interface exists. This may be overly cautious, but it prevents any possibility of multiple connections being created if, for example, a second micro:bit was plugged in. I learned how to implement the singleton pattern the usual way, by reading answers in Stack Overflow [7]. I also contribute to Stack Overflow, so I try to pay back some of the help I receive. I use the method that implements the singleton pattern as a decorator for the serial connection class.

— ADVERTISMENT—

Advertise Here

If the micro:bit is removed during use, then an exception is generated when the program tries to send a message to the micro:bit over the now non-existent serial connection. This exception is caught, and the inner loop in the main method exits. The outer loop then re-establishes the connection to the micro:bit when the micro:bit is re-connected. Figure 4 shows “Give Me a Minute” being tested at Beaumont College, Lancaster, UK. The tester was so pleased with the system that she took it away with her—which I had not anticipated!

FIGURE 4 – “Give Me a Minute” in use at Beaumont College, Lancaster, UK
PROJECT 2: “HANDSHAKE”

“HandShake” is designed to enable people to interact with switchable AAC software if they cannot use physical controllers, such as buttons or joysticks, but can make an intentional hand or arm movement.

Regular readers might remember that article I mentioned earlier—”Full-Stack Python” [3]—in which I presented a prototype of this system [3]. For that project, I used several boards to achieve what I do here with a single micro:bit. The micro:bit eliminates the need to interface a MCU board with an accelerometer board, battery management board and an XBee wireless communication board.

HandShake consists of two micro:bit boards programmed using MicroPython. One micro:bit is configured as a transmitter that is worn on the wrist (Figure 5). The second is the receiver, which is attached with a micro-USB cable to the communication device running the AAC software. Software written in Python is installed on the AAC device. The acceleration with which the student’s hand is moving is constantly monitored, using the micro:bit on the wrist. When the acceleration exceeds an adjustable limit, a key-press command is sent to the student’s AAC software to control it.

FIGURE 5 – BBC micro:bit worn on the wrist as part of the HandShake system

The threshold of acceleration at which the system triggers can be adjusted using the buttons on either of the micro:bit boards. This allows the system to be adjusted to allow both gentle and energetic motion characteristics. Since either of the micro:bits can be used to adjust the threshold, the person wearing the transmitter does not have to be disturbed.

The micro:bits have a simple operating system that allows persistent files to be saved. I leveraged this to save the shake detection threshold to a file. This value is loaded by the software running on the transmitter when it is turned on.

ACCELERATION THRESHOLD

A simple orientation-invariant algorithm for acceleration threshold recognition is implemented on the transmitter. Because the algorithm is orientation invariant, the system is not restricted in use to when the student is upright. For example, the system can be used when lying in bed, perhaps to turn on a light or to signal for attention.

Initially, the transmitter micro:bit was worn on the wrist, using a converted armband holder from a smartphone. Now, I use an off-the-shelf iPod Classic armband, so I don’t have to do any sewing. The only additions to the armband are some tabs of back-to-back “hook side” Velcro. They keep the strap from dangling when worn on the wrist, since the armbands are long enough to wrap around an arm. In initial testing at Beaumont College, two volunteers from the target user group used the system to turn on an LED. A photo of the testing is shown in Figure 6.

FIGURE 6 – Initial HandShake testing, using the system to turn on an LED

The student for whom the system was developed managed to improve his coordination, so he is now able to use contact devices, such as switches. This means that he does not need to use HandShake. Beaumont College is looking to see if any of their other students could benefit from the system. Some other organizations are interested in testing the system. This was made a bit difficult at the time of writing this article, with the ongoing COVID-19 pandemic.

Having a system that gives a visible signal in response to an intentional arm movement is useful in itself. It can help to assess if students are aware that their arm motion causes an external change. I tested the battery life by simulating one shake detection per second. The battery life exceeds 24 hours, using rechargeable batteries. The Python software on the AAC implements the same “hot-swappable” algorithm as the “Give Me a Minute” software. This means that the receiver micro:bit can be unplugged and replaced without crashing the system.

DISCUSSION

The micro:bit is a versatile platform that lends itself to real-world applications, such as the ones presented in this article. Version 2 of the board has a few new features, including a speaker and microphone, that could be valuable in the assistive technology field.

I set up GitHub repositories [2] for my software development, and recommend the system. If I want to explore a new idea, I create a new branch for the code. If the idea works out, I merge this branch into the main branch, which I then push to my GitHub site. Using Git helps prevent losing ideas I had in earlier versions of code, then deleted. I can roll back any part of the code base to an earlier saved version.

I put links to the full documentation onto my website [8], and produced “how-to” videos for both projects, along with a short video explaining the projects [2] for the TechAbility assistive technology conference. I recently ported all of the documentation to my GitHub site using the MkDocs tool. This makes the documentation open source as well as the code.

If anybody would like to implement either of the projects presented here or has ideas on how to improve them, please get in touch with me. 

RESOURCES

References:
[1] BBC micro:bit. Retrieved March 5, 2018, from https://www.microbit.co.uk/home
[2]  GitHub Repositories for micro:bit HandShake
https://github.com/hardwaremonkey/microbit_hand_shake
[3] Oppenheim “Full-Stack Python,” Circuit Cellar 319, February, 2017.
[4] MicroPython  https://tech.microbit.org/software/micropython
[5] uFlash command line utility  https://uflash.readthedocs.io/en/latest
[6] uFlash https://www.seismicmatt.com/2020/08/23/automating-loading-micropython-code-to-the-bbc-microbit-in-linux
[7] Singleton Pattern, Stack Overflow  https://stackoverflow.com/questions/3319434/singleton-pattern
[8] Oppenheim Project Website https://www.seismicmatt.com/handshake

Communication Matters:  https://www.communicationmatters.org.uk

Bill of materials:
micro:bit board
[uflash] https://www.seismicmatt.com/2020/08/23/automating-loading-micropython-code-to-the-bbc-microbit-in-linux

Code Blocks | www.codeblocks.org
Mbed | https://os.mbed.com
Micro:bit Educational Foundation | www.microbit.org
Nordic Semiconductor | www.nordicsemi.com
Stack Overflow | www.stackoverflow.com

PUBLISHED IN CIRCUIT CELLAR MAGAZINE • JULY 2021 #372 – Get a PDF of the issue

Keep up-to-date with our FREE Weekly Newsletter!

Don't miss out on upcoming issues of Circuit Cellar.


Note: We’ve made the Dec 2022 issue of Circuit Cellar available as a free sample issue. In it, you’ll find a rich variety of the kinds of articles and information that exemplify a typical issue of the current magazine.

Would you like to write for Circuit Cellar? We are always accepting articles/posts from the technical community. Get in touch with us and let's discuss your ideas.

Sponsor this Article

Supporting Companies

Upcoming Events


Copyright © KCK Media Corp.
All Rights Reserved

Copyright © 2023 KCK Media Corp.

Devices Aid Speech for People with Disabilities

by Matthew Oppenheim time to read: 15 min