Skip to content
Switch branches/tags

Latest commit


Git stats


Failed to load latest commit information.
Latest commit message
Commit time


Inkblot is a gem for interacting with Waveshare's line of E-paper displays on the Raspberry Pi using ruby. It includes components for outputting images, text, and menus to the EPD as well as reading input from HAT buttons on the side.

Hello from Inkblot


Add this line to your application's Gemfile:

gem 'inkblot', github: 'jtp184/inkblot'

You can also download and install it globally with

git clone
cd inkblot
rake install

For help setting up your pi for use with this gem, check out the RASPI_SETUP instructions.


This gem was developed using Waveshare's 2.7 inch e-Paper HAT on the Raspberry Pi 3+ and 4. Component aspect ratios and button pinouts are based on this screen size, but you can override them with module level writers. Be sure to set these before using any of the relevant functions.

Inkblot.screen_size # => Defaults to { width: 264, height: 176 }
Inkblot.screen_size = { width: 200, height: 200 }

Inkblot.button_pinout # => Defaults to [5, 6, 13, 19]
Inkblot.button_pinout = [6, 11, 12, 14] # Uses different pins



All methods and classes are RDoc documented at


The Display class handles outputting to the screen. It accepts an argument to its .show method to output to the screen, which can be any number of displayable objects.

items = []

# Primarily used to display Components and Converters
items << 'Hello')
items << '/home/pi/img.png')
# Can also display (already correct bmp) files directly or by path
items <<'/home/pi/bump.bmp') # Or a TempFile
items << '/home/pi/bump.bmp'

# call, show, and [] all work[0])[1])

# Plaintext also works
Inkblot::Display.("Show Me")

One of the easiest ways to start using Inkblot is to add a #to_display method on your object which returns a component.

class SomethingToShow
  def initialize(smth)
    @something = smth.to_s

  def to_display @something)
end"Cinnamon Buns"))

Color Depth

The EPD can operate in 1-deep and 4-deep color modes, with an accessor on the module to swap between them.

Inkblot.color_depth # => Defaults to 1

Inkblot.color_depth = 4 # For Init_4Gray


The Buttons class deals with input from the HAT buttons. The buttons communicate over the Pi's GPIO pins, and can be used for a broad variety of inputs.

# Setting up the buttons for use. This only needs to be run once at the start
# You can check whether the buttons have been initialized with
Inkblot::Buttons.ready? # => true

# Basic input from buttons. With no argument, blocks until you press a button then returns an index
Inkblot::Buttons.get_input # => 0

# With an argument, it will timeout if no button is pressed.
Inkblot::Buttons.get_input(10) # => nil if no button is pressed for 10 seconds

# It's also possible to get chords instead of single presses.
# Specify the chord length and wait for simultaneous presses.

Inkblot::Buttons.get_multi_input(2) # => [1, 2]
Inkblot::Buttons.get_multi_input(3, 10) # => nil || [0, 1, 3]

# You can also record all button activity within a timeframe

Inkblot::Buttons.get_raw_input(3) # => [[], [0], [0, 1]...]

# Unexports GPIO and releases it for other use

Usually, you'll use the buttons with a display component. Components (and other classes) may have their own defined button actions, and the Buttons class can access them.

class DummyComponent
  def to_display "Dummy")

  def button_actions
      -> { puts "Key1" },
      -> { puts "Key2" },
      -> { puts "Key3" },
      -> { puts "Key4" }

c =

# First we have to display the component

# Gets input, and runs the proc at the associated index.
# e.g. Pressing "Key1" on the HAT runs c.button_actions[0].call


The Converter class and its subclasses are used to transform input data into other forms, with the end goal of being able to display it on the EPD. Converters are used internally by Components to render their HTML, and are first-class displayables in and of themselves.


Abstract parent class for common functionality. It defines the overridable #convert method which is used by the #convert! method to produce the output.


The ImageConverter class can take images on disk, and through ImageMagick resize and convert them into 1-deep .bmp files suitable for display on the EPD.

# Paths
i =
  input: "/home/pi/img.png"

# File and Tempfile objects
j =

# Binary image data
k =

i.convert! && i.output # => Tempfile


The DataUrlConverter converts an image input into a base64 encoded data url

# Paths
d =
  input: "/home/pi/img.png",
  format: :path

# File and Tempfile objects
e =
  format: :file

# Binary image data
f =
  format: :binary

# Base64 image data
g =
  input: SomeCode.base64_image,
  format: :base64

d.convert! && d.output # => "..."


The HtmlConverter takes in an HTML doc as input, and transforms it into an image using puppeteer. This image is then piped through an ImageConverter so that it can be rendered on the EPD

html_doc = <<DOC
    <p>Happy Meal with Extra Happy</p>

h = html_doc)

h.image_contents # => \"BM\\xFE\\u0018\\u0000\\u0000\\u0000..."


Components are composable views. They're written as ultra-basic ERB/HTML templates, with scaling applied to look good at the resolution of the EPD. These are then passed into the HtmlConverter. No writing or comprehension of HTML is necessary though, as you merely pass in options to the constructor.

You can provide options to the constructor with standard or block syntax. "Works this way") do |st|
  st.text = "Or this way"

Passed in options can be retrieved through the options method, or defined accessors

c = "Text")

c.options[:text] # => "Text"

Passing a second hash as a positional argument sets the keys/values as instance variables

i ={body: "I have a name"}, { name: "Rachel" })

i.inspect # => #<Inkblot::Components::Component:0x... @name="Rachel", @options={:body=>"I have a name"}>

You can also compose components by passing them to Component.create method, which will combine the fragments together into a single page, top to bottom.

Inkblot::Components::Component.create do |cpt|
  cpt << "Several")
  cpt << "Different")
  cpt << "Components")
<!DOCTYPE html>

  <body style="height: 176px; width: 264px;">
    <div style="...">
      <h1 style="font-family: monospace; ">Several</h1>

    <div style="...">
      <h1 style="font-family: monospace; ">Different</h1>

    <div style="...">
      <h1 style="font-family: monospace; ">Components</h1>

Built-in Components

Base Component

The Component class is the superclass of all other components, as well as a component itself. Components have a #options method, a hash containing their customization options. The base component has only one relevant option, its body. %q(<h1>A Simple Component</h1>))

# You can also set width and height on any component like so do |c|
  c.body = %q(<p>small text</p>)

  c.div_height = 100 # height: 100%
  c.div_width = '500px' # width: 500px
  c.div_height = :full # height = Display.size[:height] 
  c.fullscreen = true # Same as c.div_height = c.div_width = :full

Other components subclass from this, and have their own customization options. Composed components from Component.create are instances of the base component class.


The SimpleText class allows you to do just that, simple text. Sizing can be set explicitly or automatically, as can fonts.

Inkblot::Components::SimpleText do |st|
  st.text = "Example Text"
  st.border_size = 10 # Adds an inner border
  st.size = :large # :tiny, :small, :medium, :large or an integer for px
  # Takes in an array of font names and includes them
  st.gfonts = ['Roboto', 'Open Sans', 'Jost', 'Pangolin']
  # Also works with built in fonts like Helvetica
  st.font = "Pangolin"


The FullScreenImage class displays images. You can pass a variety of image sources in, and it works with other components for nesting and resizing. Images are automatically resized and downsampled.

# Sets the img tag src to the url do |fsi| 
  fsi.url = ""

# Sets the img tag src to the absolute version of the path "/home/pi/img.jpg")

# Sets the img tag src as a data url from reading the file"/home/pi/img.jpg"))

# Sets the img tag src as a data url from the file contents themselves"/home/pi/img.jpg"))


The QrCode class can generate and render QR Codes to the screen, great for small area and two-channel color. do |qr| 
  qr.message = ""
  qr.margin_top = qr.margin_left = -5 # adds a -5% margin


The BarCode class can generate and render EAN-13 style barcodes. "123123123123")


The TableList class displays up to 4 line items in a horizontal table view. If you supply fewer than 4 items, blank entries are produced to keep 4 lines. do |tl|
  tl.items = ['Apples', 'Pears', 'Oranges', 'Bananas']


An IconGroup is a grid of icons / images to display horizontally next to each other. Column count is customizable, rows are subsequently determined by amount of content. do |ig|
  ig.icons = []
  # Standard HTML symbols work
  ig.icons << :uarr 
  # Google Material Icons symbols as well
  ig.icons << :android 
  # Simple strings work
  ig.icons << 'A' 
  # As do components, especially FullScreenImages
  ig.icons <<
    path: Inkblot.vendor_path('chris_kim.bmp')

  ig.icon_size = 40 # Default
  ig.font = 'Material Icons, monospace' # Default
  ig.columns = 4 # Default


An IconPane is a two-column layout with icons on the left, and a large rectangular frame on the right. Other components can be placed in the right-hand frame. do |ic|
  # This symbol is equivalent to %i[nwarr larr swarr swarr]
  # Other icon group presets include :arrows_out, :select, and :agree / :cancel
  # You can also pass strings, or symbols which are considered html symbols
  ic.icons = :arrows_in 

  # A single object or an array works here. Arrays are composed with `Component.create`
  ic.frame_contents = do |fc|
    fc.path = Inkblot.vendor_path('chris_kim.bmp')


The ScrollMenu can be used for selecting from a list of items, and includes a view and button actions to accomplish this

scroll_menu = do |sc|
  sc.items = (1..10).map { |x| "Option #{x}" }

# Scroll menu has 4 states: :scroll, :select, :answered, and :canceled
scroll_menu.state # => :scroll

# Scroll menu also has pages of content divided into fours
scroll_menu.current_page # => 0
scroll_menu.page_count # => 3

# Using the menu as User prompting is simple # Display the component initially

until scroll_menu.concluded? # True when canceled or answered
  # Contains the 4 presented options
  puts "User can see #{ { |x| %Q("#{x}") }.join(',') }"
  Inkblot::Buttons.get_press # Defers to ScrollMenu's #button_actions method
  Inkblot::Display.again unless scroll_menu.concluded? # Show it again

scroll_menu.answer # => One of the options, or nil if canceled

Creating new Components

Components are subclasses of Inkblot::Components::Component, which gives them all common behavior around generating their HTML representations. You can create subclasses of Component if you want to create customized views and behaviors, or if you want to go outside of the HTML template model but still use the Display / Buttons functions.

Components optionally override the #computed method. This method allows options to be calculated at render time by returning a hash of program-defined options merged into the user defined ones.

def computed
    element_name: lookup_name(options[:element]),
    element_weight: lookup_weight(options[:element])


A Template is an ERB file, like a rails view partial. The built-in components store their templates in a common directory within the gem's vendor directory, and name them after the class. You can load custom templates either by passing them in as class arguments

# Pass a base folder as a class option to a generic component
class MyUniqueComponent < Inkblot::Components::Component
  # ...

user_info = { first_name: "Erek", last_name: "King"}

muc =, { template_base_path: "/path/to/my/templates" }) do |mc|
  mc.full_name = [mc.first_name, mc.last_name].join(' ')

muc.send(:template_full_path) # => "/path/to/my/templates/MyUniqueComponent.html.erb"

# You can also override the filename

muc2 =
    template_base_path: "/new/templates"
    template_filename: "BetterTemplate.html.erb"

muc2.send(:template_full_path) # => "/new/templates/BetterTemplate.html.erb"

# You can also override the file path directly
muc3 ={}, template_full_path: "/somewhere/else/UniqueTemplate.html.erb")

muc2.send(:template_full_path) # => "/somewhere/else/UniqueTemplate.html.erb"

Or by creating a new root object to base your components off of

class ParentComponent < Inkblot::Components::Component

  def template_base_path
    @template_base_path ||= "/path/to/my/templates"

class MyFirstComponent < ParentComponent
  # ...

class MySecondComponent < ParentComponent
  # ...
end # => "/path/to/my/templates/MyFirstComponent.html.erb" # => "/path/to/my/templates/MySecondComponent.html.erb"{}, { template_filename: "SpecialComponent.html.erb"}).send(:template_path)
# => "/path/to/my/templates/SpecialComponent.html.erb"
# Overriding the template_full_path method entirely
class MyNewComponent < Inkblot::Components::Component
  def template_path

mnc_tmp =
fil ="/path/to/my/templates/NewComponentTemplate.html.erb")

mnc_tmp == fil # => true

Components expect that they come with an HTML Template and should produce a bmp version of it for the display. If your components don't intend to use HTML templates, you can still output them to the screen by overriding the #convert function

class PhotoFrame < Inkblot::Components::Component
  # ...
  def convert
    # Return any converter, or elsewise displayable object here binary_img_data)


Helper classes share common behavior across components. They're nothing more than modules, with extendable and inheritable methods for Components.


The Icons helper assists with converting symbols to icons, both standard HTML symbols and Google Material Design Icons codepoints.

require 'inkblot/components/helpers/icons'

class AccessIndicator < Inkblot::Components::Component
  include Inkblot::Components::Helpers::Icons

  def to_display do |im|
      im.fullscreen = true
      im.size = 80
      im.font = 'Material Icons'
      im.text = icon_span

  def lock
    options[:locked] = true

  def unlock
    options[:locked] = false

  def locked?

  def display_icon
    icn = locked? ? :lock : :lock_open

  def icon_span

indi =

indi.locked? # => false
indi.icon_span # => <span>&#xE898;</span>

indi.lock.locked? # => true
indi.icon_span # => <span>&#xE897;</span>

The Paginated helper deals with content that has multiple subsequent views. You must define a method which sets up this pagination, and returns an integer page count

require 'inkblot/components/helpers/paginated'

class SlideShow < Inkblot::Components::Component
  include Inkblot::Components::Helpers::Paginated

  # Must return the page count
  paginate_with :pagify
  # Defaults to zero, but can be overridden
  start_page 0

  # More work could be done here to set up the pages themselves
  def pagify

  # Override to display different content depending on the page
  def to_display

s = do |sh|
  sh.fullscreen = true
  sh.slides = do |n| "Slide #{n}", border_size: 10, div_height: 95, div_width: 95)

s.page_count # => 10
s.current_page # => 0

# Page navigation
s.prev_page && s.current_page # => 0, won't go negative
s.next_page && s.current_page # => 1
s.current_page = (s.page_count - 1) # Can directly set as well
s.next_page && s.current_page # => 9, won't go out of bounds

The MultiState helper assists with defining and transitioning between enumerated states for content. States are defined via a class method, and the current state can be read/written. A different collection is maintained for each state.

require 'inkblot/components/helpers/multi_state'

class TrafficLight < Inkblot::Components::Component
  def_states :green, :yellow, :red
  default_state :red

  def initialize
    # "Setter" mode, set the content to the block's return value
    content_for_state(:green) { "#00FF00" }
    content_for_state(:yellow) { "#FFFF00" }
    content_for_state(:red) { "#FF0000" }

  def transition_state
    case state
    when :green
      state = :yellow
    when :yellow
      state = :red
    when :red
      state = :green



  def computed
    # "Getter" mode, implicitly uses the state method if no state is given
    { hex_color: content_for_state }

t =

t.content_for_state(:green) # => "#00FF00"
t.content_for_state(:yellow) # => "#FFFF00"
t.content_for_state(:red) # => "#FF0000"

t.state # => :red

t.transition_state.state # => :green
t.transition_state.state # => :yellow
t.transition_state.state # => :red


Bug reports, feature ideas, and pull requests are welcome on GitHub at


Ruby for connecting to Waveshare EPD screen



No releases published


No packages published