376 lines
10 KiB
C++
376 lines
10 KiB
C++
/*
|
|
This file is part of Telegram Desktop,
|
|
the official desktop application for the Telegram messaging service.
|
|
|
|
For license and copyright information please follow this link:
|
|
https://github.com/telegramdesktop/tdesktop/blob/master/LEGAL
|
|
*/
|
|
#include "settings/business/settings_away_message.h"
|
|
|
|
#include "base/unixtime.h"
|
|
#include "core/application.h"
|
|
#include "data/business/data_business_info.h"
|
|
#include "data/business/data_shortcut_messages.h"
|
|
#include "data/data_session.h"
|
|
#include "lang/lang_keys.h"
|
|
#include "main/main_session.h"
|
|
#include "settings/business/settings_recipients_helper.h"
|
|
#include "settings/business/settings_shortcut_messages.h"
|
|
#include "ui/boxes/choose_date_time.h"
|
|
#include "ui/text/text_utilities.h"
|
|
#include "ui/toast/toast.h"
|
|
#include "ui/widgets/buttons.h"
|
|
#include "ui/widgets/checkbox.h"
|
|
#include "ui/wrap/slide_wrap.h"
|
|
#include "ui/wrap/vertical_layout.h"
|
|
#include "ui/vertical_list.h"
|
|
#include "window/window_session_controller.h"
|
|
#include "styles/style_layers.h"
|
|
#include "styles/style_settings.h"
|
|
|
|
namespace Settings {
|
|
namespace {
|
|
|
|
class AwayMessage : public BusinessSection<AwayMessage> {
|
|
public:
|
|
AwayMessage(
|
|
QWidget *parent,
|
|
not_null<Window::SessionController*> controller);
|
|
~AwayMessage();
|
|
|
|
[[nodiscard]] bool closeByOutsideClick() const override;
|
|
[[nodiscard]] rpl::producer<QString> title() override;
|
|
|
|
private:
|
|
void setupContent(not_null<Window::SessionController*> controller);
|
|
void save();
|
|
|
|
rpl::variable<bool> _canHave;
|
|
rpl::event_stream<> _deactivateOnAttempt;
|
|
rpl::variable<Data::BusinessRecipients> _recipients;
|
|
rpl::variable<Data::AwaySchedule> _schedule;
|
|
rpl::variable<bool> _offlineOnly;
|
|
rpl::variable<bool> _enabled;
|
|
|
|
};
|
|
|
|
[[nodiscard]] TimeId StartTimeMin() {
|
|
// Telegram was launched in August 2013 :)
|
|
return base::unixtime::serialize(QDateTime(QDate(2013, 8, 1), QTime(0, 0)));
|
|
}
|
|
|
|
[[nodiscard]] TimeId EndTimeMin() {
|
|
return StartTimeMin() + 3600;
|
|
}
|
|
|
|
[[nodiscard]] bool BadCustomInterval(const Data::WorkingInterval &interval) {
|
|
return !interval
|
|
|| (interval.start < StartTimeMin())
|
|
|| (interval.end < EndTimeMin());
|
|
}
|
|
|
|
struct AwayScheduleSelectorDescriptor {
|
|
not_null<Window::SessionController*> controller;
|
|
not_null<rpl::variable<Data::AwaySchedule>*> data;
|
|
};
|
|
void AddAwayScheduleSelector(
|
|
not_null<Ui::VerticalLayout*> container,
|
|
AwayScheduleSelectorDescriptor &&descriptor) {
|
|
using Type = Data::AwayScheduleType;
|
|
using namespace rpl::mappers;
|
|
|
|
const auto controller = descriptor.controller;
|
|
const auto data = descriptor.data;
|
|
|
|
Ui::AddSubsectionTitle(container, tr::lng_away_schedule());
|
|
const auto group = std::make_shared<Ui::RadioenumGroup<Type>>(
|
|
data->current().type);
|
|
|
|
const auto add = [&](Type type, const QString &label) {
|
|
container->add(
|
|
object_ptr<Ui::Radioenum<Type>>(
|
|
container,
|
|
group,
|
|
type,
|
|
label),
|
|
st::boxRowPadding + st::settingsAwaySchedulePadding);
|
|
};
|
|
add(Type::Always, tr::lng_away_schedule_always(tr::now));
|
|
add(Type::OutsideWorkingHours, tr::lng_away_schedule_outside(tr::now));
|
|
add(Type::Custom, tr::lng_away_schedule_custom(tr::now));
|
|
|
|
const auto customWrap = container->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
container,
|
|
object_ptr<Ui::VerticalLayout>(container)));
|
|
const auto customInner = customWrap->entity();
|
|
customWrap->toggleOn(group->value() | rpl::map(_1 == Type::Custom));
|
|
|
|
group->changes() | rpl::start_with_next([=](Type value) {
|
|
auto copy = data->current();
|
|
copy.type = value;
|
|
*data = copy;
|
|
}, customWrap->lifetime());
|
|
|
|
const auto chooseDate = [=](
|
|
rpl::producer<QString> title,
|
|
TimeId now,
|
|
Fn<TimeId()> min,
|
|
Fn<TimeId()> max,
|
|
Fn<void(TimeId)> done) {
|
|
using namespace Ui;
|
|
const auto box = std::make_shared<QPointer<Ui::BoxContent>>();
|
|
const auto save = [=](TimeId time) {
|
|
done(time);
|
|
if (const auto strong = box->data()) {
|
|
strong->closeBox();
|
|
}
|
|
};
|
|
*box = controller->show(Box(ChooseDateTimeBox, ChooseDateTimeBoxArgs{
|
|
.title = std::move(title),
|
|
.submit = tr::lng_settings_save(),
|
|
.done = save,
|
|
.min = min,
|
|
.time = now,
|
|
.max = max,
|
|
}));
|
|
};
|
|
|
|
Ui::AddSkip(customInner);
|
|
Ui::AddDivider(customInner);
|
|
Ui::AddSkip(customInner);
|
|
|
|
auto startLabel = data->value(
|
|
) | rpl::map([=](const Data::AwaySchedule &value) {
|
|
return langDateTime(
|
|
base::unixtime::parse(value.customInterval.start));
|
|
});
|
|
AddButtonWithLabel(
|
|
customInner,
|
|
tr::lng_away_custom_start(),
|
|
std::move(startLabel),
|
|
st::settingsButtonNoIcon
|
|
)->setClickedCallback([=] {
|
|
chooseDate(
|
|
tr::lng_away_custom_start(),
|
|
data->current().customInterval.start,
|
|
StartTimeMin,
|
|
[=] { return data->current().customInterval.end - 1; },
|
|
[=](TimeId time) {
|
|
auto copy = data->current();
|
|
copy.customInterval.start = time;
|
|
*data = copy;
|
|
});
|
|
});
|
|
|
|
auto endLabel = data->value(
|
|
) | rpl::map([=](const Data::AwaySchedule &value) {
|
|
return langDateTime(
|
|
base::unixtime::parse(value.customInterval.end));
|
|
});
|
|
AddButtonWithLabel(
|
|
customInner,
|
|
tr::lng_away_custom_end(),
|
|
std::move(endLabel),
|
|
st::settingsButtonNoIcon
|
|
)->setClickedCallback([=] {
|
|
chooseDate(
|
|
tr::lng_away_custom_end(),
|
|
data->current().customInterval.end,
|
|
[=] { return data->current().customInterval.start + 1; },
|
|
nullptr,
|
|
[=](TimeId time) {
|
|
auto copy = data->current();
|
|
copy.customInterval.end = time;
|
|
*data = copy;
|
|
});
|
|
});
|
|
}
|
|
|
|
AwayMessage::AwayMessage(
|
|
QWidget *parent,
|
|
not_null<Window::SessionController*> controller)
|
|
: BusinessSection(parent, controller) {
|
|
setupContent(controller);
|
|
}
|
|
|
|
AwayMessage::~AwayMessage() {
|
|
if (!Core::Quitting()) {
|
|
save();
|
|
}
|
|
}
|
|
|
|
bool AwayMessage::closeByOutsideClick() const {
|
|
return false;
|
|
}
|
|
|
|
rpl::producer<QString> AwayMessage::title() {
|
|
return tr::lng_away_title();
|
|
}
|
|
|
|
void AwayMessage::setupContent(
|
|
not_null<Window::SessionController*> controller) {
|
|
using namespace Data;
|
|
using namespace rpl::mappers;
|
|
|
|
const auto content = Ui::CreateChild<Ui::VerticalLayout>(this);
|
|
const auto info = &controller->session().data().businessInfo();
|
|
const auto current = info->awaySettings();
|
|
const auto disabled = (current.schedule.type == AwayScheduleType::Never);
|
|
|
|
_recipients = disabled
|
|
? Data::BusinessRecipients{ .allButExcluded = true }
|
|
: current.recipients;
|
|
auto initialSchedule = disabled ? AwaySchedule{
|
|
.type = AwayScheduleType::Always,
|
|
} : current.schedule;
|
|
if (BadCustomInterval(initialSchedule.customInterval)) {
|
|
const auto now = base::unixtime::now();
|
|
initialSchedule.customInterval = WorkingInterval{
|
|
.start = now,
|
|
.end = now + 24 * 60 * 60,
|
|
};
|
|
}
|
|
_schedule = initialSchedule;
|
|
|
|
AddDividerTextWithLottie(content, {
|
|
.lottie = u"sleep"_q,
|
|
.lottieSize = st::settingsCloudPasswordIconSize,
|
|
.lottieMargins = st::peerAppearanceIconPadding,
|
|
.showFinished = showFinishes(),
|
|
.about = tr::lng_away_about(Ui::Text::WithEntities),
|
|
.aboutMargins = st::peerAppearanceCoverLabelMargin,
|
|
});
|
|
|
|
const auto session = &controller->session();
|
|
_canHave = rpl::combine(
|
|
ShortcutsCountValue(session),
|
|
ShortcutsLimitValue(session),
|
|
ShortcutExistsValue(session, u"away"_q),
|
|
(_1 < _2) || _3);
|
|
|
|
Ui::AddSkip(content);
|
|
const auto enabled = content->add(object_ptr<Ui::SettingsButton>(
|
|
content,
|
|
tr::lng_away_enable(),
|
|
st::settingsButtonNoIcon
|
|
))->toggleOn(rpl::single(
|
|
!disabled
|
|
) | rpl::then(rpl::merge(
|
|
_canHave.value() | rpl::filter(!_1),
|
|
_deactivateOnAttempt.events() | rpl::map_to(false)
|
|
)));
|
|
|
|
_enabled = enabled->toggledValue();
|
|
_enabled.value() | rpl::filter(_1) | rpl::start_with_next([=] {
|
|
if (!_canHave.current()) {
|
|
controller->showToast({
|
|
.text = { tr::lng_away_limit_reached(tr::now) },
|
|
.adaptive = true,
|
|
});
|
|
_deactivateOnAttempt.fire({});
|
|
}
|
|
}, lifetime());
|
|
|
|
const auto wrap = content->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
content,
|
|
object_ptr<Ui::VerticalLayout>(content)));
|
|
const auto inner = wrap->entity();
|
|
|
|
Ui::AddSkip(inner);
|
|
Ui::AddDivider(inner);
|
|
|
|
const auto createWrap = inner->add(
|
|
object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
|
|
inner,
|
|
object_ptr<Ui::VerticalLayout>(inner)));
|
|
const auto createInner = createWrap->entity();
|
|
Ui::AddSkip(createInner);
|
|
const auto create = AddButtonWithLabel(
|
|
createInner,
|
|
rpl::conditional(
|
|
ShortcutExistsValue(session, u"away"_q),
|
|
tr::lng_business_edit_messages(),
|
|
tr::lng_away_create()),
|
|
ShortcutMessagesCountValue(
|
|
session,
|
|
u"away"_q
|
|
) | rpl::map([=](int count) {
|
|
return count
|
|
? tr::lng_forum_messages(tr::now, lt_count, count)
|
|
: QString();
|
|
}),
|
|
st::settingsButtonLightNoIcon);
|
|
create->setClickedCallback([=] {
|
|
const auto owner = &controller->session().data();
|
|
const auto id = owner->shortcutMessages().emplaceShortcut("away");
|
|
showOther(ShortcutMessagesId(id));
|
|
});
|
|
Ui::AddSkip(createInner);
|
|
Ui::AddDivider(createInner);
|
|
|
|
createWrap->toggleOn(rpl::single(true));
|
|
|
|
Ui::AddSkip(inner);
|
|
AddAwayScheduleSelector(inner, {
|
|
.controller = controller,
|
|
.data = &_schedule,
|
|
});
|
|
Ui::AddSkip(inner);
|
|
Ui::AddDivider(inner);
|
|
Ui::AddSkip(inner);
|
|
|
|
const auto offlineOnly = inner->add(
|
|
object_ptr<Ui::SettingsButton>(
|
|
inner,
|
|
tr::lng_away_offline_only(),
|
|
st::settingsButtonNoIcon)
|
|
)->toggleOn(rpl::single(current.offlineOnly));
|
|
_offlineOnly = offlineOnly->toggledValue();
|
|
|
|
Ui::AddSkip(inner);
|
|
Ui::AddDividerText(inner, tr::lng_away_offline_only_about());
|
|
|
|
AddBusinessRecipientsSelector(inner, {
|
|
.controller = controller,
|
|
.title = tr::lng_away_recipients(),
|
|
.data = &_recipients,
|
|
});
|
|
|
|
Ui::AddSkip(inner, st::settingsChatbotsAccessSkip);
|
|
|
|
wrap->toggleOn(enabled->toggledValue());
|
|
wrap->finishAnimating();
|
|
|
|
Ui::ResizeFitChild(this, content);
|
|
}
|
|
|
|
void AwayMessage::save() {
|
|
const auto show = controller()->uiShow();
|
|
const auto session = &controller()->session();
|
|
const auto fail = [=](QString error) {
|
|
if (error == u"BUSINESS_RECIPIENTS_EMPTY"_q) {
|
|
show->showToast(tr::lng_greeting_recipients_empty(tr::now));
|
|
} else if (error != u"SHORTCUT_INVALID"_q) {
|
|
show->showToast(error);
|
|
}
|
|
};
|
|
session->data().businessInfo().saveAwaySettings(
|
|
_enabled.current() ? Data::AwaySettings{
|
|
.recipients = _recipients.current(),
|
|
.schedule = _schedule.current(),
|
|
.shortcutId = LookupShortcutId(session, u"away"_q),
|
|
.offlineOnly = _offlineOnly.current(),
|
|
} : Data::AwaySettings(),
|
|
fail);
|
|
}
|
|
|
|
} // namespace
|
|
|
|
Type AwayMessageId() {
|
|
return AwayMessage::Id();
|
|
}
|
|
|
|
} // namespace Settings
|