Skip to content

Data API 2: Binding Package

Stuart Scott edited this page Apr 13, 2020 · 35 revisions

The objective of this page is to define a Data Binding API that could be used by all widgets that need to update data and also be updated whenever the data is changed;

  • Button
  • Check
  • Entry
  • Hyperlink
  • Icon
  • Label
  • List*
  • ProgressBar
  • Radio
  • Select
  • Slider
  • Table*
  • Tree*

*The List, Table, and Tree widgets have not been designed / implemented yet, however an initial proposal for list is available (List Proposal).

Previous Work

In a previous iteration (Data API) the reflection package was used to establish a binding between a widget and a piece of data, however this had some shortcoming that this new API is intended to resolve;

  • Implicity < Explicity - reflection was used to establish a binding by setting a function field in a widget and would perform type conversion behind the scenes. This opacity made it difficult for developers to understand, and harder to test and debug.
  • Single Binding Only - since reflection would lookup the function field by name, it was only possible to have one binding per widget.
  • Performance and Deadlock Susceptibility - when a binding was updated it would trigger all listeners on the same thread, and a circular dependency could cause locking issues and race conditions.

Notifiable Interface

Notifiable is a simple interface that represents an object that can be notified.

type Notifiable interface {
	Notify(Binding)
}

NotifyFunction

NotifyFunction is a type that implements Notifiable by called the encompassed function field, which makes it possible for a function closure to listen to a binding.

Binding Interface

Binding is the base interface of the Data Binding API and defines the functions to add and remove the listeners notified when the bound data changes.

type Binding interface {
	AddListener(Notifiable)
	DeleteListener(Notifiable)
}

Base

Base implements the basic behaviour of a binding, including managing and notifying listeners.

type Base struct {...}
func (b *Base) AddListenerFunction(listener func(Binding)) *NotifyFunction
func (b *Base) AddListener(listener Notifiable)
func (b *Base) DeleteListener(listener Notifiable)

Generated Type-Specific Bindings

Base is extended by Typed Bindings to provide type-specific constructors, getters, setters, and listeners for a single data item.

type String struct {...}
func NewString(value string) *String
func (b *String) Get() string
func (b *String) Set(string)
func (b *String) AddStringListener(func(string)) *NotifyFunction

Supported Types

  • Bool
  • Float64
  • Int
  • Int64
  • fyne.Resource
  • Rune
  • String
  • *url.URL

Proposed Types

  • Byte
  • Float32
  • Int8, Int16, Int32
  • Uint, Uint8, Uint16, Uint32, Uint64

List

List encompasses a list of bindings and provides methods to get the Length of the list, Append to the list, and get/set individual elements of the list. List will notify its listeners whenever the list is append to, or an index is set. List will not notify its listeners when an individual element is updated, instead the UI is expected to bind to List as well as the individual element bindings within it.

type List struct {...}
func (b *List) Length() int
func (b *List) Get(int) Binding
func (b *List) Append(Binding)
func (b *List) Set(int, Binding)

Map

Map is similar to List but encompasses a map of string to binding and provides methods to get the Length of the map, and get/set individual elements of the map. Map will notify its listeners whenever a key/value is set. Map will not notify its listeners when an individual element is updated, instead the UI is expected to bind to Map as well as the individual element bindings within it.

type Map struct {...}
func (b *Map) Length() int
func (b *Map) Get(string) Binding
func (b *Map) Set(string, Binding)

Impact on existing widgets

The Data Binding API extends the current Widget API to provide Bind methods for individual parts of the Widget. For example, Radio has BindOptions(List) and BindSelected(String), ProgressBar has BindMin(Float64), BindMax(Float64), BindStep(Float64), and BindValue(Float64).

Example

This example demonstrates how to bind an Entry to a Label so that the Label displays the number of characters in the Entry.

text := binding.NewString("")
entry := widget.NewEntry().BindText(text)
label := widget.NewLabel("0")
text.AddStringListener(func(s string) {
	label.SetText(strconv.Itoa(len(s)))
})

Welcome to the Fyne wiki.

Project documentation

Getting involved

Platform details

Clone this wiki locally