Cheatsheet

View Source

This is a very brief overview over the most important commands in the Fledex language

use Fledex library

Before you use any of the structures below, you first need to install the library and then use it.

Installation

Before you can use Fledex, you need to install the library. Either as a dependency in your mix.exs file, or as part of your setup in your livebook by calling

Mix.install([
  {:fledex, "~>0.6"}
])

Example

use Fledex

Description

By use-ing the library, it is possible to use the structures specified below. It is safe to use this even in a functional context, i.e. it does not need to be part of a module.

In addition, the following modules will be imported:

In addition a Fledex.Animation.Manager will be started in the background (if there is none running) which will manage all animations.

The led_strip

Example

led_strip :name, Kino do
  animation :seconds do
    %{seconds: seconds} -> leds(60) |> light(:red, offset: seconds)
    _ -> leds()
  end
  static :indicators do
    leds(5) |> light(:white) |> repeat(12)
    end
  end
  job :clock, ~e[* * * * * * *]e do
    date_time = DateTime.utc_now()
    broadcast_trigger(%{seconds: date_time.seconds})
  end
end

Parameters

The led_strip takes the following parameters:

ParameterDescription
name(mandatory) Every strip needs a unique name. This needs to be an atom
driver(mandatory) The driver that should be used to display the LEDs. The most common options are: Kino (for livebook), and Spi (for real hardware). The special driver :config returns the defined config, instead of configuring a driver.

The do ... end block

The following structures inside the do ... end block are allowed:

StructureDescription
animationDefines an animation that can change over time
staticDefines a certain pattern, it cannot change over time
effectDefines an effect that gets applied to an animation
jobDoes not define any LEDs, but some job that should be executed in regular intervals
componentDefines a component. It's equivalent to a set of static, animation, job, including effects. It allows to reuse certain aspects

See below for more details about them

The animation

Example

animation name, options do
  leds(10)
end

Parameters

The animation takes the following paramerters

ParameterDescription
name(mandatory) Every animation needs a unique name. This must be an atom
options(optional) A set of options, that is rarely used

The do ... end block

The structure inside the do ... end block can be any Elixir code, but it needs to return (at least) an Fledex.Leds structure (see below).

Caution: This function gets executed quite frequently and you need to ensure to keep it lightweight.

The code block is actually a function and it is possible to pattern match on a parameter (which is a map) and to extract information from the paramerter:

animation :name do
  %{trigger_name: value} -> leds(value)
  _ -> leds()
end

Note: it's important to notice that the map might not contain the information you expect especially during initialization phase. Make sure you have a "catch all" clause.

The static

The static structure is very similar to the animation structure, except that won't be updated and changed over time. Therefore it's also not possible to use the optional parameter in the do ... end block.

Example

static name, options do
  leds(10) |> light(:red)
end

Parameters

The static structure takes the following parameters:

ParameterDescription
name(mandatory) Every animation needs a unique name. This must be an atom
options(optional) A set of options, that is rarely used

The do ... end block

The structure inside the do ... end block can be any Elixir code, but it needs to return (at least) an Leds structure (see below).

Contrary to the animation the static structure only get called once and does not get called repeatedly.

The effect

Any animation can be wrapped into an effect that impacts how the animation is perceived.

Example

effect Fledex.Effect.Rotation, trigger_name: :john do
  animation :seconds do:
    ...
  end
  animation :minutes do:
    ...
  end
end

As can be seen in the example, the effect encapsulates one (or more) animations that it will impact.

Parameters

ParameterDescription
effect(mandatory) A module that implements the Fledex.Effect.Interface. You cand find some examples in the Fledex.Effect namespace, like: Dimming, Rotation, Offset, Wanish.
options(optional) A list of key value pairs that configure the effect. The settings are very specific to each effect.

The do ... end block

The do ... end block can contain, similar to the led_strip any of the fledex macros, but it only has a real effect on animations. Effects can even be nested, i.e. to have a Dimming around a Rotation effect which contains an animation.

The job

As mentioned above, very little work should be done as part of an animation. Any heavy lifting should be done in a separate process, and you probably don't want to do it very frequently anyway.

A job is there to shedule a task in regular intervals and to publish it to the animations. The animation can extract that information from the parameter that gets passed to the do ... end block (see above).

Example

job :name, ~e[* * * * * * *]e do
  date_time = DateTime.utc_now()
  broadcast_trigger(%{seconds: date_time.seconds})
end

Parameters

The job takes a couple of parameters:

ParameterDescription
name(mandatory) Every job needs to get a unique name. This must be an atom
pattern(mandatory) You need to specify a pattern when the job should run. The pattern can eithe rbe a normal crontab pattern (e.g. ~e[*/15]) with minute precision or an extended one (e.g. ~e[*/15]e) with second precision. More details can be found in this cheatsheet.
options(optional) one or more options can be specified (see below for details)

Options

The following options are recognized:

OptionDescription
:run_onceto run the directly after configuration, independent from the actual schedule
:timezonespecify a different timezone than the usual UTC
:overlapWhether different runs should overlap. Usually this is not desired

The do ... end block

The do ... end block can consist of any valid Elixir code. The code will be called without any arguments. Usually you want to use at the end a broadcast_trigger/1 call to publish data towards the animations that react to data and change their look.

The component

The component allows to encapsulate several animations and to reuse them into a single component. Examples are: Fledex.Component.Clock, Fledex.Component.Dot, and Fledex.Component.Thermometer.

Example

  component(:thermo, Thermometer,
    range: -20..40,
    trigger: :temperature,
    negative: :blue,
    null: :may_green,
    positive: :red,
    marker: :davy_s_grey
  )

Parameters

The component takes the following parameters:

ParametersDescription
name(mandatory) Every component needs a name, but since a component consists of potentially several animations, the component could also take a tuple of names that are then assigned to the different animations (otherwise they are generated from the base name passed in)
component(mandatory) The name of the module that defines the component (that implements the Fledex.Component.Interface behaviour)
options(mandatory) A lit of options. The exact options are dependent on the component

The Fledex.Leds struct

The Fledex.Leds module allows to define a sequence of leds

Example

leds(3)
  |> light(:red)
  |> light(:green)
  |> light(:blue)
  |> repeat(5)

Creation

FunctionDescription
leds/0Creates an led sequence with zero length. This sounds maybe stupid but is useful if you don't want to define an animation (maybe because you are maybe still not fully initialized)
leds/1Creates an led sequence with a specified length
leds/2(rarely used) Creates an led sequence with a specific length and some options (important for the send function).
leds/3Creates an led sequence with a list of colors that will be assigned to the leds. Pass an empty map (%{}) as the last argument
leds/4(rarely used) Creates an led sequence with all the bells and whisles: count, list of colors, options, and some meta information

Lights

FunctionDescription
light/2This sets the color of the next led. Whenever you call this function an internal counter will be updated to the next led. Therefore you can nicely sequence it. Note:. The "color" can also be another led sequence that will be added at the corresponding position.
light/3This sets a color with some options, like to offset and whether it repeats. Note: the offset is 1 indexed and must be greater than zero.

Other Functions

The above functions would be enough, but some special functions make life easier:

FunctionDescription
repeat/2This function does what it says. It will repeat the current sequence n-times and make the whole sequence n-times longer.
rainbow/2This function creates a rainbow over the leds. It takes a couple of parameters to control the rainbow effect.
gradient/4This function creates a gradient between the two colors specified, with a couple of options to control the exact behaviour.

The colors

Fledex has a very extensive color list as can be found on Wikipedia. Take a look at Fledex.Color.Names for the full list of colors, but here the most important colors:

Examples

red
orange
yellow
green
cyan
blue
violet

Usage

There are several ways on how a color can be applied to an Fledex.Leds struct:

Atom

leds(1) |> light(:red)

Function

leds(1) |> red()