summaryrefslogtreecommitdiff
path: root/indra/newview/llpanelmediasettingsgeneral.cpp
blob: 85c173a418c06524f75a74cb9e5ab81dd7a41cbf (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
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
/**
 * @file llpanelmediasettingsgeneral.cpp
 * @brief LLPanelMediaSettingsGeneral class implementation
 *
 * $LicenseInfo:firstyear=2009&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$
 */

#include "llviewerprecompiledheaders.h"

#include "llpanelmediasettingsgeneral.h"

// library includes
#include "llcombobox.h"
#include "llcheckboxctrl.h"
#include "llnotificationsutil.h"
#include "llspinctrl.h"
#include "lluictrlfactory.h"

// project includes
#include "llagent.h"
#include "llviewerwindow.h"
#include "llviewermedia.h"
#include "llvovolume.h"
#include "llsdutil.h"
#include "llselectmgr.h"
#include "llbutton.h"
#include "lltexturectrl.h"
#include "llurl.h"
#include "llwindow.h"
#include "llmediaentry.h"
#include "llmediactrl.h"
#include "llpanelcontents.h"
#include "llpermissions.h"
#include "llpluginclassmedia.h"
#include "llfloatermediasettings.h"
#include "llfloatertools.h"
#include "lltrans.h"
#include "lltextbox.h"
#include "llpanelmediasettingssecurity.h"

const char *CHECKERBOARD_DATA_URL = "data:image/svg+xml,%3Csvg xmlns=%22http://www.w3.org/2000/svg%22 width=%22100%%22 height=%22100%%22 %3E%3Cdefs%3E%3Cpattern id=%22checker%22 patternUnits=%22userSpaceOnUse%22 x=%220%22 y=%220%22 width=%22128%22 height=%22128%22 viewBox=%220 0 128 128%22 %3E%3Crect x=%220%22 y=%220%22 width=%2264%22 height=%2264%22 fill=%22#ddddff%22 /%3E%3Crect x=%2264%22 y=%2264%22 width=%2264%22 height=%2264%22 fill=%22#ddddff%22 /%3E%3C/pattern%3E%3C/defs%3E%3Crect x=%220%22 y=%220%22 width=%22100%%22 height=%22100%%22 fill=%22url(#checker)%22 /%3E%3C/svg%3E";

////////////////////////////////////////////////////////////////////////////////
//
LLPanelMediaSettingsGeneral::LLPanelMediaSettingsGeneral() :
    mAutoLoop( NULL ),
    mFirstClick( NULL ),
    mAutoZoom( NULL ),
    mAutoPlay( NULL ),
    mAutoScale( NULL ),
    mWidthPixels( NULL ),
    mHeightPixels( NULL ),
    mHomeURL( NULL ),
    mCurrentURL( NULL ),
    mParent( NULL ),
    mMediaEditable(false)
{
    // build dialog from XML
    buildFromFile( "panel_media_settings_general.xml");
}

////////////////////////////////////////////////////////////////////////////////
//
bool LLPanelMediaSettingsGeneral::postBuild()
{
    // connect member vars with UI widgets
    mAutoLoop = getChild< LLCheckBoxCtrl >( LLMediaEntry::AUTO_LOOP_KEY );
    mAutoPlay = getChild< LLCheckBoxCtrl >( LLMediaEntry::AUTO_PLAY_KEY );
    mAutoScale = getChild< LLCheckBoxCtrl >( LLMediaEntry::AUTO_SCALE_KEY );
    mAutoZoom = getChild< LLCheckBoxCtrl >( LLMediaEntry::AUTO_ZOOM_KEY );
    mCurrentURL = getChild< LLTextBox >( LLMediaEntry::CURRENT_URL_KEY );
    mFirstClick = getChild< LLCheckBoxCtrl >( LLMediaEntry::FIRST_CLICK_INTERACT_KEY );
    mHeightPixels = getChild< LLSpinCtrl >( LLMediaEntry::HEIGHT_PIXELS_KEY );
    mHomeURL = getChild< LLLineEditor >( LLMediaEntry::HOME_URL_KEY );
    mWidthPixels = getChild< LLSpinCtrl >( LLMediaEntry::WIDTH_PIXELS_KEY );
    mPreviewMedia = getChild<LLMediaCtrl>("preview_media");
    mFailWhiteListText = getChild<LLTextBox>( "home_fails_whitelist_label" );

    // watch commit action for HOME URL
    childSetCommitCallback( LLMediaEntry::HOME_URL_KEY, onCommitHomeURL, this);
    childSetCommitCallback( "current_url_reset_btn",onBtnResetCurrentUrl, this);

    return true;
}

////////////////////////////////////////////////////////////////////////////////
// virtual
LLPanelMediaSettingsGeneral::~LLPanelMediaSettingsGeneral()
{
}

////////////////////////////////////////////////////////////////////////////////
// static
void LLPanelMediaSettingsGeneral::draw()
{
    // housekeeping
    LLPanel::draw();

    // TODO: we need to call this repeatedly until the floater panels are fully
    // created but once we have a valid answer, we should stop looking here - the
    // commit callback will handle it
    checkHomeUrlPassesWhitelist();

    // enable/disable pixel values image entry based on auto scale checkbox
    if (!mAutoScale->getValue().asBoolean())
    {
        getChildView( LLMediaEntry::WIDTH_PIXELS_KEY )->setEnabled( true );
        getChildView( LLMediaEntry::HEIGHT_PIXELS_KEY )->setEnabled( true );
    }
    else
    {
        getChildView( LLMediaEntry::WIDTH_PIXELS_KEY )->setEnabled( false );
        getChildView( LLMediaEntry::HEIGHT_PIXELS_KEY )->setEnabled( false );
    };

    // enable/disable UI based on type of media
    bool reset_button_is_active = true;
    if (mPreviewMedia)
    {
        if (LLPluginClassMedia* media_plugin = mPreviewMedia->getMediaPlugin())
        {
            // turn off volume (if we can) for preview. Note: this really only
            // works for QuickTime movies right now - no way to control the
            // volume of a flash app embedded in a page for example
            media_plugin->setVolume( 0 );

            // some controls are only appropriate for time or browser type plugins
            // so we selectively enable/disable them - need to do it in draw
            // because the information from plugins arrives assynchronously
            if (media_plugin->pluginSupportsMediaTime())
            {
                getChildView( LLMediaEntry::CURRENT_URL_KEY )->setEnabled( false );
                reset_button_is_active = false;
                getChildView("current_url_label")->setEnabled(false );
                getChildView( LLMediaEntry::AUTO_LOOP_KEY )->setEnabled( true );
            }
            else
            {
                getChildView( LLMediaEntry::CURRENT_URL_KEY )->setEnabled( true );
                reset_button_is_active = true;
                getChildView("current_url_label")->setEnabled(true );
                getChildView( LLMediaEntry::AUTO_LOOP_KEY )->setEnabled( false );
            };
        };
    };

    // current URL can change over time, update it here
    updateCurrentUrl();

    LLPermissions perm;
    bool user_can_press_reset = mMediaEditable;

    // several places modify this widget so we must collect states in one place
    if ( reset_button_is_active )
    {
        // user has perms to press reset button and it is active
        if ( user_can_press_reset )
        {
            getChildView("current_url_reset_btn")->setEnabled(true );
        }
        // user does not has perms to press reset button and it is active
        else
        {
            getChildView("current_url_reset_btn")->setEnabled(false );
        };
    }
    else
    // reset button is inactive so we just slam it to off - other states don't matter
    {
        getChildView("current_url_reset_btn")->setEnabled(false );
    };
}

////////////////////////////////////////////////////////////////////////////////
// static
void LLPanelMediaSettingsGeneral::clearValues( void* userdata, bool editable, bool update_preview)
{
    LLPanelMediaSettingsGeneral *self =(LLPanelMediaSettingsGeneral *)userdata;
    self->mAutoLoop->clear();
    self->mAutoPlay->clear();
    self->mAutoScale->clear();
    self->mAutoZoom ->clear();
    self->mCurrentURL->clear();
    self->mFirstClick->clear();
    self->mHeightPixels->clear();
    self->mHomeURL->clear();
    self->mWidthPixels->clear();
    self->mAutoLoop ->setEnabled(editable);
    self->mAutoPlay ->setEnabled(editable);
    self->mAutoScale ->setEnabled(editable);
    self->mAutoZoom  ->setEnabled(editable);
    self->mCurrentURL ->setEnabled(editable);
    self->mFirstClick ->setEnabled(editable);
    self->mHeightPixels ->setEnabled(editable);
    self->mHomeURL ->setEnabled(editable);
    self->mWidthPixels ->setEnabled(editable);
    if (update_preview)
    {
        self->updateMediaPreview();
    }
}

// static
bool LLPanelMediaSettingsGeneral::isMultiple()
{
    // IF all the faces have media (or all dont have media)
    if ( LLFloaterMediaSettings::getInstance()->mIdenticalHasMediaInfo )
    {
        if(LLFloaterMediaSettings::getInstance()->mMultipleMedia)
        {
            return true;
        }

    }
    else
    {
        if(LLFloaterMediaSettings::getInstance()->mMultipleValidMedia)
        {
            return true;
        }
    }
    return false;
}

////////////////////////////////////////////////////////////////////////////////
// static
void LLPanelMediaSettingsGeneral::initValues( void* userdata, const LLSD& _media_settings, bool editable)
{
    LLPanelMediaSettingsGeneral *self =(LLPanelMediaSettingsGeneral *)userdata;
    self->mMediaEditable = editable;

    LLSD media_settings = _media_settings;

    if ( LLPanelMediaSettingsGeneral::isMultiple() )
    {
        // *HACK:  "edit" the incoming media_settings
        media_settings[LLMediaEntry::CURRENT_URL_KEY] = LLTrans::getString("Multiple Media");
        media_settings[LLMediaEntry::HOME_URL_KEY] = LLTrans::getString("Multiple Media");
    }

    std::string base_key( "" );
    std::string tentative_key( "" );

    struct
    {
        std::string key_name;
        LLUICtrl* ctrl_ptr;
        std::string ctrl_type;

    } data_set [] =
    {
        { LLMediaEntry::AUTO_LOOP_KEY,              self->mAutoLoop,        "LLCheckBoxCtrl" },
        { LLMediaEntry::AUTO_PLAY_KEY,              self->mAutoPlay,        "LLCheckBoxCtrl" },
        { LLMediaEntry::AUTO_SCALE_KEY,             self->mAutoScale,       "LLCheckBoxCtrl" },
        { LLMediaEntry::AUTO_ZOOM_KEY,              self->mAutoZoom,        "LLCheckBoxCtrl" },
        { LLMediaEntry::CURRENT_URL_KEY,            self->mCurrentURL,      "LLTextBox" },
        { LLMediaEntry::HEIGHT_PIXELS_KEY,          self->mHeightPixels,    "LLSpinCtrl" },
        { LLMediaEntry::HOME_URL_KEY,               self->mHomeURL,         "LLLineEditor" },
        { LLMediaEntry::FIRST_CLICK_INTERACT_KEY,   self->mFirstClick,      "LLCheckBoxCtrl" },
        { LLMediaEntry::WIDTH_PIXELS_KEY,           self->mWidthPixels,     "LLSpinCtrl" },
        { "", NULL , "" }
    };

    for( int i = 0; data_set[ i ].key_name.length() > 0; ++i )
    {
        base_key = std::string( data_set[ i ].key_name );
        tentative_key = base_key + std::string( LLPanelContents::TENTATIVE_SUFFIX );
        // TODO: CP - I bet there is a better way to do this using Boost
        if ( media_settings[ base_key ].isDefined() )
        {
            if ( data_set[ i ].ctrl_type == "LLLineEditor" )
            {
                static_cast< LLLineEditor* >( data_set[ i ].ctrl_ptr )->
                    setText( media_settings[ base_key ].asString() );
            }
            else
            if ( data_set[ i ].ctrl_type == "LLCheckBoxCtrl" )
                static_cast< LLCheckBoxCtrl* >( data_set[ i ].ctrl_ptr )->
                    setValue( media_settings[ base_key ].asBoolean() );
            else
            if ( data_set[ i ].ctrl_type == "LLComboBox" )
                static_cast< LLComboBox* >( data_set[ i ].ctrl_ptr )->
                    setCurrentByIndex( media_settings[ base_key ].asInteger() );
            else
            if ( data_set[ i ].ctrl_type == "LLSpinCtrl" )
                static_cast< LLSpinCtrl* >( data_set[ i ].ctrl_ptr )->
                    setValue( media_settings[ base_key ].asInteger() );

            data_set[ i ].ctrl_ptr->setEnabled(self->mMediaEditable);
            data_set[ i ].ctrl_ptr->setTentative( media_settings[ tentative_key ].asBoolean() );
        };
    };
}

////////////////////////////////////////////////////////////////////////////////
// Helper to set media control to media URL as required
void LLPanelMediaSettingsGeneral::updateMediaPreview()
{
    if(LLTrans::getString("Multiple Media") == mHomeURL->getValue().asString())
    {
        return;
    }
    if ( mHomeURL->getValue().asString().length() > 0 )
    {
        if(mPreviewMedia->getCurrentNavUrl() != mHomeURL->getValue().asString())
        {
            mPreviewMedia->navigateTo( mHomeURL->getValue().asString() );
        }
    }
    else
    // new home URL will be empty if media is deleted so display a
    // "preview goes here" data url page
    {
        if(mPreviewMedia->getCurrentNavUrl() != CHECKERBOARD_DATA_URL)
        {
            mPreviewMedia->navigateTo( CHECKERBOARD_DATA_URL );
        }
    };
}

////////////////////////////////////////////////////////////////////////////////

// virtual
void LLPanelMediaSettingsGeneral::onClose(bool app_quitting)
{
    if(mPreviewMedia)
    {
        mPreviewMedia->unloadMediaSource();
    }
}

////////////////////////////////////////////////////////////////////////////////
//
void LLPanelMediaSettingsGeneral::checkHomeUrlPassesWhitelist()
{
    // parent floater has not constructed the security panel yet
    if ( mParent->getPanelSecurity() == 0 )
        return;

    std::string home_url = getHomeUrl();
    if ( home_url.empty() || mParent->getPanelSecurity()->urlPassesWhiteList( home_url ) )
    {
        // Home URL is empty or passes the white list so hide the warning message
        mFailWhiteListText->setVisible( false );
    }
    else
    {
        // Home URL does not pass the white list so show the warning message
        mFailWhiteListText->setVisible( true );
    };
}

////////////////////////////////////////////////////////////////////////////////
// static
void LLPanelMediaSettingsGeneral::onCommitHomeURL( LLUICtrl* ctrl, void *userdata )
{
    LLPanelMediaSettingsGeneral* self =(LLPanelMediaSettingsGeneral *)userdata;

    // check home url passes whitelist and display warning if not
    self->checkHomeUrlPassesWhitelist();

    self->updateMediaPreview();
}

////////////////////////////////////////////////////////////////////////////////
// static
void LLPanelMediaSettingsGeneral::onBtnResetCurrentUrl(LLUICtrl* ctrl, void *userdata)
{
    LLPanelMediaSettingsGeneral* self =(LLPanelMediaSettingsGeneral *)userdata;
    self->navigateHomeSelectedFace(false);
}

////////////////////////////////////////////////////////////////////////////////
//
void LLPanelMediaSettingsGeneral::preApply()
{
    // Make sure the home URL entry is committed
    mHomeURL->onCommit();
}

////////////////////////////////////////////////////////////////////////////////
//
void LLPanelMediaSettingsGeneral::getValues( LLSD &fill_me_in, bool include_tentative )
{
    if (include_tentative || !mAutoLoop->getTentative()) fill_me_in[LLMediaEntry::AUTO_LOOP_KEY] = (LLSD::Boolean)mAutoLoop->getValue();
    if (include_tentative || !mAutoPlay->getTentative()) fill_me_in[LLMediaEntry::AUTO_PLAY_KEY] = (LLSD::Boolean)mAutoPlay->getValue();
    if (include_tentative || !mAutoScale->getTentative()) fill_me_in[LLMediaEntry::AUTO_SCALE_KEY] = (LLSD::Boolean)mAutoScale->getValue();
    if (include_tentative || !mAutoZoom->getTentative()) fill_me_in[LLMediaEntry::AUTO_ZOOM_KEY] = (LLSD::Boolean)mAutoZoom->getValue();
    //Don't fill in current URL: this is only supposed to get changed via navigate
    // if (include_tentative || !mCurrentURL->getTentative()) fill_me_in[LLMediaEntry::CURRENT_URL_KEY] = mCurrentURL->getValue();
    if (include_tentative || !mHeightPixels->getTentative()) fill_me_in[LLMediaEntry::HEIGHT_PIXELS_KEY] = (LLSD::Integer)mHeightPixels->getValue();
    // Don't fill in the home URL if it is the special "Multiple Media" string!
    if ((include_tentative || !mHomeURL->getTentative())
        && LLTrans::getString("Multiple Media") != mHomeURL->getValue())
            fill_me_in[LLMediaEntry::HOME_URL_KEY] = (LLSD::String)mHomeURL->getValue();
    if (include_tentative || !mFirstClick->getTentative()) fill_me_in[LLMediaEntry::FIRST_CLICK_INTERACT_KEY] = (LLSD::Boolean)mFirstClick->getValue();
    if (include_tentative || !mWidthPixels->getTentative()) fill_me_in[LLMediaEntry::WIDTH_PIXELS_KEY] = (LLSD::Integer)mWidthPixels->getValue();
}

////////////////////////////////////////////////////////////////////////////////
//
void LLPanelMediaSettingsGeneral::postApply()
{
    // Make sure to navigate to the home URL if the current URL is empty and
    // autoplay is on
    navigateHomeSelectedFace(true);
}


////////////////////////////////////////////////////////////////////////////////
//
void LLPanelMediaSettingsGeneral::setParent( LLFloaterMediaSettings* parent )
{
    mParent = parent;
};

////////////////////////////////////////////////////////////////////////////////
//
bool LLPanelMediaSettingsGeneral::navigateHomeSelectedFace(bool only_if_current_is_empty)
{
    struct functor_navigate_media : public LLSelectedTEGetFunctor< bool>
    {
        functor_navigate_media(bool flag) : only_if_current_is_empty(flag) {}
        bool get( LLViewerObject* object, S32 face )
        {
            if ( object && object->getTE(face) && object->permModify() )
            {
                const LLMediaEntry *media_data = object->getTE(face)->getMediaData();
                if ( media_data )
                {
                    if (!only_if_current_is_empty || (media_data->getCurrentURL().empty() && media_data->getAutoPlay()))
                    {
                        viewer_media_t media_impl =
                            LLViewerMedia::getInstance()->getMediaImplFromTextureID(object->getTE(face)->getMediaData()->getMediaID());
                        if (media_impl)
                        {
                            media_impl->setPriority(LLPluginClassMedia::PRIORITY_NORMAL);
                            media_impl->navigateHome();

                            if (!only_if_current_is_empty)
                            {
                                LLSD media_data;
                                media_data[LLMediaEntry::CURRENT_URL_KEY] = std::string();
                                object->getTE(face)->mergeIntoMediaData(media_data);
                            }
                            return true;
                        }
                    }
                }
            }
            return false;
        };
        bool only_if_current_is_empty;

    } functor_navigate_media(only_if_current_is_empty);

    bool all_face_media_navigated = false;
    LLObjectSelectionHandle selected_objects =LLSelectMgr::getInstance()->getSelection();
    selected_objects->getSelectedTEValue( &functor_navigate_media, all_face_media_navigated );

    if (all_face_media_navigated)
    {
        struct functor_sync_to_server : public LLSelectedObjectFunctor
        {
            virtual bool apply(LLViewerObject* object)
            {
                LLVOVolume *volume = dynamic_cast<LLVOVolume*>(object);
                if (volume)
                {
                    volume->sendMediaDataUpdate();
                }
                return true;
            }
        } sendfunc;
        selected_objects->applyToObjects(&sendfunc);
    }

    // Note: we don't update the 'current URL' field until the media data itself changes

    return all_face_media_navigated;
}

////////////////////////////////////////////////////////////////////////////////
//
const std::string LLPanelMediaSettingsGeneral::getHomeUrl()
{
    return mHomeURL->getValue().asString();
}

////////////////////////////////////////////////////////////////////////////////
//
void LLPanelMediaSettingsGeneral::updateCurrentUrl()
{
    // Get the current URL from the selection
    const LLMediaEntry default_media_data;
    std::string value_str = default_media_data.getCurrentURL();
    struct functor_getter_current_url : public LLSelectedTEGetFunctor< std::string >
    {
        functor_getter_current_url(const LLMediaEntry& entry): mMediaEntry(entry) {}

        std::string get( LLViewerObject* object, S32 face )
        {
            if ( object )
                if ( object->getTE(face) )
                    if ( object->getTE(face)->getMediaData() )
                        return object->getTE(face)->getMediaData()->getCurrentURL();
            return mMediaEntry.getCurrentURL();
        };

        const LLMediaEntry &  mMediaEntry;

    } func_current_url(default_media_data);
    bool identical = LLSelectMgr::getInstance()->getSelection()->getSelectedTEValue( &func_current_url, value_str );
    mCurrentURL->setText(value_str);
    mCurrentURL->setTentative(identical);

    if ( LLPanelMediaSettingsGeneral::isMultiple() )
    {
        mCurrentURL->setText(LLTrans::getString("Multiple Media"));
    }
}