2014-02-14 11:16:27 -06:00
// 14 february 2014
2014-03-12 20:55:45 -05:00
2014-02-19 10:41:10 -06:00
package ui
2014-02-14 11:16:27 -06:00
import (
2014-03-07 14:01:42 -06:00
"fmt"
2014-02-14 11:16:27 -06:00
"sync"
)
2014-03-07 18:28:25 -06:00
// A Combobox is a drop-down list of items, of which at most one can be selected at any given time. You may optionally make the combobox editable to allow custom items. Initially, no item will be selected (and no text entered in an editable Combobox's entry field).
2014-02-14 11:16:27 -06:00
type Combobox struct {
// TODO Select event
lock sync . Mutex
created bool
sysData * sysData
initItems [ ] string
}
2014-02-25 14:13:37 -06:00
func newCombobox ( editable bool , items ... string ) ( c * Combobox ) {
2014-02-14 11:16:27 -06:00
c = & Combobox {
sysData : mksysdata ( c_combobox ) ,
initItems : items ,
}
2014-02-14 14:54:56 -06:00
c . sysData . alternate = editable
2014-02-14 11:16:27 -06:00
return c
}
2014-02-25 14:13:37 -06:00
// NewCombobox makes a new Combobox with the given items.
func NewCombobox ( items ... string ) * Combobox {
return newCombobox ( false , items ... )
}
// NewEditableCombobox makes a new editable Combobox with the given items.
func NewEditableCombobox ( items ... string ) * Combobox {
return newCombobox ( true , items ... )
}
2014-03-07 14:01:42 -06:00
// Append adds items to the end of the Combobox's list.
2014-03-09 15:02:17 -05:00
// Append will panic if something goes wrong on platforms that do not abort themselves.
func ( c * Combobox ) Append ( what ... string ) {
2014-03-18 10:50:56 -05:00
c . lock . Lock ( )
defer c . lock . Unlock ( )
2014-02-15 16:52:33 -06:00
if c . created {
2014-03-09 15:56:45 -05:00
for _ , s := range what {
2014-03-09 15:02:17 -05:00
c . sysData . append ( s )
2014-03-07 14:01:42 -06:00
}
2014-03-09 15:02:17 -05:00
return
2014-02-15 16:52:33 -06:00
}
2014-03-07 14:01:42 -06:00
c . initItems = append ( c . initItems , what ... )
2014-02-15 16:52:33 -06:00
}
2014-03-09 09:41:07 -05:00
// InsertBefore inserts a new item in the Combobox before the item at the given position. It panics if the given index is out of bounds.
2014-03-09 15:02:17 -05:00
// InsertBefore will also panic if something goes wrong on platforms that do not abort themselves.
func ( c * Combobox ) InsertBefore ( what string , before int ) {
2014-03-18 10:50:56 -05:00
c . lock . Lock ( )
defer c . lock . Unlock ( )
2014-03-09 09:41:07 -05:00
var m [ ] string
2014-02-15 16:52:33 -06:00
if c . created {
2014-03-09 09:41:07 -05:00
if before < 0 || before >= c . sysData . len ( ) {
goto badrange
}
2014-03-09 15:02:17 -05:00
c . sysData . insertBefore ( what , before )
return
2014-02-15 16:52:33 -06:00
}
2014-03-09 09:41:07 -05:00
if before < 0 || before >= len ( c . initItems ) {
goto badrange
}
m = make ( [ ] string , 0 , len ( c . initItems ) + 1 )
2014-02-15 16:52:33 -06:00
m = append ( m , c . initItems [ : before ] ... )
m = append ( m , what )
c . initItems = append ( m , c . initItems [ before : ] ... )
2014-03-09 15:02:17 -05:00
return
2014-03-09 09:41:07 -05:00
badrange :
panic ( fmt . Errorf ( "index %d out of range in Combobox.InsertBefore()" , before ) )
2014-02-15 16:52:33 -06:00
}
2014-03-09 09:41:07 -05:00
// Delete removes the given item from the Combobox. It panics if the given index is out of bounds.
2014-03-11 12:50:02 -05:00
func ( c * Combobox ) Delete ( index int ) {
2014-03-18 10:50:56 -05:00
c . lock . Lock ( )
defer c . lock . Unlock ( )
2014-02-15 17:14:43 -06:00
if c . created {
2014-03-09 09:41:07 -05:00
if index < 0 || index >= c . sysData . len ( ) {
goto badrange
}
2014-03-11 12:50:02 -05:00
c . sysData . delete ( index )
return
2014-02-15 17:14:43 -06:00
}
2014-03-09 09:41:07 -05:00
if index < 0 || index >= len ( c . initItems ) {
goto badrange
}
2014-02-15 17:14:43 -06:00
c . initItems = append ( c . initItems [ : index ] , c . initItems [ index + 1 : ] ... )
2014-03-11 12:50:02 -05:00
return
2014-03-09 09:41:07 -05:00
badrange :
panic ( fmt . Errorf ( "index %d out of range in Combobox.Delete()" , index ) )
2014-02-15 17:14:43 -06:00
}
2014-02-14 11:16:27 -06:00
// Selection returns the current selection.
2014-02-15 12:36:24 -06:00
func ( c * Combobox ) Selection ( ) string {
2014-03-18 10:50:56 -05:00
c . lock . Lock ( )
defer c . lock . Unlock ( )
2014-02-14 19:41:36 -06:00
if c . created {
return c . sysData . text ( )
}
2014-02-15 12:36:24 -06:00
return ""
2014-02-14 11:16:27 -06:00
}
2014-02-15 17:27:34 -06:00
// SelectedIndex returns the index of the current selection in the Combobox. It returns -1 either if no selection was made or if text was manually entered in an editable Combobox.
func ( c * Combobox ) SelectedIndex ( ) int {
2014-03-18 10:50:56 -05:00
c . lock . Lock ( )
defer c . lock . Unlock ( )
2014-02-15 17:27:34 -06:00
if c . created {
return c . sysData . selectedIndex ( )
}
return - 1
}
2014-02-14 11:16:27 -06:00
2014-03-08 15:42:57 -06:00
// Len returns the number of items in the Combobox.
//
// On platforms for which this function may return an error, it panics if one is returned.
func ( c * Combobox ) Len ( ) int {
2014-03-18 10:50:56 -05:00
c . lock . Lock ( )
defer c . lock . Unlock ( )
2014-03-08 15:42:57 -06:00
if c . created {
return c . sysData . len ( )
}
return len ( c . initItems )
}
2014-02-14 11:16:27 -06:00
func ( c * Combobox ) make ( window * sysData ) ( err error ) {
c . lock . Lock ( )
defer c . lock . Unlock ( )
2014-04-01 15:43:56 -05:00
err = c . sysData . make ( window )
2014-02-14 11:16:27 -06:00
if err != nil {
return err
}
for _ , s := range c . initItems {
2014-03-09 15:56:45 -05:00
c . sysData . append ( s )
2014-02-14 11:16:27 -06:00
}
2014-02-14 19:41:36 -06:00
c . created = true
2014-02-14 11:16:27 -06:00
return nil
}
2014-03-17 20:09:03 -05:00
func ( c * Combobox ) setRect ( x int , y int , width int , height int , rr * [ ] resizerequest ) {
* rr = append ( * rr , resizerequest {
2014-03-17 19:42:36 -05:00
sysData : c . sysData ,
x : x ,
y : y ,
width : width ,
height : height ,
2014-03-17 20:09:03 -05:00
} )
2014-02-14 11:16:27 -06:00
}
2014-02-24 09:56:35 -06:00
2014-03-09 18:44:41 -05:00
func ( c * Combobox ) preferredSize ( ) ( width int , height int ) {
return c . sysData . preferredSize ( )
2014-02-24 09:56:35 -06:00
}