2012-07-30 19:55:21 -04:00
|
|
|
/****************************************************************
|
|
|
|
* RetroShare is distributed under the following license:
|
|
|
|
*
|
|
|
|
* Copyright (C) 2008 Robert Fernie
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program 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 General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor,
|
|
|
|
* Boston, MA 02110-1301, USA.
|
|
|
|
****************************************************************/
|
|
|
|
|
|
|
|
#include <QMessageBox>
|
|
|
|
|
|
|
|
#include "util/misc.h"
|
|
|
|
#include "GxsGroupDialog.h"
|
|
|
|
#include "gui/common/PeerDefs.h"
|
2012-10-21 15:45:35 -04:00
|
|
|
#include "gxs/rsgxsflags.h"
|
2012-07-30 19:55:21 -04:00
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
|
|
|
|
#include <retroshare/rspeers.h>
|
|
|
|
|
|
|
|
#include <iostream>
|
|
|
|
|
2012-10-21 15:45:35 -04:00
|
|
|
// Control of Publish Signatures.
|
|
|
|
#define RSGXS_GROUP_SIGN_PUBLISH_MASK 0x000000ff
|
|
|
|
#define RSGXS_GROUP_SIGN_PUBLISH_ENCRYPTED 0x00000001
|
|
|
|
#define RSGXS_GROUP_SIGN_PUBLISH_ALLSIGNED 0x00000002
|
|
|
|
#define RSGXS_GROUP_SIGN_PUBLISH_THREADHEAD 0x00000004
|
|
|
|
#define RSGXS_GROUP_SIGN_PUBLISH_NONEREQ 0x00000008
|
|
|
|
|
|
|
|
// Author Signature.
|
|
|
|
#define RSGXS_GROUP_SIGN_AUTHOR_MASK 0x0000ff00
|
|
|
|
#define RSGXS_GROUP_SIGN_AUTHOR_GPG 0x00000100
|
|
|
|
#define RSGXS_GROUP_SIGN_AUTHOR_REQUIRED 0x00000200
|
|
|
|
#define RSGXS_GROUP_SIGN_AUTHOR_IFNOPUBSIGN 0x00000400
|
|
|
|
#define RSGXS_GROUP_SIGN_AUTHOR_NONE 0x00000800
|
|
|
|
|
2012-07-30 19:55:21 -04:00
|
|
|
#define GXSGROUP_NEWGROUPID 1
|
|
|
|
#define GXSGROUP_LOADGROUP 2
|
|
|
|
|
|
|
|
/** Constructor */
|
2012-11-07 18:24:23 -05:00
|
|
|
GxsGroupDialog::GxsGroupDialog(TokenQueue *tokenQueue, uint32_t enableFlags, uint16_t defaultFlags, QWidget *parent)
|
|
|
|
: QDialog(parent), mTokenQueue(tokenQueue), mMode(GXS_GROUP_DIALOG_CREATE_MODE), mEnabledFlags(enableFlags), mDefaultsFlags(defaultFlags), mReadonlyFlags(0)
|
2012-07-30 19:55:21 -04:00
|
|
|
{
|
2012-10-29 18:41:41 -04:00
|
|
|
/* Invoke the Qt Designer generated object setup routine */
|
|
|
|
ui.setupUi(this);
|
2012-07-30 19:55:21 -04:00
|
|
|
|
2012-10-29 18:41:41 -04:00
|
|
|
// connect up the buttons.
|
|
|
|
connect( ui.cancelButton, SIGNAL( clicked ( bool ) ), this, SLOT( cancelDialog( ) ) );
|
|
|
|
connect( ui.createButton, SIGNAL( clicked ( bool ) ), this, SLOT( submitGroup( ) ) );
|
|
|
|
connect( ui.pubKeyShare_cb, SIGNAL( clicked() ), this, SLOT( setShareList( ) ));
|
2012-07-30 19:55:21 -04:00
|
|
|
|
2012-10-29 18:41:41 -04:00
|
|
|
connect( ui.groupLogo, SIGNAL(clicked() ), this , SLOT(addGroupLogo()));
|
2012-11-16 11:41:59 -05:00
|
|
|
connect( ui.addLogoButton, SIGNAL(clicked() ), this , SLOT(addGroupLogo()));
|
|
|
|
|
|
|
|
//ui.headerFrame->setHeaderImage(QPixmap(":/WikiPoos/images/resource-group-new_48.png"));
|
|
|
|
//ui.headerFrame->setHeaderText(tr("Create Wiki Group"));
|
2012-07-30 19:55:21 -04:00
|
|
|
|
2012-11-07 18:24:23 -05:00
|
|
|
if (!ui.pubKeyShare_cb->isChecked())
|
|
|
|
{
|
2012-10-29 18:41:41 -04:00
|
|
|
ui.contactsdockWidget->hide();
|
|
|
|
this->resize(this->size().width() - ui.contactsdockWidget->size().width(), this->size().height());
|
|
|
|
}
|
2012-07-30 19:55:21 -04:00
|
|
|
|
2012-10-29 18:41:41 -04:00
|
|
|
/* initialize key share list */
|
|
|
|
ui.keyShareList->setHeaderText(tr("Contacts:"));
|
|
|
|
ui.keyShareList->setModus(FriendSelectionWidget::MODUS_CHECK);
|
|
|
|
ui.keyShareList->start();
|
2012-07-30 19:55:21 -04:00
|
|
|
|
2012-10-29 18:41:41 -04:00
|
|
|
/* Setup Reasonable Defaults */
|
2012-07-30 19:55:21 -04:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2012-11-07 18:24:23 -05:00
|
|
|
GxsGroupDialog::GxsGroupDialog(const RsGroupMetaData &grpMeta, uint32_t mode, QWidget *parent)
|
|
|
|
: QDialog(parent), mMode(mode), mGrpMeta(grpMeta) {
|
|
|
|
|
2012-07-30 19:55:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void GxsGroupDialog::setMode(uint32_t mode)
|
|
|
|
{
|
2012-10-29 18:41:41 -04:00
|
|
|
switch(mMode)
|
|
|
|
{
|
|
|
|
case GXS_GROUP_DIALOG_CREATE_MODE:
|
|
|
|
{
|
|
|
|
ui.createButton->setText(tr("Create Group"));
|
|
|
|
newGroup();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
case GXS_GROUP_DIALOG_SHOW_MODE:
|
|
|
|
{
|
|
|
|
ui.cancelButton->setVisible(false);
|
|
|
|
ui.createButton->setText(tr("Close"));
|
|
|
|
}
|
|
|
|
break;
|
2012-11-07 18:24:23 -05:00
|
|
|
//TODO
|
2012-10-29 18:41:41 -04:00
|
|
|
// case GXS_GROUP_DIALOG_EDIT_MODE:
|
|
|
|
// {
|
|
|
|
// ui.createButton->setText(tr("Submit Changes"));
|
|
|
|
// }
|
|
|
|
// break;
|
|
|
|
}
|
2012-07-30 19:55:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void GxsGroupDialog::clearForm()
|
|
|
|
{
|
|
|
|
ui.groupName->clear();
|
|
|
|
ui.groupDesc->clear();
|
|
|
|
ui.groupName->setFocus();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void GxsGroupDialog::setupDefaults()
|
|
|
|
{
|
|
|
|
/* Enable / Show Parts based on Flags */
|
|
|
|
|
2012-10-29 18:41:41 -04:00
|
|
|
if (mDefaultsFlags & GXS_GROUP_DEFAULTS_DISTRIB_MASK)
|
|
|
|
{
|
|
|
|
if (mDefaultsFlags & GXS_GROUP_DEFAULTS_DISTRIB_PUBLIC)
|
|
|
|
{
|
|
|
|
ui.typePublic->setChecked(true);
|
|
|
|
}
|
|
|
|
else if (mDefaultsFlags & GXS_GROUP_DEFAULTS_DISTRIB_GROUP)
|
|
|
|
{
|
|
|
|
ui.typeGroup->setChecked(true);
|
|
|
|
}
|
|
|
|
else if (mDefaultsFlags & GXS_GROUP_DEFAULTS_DISTRIB_LOCAL)
|
|
|
|
{
|
|
|
|
ui.typeLocal->setChecked(true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// default
|
|
|
|
ui.typePublic->setChecked(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mDefaultsFlags & GXS_GROUP_DEFAULTS_PUBLISH_MASK)
|
|
|
|
{
|
|
|
|
if (mDefaultsFlags & GXS_GROUP_DEFAULTS_PUBLISH_ENCRYPTED)
|
|
|
|
{
|
|
|
|
ui.publish_encrypt->setChecked(true);
|
|
|
|
}
|
|
|
|
else if (mDefaultsFlags & GXS_GROUP_DEFAULTS_PUBLISH_REQUIRED)
|
|
|
|
{
|
|
|
|
ui.publish_required->setChecked(true);
|
|
|
|
}
|
|
|
|
else if (mDefaultsFlags & GXS_GROUP_DEFAULTS_PUBLISH_THREADS)
|
|
|
|
{
|
|
|
|
ui.publish_threads->setChecked(true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// default
|
|
|
|
ui.publish_open->setChecked(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mDefaultsFlags & GXS_GROUP_DEFAULTS_PERSONAL_MASK)
|
|
|
|
{
|
|
|
|
if (mDefaultsFlags & GXS_GROUP_DEFAULTS_PERSONAL_PGP)
|
|
|
|
{
|
|
|
|
ui.personal_pgp->setChecked(true);
|
|
|
|
}
|
|
|
|
else if (mDefaultsFlags & GXS_GROUP_DEFAULTS_PERSONAL_REQUIRED)
|
|
|
|
{
|
|
|
|
ui.personal_required->setChecked(true);
|
|
|
|
}
|
|
|
|
else if (mDefaultsFlags & GXS_GROUP_DEFAULTS_PERSONAL_IFNOPUB)
|
|
|
|
{
|
|
|
|
ui.personal_ifnopub->setChecked(true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// default
|
|
|
|
ui.personal_ifnopub->setChecked(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (mDefaultsFlags & GXS_GROUP_DEFAULTS_COMMENTS_MASK)
|
|
|
|
{
|
|
|
|
if (mDefaultsFlags & GXS_GROUP_DEFAULTS_COMMENTS_YES)
|
|
|
|
{
|
|
|
|
ui.comments_allowed->setChecked(true);
|
|
|
|
}
|
|
|
|
else if (mDefaultsFlags & GXS_GROUP_DEFAULTS_COMMENTS_NO)
|
|
|
|
{
|
|
|
|
ui.comments_no->setChecked(true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// default
|
|
|
|
ui.comments_no->setChecked(true);
|
|
|
|
}
|
|
|
|
}
|
2012-07-30 19:55:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void GxsGroupDialog::setupVisibility()
|
|
|
|
{
|
2012-10-29 18:41:41 -04:00
|
|
|
{
|
|
|
|
ui.groupLogo->setVisible(mEnabledFlags & GXS_GROUP_FLAGS_ICON);
|
|
|
|
ui.addLogoButton->setVisible(mEnabledFlags & GXS_GROUP_FLAGS_ICON);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
ui.groupDesc->setVisible(mEnabledFlags & GXS_GROUP_FLAGS_DESCRIPTION);
|
|
|
|
ui.groupDescLabel->setVisible(mEnabledFlags & GXS_GROUP_FLAGS_DESCRIPTION);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
ui.distribGroupBox->setVisible(mEnabledFlags & GXS_GROUP_FLAGS_DISTRIBUTION);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
ui.publishGroupBox->setVisible(mEnabledFlags & GXS_GROUP_FLAGS_PUBLISHSIGN);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
ui.pubKeyShare_cb->setVisible(mEnabledFlags & GXS_GROUP_FLAGS_SHAREKEYS);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
ui.personalGroupBox->setVisible(mEnabledFlags & GXS_GROUP_FLAGS_PERSONALSIGN);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
ui.commentGroupBox->setVisible(mEnabledFlags & GXS_GROUP_FLAGS_COMMENTS);
|
|
|
|
}
|
|
|
|
|
|
|
|
{
|
|
|
|
ui.extraFrame->setVisible(mEnabledFlags & GXS_GROUP_FLAGS_EXTRA);
|
|
|
|
}
|
2012-07-30 19:55:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void GxsGroupDialog::newGroup()
|
|
|
|
{
|
2012-10-29 18:41:41 -04:00
|
|
|
setupDefaults();
|
|
|
|
setupVisibility();
|
|
|
|
clearForm();
|
2012-07-30 19:55:21 -04:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
void GxsGroupDialog::submitGroup()
|
|
|
|
{
|
2012-10-29 18:41:41 -04:00
|
|
|
std::cerr << "GxsGroupDialog::submitGroup()";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
|
|
|
|
/* switch depending on mode */
|
|
|
|
switch(mMode)
|
|
|
|
{
|
|
|
|
case GXS_GROUP_DIALOG_CREATE_MODE:
|
|
|
|
{
|
|
|
|
/* just close if down */
|
|
|
|
createGroup();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
case GXS_GROUP_DIALOG_SHOW_MODE:
|
|
|
|
{
|
|
|
|
/* just close if down */
|
|
|
|
cancelDialog();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
|
|
|
|
case GXS_GROUP_DIALOG_EDIT_MODE:
|
|
|
|
{
|
|
|
|
/* TEMP: just close if down */
|
|
|
|
cancelDialog();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
2012-07-30 19:55:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void GxsGroupDialog::createGroup()
|
|
|
|
{
|
2012-11-12 15:47:55 -05:00
|
|
|
std::cerr << "GxsGroupDialog::createGroup()";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
|
2012-10-29 18:41:41 -04:00
|
|
|
QString name = misc::removeNewLine(ui.groupName->text());
|
|
|
|
uint32_t flags = 0;
|
|
|
|
|
|
|
|
if(name.isEmpty())
|
|
|
|
{
|
|
|
|
/* error message */
|
|
|
|
QMessageBox::warning(this, "RetroShare", tr("Please add a Name"), QMessageBox::Ok, QMessageBox::Ok);
|
|
|
|
return; //Don't add a empty name!!
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t token;
|
|
|
|
RsGroupMetaData meta;
|
|
|
|
|
|
|
|
// Fill in the MetaData as best we can.
|
|
|
|
meta.mGroupName = std::string(name.toUtf8());
|
|
|
|
|
|
|
|
meta.mGroupFlags = flags;
|
|
|
|
meta.mSignFlags = getGroupSignFlags();
|
|
|
|
|
|
|
|
if (service_CreateGroup(token, meta))
|
|
|
|
{
|
|
|
|
// get the Queue to handle response.
|
|
|
|
if(mTokenQueue != NULL)
|
|
|
|
mTokenQueue->queueRequest(token, TOKENREQ_GROUPINFO, RS_TOKREQ_ANSTYPE_ACK, GXSGROUP_NEWGROUPID);
|
|
|
|
}
|
|
|
|
close();
|
2012-07-30 19:55:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t GxsGroupDialog::getGroupSignFlags()
|
|
|
|
{
|
2012-10-29 18:41:41 -04:00
|
|
|
/* grab from the ui options -> */
|
|
|
|
uint32_t signFlags = 0;
|
|
|
|
if (ui.publish_encrypt->isChecked()) {
|
|
|
|
signFlags |= RSGXS_GROUP_SIGN_PUBLISH_ENCRYPTED;
|
|
|
|
} else if (ui.publish_required->isChecked()) {
|
|
|
|
signFlags |= RSGXS_GROUP_SIGN_PUBLISH_ALLSIGNED;
|
|
|
|
} else if (ui.publish_threads->isChecked()) {
|
|
|
|
signFlags |= RSGXS_GROUP_SIGN_PUBLISH_THREADHEAD;
|
|
|
|
} else { // publish_open (default).
|
|
|
|
signFlags |= RSGXS_GROUP_SIGN_PUBLISH_NONEREQ;
|
|
|
|
}
|
2012-07-30 19:55:21 -04:00
|
|
|
|
|
|
|
// Author Signature.
|
2012-10-29 18:41:41 -04:00
|
|
|
if (ui.personal_pgp->isChecked()) {
|
|
|
|
signFlags |= RSGXS_GROUP_SIGN_AUTHOR_GPG;
|
|
|
|
} else if (ui.personal_required->isChecked()) {
|
|
|
|
signFlags |= RSGXS_GROUP_SIGN_AUTHOR_REQUIRED;
|
|
|
|
} else if (ui.personal_ifnopub->isChecked()) {
|
|
|
|
signFlags |= RSGXS_GROUP_SIGN_AUTHOR_IFNOPUBSIGN;
|
|
|
|
} else { // shouldn't allow this one.
|
|
|
|
signFlags |= RSGXS_GROUP_SIGN_AUTHOR_NONE;
|
|
|
|
}
|
|
|
|
return signFlags;
|
2012-07-30 19:55:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void GxsGroupDialog::setGroupSignFlags(uint32_t signFlags)
|
|
|
|
{
|
|
|
|
|
2012-10-29 18:41:41 -04:00
|
|
|
if (signFlags & RSGXS_GROUP_SIGN_PUBLISH_ENCRYPTED) {
|
|
|
|
ui.publish_encrypt->setChecked(true);
|
|
|
|
} else if (signFlags & RSGXS_GROUP_SIGN_PUBLISH_ALLSIGNED) {
|
|
|
|
ui.publish_required->setChecked(true);
|
|
|
|
} else if (signFlags & RSGXS_GROUP_SIGN_PUBLISH_THREADHEAD) {
|
|
|
|
ui.publish_threads->setChecked(true);
|
|
|
|
} else if (signFlags & RSGXS_GROUP_SIGN_PUBLISH_NONEREQ) {
|
|
|
|
ui.publish_open->setChecked(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (signFlags & RSGXS_GROUP_SIGN_AUTHOR_GPG) {
|
|
|
|
ui.personal_pgp->setChecked(true);
|
|
|
|
} else if (signFlags & RSGXS_GROUP_SIGN_AUTHOR_REQUIRED) {
|
|
|
|
ui.personal_required->setChecked(true);
|
|
|
|
} else if (signFlags & RSGXS_GROUP_SIGN_AUTHOR_IFNOPUBSIGN) {
|
|
|
|
ui.personal_ifnopub->setChecked(true);
|
|
|
|
} else if (signFlags & RSGXS_GROUP_SIGN_AUTHOR_NONE) {
|
|
|
|
// Its the same... but not quite.
|
|
|
|
//ui.personal_noifpub->setChecked();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* guess at comments */
|
|
|
|
if ((signFlags & RSGXS_GROUP_SIGN_PUBLISH_THREADHEAD)
|
|
|
|
&& (signFlags & RSGXS_GROUP_SIGN_AUTHOR_IFNOPUBSIGN))
|
|
|
|
{
|
|
|
|
ui.comments_allowed->setChecked(true);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
ui.comments_no->setChecked(true);
|
|
|
|
}
|
2012-07-30 19:55:21 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void GxsGroupDialog::cancelDialog()
|
|
|
|
{
|
|
|
|
std::cerr << "GxsGroupDialog::cancelDialog() Should Close!";
|
|
|
|
std::cerr << std::endl;
|
|
|
|
|
|
|
|
close();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void GxsGroupDialog::addGroupLogo()
|
|
|
|
{
|
|
|
|
QPixmap img = misc::getOpenThumbnailedPicture(this, tr("Load Group Logo"), 64, 64);
|
|
|
|
|
|
|
|
if (img.isNull())
|
|
|
|
return;
|
|
|
|
|
|
|
|
picture = img;
|
|
|
|
|
|
|
|
// to show the selected
|
|
|
|
ui.groupLogo->setIcon(picture);
|
|
|
|
}
|
|
|
|
|
2012-11-11 18:45:22 -05:00
|
|
|
QPixmap GxsGroupDialog::getLogo()
|
|
|
|
{
|
|
|
|
return picture;
|
|
|
|
}
|
2012-07-30 19:55:21 -04:00
|
|
|
|
2012-11-11 18:45:22 -05:00
|
|
|
QString GxsGroupDialog::getDescription()
|
|
|
|
{
|
|
|
|
return ui.groupDesc->document()->toPlainText();
|
|
|
|
}
|
2012-07-30 19:55:21 -04:00
|
|
|
|
|
|
|
/***********************************************************************************
|
|
|
|
Share Lists.
|
|
|
|
***********************************************************************************/
|
|
|
|
|
|
|
|
void GxsGroupDialog::sendShareList(std::string groupId)
|
|
|
|
{
|
|
|
|
close();
|
|
|
|
}
|
|
|
|
|
|
|
|
void GxsGroupDialog::setShareList()
|
|
|
|
{
|
2012-10-29 18:41:41 -04:00
|
|
|
if (ui.pubKeyShare_cb->isChecked()){
|
|
|
|
this->resize(this->size().width() + ui.contactsdockWidget->size().width(), this->size().height());
|
|
|
|
ui.contactsdockWidget->show();
|
|
|
|
} else { // hide share widget
|
|
|
|
ui.contactsdockWidget->hide();
|
|
|
|
this->resize(this->size().width() - ui.contactsdockWidget->size().width(), this->size().height());
|
|
|
|
}
|
2012-07-30 19:55:21 -04:00
|
|
|
}
|
|
|
|
|