summaryrefslogtreecommitdiff
path: root/indra/llui/llcheckboxctrl.cpp
blob: 8ba37a80758f44c89a7ce08e1578c646ec36ad4e (plain)
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
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
/**
 * @file llcheckboxctrl.cpp
 * @brief LLCheckBoxCtrl base class
 *
 * $LicenseInfo:firstyear=2001&license=viewerlgpl$
 * Second Life Viewer Source Code
 * Copyright (C) 2010, Linden Research, Inc.
 *
 * This library is free software; you can redistribute it and/or
 * modify it under the terms of the GNU Lesser General Public
 * License as published by the Free Software Foundation;
 * version 2.1 of the License only.
 *
 * This library is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
 * Lesser General Public License for more details.
 *
 * You should have received a copy of the GNU Lesser General Public
 * License along with this library; if not, write to the Free Software
 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
 *
 * Linden Research, Inc., 945 Battery Street, San Francisco, CA  94111  USA
 * $/LicenseInfo$
 */

// The mutants are coming!
#include "linden_common.h"

#define LLCHECKBOXCTRL_CPP
#include "llcheckboxctrl.h"

#include "llgl.h"
#include "llui.h"
#include "lluiconstants.h"
#include "lluictrlfactory.h"
#include "llcontrol.h"

#include "llstring.h"
#include "llfontgl.h"
#include "lltextbox.h"
#include "llkeyboard.h"

static LLDefaultChildRegistry::Register<LLCheckBoxCtrl> r("check_box");

// Compiler optimization, generate extern template
template class LLCheckBoxCtrl* LLView::getChild<class LLCheckBoxCtrl>(
    const std::string& name, BOOL recurse) const;

void LLCheckBoxCtrl::WordWrap::declareValues()
{
    declare("none", EWordWrap::WRAP_NONE);
    declare("down", EWordWrap::WRAP_DOWN);
    declare("up", EWordWrap::WRAP_UP);
}

LLCheckBoxCtrl::Params::Params()
:   initial_value("initial_value", false),
    label_text("label_text"),
    check_button("check_button"),
    word_wrap("word_wrap", EWordWrap::WRAP_NONE),
    radio_style("radio_style")
{}


LLCheckBoxCtrl::LLCheckBoxCtrl(const LLCheckBoxCtrl::Params& p)
:   LLUICtrl(p),
    mTextEnabledColor(p.label_text.text_color()),
    mTextDisabledColor(p.label_text.text_readonly_color()),
    mFont(p.font()),
    mWordWrap(p.word_wrap)
{
    mViewModel->setValue(LLSD(p.initial_value));
    mViewModel->resetDirty();
    static LLUICachedControl<S32> llcheckboxctrl_spacing ("UICheckboxctrlSpacing", 0);
    static LLUICachedControl<S32> llcheckboxctrl_hpad ("UICheckboxctrlHPad", 0);
    static LLUICachedControl<S32> llcheckboxctrl_vpad ("UICheckboxctrlVPad", 0);

    // must be big enough to hold all children
    setUseBoundingRect(TRUE);

    // *HACK Get rid of this with SL-55508...
    // this allows blank check boxes and radio boxes for now
    std::string local_label = p.label;
    if(local_label.empty())
    {
        local_label = " ";
    }

    LLTextBox::Params tbparams = p.label_text;
    tbparams.initial_value(local_label);
    if (p.font.isProvided())
    {
        tbparams.font(p.font);
    }

    mLabel = LLUICtrlFactory::create<LLTextBox>(tbparams);
    if (mWordWrap != WRAP_NONE)
    {
        // Not setWordWrap(mWordWrap != WRAP_NONE) because there might be some old lurking code that sets it manually
        mLabel->setWordWrap(true);
        S32 new_width = getRect().getWidth() - p.check_button.rect().getWidth() - llcheckboxctrl_hpad;
        LLRect label_rect = mLabel->getRect();
        label_rect.mRight = label_rect.mLeft + new_width;
        mLabel->setRect(label_rect);
    }
    mLabel->reshapeToFitText();

    LLRect label_rect = mLabel->getRect();
    if (mLabel->getLineCount() > 1)
    {
        if (mWordWrap == WRAP_DOWN)
        {
            // reshapeToFitText uses LLView::reshape() which always reshapes
            // from bottom to top, but we want to extend the bottom
            // Note: might be better idea to use getRect().mTop of LLCheckBoxCtrl (+pad) as top point of new rect
            S32 delta = ll_round((F32)mLabel->getFont()->getLineHeight() * mLabel->getLineSpacingMult()) - label_rect.getHeight();
            label_rect.translate(0, delta);
            mLabel->setRect(label_rect);
        }
        // else
        // WRAP_UP is essentially done by reshapeToFitText() (extends from bottom to top)
        // howhever it doesn't respect rect of checkbox
        // todo: this should be fixed, but there are at least couple checkboxes that use this feature as is.
    }

    addChild(mLabel);

    // Button
    // Note: button cover the label by extending all the way to the right and down.
    LLRect btn_rect = p.check_button.rect();
    btn_rect.setOriginAndSize(
        btn_rect.mLeft,
        llmin(btn_rect.mBottom, label_rect.mBottom),
        llmax(btn_rect.mRight, label_rect.mRight - btn_rect.mLeft),
        llmax(label_rect.getHeight(), btn_rect.mTop));
    std::string active_true_id, active_false_id;
    std::string inactive_true_id, inactive_false_id;

    LLButton::Params params = p.check_button;
    params.rect(btn_rect);
    //params.control_name(p.control_name);
    params.click_callback.function(boost::bind(&LLCheckBoxCtrl::onCommit, this));
    params.commit_on_return(false);
    // Checkboxes only allow boolean initial values, but buttons can
    // take any LLSD.
    params.initial_value(LLSD(p.initial_value));
    params.follows.flags(FOLLOWS_LEFT | FOLLOWS_BOTTOM);

    mButton = LLUICtrlFactory::create<LLButton>(params);
    addChild(mButton);
}

LLCheckBoxCtrl::~LLCheckBoxCtrl()
{
    // Children all cleaned up by default view destructor.
}

void LLCheckBoxCtrl::onCommit()
{
    if( getEnabled() )
    {
        setTentative(FALSE);
        setControlValue(getValue());
        LLUICtrl::onCommit();
    }
}

void LLCheckBoxCtrl::setEnabled(BOOL b)
{
    LLView::setEnabled(b);

    if (b)
    {
        mLabel->setColor( mTextEnabledColor.get() );
    }
    else
    {
        mLabel->setColor( mTextDisabledColor.get() );
    }
}

void LLCheckBoxCtrl::clear()
{
    setValue( FALSE );
}

void LLCheckBoxCtrl::reshape(S32 width, S32 height, BOOL called_from_parent)
{
    LLRect rect = getRect();
    S32 delta_width = width - rect.getWidth();
    S32 delta_height = height - rect.getHeight();

    if (delta_width || delta_height)
    {
        // adjust our rectangle
        rect.mRight = getRect().mLeft + width;
        rect.mTop = getRect().mBottom + height;
        setRect(rect);
    }

    // reshapeToFitText reshapes label to minimal size according to last bounding box
    // it will work fine in case of decrease of space, but if we get more space or text
    // becomes longer, label will fail to grow so reinit label's dimentions.

    LLRect label_rect = mLabel->getRect();
    S32 new_width = rect.getWidth() - label_rect.mLeft;
    mLabel->reshape(new_width, label_rect.getHeight(), TRUE);

    S32 label_top = label_rect.mTop;
    mLabel->reshapeToFitText(TRUE);

    label_rect = mLabel->getRect();
    if (label_top != label_rect.mTop && mWordWrap == WRAP_DOWN)
    {
        // reshapeToFitText uses LLView::reshape() which always reshapes
        // from bottom to top, but we want to extend the bottom so
        // reposition control
        S32 delta = label_top - label_rect.mTop;
        label_rect.translate(0, delta);
        mLabel->setRect(label_rect);
    }

    // Button
    // Note: button cover the label by extending all the way to the right and down.
    LLRect btn_rect = mButton->getRect();
    btn_rect.setOriginAndSize(
        btn_rect.mLeft,
        llmin(btn_rect.mBottom, label_rect.mBottom),
        llmax(btn_rect.getWidth(), label_rect.mRight - btn_rect.mLeft),
        llmax(label_rect.mTop - btn_rect.mBottom, btn_rect.getHeight()));
    mButton->setShape(btn_rect);

    updateBoundingRect();
}

//virtual
void LLCheckBoxCtrl::setValue(const LLSD& value )
{
    mButton->setValue( value );
}

//virtual
LLSD LLCheckBoxCtrl::getValue() const
{
    return mButton->getValue();
}

//virtual
void LLCheckBoxCtrl::setTentative(BOOL b)
{
    mButton->setTentative(b);
}

//virtual
BOOL LLCheckBoxCtrl::getTentative() const
{
    return mButton->getTentative();
}

void LLCheckBoxCtrl::setLabel( const LLStringExplicit& label )
{
    mLabel->setText( label );
    reshape(getRect().getWidth(), getRect().getHeight(), FALSE);
}

std::string LLCheckBoxCtrl::getLabel() const
{
    return mLabel->getText();
}

BOOL LLCheckBoxCtrl::setLabelArg( const std::string& key, const LLStringExplicit& text )
{
    BOOL res = mLabel->setTextArg(key, text);
    reshape(getRect().getWidth(), getRect().getHeight(), FALSE);
    return res;
}

// virtual
void LLCheckBoxCtrl::setControlName(const std::string& control_name, LLView* context)
{
    mButton->setControlName(control_name, context);
}


// virtual      Returns TRUE if the user has modified this control.
BOOL     LLCheckBoxCtrl::isDirty() const
{
    if ( mButton )
    {
        return mButton->isDirty();
    }
    return FALSE;       // Shouldn't get here
}


// virtual          Clear dirty state
void    LLCheckBoxCtrl::resetDirty()
{
    if ( mButton )
    {
        mButton->resetDirty();
    }
}