1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License in the LICENSE-APACHE file or at:
//     https://www.apache.org/licenses/LICENSE-2.0

//! Display widgets show information but are not interactive
// TODO: except `Entry`?

use std::fmt::{self, Debug};

use crate::macros::Widget;
use crate::event::{Action, Handler, NoResponse, err_num, err_unhandled};
use crate::{Class, Core, CoreData, HasText, Editable, TkWidget};

/// A simple, static text label
#[widget(class = Class::Text(self))]
#[handler(response = NoResponse)]
#[derive(Clone, Default, Debug, Widget)]
pub struct Text {
    #[core] core: CoreData,
    text: String,
}

impl Text {
    /// Construct a new, empty instance
    pub fn new() -> Self {
        Text {
            core: Default::default(),
            text: String::new()
        }
    }
}

impl<T> From<T> for Text where String: From<T> {
    fn from(text: T) -> Self {
        Text {
            core: Default::default(),
            text: String::from(text)
        }
    }
}

impl HasText for Text {
    fn get_text(&self) -> &str {
        &self.text
    }
    
    fn set_text(&mut self, tk: &TkWidget, text: &str) {
        tk.set_label(self.tkd(), text);
        self.text = text.into();
    }
}


/// An editable, single-line text box.
#[widget(class = Class::Entry(self))]
#[derive(Clone, Default, Widget)]
pub struct Entry<H: 'static> {
    #[core] core: CoreData,
    editable: bool,
    text: String,
    handler: H,
}

impl<H> Debug for Entry<H> {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        write!(f, "Entry {{ core: {:?}, editable: {:?}, text: {:?}, handler: <omitted> }}",
            self.core, self.editable, self.text)
    }
}

impl<R, H: Fn() -> R> Entry<H> {
    /// Construct an `Entry` with the given initial `text`.
    /// 
    /// The `handler` is called when [`Action::Activate`] is received
    /// (when the "enter" key is pressed)
    /// and its result is returned from the event handler.
    /// 
    /// [`Action::Activate`]: kas::event::Action::Activate
    pub fn new_on_activate(text: String, handler: H) -> Self {
        Entry {
            core: Default::default(),
            editable: true,
            text,
            handler,
        }
    }
}

impl Entry<()> {
    /// Construct an `Entry` which is optionally `editable`, and has the given
    /// inital `text`.
    pub fn new(editable: bool, text: String) -> Self {
        Entry {
            core: Default::default(),
            editable,
            text,
            handler: (),
        }
    }
}

impl<H> HasText for Entry<H> {
    fn get_text(&self) -> &str {
        &self.text
    }
    
    fn set_text(&mut self, tk: &TkWidget, text: &str) {
        tk.set_label(self.tkd(), text);
        self.text = text.into();
    }
}

impl<H> Editable for Entry<H> {
    fn is_editable(&self) -> bool {
        self.editable
    }
}

impl Handler for Entry<()> {
    type Response = NoResponse;
    
    fn handle_action(&mut self, _tk: &TkWidget, action: Action, num: u32) -> Self::Response {
        if num != self.number() {
            return err_num()
        }
        
        match action {
            Action::Activate => {
                NoResponse
            }
            a @ _ => err_unhandled(a)
        }
    }
}

impl<R: From<NoResponse>, H: Fn() -> R> Handler for Entry<H> {
    type Response = R;
    
    fn handle_action(&mut self, _tk: &TkWidget, action: Action, num: u32) -> Self::Response {
        if num != self.number() {
            return err_num()
        }
        
        match action {
            Action::Activate => {
                (self.handler)()
            }
            a @ _ => err_unhandled(a)
        }
    }
}