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 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()) {
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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) {
|
||||||
|
|
|
@ -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();
|
||||||
|
|
|
@ -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 "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
|
||||||
|
|
|
@ -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;
|
||||||
|
|
||||||
|
|
|
@ -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<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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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<>>(
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
||||||
|
|
|
@ -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
|
||||||
|
|
Loading…
Reference in New Issue