mirror of https://github.com/procxx/kepka.git
Add tabs and other types links to Info::Media.
This commit is contained in:
parent
335704e176
commit
7905694b31
|
@ -1579,7 +1579,7 @@ void HistoryInner::keyPressEvent(QKeyEvent *e) {
|
|||
}
|
||||
}
|
||||
|
||||
void HistoryInner::recountHeight() {
|
||||
void HistoryInner::recountHistoryGeometry() {
|
||||
int visibleHeight = _scroll->height();
|
||||
int oldHistoryPaddingTop = qMax(visibleHeight - historyHeight() - st::historyPaddingBottom, 0);
|
||||
if (_botAbout && !_botAbout->info->text.isEmpty()) {
|
||||
|
|
|
@ -47,7 +47,7 @@ public:
|
|||
void touchScrollUpdated(const QPoint &screenPos);
|
||||
QPoint mapPointToItem(QPoint p, HistoryItem *item);
|
||||
|
||||
void recountHeight();
|
||||
void recountHistoryGeometry();
|
||||
void updateSize();
|
||||
|
||||
void repaintItem(const HistoryItem *item);
|
||||
|
|
|
@ -4785,7 +4785,7 @@ int HistoryWidget::countAutomaticScrollTop() {
|
|||
void HistoryWidget::updateHistoryGeometry(bool initial, bool loadedDown, const ScrollChange &change) {
|
||||
if (!_history || (initial && _historyInited) || (!initial && !_historyInited)) return;
|
||||
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();
|
||||
|
@ -4868,7 +4868,7 @@ void HistoryWidget::updateHistoryGeometry(bool initial, bool loadedDown, const S
|
|||
}
|
||||
|
||||
void HistoryWidget::updateListSize() {
|
||||
_list->recountHeight();
|
||||
_list->recountHistoryGeometry();
|
||||
auto washidden = _scroll->isHidden();
|
||||
if (washidden) {
|
||||
_scroll->show();
|
||||
|
|
|
@ -173,9 +173,15 @@ infoIconMembers: icon {{ "info_members", infoIconFg }};
|
|||
infoIconNotifications: icon {{ "info_notifications", infoIconFg }};
|
||||
infoIconActions: icon {{ "info_actions", 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);
|
||||
infoNotificationsIconPosition: point(20px, 5px);
|
||||
infoSharedMediaIconPosition: point(20px, 24px);
|
||||
infoSharedMediaButtonIconPosition: point(20px, 3px);
|
||||
infoIconPosition: point(20px, 15px);
|
||||
|
||||
infoLabeledOneLine: FlatLabel(defaultFlatLabel) {
|
||||
|
|
|
@ -57,6 +57,7 @@ LayerWidget::LayerWidget(
|
|||
void LayerWidget::setupHeightConsumers() {
|
||||
_content->desiredHeightValue()
|
||||
| rpl::start_with_next([this](int height) {
|
||||
if (!_content) return;
|
||||
accumulate_max(_desiredHeight, height);
|
||||
resizeToWidth(width());
|
||||
_content->forceContentRepaint();
|
||||
|
|
|
@ -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
|
|
@ -20,28 +20,160 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
|||
*/
|
||||
#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 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(
|
||||
QWidget *parent,
|
||||
rpl::producer<Wrap> &&wrap,
|
||||
not_null<Window::Controller*> controller,
|
||||
not_null<PeerData*> peer,
|
||||
Type type)
|
||||
: RpWidget(parent)
|
||||
, _peer(peer)
|
||||
, _type(type) {
|
||||
auto text = qsl("Media Overview\n\n");
|
||||
auto label = object_ptr<Ui::FlatLabel>(this);
|
||||
label->setText(text.repeated(50));
|
||||
widthValue() | rpl::start_with_next([inner = label.data()](int w) {
|
||||
inner->resizeToWidth(w);
|
||||
}, lifetime());
|
||||
label->heightValue() | rpl::start_with_next([this](int h) {
|
||||
_rowsHeightFake = h;
|
||||
resizeToWidth(width());
|
||||
}, lifetime());
|
||||
: RpWidget(parent) {
|
||||
_list = setupList(controller, peer, type);
|
||||
setupOtherTypes(std::move(wrap));
|
||||
}
|
||||
|
||||
void InnerWidget::setupOtherTypes(rpl::producer<Wrap> &&wrap) {
|
||||
std::move(wrap)
|
||||
| rpl::start_with_next([this](Wrap value) {
|
||||
if (value == Wrap::Side
|
||||
&& TypeToTabIndex(type())) {
|
||||
createOtherTypes();
|
||||
} else {
|
||||
_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(
|
||||
|
@ -51,6 +183,50 @@ void InnerWidget::visibleTopBottomUpdated(
|
|||
_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) {
|
||||
}
|
||||
|
||||
|
@ -58,7 +234,34 @@ void InnerWidget::restoreState(not_null<Memento*> memento) {
|
|||
}
|
||||
|
||||
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
|
||||
|
|
|
@ -23,23 +23,31 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
|||
#include "ui/rp_widget.h"
|
||||
#include "info/media/info_media_widget.h"
|
||||
|
||||
namespace Ui {
|
||||
class SettingsSlider;
|
||||
class VerticalLayout;
|
||||
} // namespace Ui
|
||||
|
||||
namespace Info {
|
||||
namespace Media {
|
||||
|
||||
class Memento;
|
||||
class ListWidget;
|
||||
|
||||
class InnerWidget final : public Ui::RpWidget {
|
||||
public:
|
||||
using Type = Widget::Type;
|
||||
InnerWidget(
|
||||
QWidget *parent,
|
||||
rpl::producer<Wrap> &&wrap,
|
||||
not_null<Window::Controller*> controller,
|
||||
not_null<PeerData*> peer,
|
||||
Type type);
|
||||
|
||||
not_null<PeerData*> peer() const {
|
||||
return _peer;
|
||||
}
|
||||
Type type() const {
|
||||
return _type;
|
||||
}
|
||||
not_null<PeerData*> peer() const;
|
||||
Type type() const;
|
||||
|
||||
bool showInternal(not_null<Memento*> memento);
|
||||
|
||||
void saveState(not_null<Memento*> memento);
|
||||
void restoreState(not_null<Memento*> memento);
|
||||
|
@ -51,10 +59,27 @@ protected:
|
|||
int visibleBottom) override;
|
||||
|
||||
private:
|
||||
not_null<PeerData*> _peer;
|
||||
Type _type = Type::Photo;
|
||||
int recountHeight();
|
||||
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 _visibleBottom = 0;
|
||||
|
||||
|
|
|
@ -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
|
|
@ -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
|
|
@ -47,8 +47,13 @@ Widget::Widget(
|
|||
not_null<Window::Controller*> controller,
|
||||
not_null<PeerData*> peer,
|
||||
Type type)
|
||||
: ContentWidget(parent, std::move(wrap), controller, peer) {
|
||||
_inner = setInnerWidget(object_ptr<InnerWidget>(this, peer, type));
|
||||
: ContentWidget(parent, rpl::duplicate(wrap), controller, peer) {
|
||||
_inner = setInnerWidget(object_ptr<InnerWidget>(
|
||||
this,
|
||||
std::move(wrap),
|
||||
controller,
|
||||
peer,
|
||||
type));
|
||||
}
|
||||
|
||||
Section Widget::section() const {
|
||||
|
@ -61,8 +66,7 @@ Widget::Type Widget::type() const {
|
|||
|
||||
bool Widget::showInternal(not_null<ContentMemento*> memento) {
|
||||
if (auto mediaMemento = dynamic_cast<Memento*>(memento.get())) {
|
||||
if (mediaMemento->peerId() == peer()->id) {
|
||||
restoreState(mediaMemento);
|
||||
if (_inner->showInternal(mediaMemento)) {
|
||||
return true;
|
||||
}
|
||||
}
|
||||
|
|
|
@ -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_icon.h"
|
||||
#include "info/profile/info_profile_members.h"
|
||||
#include "info/media/info_media_buttons.h"
|
||||
#include "boxes/abstract_box.h"
|
||||
#include "boxes/add_contact_box.h"
|
||||
#include "boxes/confirm_box.h"
|
||||
|
@ -151,7 +152,7 @@ object_ptr<Ui::RpWidget> InnerWidget::setupDetails(
|
|||
object_ptr<Ui::RpWidget> InnerWidget::setupInfo(
|
||||
RpWidget *parent) const {
|
||||
auto result = object_ptr<Ui::VerticalLayout>(parent);
|
||||
auto tracker = MultiLineTracker();
|
||||
auto tracker = Ui::MultiSlideTracker();
|
||||
auto addInfoLine = [&](
|
||||
LangKey label,
|
||||
rpl::producer<TextWithEntities> &&text,
|
||||
|
@ -222,7 +223,7 @@ void InnerWidget::setupUserButtons(
|
|||
Ui::VerticalLayout *wrap,
|
||||
not_null<UserData*> user) const {
|
||||
using namespace rpl::mappers;
|
||||
auto tracker = MultiLineTracker();
|
||||
auto tracker = Ui::MultiSlideTracker();
|
||||
auto topSkip = wrap->add(createSlideSkipWidget(wrap));
|
||||
auto addButton = [&](auto &&text) {
|
||||
auto result = wrap->add(object_ptr<Ui::SlideWrap<Button>>(
|
||||
|
@ -264,67 +265,26 @@ void InnerWidget::setupUserButtons(
|
|||
object_ptr<Ui::RpWidget> InnerWidget::setupSharedMedia(
|
||||
RpWidget *parent) {
|
||||
using namespace rpl::mappers;
|
||||
using MediaType = Media::Type;
|
||||
|
||||
auto content = object_ptr<Ui::VerticalLayout>(parent);
|
||||
auto tracker = MultiLineTracker();
|
||||
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 tracker = Ui::MultiSlideTracker();
|
||||
auto addMediaButton = [&](MediaType type) {
|
||||
return addButton(
|
||||
SharedMediaCountValue(_peer, type),
|
||||
[phrase = mediaText(type)](int count) {
|
||||
return phrase(lt_count, count);
|
||||
}
|
||||
)->entity()->addClickHandler([this, peer = _peer, type] {
|
||||
_controller->showSection(
|
||||
Info::Memento(peer->id, Section(type)));
|
||||
});
|
||||
return Media::AddButton(
|
||||
content,
|
||||
_controller,
|
||||
peer(),
|
||||
type,
|
||||
tracker);
|
||||
};
|
||||
auto addCommonGroupsButton = [&](not_null<UserData*> user) {
|
||||
return addButton(
|
||||
CommonGroupsCountValue(user),
|
||||
[](int count) {
|
||||
return lng_profile_common_groups(lt_count, count);
|
||||
}
|
||||
)->entity()->addClickHandler([this, peer = _peer] {
|
||||
_controller->showSection(
|
||||
::Profile::CommonGroups::SectionMemento(
|
||||
peer->asUser()));
|
||||
});
|
||||
return Media::AddCommonGroupsButton(
|
||||
content,
|
||||
_controller,
|
||||
user,
|
||||
tracker);
|
||||
};
|
||||
|
||||
addMediaButton(MediaType::Photo);
|
||||
addMediaButton(MediaType::Video);
|
||||
addMediaButton(MediaType::File);
|
||||
|
@ -343,8 +303,9 @@ object_ptr<Ui::RpWidget> InnerWidget::setupSharedMedia(
|
|||
result->toggleOn(tracker.atLeastOneShownValue());
|
||||
auto layout = result->entity();
|
||||
|
||||
layout->add(object_ptr<BoxContentDivider>(result));
|
||||
_sharedMediaCover = layout->add(object_ptr<SharedMediaCover>(layout));
|
||||
layout->add(object_ptr<BoxContentDivider>(layout));
|
||||
_sharedMediaCover = layout->add(
|
||||
object_ptr<SharedMediaCover>(layout));
|
||||
if (canHideDetailsEver()) {
|
||||
_sharedMediaCover->setToggleShown(canHideDetails());
|
||||
_sharedMediaWrap = layout->add(object_ptr<Ui::SlideWrap<>>(
|
||||
|
|
|
@ -240,18 +240,5 @@ rpl::producer<bool> CanAddMemberValue(
|
|||
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 Info
|
||||
|
|
|
@ -77,19 +77,5 @@ rpl::producer<int> CommonGroupsCountValue(
|
|||
rpl::producer<bool> CanAddMemberValue(
|
||||
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 Info
|
||||
|
|
|
@ -20,6 +20,8 @@ Copyright (c) 2014-2017 John Preston, https://desktop.telegram.org
|
|||
*/
|
||||
#include "ui/wrap/slide_wrap.h"
|
||||
|
||||
#include <rpl/combine.h>
|
||||
|
||||
namespace Ui {
|
||||
|
||||
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
|
||||
|
||||
|
|
|
@ -136,5 +136,19 @@ inline object_ptr<SlideWrap<>> CreateSlideSkipWidget(
|
|||
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
|
||||
|
||||
|
|
|
@ -221,8 +221,11 @@
|
|||
<(src_loc)/info/info_top_bar.h
|
||||
<(src_loc)/info/info_wrap_widget.cpp
|
||||
<(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.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.h
|
||||
<(src_loc)/info/profile/info_profile_button.cpp
|
||||
|
|
Loading…
Reference in New Issue