mirror of https://github.com/anoshenko/rui.git
405 lines
11 KiB
Go
405 lines
11 KiB
Go
package rui
|
|
|
|
import (
|
|
"fmt"
|
|
"strconv"
|
|
"strings"
|
|
"time"
|
|
)
|
|
|
|
const (
|
|
TimeChangedEvent = "time-changed"
|
|
TimePickerMin = "time-picker-min"
|
|
TimePickerMax = "time-picker-max"
|
|
TimePickerStep = "time-picker-step"
|
|
TimePickerValue = "time-picker-value"
|
|
timeFormat = "15:04:05"
|
|
)
|
|
|
|
// TimePicker - TimePicker view
|
|
type TimePicker interface {
|
|
View
|
|
}
|
|
|
|
type timePickerData struct {
|
|
viewData
|
|
timeChangedListeners []func(TimePicker, time.Time)
|
|
}
|
|
|
|
// NewTimePicker create new TimePicker object and return it
|
|
func NewTimePicker(session Session, params Params) TimePicker {
|
|
view := new(timePickerData)
|
|
view.init(session)
|
|
setInitParams(view, params)
|
|
return view
|
|
}
|
|
|
|
func newTimePicker(session Session) View {
|
|
return NewTimePicker(session, nil)
|
|
}
|
|
|
|
func (picker *timePickerData) init(session Session) {
|
|
picker.viewData.init(session)
|
|
picker.tag = "TimePicker"
|
|
picker.timeChangedListeners = []func(TimePicker, time.Time){}
|
|
}
|
|
|
|
func (picker *timePickerData) String() string {
|
|
return getViewString(picker)
|
|
}
|
|
|
|
func (picker *timePickerData) Focusable() bool {
|
|
return true
|
|
}
|
|
|
|
func (picker *timePickerData) normalizeTag(tag string) string {
|
|
tag = strings.ToLower(tag)
|
|
switch tag {
|
|
case Type, Min, Max, Step, Value:
|
|
return "time-picker-" + tag
|
|
}
|
|
|
|
return tag
|
|
}
|
|
|
|
func (picker *timePickerData) Remove(tag string) {
|
|
picker.remove(picker.normalizeTag(tag))
|
|
}
|
|
|
|
func (picker *timePickerData) remove(tag string) {
|
|
switch tag {
|
|
case TimeChangedEvent:
|
|
if len(picker.timeChangedListeners) > 0 {
|
|
picker.timeChangedListeners = []func(TimePicker, time.Time){}
|
|
picker.propertyChangedEvent(tag)
|
|
}
|
|
return
|
|
|
|
case TimePickerMin:
|
|
delete(picker.properties, TimePickerMin)
|
|
if picker.created {
|
|
removeProperty(picker.htmlID(), Min, picker.session)
|
|
}
|
|
|
|
case TimePickerMax:
|
|
delete(picker.properties, TimePickerMax)
|
|
if picker.created {
|
|
removeProperty(picker.htmlID(), Max, picker.session)
|
|
}
|
|
|
|
case TimePickerStep:
|
|
delete(picker.properties, TimePickerStep)
|
|
if picker.created {
|
|
removeProperty(picker.htmlID(), Step, picker.session)
|
|
}
|
|
|
|
case TimePickerValue:
|
|
if _, ok := picker.properties[TimePickerValue]; ok {
|
|
delete(picker.properties, TimePickerValue)
|
|
time := GetTimePickerValue(picker)
|
|
if picker.created {
|
|
picker.session.runScript(fmt.Sprintf(`setInputValue('%s', '%s')`, picker.htmlID(), time.Format(timeFormat)))
|
|
}
|
|
for _, listener := range picker.timeChangedListeners {
|
|
listener(picker, time)
|
|
}
|
|
} else {
|
|
return
|
|
}
|
|
|
|
default:
|
|
picker.viewData.remove(tag)
|
|
return
|
|
}
|
|
picker.propertyChangedEvent(tag)
|
|
}
|
|
|
|
func (picker *timePickerData) Set(tag string, value any) bool {
|
|
return picker.set(picker.normalizeTag(tag), value)
|
|
}
|
|
|
|
func (picker *timePickerData) set(tag string, value any) bool {
|
|
if value == nil {
|
|
picker.remove(tag)
|
|
return true
|
|
}
|
|
|
|
setTimeValue := func(tag string) (time.Time, bool) {
|
|
switch value := value.(type) {
|
|
case time.Time:
|
|
picker.properties[tag] = value
|
|
return value, true
|
|
|
|
case string:
|
|
if text, ok := picker.Session().resolveConstants(value); ok {
|
|
lowText := strings.ToLower(text)
|
|
pm := strings.HasSuffix(lowText, "pm") || strings.HasSuffix(lowText, "am")
|
|
|
|
var format string
|
|
switch len(strings.Split(text, ":")) {
|
|
case 2:
|
|
if pm {
|
|
format = "3:04 PM"
|
|
} else {
|
|
format = "15:04"
|
|
}
|
|
|
|
default:
|
|
if pm {
|
|
format = "03:04:05 PM"
|
|
} else {
|
|
format = "15:04:05"
|
|
}
|
|
}
|
|
|
|
if time, err := time.Parse(format, text); err == nil {
|
|
picker.properties[tag] = value
|
|
return time, true
|
|
} else {
|
|
ErrorLog(err.Error())
|
|
}
|
|
return time.Now(), false
|
|
}
|
|
}
|
|
|
|
notCompatibleType(tag, value)
|
|
return time.Now(), false
|
|
}
|
|
|
|
switch tag {
|
|
case TimePickerMin:
|
|
old, oldOK := getTimeProperty(picker, TimePickerMin, Min)
|
|
if time, ok := setTimeValue(TimePickerMin); ok {
|
|
if !oldOK || time != old {
|
|
if picker.created {
|
|
updateProperty(picker.htmlID(), Min, time.Format(timeFormat), picker.session)
|
|
}
|
|
picker.propertyChangedEvent(tag)
|
|
}
|
|
return true
|
|
}
|
|
|
|
case TimePickerMax:
|
|
old, oldOK := getTimeProperty(picker, TimePickerMax, Max)
|
|
if time, ok := setTimeValue(TimePickerMax); ok {
|
|
if !oldOK || time != old {
|
|
if picker.created {
|
|
updateProperty(picker.htmlID(), Max, time.Format(timeFormat), picker.session)
|
|
}
|
|
picker.propertyChangedEvent(tag)
|
|
}
|
|
return true
|
|
}
|
|
|
|
case TimePickerStep:
|
|
oldStep := GetTimePickerStep(picker)
|
|
if picker.setIntProperty(TimePickerStep, value) {
|
|
if step := GetTimePickerStep(picker); oldStep != step {
|
|
if picker.created {
|
|
if step > 0 {
|
|
updateProperty(picker.htmlID(), Step, strconv.Itoa(step), picker.session)
|
|
} else {
|
|
removeProperty(picker.htmlID(), Step, picker.session)
|
|
}
|
|
}
|
|
picker.propertyChangedEvent(tag)
|
|
}
|
|
return true
|
|
}
|
|
|
|
case TimePickerValue:
|
|
oldTime := GetTimePickerValue(picker)
|
|
if time, ok := setTimeValue(TimePickerValue); ok {
|
|
if time != oldTime {
|
|
if picker.created {
|
|
picker.session.runScript(fmt.Sprintf(`setInputValue('%s', '%s')`, picker.htmlID(), time.Format(timeFormat)))
|
|
}
|
|
for _, listener := range picker.timeChangedListeners {
|
|
listener(picker, time)
|
|
}
|
|
picker.propertyChangedEvent(tag)
|
|
}
|
|
return true
|
|
}
|
|
|
|
case TimeChangedEvent:
|
|
listeners, ok := valueToEventListeners[TimePicker, time.Time](value)
|
|
if !ok {
|
|
notCompatibleType(tag, value)
|
|
return false
|
|
} else if listeners == nil {
|
|
listeners = []func(TimePicker, time.Time){}
|
|
}
|
|
picker.timeChangedListeners = listeners
|
|
picker.propertyChangedEvent(tag)
|
|
return true
|
|
|
|
default:
|
|
return picker.viewData.set(tag, value)
|
|
}
|
|
return false
|
|
}
|
|
|
|
func (picker *timePickerData) Get(tag string) any {
|
|
return picker.get(picker.normalizeTag(tag))
|
|
}
|
|
|
|
func (picker *timePickerData) get(tag string) any {
|
|
switch tag {
|
|
case TimeChangedEvent:
|
|
return picker.timeChangedListeners
|
|
|
|
default:
|
|
return picker.viewData.get(tag)
|
|
}
|
|
}
|
|
|
|
func (picker *timePickerData) htmlTag() string {
|
|
return "input"
|
|
}
|
|
|
|
func (picker *timePickerData) htmlProperties(self View, buffer *strings.Builder) {
|
|
picker.viewData.htmlProperties(self, buffer)
|
|
|
|
buffer.WriteString(` type="time"`)
|
|
|
|
if min, ok := getTimeProperty(picker, TimePickerMin, Min); ok {
|
|
buffer.WriteString(` min="`)
|
|
buffer.WriteString(min.Format(timeFormat))
|
|
buffer.WriteByte('"')
|
|
}
|
|
|
|
if max, ok := getTimeProperty(picker, TimePickerMax, Max); ok {
|
|
buffer.WriteString(` max="`)
|
|
buffer.WriteString(max.Format(timeFormat))
|
|
buffer.WriteByte('"')
|
|
}
|
|
|
|
if step, ok := intProperty(picker, TimePickerStep, picker.Session(), 0); ok && step > 0 {
|
|
buffer.WriteString(` step="`)
|
|
buffer.WriteString(strconv.Itoa(step))
|
|
buffer.WriteByte('"')
|
|
}
|
|
|
|
buffer.WriteString(` value="`)
|
|
buffer.WriteString(GetTimePickerValue(picker).Format(timeFormat))
|
|
buffer.WriteByte('"')
|
|
|
|
buffer.WriteString(` oninput="editViewInputEvent(this)"`)
|
|
if picker.getRaw(ClickEvent) == nil {
|
|
buffer.WriteString(` onclick="stopEventPropagation(this, event)"`)
|
|
}
|
|
}
|
|
|
|
func (picker *timePickerData) htmlDisabledProperties(self View, buffer *strings.Builder) {
|
|
if IsDisabled(self) {
|
|
buffer.WriteString(` disabled`)
|
|
}
|
|
picker.viewData.htmlDisabledProperties(self, buffer)
|
|
}
|
|
|
|
func (picker *timePickerData) handleCommand(self View, command string, data DataObject) bool {
|
|
switch command {
|
|
case "textChanged":
|
|
if text, ok := data.PropertyValue("text"); ok {
|
|
if value, err := time.Parse(timeFormat, text); err == nil {
|
|
oldValue := GetTimePickerValue(picker)
|
|
picker.properties[TimePickerValue] = value
|
|
if value != oldValue {
|
|
for _, listener := range picker.timeChangedListeners {
|
|
listener(picker, value)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return true
|
|
}
|
|
|
|
return picker.viewData.handleCommand(self, command, data)
|
|
}
|
|
|
|
func getTimeProperty(view View, mainTag, shortTag string) (time.Time, bool) {
|
|
valueToTime := func(value any) (time.Time, bool) {
|
|
if value != nil {
|
|
switch value := value.(type) {
|
|
case time.Time:
|
|
return value, true
|
|
|
|
case string:
|
|
if text, ok := view.Session().resolveConstants(value); ok {
|
|
if result, err := time.Parse(timeFormat, text); err == nil {
|
|
return result, true
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return time.Now(), false
|
|
}
|
|
|
|
if view != nil {
|
|
if result, ok := valueToTime(view.getRaw(mainTag)); ok {
|
|
return result, true
|
|
}
|
|
|
|
if value := valueFromStyle(view, shortTag); value != nil {
|
|
if result, ok := valueToTime(value); ok {
|
|
return result, true
|
|
}
|
|
}
|
|
}
|
|
|
|
return time.Now(), false
|
|
}
|
|
|
|
// GetTimePickerMin returns the min time of TimePicker subview and "true" as the second value if the min time is set,
|
|
// "false" as the second value otherwise.
|
|
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
|
|
func GetTimePickerMin(view View, subviewID ...string) (time.Time, bool) {
|
|
if len(subviewID) > 0 && subviewID[0] != "" {
|
|
view = ViewByID(view, subviewID[0])
|
|
}
|
|
if view != nil {
|
|
return getTimeProperty(view, TimePickerMin, Min)
|
|
}
|
|
return time.Now(), false
|
|
}
|
|
|
|
// GetTimePickerMax returns the max time of TimePicker subview and "true" as the second value if the min time is set,
|
|
// "false" as the second value otherwise.
|
|
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
|
|
func GetTimePickerMax(view View, subviewID ...string) (time.Time, bool) {
|
|
if len(subviewID) > 0 && subviewID[0] != "" {
|
|
view = ViewByID(view, subviewID[0])
|
|
}
|
|
if view != nil {
|
|
return getTimeProperty(view, TimePickerMax, Max)
|
|
}
|
|
return time.Now(), false
|
|
}
|
|
|
|
// GetTimePickerStep returns the time changing step in seconds of TimePicker subview.
|
|
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
|
|
func GetTimePickerStep(view View, subviewID ...string) int {
|
|
return intStyledProperty(view, subviewID, TimePickerStep, 60)
|
|
}
|
|
|
|
// GetTimePickerValue returns the time of TimePicker subview.
|
|
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
|
|
func GetTimePickerValue(view View, subviewID ...string) time.Time {
|
|
if len(subviewID) > 0 && subviewID[0] != "" {
|
|
view = ViewByID(view, subviewID[0])
|
|
}
|
|
if view == nil {
|
|
return time.Now()
|
|
}
|
|
time, _ := getTimeProperty(view, TimePickerValue, Value)
|
|
return time
|
|
}
|
|
|
|
// GetTimeChangedListeners returns the TimeChangedListener list of an TimePicker subview.
|
|
// If there are no listeners then the empty list is returned
|
|
// If the second argument (subviewID) is not specified or it is "" then a value from the first argument (view) is returned.
|
|
func GetTimeChangedListeners(view View, subviewID ...string) []func(TimePicker, time.Time) {
|
|
return getEventListeners[TimePicker, time.Time](view, subviewID, TimeChangedEvent)
|
|
}
|