Add tabs and other types links to Info::Media.

This commit is contained in:
John Preston 2017-10-04 13:39:59 +01:00
parent 335704e176
commit 7905694b31
17 changed files with 550 additions and 118 deletions

View File

@ -1579,7 +1579,7 @@ void HistoryInner::keyPressEvent(QKeyEvent *e) {
} }
} }
void HistoryInner::recountHeight() { void HistoryInner::recountHistoryGeometry() {
int visibleHeight = _scroll->height(); int visibleHeight = _scroll->height();
int oldHistoryPaddingTop = qMax(visibleHeight - historyHeight() - st::historyPaddingBottom, 0); int oldHistoryPaddingTop = qMax(visibleHeight - historyHeight() - st::historyPaddingBottom, 0);
if (_botAbout && !_botAbout->info->text.isEmpty()) { if (_botAbout && !_botAbout->info->text.isEmpty()) {

View File

@ -47,7 +47,7 @@ public:
void touchScrollUpdated(const QPoint &screenPos); void touchScrollUpdated(const QPoint &screenPos);
QPoint mapPointToItem(QPoint p, HistoryItem *item); QPoint mapPointToItem(QPoint p, HistoryItem *item);
void recountHeight(); void recountHistoryGeometry();
void updateSize(); void updateSize();
void repaintItem(const HistoryItem *item); void repaintItem(const HistoryItem *item);

View File

@ -4785,7 +4785,7 @@ int HistoryWidget::countAutomaticScrollTop() {
void HistoryWidget::updateHistoryGeometry(bool initial, bool loadedDown, const ScrollChange &change) { void HistoryWidget::updateHistoryGeometry(bool initial, bool loadedDown, const ScrollChange &change) {
if (!_history || (initial && _historyInited) || (!initial && !_historyInited)) return; if (!_history || (initial && _historyInited) || (!initial && !_historyInited)) return;
if (_firstLoadRequest || _a_show.animating()) { if (_firstLoadRequest || _a_show.animating()) {
return; // scrollTopMax etc are not working after recountHeight() return; // scrollTopMax etc are not working after recountHistoryGeometry()
} }
auto newScrollHeight = height() - _topBar->height(); auto newScrollHeight = height() - _topBar->height();
@ -4868,7 +4868,7 @@ void HistoryWidget::updateHistoryGeometry(bool initial, bool loadedDown, const S
} }
void HistoryWidget::updateListSize() { void HistoryWidget::updateListSize() {
_list->recountHeight(); _list->recountHistoryGeometry();
auto washidden = _scroll->isHidden(); auto washidden = _scroll->isHidden();
if (washidden) { if (washidden) {
_scroll->show(); _scroll->show();

View File

@ -173,9 +173,15 @@ infoIconMembers: icon {{ "info_members", infoIconFg }};
infoIconNotifications: icon {{ "info_notifications", infoIconFg }}; infoIconNotifications: icon {{ "info_notifications", infoIconFg }};
infoIconActions: icon {{ "info_actions", infoIconFg }}; infoIconActions: icon {{ "info_actions", infoIconFg }};
infoIconMediaPhoto: icon {{ "info_media_photo", infoIconFg }}; infoIconMediaPhoto: icon {{ "info_media_photo", infoIconFg }};
infoIconMediaAudio: icon {{ "info_media_audio", infoIconFg }};
infoIconMediaLink: icon {{ "info_media_link", infoIconFg }};
infoIconMediaGroup: icon {{ "info_common_groups", infoIconFg }};
infoIconMediaVoice: icon {{ "info_media_voice", infoIconFg }};
infoIconMediaRound: icon {{ "info_media_round", infoIconFg }};
infoInformationIconPosition: point(25px, 12px); infoInformationIconPosition: point(25px, 12px);
infoNotificationsIconPosition: point(20px, 5px); infoNotificationsIconPosition: point(20px, 5px);
infoSharedMediaIconPosition: point(20px, 24px); infoSharedMediaIconPosition: point(20px, 24px);
infoSharedMediaButtonIconPosition: point(20px, 3px);
infoIconPosition: point(20px, 15px); infoIconPosition: point(20px, 15px);
infoLabeledOneLine: FlatLabel(defaultFlatLabel) { infoLabeledOneLine: FlatLabel(defaultFlatLabel) {

View File

@ -57,6 +57,7 @@ LayerWidget::LayerWidget(
void LayerWidget::setupHeightConsumers() { void LayerWidget::setupHeightConsumers() {
_content->desiredHeightValue() _content->desiredHeightValue()
| rpl::start_with_next([this](int height) { | rpl::start_with_next([this](int height) {
if (!_content) return;
accumulate_max(_desiredHeight, height); accumulate_max(_desiredHeight, height);
resizeToWidth(width()); resizeToWidth(width());
_content->forceContentRepaint(); _content->forceContentRepaint();

View File

@ -0,0 +1,129 @@
/*
This file is part of Telegram Desktop,
the official desktop version of Telegram messaging app, see https://telegram.org
Telegram Desktop 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 3 of the License, or
(at your option) any later version.
It 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.
In addition, as a special exception, the copyright holders give permission
to link the code of portions of this program with the OpenSSL library.
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
*/
#pragma once
#include <rpl/mappers.h>
#include <rpl/map.h>
#include "lang/lang_keys.h"
#include "storage/storage_shared_media.h"
#include "info/info_memento.h"
#include "info/profile/info_profile_button.h"
#include "info/profile/info_profile_values.h"
#include "ui/wrap/slide_wrap.h"
#include "ui/wrap/vertical_layout.h"
#include "window/window_controller.h"
#include "styles/style_info.h"
namespace Info {
namespace Media {
using Type = Storage::SharedMediaType;
inline auto MediaTextPhrase(Type type) {
switch (type) {
case Type::Photo: return lng_profile_photos;
case Type::Video: return lng_profile_videos;
case Type::File: return lng_profile_files;
case Type::MusicFile: return lng_profile_songs;
case Type::Link: return lng_profile_shared_links;
case Type::VoiceFile: return lng_profile_audios;
case Type::RoundFile: return lng_profile_rounds;
}
Unexpected("Type in setupSharedMedia()");
};
inline auto MediaText(Type type) {
return [phrase = MediaTextPhrase(type)](int count) {
return phrase(lt_count, count);
};
}
template <typename Count, typename Text>
inline auto AddCountedButton(
Ui::VerticalLayout *parent,
Count &&count,
Text &&textFromCount,
Ui::MultiSlideTracker &tracker) {
using namespace rpl::mappers;
using Button = Profile::Button;
auto forked = std::move(count)
| start_spawning(parent->lifetime());
auto text = rpl::duplicate(forked)
| rpl::map([textFromCount](int count) {
return (count > 0)
? textFromCount(count)
: QString();
});
auto button = parent->add(object_ptr<Ui::SlideWrap<Button>>(
parent,
object_ptr<Button>(
parent,
std::move(text),
st::infoSharedMediaButton))
)->toggleOn(
rpl::duplicate(forked)
| rpl::map($1 > 0));
tracker.track(button);
return button;
};
inline auto AddButton(
Ui::VerticalLayout *parent,
not_null<Window::Controller*> controller,
not_null<PeerData*> peer,
Type type,
Ui::MultiSlideTracker &tracker) {
auto result = AddCountedButton(
parent,
Profile::SharedMediaCountValue(peer, type),
MediaText(type),
tracker)->entity();
result->addClickHandler([controller, peer, type] {
controller->showSection(
Info::Memento(peer->id, Section(type)));
});
return std::move(result);
};
inline auto AddCommonGroupsButton(
Ui::VerticalLayout *parent,
not_null<Window::Controller*> controller,
not_null<UserData*> user,
Ui::MultiSlideTracker &tracker) {
auto result = AddCountedButton(
parent,
Profile::CommonGroupsCountValue(user),
[](int count) {
return lng_profile_common_groups(lt_count, count);
},
tracker)->entity();
result->addClickHandler([controller, user] {
controller->showSection(
Info::Memento(
user->id,
Section::Type::CommonGroups));
});
return std::move(result);
};
} // namespace Media
} // namespace Info

View File

@ -20,28 +20,160 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
*/ */
#include "info/media/info_media_inner_widget.h" #include "info/media/info_media_inner_widget.h"
#include "ui/widgets/labels.h" #include "boxes/abstract_box.h"
#include "info/media/info_media_list_widget.h"
#include "info/media/info_media_buttons.h"
#include "info/profile/info_profile_button.h"
#include "info/profile/info_profile_icon.h"
#include "ui/widgets/discrete_sliders.h"
#include "ui/wrap/vertical_layout.h"
#include "styles/style_info.h"
#include "lang/lang_keys.h"
namespace Info { namespace Info {
namespace Media { namespace Media {
namespace {
using Type = InnerWidget::Type;
base::optional<int> TypeToTabIndex(Type type) {
switch (type) {
case Type::Photo: return 0;
case Type::Video: return 1;
case Type::File: return 2;
}
return base::none;
}
Type TabIndexToType(int index) {
switch (index) {
case 0: return Type::Photo;
case 1: return Type::Video;
case 2: return Type::File;
}
Unexpected("Index in Info::Media::TabIndexToType()");
}
} // namespace
InnerWidget::InnerWidget( InnerWidget::InnerWidget(
QWidget *parent, QWidget *parent,
rpl::producer<Wrap> &&wrap,
not_null<Window::Controller*> controller,
not_null<PeerData*> peer, not_null<PeerData*> peer,
Type type) Type type)
: RpWidget(parent) : RpWidget(parent) {
, _peer(peer) _list = setupList(controller, peer, type);
, _type(type) { setupOtherTypes(std::move(wrap));
auto text = qsl("Media Overview\n\n"); }
auto label = object_ptr<Ui::FlatLabel>(this);
label->setText(text.repeated(50)); void InnerWidget::setupOtherTypes(rpl::producer<Wrap> &&wrap) {
widthValue() | rpl::start_with_next([inner = label.data()](int w) { std::move(wrap)
inner->resizeToWidth(w); | rpl::start_with_next([this](Wrap value) {
}, lifetime()); if (value == Wrap::Side
label->heightValue() | rpl::start_with_next([this](int h) { && TypeToTabIndex(type())) {
_rowsHeightFake = h; createOtherTypes();
resizeToWidth(width()); } else {
}, lifetime()); _otherTabs = nullptr;
_otherTypes.destroy();
refreshHeight();
}
}, lifetime());
}
void InnerWidget::createOtherTypes() {
_otherTabs = nullptr;
_otherTypes.create(this);
createTypeButtons();
_otherTypes->add(object_ptr<BoxContentDivider>(_otherTypes));
createTabs();
_otherTypes->heightValue()
| rpl::start_with_next(
[this] { refreshHeight(); },
_otherTypes->lifetime());
}
void InnerWidget::createTypeButtons() {
auto wrap = _otherTypes->add(object_ptr<Ui::SlideWrap<Ui::VerticalLayout>>(
_otherTypes,
object_ptr<Ui::VerticalLayout>(_otherTypes)));
auto content = wrap->entity();
content->add(object_ptr<Ui::FixedHeightWidget>(
content,
st::infoProfileSkip));
auto tracker = Ui::MultiSlideTracker();
auto addMediaButton = [&](
Type type,
const style::icon &icon) {
auto result = AddButton(
content,
controller(),
peer(),
type,
tracker);
object_ptr<Profile::FloatingIcon>(
result,
icon,
st::infoSharedMediaButtonIconPosition);
};
auto addCommonGroupsButton = [&](
not_null<UserData*> user,
const style::icon &icon) {
auto result = AddCommonGroupsButton(
content,
controller(),
user,
tracker);
object_ptr<Profile::FloatingIcon>(
result,
icon,
st::infoSharedMediaButtonIconPosition);
};
addMediaButton(Type::MusicFile, st::infoIconMediaAudio);
addMediaButton(Type::Link, st::infoIconMediaLink);
if (auto user = peer()->asUser()) {
addCommonGroupsButton(user, st::infoIconMediaGroup);
}
addMediaButton(Type::VoiceFile, st::infoIconMediaVoice);
addMediaButton(Type::RoundFile, st::infoIconMediaRound);
content->add(object_ptr<Ui::FixedHeightWidget>(
content,
st::infoProfileSkip));
wrap->toggleOn(tracker.atLeastOneShownValue());
wrap->finishAnimating();
}
void InnerWidget::createTabs() {
_otherTabs = _otherTypes->add(object_ptr<Ui::SettingsSlider>(
this,
st::infoTabs));
auto sections = QStringList();
sections.push_back(lang(lng_media_type_photos).toUpper());
sections.push_back(lang(lng_media_type_videos).toUpper());
sections.push_back(lang(lng_media_type_files).toUpper());
_otherTabs->setSections(sections);
_otherTabs->sectionActivated()
| rpl::map([](int index) { return TabIndexToType(index); })
| rpl::start_with_next(
[this](Type type) {
if (_list->type() != type) {
switchToTab(Memento(peer()->id, type));
}
},
_otherTabs->lifetime());
}
not_null<PeerData*> InnerWidget::peer() const {
return _list->peer();
}
Type InnerWidget::type() const {
return _list->type();
} }
void InnerWidget::visibleTopBottomUpdated( void InnerWidget::visibleTopBottomUpdated(
@ -51,6 +183,50 @@ void InnerWidget::visibleTopBottomUpdated(
_visibleBottom = visibleBottom; _visibleBottom = visibleBottom;
} }
bool InnerWidget::showInternal(not_null<Memento*> memento) {
if (memento->peerId() != peer()->id) {
return false;
}
auto mementoType = memento->section().mediaType();
if (mementoType == type()) {
restoreState(memento);
return true;
} else if (_otherTypes) {
if (TypeToTabIndex(mementoType)) {
switchToTab(std::move(*memento));
return true;
}
}
return false;
}
void InnerWidget::switchToTab(Memento &&memento) {
auto type = memento.section().mediaType();
_list = setupList(controller(), peer(), type);
restoreState(&memento);
_otherTabs->setActiveSection(*TypeToTabIndex(type));
}
not_null<Window::Controller*> InnerWidget::controller() const {
return _list->controller();
}
object_ptr<ListWidget> InnerWidget::setupList(
not_null<Window::Controller*> controller,
not_null<PeerData*> peer,
Type type) {
auto result = object_ptr<ListWidget>(
this,
controller,
peer,
type);
result->heightValue()
| rpl::start_with_next(
[this] { refreshHeight(); },
result->lifetime());
return result;
}
void InnerWidget::saveState(not_null<Memento*> memento) { void InnerWidget::saveState(not_null<Memento*> memento) {
} }
@ -58,7 +234,34 @@ void InnerWidget::restoreState(not_null<Memento*> memento) {
} }
int InnerWidget::resizeGetHeight(int newWidth) { int InnerWidget::resizeGetHeight(int newWidth) {
return _rowsHeightFake; _inResize = true;
auto guard = gsl::finally([this] { _inResize = false; });
if (_otherTypes) {
_otherTypes->resizeToWidth(newWidth);
}
_list->resizeToWidth(newWidth);
return recountHeight();
}
void InnerWidget::refreshHeight() {
if (_inResize) {
return;
}
resize(width(), recountHeight());
}
int InnerWidget::recountHeight() {
auto top = 0;
if (_otherTypes) {
_otherTypes->moveToLeft(0, top);
top += _otherTypes->heightNoMargins();
}
if (_list) {
_list->moveToLeft(0, top);
top += _list->heightNoMargins();
}
return top;
} }
} // namespace Media } // namespace Media

View File

@ -23,23 +23,31 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "ui/rp_widget.h" #include "ui/rp_widget.h"
#include "info/media/info_media_widget.h" #include "info/media/info_media_widget.h"
namespace Ui {
class SettingsSlider;
class VerticalLayout;
} // namespace Ui
namespace Info { namespace Info {
namespace Media { namespace Media {
class Memento;
class ListWidget;
class InnerWidget final : public Ui::RpWidget { class InnerWidget final : public Ui::RpWidget {
public: public:
using Type = Widget::Type; using Type = Widget::Type;
InnerWidget( InnerWidget(
QWidget *parent, QWidget *parent,
rpl::producer<Wrap> &&wrap,
not_null<Window::Controller*> controller,
not_null<PeerData*> peer, not_null<PeerData*> peer,
Type type); Type type);
not_null<PeerData*> peer() const { not_null<PeerData*> peer() const;
return _peer; Type type() const;
}
Type type() const { bool showInternal(not_null<Memento*> memento);
return _type;
}
void saveState(not_null<Memento*> memento); void saveState(not_null<Memento*> memento);
void restoreState(not_null<Memento*> memento); void restoreState(not_null<Memento*> memento);
@ -51,10 +59,27 @@ protected:
int visibleBottom) override; int visibleBottom) override;
private: private:
not_null<PeerData*> _peer; int recountHeight();
Type _type = Type::Photo; void refreshHeight();
void setupOtherTypes(rpl::producer<Wrap> &&wrap);
void createOtherTypes();
void createTypeButtons();
void createTabs();
void switchToTab(Memento &&memento);
not_null<Window::Controller*> controller() const;
object_ptr<ListWidget> setupList(
not_null<Window::Controller*> controller,
not_null<PeerData*> peer,
Type type);
bool _inResize = false;
Ui::SettingsSlider *_otherTabs = nullptr;
object_ptr<Ui::VerticalLayout> _otherTypes = { nullptr };
object_ptr<ListWidget> _list = { nullptr };
int _rowsHeightFake = 0;
int _visibleTop = 0; int _visibleTop = 0;
int _visibleBottom = 0; int _visibleBottom = 0;

View File

@ -0,0 +1,38 @@
/*
This file is part of Telegram Desktop,
the official desktop version of Telegram messaging app, see https://telegram.org
Telegram Desktop 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 3 of the License, or
(at your option) any later version.
It 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.
In addition, as a special exception, the copyright holders give permission
to link the code of portions of this program with the OpenSSL library.
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
*/
#include "info/media/info_media_list_widget.h"
namespace Info {
namespace Media {
ListWidget::ListWidget(
QWidget *parent,
not_null<Window::Controller*> controller,
not_null<PeerData*> peer,
Type type)
: RpWidget(parent)
, _controller(controller)
, _peer(peer)
, _type(type) {
}
} // namespace Media
} // namespace Info

View File

@ -0,0 +1,60 @@
/*
This file is part of Telegram Desktop,
the official desktop version of Telegram messaging app, see https://telegram.org
Telegram Desktop 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 3 of the License, or
(at your option) any later version.
It 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.
In addition, as a special exception, the copyright holders give permission
to link the code of portions of this program with the OpenSSL library.
Full license: https://github.com/telegramdesktop/tdesktop/blob/master/LICENSE
Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
*/
#pragma once
#include "ui/rp_widget.h"
#include "info/media/info_media_widget.h"
namespace Window {
class Controller;
} // namespace Window
namespace Info {
namespace Media {
class ListWidget : public Ui::RpWidget {
public:
using Type = Widget::Type;
ListWidget(
QWidget *parent,
not_null<Window::Controller*> controller,
not_null<PeerData*> peer,
Type type);
not_null<Window::Controller*> controller() const {
return _controller;
}
not_null<PeerData*> peer() const {
return _peer;
}
Type type() const {
return _type;
}
private:
not_null<Window::Controller*> _controller;
not_null<PeerData*> _peer;
Type _type = Type::Photo;
};
} // namespace Media
} // namespace Info

View File

@ -47,8 +47,13 @@ Widget::Widget(
not_null<Window::Controller*> controller, not_null<Window::Controller*> controller,
not_null<PeerData*> peer, not_null<PeerData*> peer,
Type type) Type type)
: ContentWidget(parent, std::move(wrap), controller, peer) { : ContentWidget(parent, rpl::duplicate(wrap), controller, peer) {
_inner = setInnerWidget(object_ptr<InnerWidget>(this, peer, type)); _inner = setInnerWidget(object_ptr<InnerWidget>(
this,
std::move(wrap),
controller,
peer,
type));
} }
Section Widget::section() const { Section Widget::section() const {
@ -61,8 +66,7 @@ Widget::Type Widget::type() const {
bool Widget::showInternal(not_null<ContentMemento*> memento) { bool Widget::showInternal(not_null<ContentMemento*> memento) {
if (auto mediaMemento = dynamic_cast<Memento*>(memento.get())) { if (auto mediaMemento = dynamic_cast<Memento*>(memento.get())) {
if (mediaMemento->peerId() == peer()->id) { if (_inner->showInternal(mediaMemento)) {
restoreState(mediaMemento);
return true; return true;
} }
} }

View File

@ -30,6 +30,7 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
#include "info/profile/info_profile_cover.h" #include "info/profile/info_profile_cover.h"
#include "info/profile/info_profile_icon.h" #include "info/profile/info_profile_icon.h"
#include "info/profile/info_profile_members.h" #include "info/profile/info_profile_members.h"
#include "info/media/info_media_buttons.h"
#include "boxes/abstract_box.h" #include "boxes/abstract_box.h"
#include "boxes/add_contact_box.h" #include "boxes/add_contact_box.h"
#include "boxes/confirm_box.h" #include "boxes/confirm_box.h"
@ -151,7 +152,7 @@ object_ptr<Ui::RpWidget> InnerWidget::setupDetails(
object_ptr<Ui::RpWidget> InnerWidget::setupInfo( object_ptr<Ui::RpWidget> InnerWidget::setupInfo(
RpWidget *parent) const { RpWidget *parent) const {
auto result = object_ptr<Ui::VerticalLayout>(parent); auto result = object_ptr<Ui::VerticalLayout>(parent);
auto tracker = MultiLineTracker(); auto tracker = Ui::MultiSlideTracker();
auto addInfoLine = [&]( auto addInfoLine = [&](
LangKey label, LangKey label,
rpl::producer<TextWithEntities> &&text, rpl::producer<TextWithEntities> &&text,
@ -222,7 +223,7 @@ void InnerWidget::setupUserButtons(
Ui::VerticalLayout *wrap, Ui::VerticalLayout *wrap,
not_null<UserData*> user) const { not_null<UserData*> user) const {
using namespace rpl::mappers; using namespace rpl::mappers;
auto tracker = MultiLineTracker(); auto tracker = Ui::MultiSlideTracker();
auto topSkip = wrap->add(createSlideSkipWidget(wrap)); auto topSkip = wrap->add(createSlideSkipWidget(wrap));
auto addButton = [&](auto &&text) { auto addButton = [&](auto &&text) {
auto result = wrap->add(object_ptr<Ui::SlideWrap<Button>>( auto result = wrap->add(object_ptr<Ui::SlideWrap<Button>>(
@ -264,67 +265,26 @@ void InnerWidget::setupUserButtons(
object_ptr<Ui::RpWidget> InnerWidget::setupSharedMedia( object_ptr<Ui::RpWidget> InnerWidget::setupSharedMedia(
RpWidget *parent) { RpWidget *parent) {
using namespace rpl::mappers; using namespace rpl::mappers;
using MediaType = Media::Type;
auto content = object_ptr<Ui::VerticalLayout>(parent); auto content = object_ptr<Ui::VerticalLayout>(parent);
auto tracker = MultiLineTracker(); auto tracker = Ui::MultiSlideTracker();
auto addButton = [&](
auto &&count,
auto textFromCount) {
auto forked = std::move(count)
| start_spawning(content->lifetime());
auto button = content->add(object_ptr<Ui::SlideWrap<Button>>(
content,
object_ptr<Button>(
content,
rpl::duplicate(forked)
| rpl::map([textFromCount](int count) {
return (count > 0)
? textFromCount(count)
: QString();
}),
st::infoSharedMediaButton))
)->toggleOn(
rpl::duplicate(forked)
| rpl::map($1 > 0));
tracker.track(button);
return button;
};
using MediaType = Storage::SharedMediaType;
auto mediaText = [](MediaType type) {
switch (type) {
case MediaType::Photo: return lng_profile_photos;
case MediaType::Video: return lng_profile_videos;
case MediaType::File: return lng_profile_files;
case MediaType::MusicFile: return lng_profile_songs;
case MediaType::Link: return lng_profile_shared_links;
case MediaType::VoiceFile: return lng_profile_audios;
case MediaType::RoundFile: return lng_profile_rounds;
}
Unexpected("Type in setupSharedMedia()");
};
auto addMediaButton = [&](MediaType type) { auto addMediaButton = [&](MediaType type) {
return addButton( return Media::AddButton(
SharedMediaCountValue(_peer, type), content,
[phrase = mediaText(type)](int count) { _controller,
return phrase(lt_count, count); peer(),
} type,
)->entity()->addClickHandler([this, peer = _peer, type] { tracker);
_controller->showSection(
Info::Memento(peer->id, Section(type)));
});
}; };
auto addCommonGroupsButton = [&](not_null<UserData*> user) { auto addCommonGroupsButton = [&](not_null<UserData*> user) {
return addButton( return Media::AddCommonGroupsButton(
CommonGroupsCountValue(user), content,
[](int count) { _controller,
return lng_profile_common_groups(lt_count, count); user,
} tracker);
)->entity()->addClickHandler([this, peer = _peer] {
_controller->showSection(
::Profile::CommonGroups::SectionMemento(
peer->asUser()));
});
}; };
addMediaButton(MediaType::Photo); addMediaButton(MediaType::Photo);
addMediaButton(MediaType::Video); addMediaButton(MediaType::Video);
addMediaButton(MediaType::File); addMediaButton(MediaType::File);
@ -343,8 +303,9 @@ object_ptr<Ui::RpWidget> InnerWidget::setupSharedMedia(
result->toggleOn(tracker.atLeastOneShownValue()); result->toggleOn(tracker.atLeastOneShownValue());
auto layout = result->entity(); auto layout = result->entity();
layout->add(object_ptr<BoxContentDivider>(result)); layout->add(object_ptr<BoxContentDivider>(layout));
_sharedMediaCover = layout->add(object_ptr<SharedMediaCover>(layout)); _sharedMediaCover = layout->add(
object_ptr<SharedMediaCover>(layout));
if (canHideDetailsEver()) { if (canHideDetailsEver()) {
_sharedMediaCover->setToggleShown(canHideDetails()); _sharedMediaCover->setToggleShown(canHideDetails());
_sharedMediaWrap = layout->add(object_ptr<Ui::SlideWrap<>>( _sharedMediaWrap = layout->add(object_ptr<Ui::SlideWrap<>>(

View File

@ -240,18 +240,5 @@ rpl::producer<bool> CanAddMemberValue(
return rpl::single(false); return rpl::single(false);
} }
rpl::producer<bool> MultiLineTracker::atLeastOneShownValue() const {
auto shown = std::vector<rpl::producer<bool>>();
shown.reserve(_widgets.size());
for (auto &widget : _widgets) {
shown.push_back(widget->toggledValue());
}
return rpl::combine(
std::move(shown),
[](const std::vector<bool> &values) {
return base::find(values, true) != values.end();
});
}
} // namespace Profile } // namespace Profile
} // namespace Info } // namespace Info

View File

@ -77,19 +77,5 @@ rpl::producer<int> CommonGroupsCountValue(
rpl::producer<bool> CanAddMemberValue( rpl::producer<bool> CanAddMemberValue(
not_null<PeerData*> peer); not_null<PeerData*> peer);
class MultiLineTracker {
public:
template <typename Widget>
void track(const Ui::SlideWrap<Widget> *wrap) {
_widgets.push_back(wrap);
}
rpl::producer<bool> atLeastOneShownValue() const;
private:
std::vector<const Ui::SlideWrap<Ui::RpWidget>*> _widgets;
};
} // namespace Profile } // namespace Profile
} // namespace Info } // namespace Info

View File

@ -20,6 +20,8 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
*/ */
#include "ui/wrap/slide_wrap.h" #include "ui/wrap/slide_wrap.h"
#include <rpl/combine.h>
namespace Ui { namespace Ui {
SlideWrap<RpWidget>::SlideWrap( SlideWrap<RpWidget>::SlideWrap(
@ -144,5 +146,18 @@ void SlideWrap<RpWidget>::wrappedSizeUpdated(QSize size) {
} }
} }
rpl::producer<bool> MultiSlideTracker::atLeastOneShownValue() const {
auto shown = std::vector<rpl::producer<bool>>();
shown.reserve(_widgets.size());
for (auto &widget : _widgets) {
shown.push_back(widget->toggledValue());
}
return rpl::combine(
std::move(shown),
[](const std::vector<bool> &values) {
return base::find(values, true) != values.end();
});
}
} // namespace Ui } // namespace Ui

View File

@ -136,5 +136,19 @@ inline object_ptr<SlideWrap<>> CreateSlideSkipWidget(
QMargins(0, 0, 0, skip)); QMargins(0, 0, 0, skip));
} }
class MultiSlideTracker {
public:
template <typename Widget>
void track(const Ui::SlideWrap<Widget> *wrap) {
_widgets.push_back(wrap);
}
rpl::producer<bool> atLeastOneShownValue() const;
private:
std::vector<const Ui::SlideWrap<Ui::RpWidget>*> _widgets;
};
} // namespace Ui } // namespace Ui

View File

@ -221,8 +221,11 @@
<(src_loc)/info/info_top_bar.h <(src_loc)/info/info_top_bar.h
<(src_loc)/info/info_wrap_widget.cpp <(src_loc)/info/info_wrap_widget.cpp
<(src_loc)/info/info_wrap_widget.h <(src_loc)/info/info_wrap_widget.h
<(src_loc)/info/media/info_media_buttons.h
<(src_loc)/info/media/info_media_inner_widget.cpp <(src_loc)/info/media/info_media_inner_widget.cpp
<(src_loc)/info/media/info_media_inner_widget.h <(src_loc)/info/media/info_media_inner_widget.h
<(src_loc)/info/media/info_media_list_widget.cpp
<(src_loc)/info/media/info_media_list_widget.h
<(src_loc)/info/media/info_media_widget.cpp <(src_loc)/info/media/info_media_widget.cpp
<(src_loc)/info/media/info_media_widget.h <(src_loc)/info/media/info_media_widget.h
<(src_loc)/info/profile/info_profile_button.cpp <(src_loc)/info/profile/info_profile_button.cpp