Fix layout update notifications in Info::Media.

This commit is contained in:
John Preston 2018-01-21 22:52:44 +03:00
parent ebd4651ac2
commit 89941a8e83
10 changed files with 69 additions and 107 deletions

View File

@ -235,6 +235,7 @@ MediaPhoto::MediaPhoto(
not_null<PhotoData*> photo) not_null<PhotoData*> photo)
: Media(parent) : Media(parent)
, _photo(photo) { , _photo(photo) {
Auth().data().registerPhotoItem(_photo, parent);
} }
MediaPhoto::MediaPhoto( MediaPhoto::MediaPhoto(
@ -244,9 +245,11 @@ MediaPhoto::MediaPhoto(
: Media(parent) : Media(parent)
, _photo(photo) , _photo(photo)
, _chat(chat) { , _chat(chat) {
Auth().data().registerPhotoItem(_photo, parent);
} }
MediaPhoto::~MediaPhoto() { MediaPhoto::~MediaPhoto() {
Auth().data().unregisterPhotoItem(_photo, parent());
} }
std::unique_ptr<Media> MediaPhoto::clone(not_null<HistoryItem*> parent) { std::unique_ptr<Media> MediaPhoto::clone(not_null<HistoryItem*> parent) {

View File

@ -107,19 +107,19 @@ void Session::animationLoadSettingsChanged() {
} }
void Session::notifyPhotoLayoutChanged(not_null<const PhotoData*> photo) { void Session::notifyPhotoLayoutChanged(not_null<const PhotoData*> photo) {
if (const auto i = _photoViews.find(photo); i != end(_photoViews)) { if (const auto i = _photoItems.find(photo); i != end(_photoItems)) {
for (const auto view : i->second) { for (const auto item : i->second) {
notifyViewLayoutChange(view); notifyItemLayoutChange(item);
} }
} }
} }
void Session::notifyDocumentLayoutChanged( void Session::notifyDocumentLayoutChanged(
not_null<const DocumentData*> document) { not_null<const DocumentData*> document) {
const auto i = _documentViews.find(document); const auto i = _documentItems.find(document);
if (i != end(_documentViews)) { if (i != end(_documentItems)) {
for (const auto view : i->second) { for (const auto item : i->second) {
notifyViewLayoutChange(view); notifyItemLayoutChange(item);
} }
} }
if (const auto items = InlineBots::Layout::documentItems()) { if (const auto items = InlineBots::Layout::documentItems()) {
@ -133,10 +133,10 @@ void Session::notifyDocumentLayoutChanged(
void Session::requestDocumentViewRepaint( void Session::requestDocumentViewRepaint(
not_null<const DocumentData*> document) { not_null<const DocumentData*> document) {
const auto i = _documentViews.find(document); const auto i = _documentItems.find(document);
if (i != end(_documentViews)) { if (i != end(_documentItems)) {
for (const auto view : i->second) { for (const auto item : i->second) {
requestViewRepaint(view); requestItemRepaint(item);
} }
} }
} }
@ -148,6 +148,17 @@ void Session::markMediaRead(not_null<const DocumentData*> document) {
} }
} }
void Session::notifyItemLayoutChange(not_null<const HistoryItem*> item) {
_itemLayoutChanges.fire_copy(item);
enumerateItemViews(item, [&](not_null<ViewElement*> view) {
notifyViewLayoutChange(view);
});
}
rpl::producer<not_null<const HistoryItem*>> Session::itemLayoutChanged() const {
return _itemLayoutChanges.events();
}
void Session::notifyViewLayoutChange(not_null<const ViewElement*> view) { void Session::notifyViewLayoutChange(not_null<const ViewElement*> view) {
_viewLayoutChanges.fire_copy(view); _viewLayoutChanges.fire_copy(view);
} }
@ -1131,38 +1142,20 @@ void Session::gameApplyFields(
notifyGameUpdateDelayed(game); notifyGameUpdateDelayed(game);
} }
void Session::registerPhotoView( void Session::registerPhotoItem(
not_null<const PhotoData*> photo, not_null<const PhotoData*> photo,
not_null<ViewElement*> view) { not_null<HistoryItem*> item) {
_photoViews[photo].insert(view); _photoItems[photo].insert(item);
} }
void Session::unregisterPhotoView( void Session::unregisterPhotoItem(
not_null<const PhotoData*> photo, not_null<const PhotoData*> photo,
not_null<ViewElement*> view) { not_null<HistoryItem*> item) {
const auto i = _photoViews.find(photo); const auto i = _photoItems.find(photo);
if (i != _photoViews.end()) { if (i != _photoItems.end()) {
auto &items = i->second; auto &items = i->second;
if (items.remove(view) && items.empty()) { if (items.remove(item) && items.empty()) {
_photoViews.erase(i); _photoItems.erase(i);
}
}
}
void Session::registerDocumentView(
not_null<const DocumentData*> document,
not_null<ViewElement*> view) {
_documentViews[document].insert(view);
}
void Session::unregisterDocumentView(
not_null<const DocumentData*> document,
not_null<ViewElement*> view) {
const auto i = _documentViews.find(document);
if (i != _documentViews.end()) {
auto &items = i->second;
if (items.remove(view) && items.empty()) {
_documentViews.erase(i);
} }
} }
} }

View File

@ -60,6 +60,8 @@ public:
}; };
void notifyItemIdChange(IdChange event); void notifyItemIdChange(IdChange event);
rpl::producer<IdChange> itemIdChanged() const; rpl::producer<IdChange> itemIdChanged() const;
void notifyItemLayoutChange(not_null<const HistoryItem*> item);
rpl::producer<not_null<const HistoryItem*>> itemLayoutChanged() const;
void notifyViewLayoutChange(not_null<const ViewElement*> view); void notifyViewLayoutChange(not_null<const ViewElement*> view);
rpl::producer<not_null<const ViewElement*>> viewLayoutChanged() const; rpl::producer<not_null<const ViewElement*>> viewLayoutChanged() const;
void requestItemRepaint(not_null<const HistoryItem*> item); void requestItemRepaint(not_null<const HistoryItem*> item);
@ -273,18 +275,12 @@ public:
not_null<GameData*> original, not_null<GameData*> original,
const MTPGame &data); const MTPGame &data);
void registerPhotoView( void registerPhotoItem(
not_null<const PhotoData*> photo, not_null<const PhotoData*> photo,
not_null<ViewElement*> view); not_null<HistoryItem*> item);
void unregisterPhotoView( void unregisterPhotoItem(
not_null<const PhotoData*> photo, not_null<const PhotoData*> photo,
not_null<ViewElement*> view); not_null<HistoryItem*> item);
void registerDocumentView(
not_null<const DocumentData*> document,
not_null<ViewElement*> view);
void unregisterDocumentView(
not_null<const DocumentData*> document,
not_null<ViewElement*> view);
void registerDocumentItem( void registerDocumentItem(
not_null<const DocumentData*> document, not_null<const DocumentData*> document,
not_null<HistoryItem*> item); not_null<HistoryItem*> item);
@ -439,6 +435,7 @@ private:
base::Observable<void> _moreChatsLoaded; base::Observable<void> _moreChatsLoaded;
base::Observable<ItemVisibilityQuery> _queryItemVisibility; base::Observable<ItemVisibilityQuery> _queryItemVisibility;
rpl::event_stream<IdChange> _itemIdChanges; rpl::event_stream<IdChange> _itemIdChanges;
rpl::event_stream<not_null<const HistoryItem*>> _itemLayoutChanges;
rpl::event_stream<not_null<const ViewElement*>> _viewLayoutChanges; rpl::event_stream<not_null<const ViewElement*>> _viewLayoutChanges;
rpl::event_stream<not_null<const HistoryItem*>> _itemRepaintRequest; rpl::event_stream<not_null<const HistoryItem*>> _itemRepaintRequest;
rpl::event_stream<not_null<const ViewElement*>> _viewRepaintRequest; rpl::event_stream<not_null<const ViewElement*>> _viewRepaintRequest;
@ -473,16 +470,13 @@ private:
std::unique_ptr<PhotoData>> _photos; std::unique_ptr<PhotoData>> _photos;
std::map< std::map<
not_null<const PhotoData*>, not_null<const PhotoData*>,
base::flat_set<not_null<ViewElement*>>> _photoViews; base::flat_set<not_null<HistoryItem*>>> _photoItems;
std::unordered_map< std::unordered_map<
DocumentId, DocumentId,
std::unique_ptr<DocumentData>> _documents; std::unique_ptr<DocumentData>> _documents;
std::map< std::map<
not_null<const DocumentData*>, not_null<const DocumentData*>,
base::flat_set<not_null<HistoryItem*>>> _documentItems; base::flat_set<not_null<HistoryItem*>>> _documentItems;
std::map<
not_null<const DocumentData*>,
base::flat_set<not_null<ViewElement*>>> _documentViews;
std::unordered_map< std::unordered_map<
WebPageId, WebPageId,
std::unique_ptr<WebPageData>> _webpages; std::unique_ptr<WebPageData>> _webpages;

View File

@ -231,6 +231,14 @@ InnerWidget::InnerWidget(
refreshItem(view); refreshItem(view);
} }
}, lifetime()); }, lifetime());
Auth().data().viewLayoutChanged(
) | rpl::start_with_next([this](auto view) {
if (view->delegate() == this) {
if (view->isUnderCursor()) {
updateSelected();
}
}
}, lifetime());
Auth().data().animationPlayInlineRequest( Auth().data().animationPlayInlineRequest(
) | rpl::start_with_next([this](auto item) { ) | rpl::start_with_next([this](auto item) {
if (const auto view = viewForItem(item)) { if (const auto view = viewForItem(item)) {

View File

@ -245,7 +245,6 @@ HistoryPhoto::HistoryPhoto(
} }
void HistoryPhoto::create(FullMsgId contextId, PeerData *chat) { void HistoryPhoto::create(FullMsgId contextId, PeerData *chat) {
Auth().data().registerPhotoView(_data, _parent);
setLinks( setLinks(
std::make_shared<PhotoOpenClickHandler>(_data, contextId, chat), std::make_shared<PhotoOpenClickHandler>(_data, contextId, chat),
std::make_shared<PhotoSaveClickHandler>(_data, contextId, chat), std::make_shared<PhotoSaveClickHandler>(_data, contextId, chat),
@ -707,28 +706,11 @@ ImagePtr HistoryPhoto::replyPreview() {
return _data->makeReplyPreview(); return _data->makeReplyPreview();
} }
HistoryPhoto::~HistoryPhoto() {
Auth().data().unregisterPhotoView(_data, _parent);
}
DocumentViewRegister::DocumentViewRegister(
not_null<HistoryView::Element*> parent,
not_null<DocumentData*> document)
: _savedParent(parent)
, _savedDocument(document) {
Auth().data().registerDocumentView(_savedDocument, _savedParent);
}
DocumentViewRegister::~DocumentViewRegister() {
Auth().data().unregisterDocumentView(_savedDocument, _savedParent);
}
HistoryVideo::HistoryVideo( HistoryVideo::HistoryVideo(
not_null<Element*> parent, not_null<Element*> parent,
not_null<HistoryItem*> realParent, not_null<HistoryItem*> realParent,
not_null<DocumentData*> document) not_null<DocumentData*> document)
: HistoryFileMedia(parent) : HistoryFileMedia(parent)
, DocumentViewRegister(parent, document)
, _data(document) , _data(document)
, _thumbw(1) , _thumbw(1)
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) { , _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
@ -1197,7 +1179,6 @@ HistoryDocument::HistoryDocument(
not_null<Element*> parent, not_null<Element*> parent,
not_null<DocumentData*> document) not_null<DocumentData*> document)
: HistoryFileMedia(parent) : HistoryFileMedia(parent)
, DocumentViewRegister(parent, document)
, _data(document) { , _data(document) {
const auto item = parent->data(); const auto item = parent->data();
auto caption = createCaption(item); auto caption = createCaption(item);
@ -1934,7 +1915,6 @@ HistoryGif::HistoryGif(
not_null<Element*> parent, not_null<Element*> parent,
not_null<DocumentData*> document) not_null<DocumentData*> document)
: HistoryFileMedia(parent) : HistoryFileMedia(parent)
, DocumentViewRegister(parent, document)
, _data(document) , _data(document)
, _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) { , _caption(st::minPhotoSize - st::msgPadding.left() - st::msgPadding.right()) {
const auto item = parent->data(); const auto item = parent->data();
@ -2782,7 +2762,6 @@ HistorySticker::HistorySticker(
not_null<Element*> parent, not_null<Element*> parent,
not_null<DocumentData*> document) not_null<DocumentData*> document)
: HistoryMedia(parent) : HistoryMedia(parent)
, DocumentViewRegister(parent, document)
, _data(document) , _data(document)
, _emoji(_data->sticker()->alt) { , _emoji(_data->sticker()->alt) {
_data->thumb->load(); _data->thumb->load();

View File

@ -200,8 +200,6 @@ public:
return _data->loaded(); return _data->loaded();
} }
~HistoryPhoto();
protected: protected:
float64 dataProgress() const override; float64 dataProgress() const override;
bool dataFinished() const override; bool dataFinished() const override;
@ -228,20 +226,7 @@ private:
}; };
class DocumentViewRegister { class HistoryVideo : public HistoryFileMedia {
public:
DocumentViewRegister(
not_null<HistoryView::Element*> parent,
not_null<DocumentData*> document);
~DocumentViewRegister();
private:
not_null<HistoryView::Element*> _savedParent;
not_null<DocumentData*> _savedDocument;
};
class HistoryVideo : public HistoryFileMedia, public DocumentViewRegister {
public: public:
HistoryVideo( HistoryVideo(
not_null<Element*> parent, not_null<Element*> parent,
@ -333,7 +318,6 @@ private:
class HistoryDocument class HistoryDocument
: public HistoryFileMedia : public HistoryFileMedia
, public DocumentViewRegister
, public RuntimeComposer<HistoryDocument> { , public RuntimeComposer<HistoryDocument> {
public: public:
HistoryDocument( HistoryDocument(
@ -415,7 +399,7 @@ private:
}; };
class HistoryGif : public HistoryFileMedia, public DocumentViewRegister { class HistoryGif : public HistoryFileMedia {
public: public:
HistoryGif( HistoryGif(
not_null<Element*> parent, not_null<Element*> parent,
@ -516,7 +500,7 @@ private:
}; };
class HistorySticker : public HistoryMedia, public DocumentViewRegister { class HistorySticker : public HistoryMedia {
public: public:
HistorySticker( HistorySticker(
not_null<Element*> parent, not_null<Element*> parent,

View File

@ -665,11 +665,9 @@ HistoryWidget::HistoryWidget(QWidget *parent, not_null<Window::Controller*> cont
}); });
Auth().data().viewLayoutChanged( Auth().data().viewLayoutChanged(
) | rpl::start_with_next([this](auto view) { ) | rpl::start_with_next([this](auto view) {
if (_peer && _list) { if (view == view->data()->mainView()) {
if (view == view->data()->mainView()) { if (view->isUnderCursor() && _list) {
if (view->isUnderCursor()) { _list->onUpdateSelected();
_list->onUpdateSelected();
}
} }
} }
}, lifetime()); }, lifetime());

View File

@ -240,6 +240,14 @@ ListWidget::ListWidget(
refreshItem(view); refreshItem(view);
} }
}, lifetime()); }, lifetime());
Auth().data().viewLayoutChanged(
) | rpl::start_with_next([this](auto view) {
if (view->delegate() == this) {
if (view->isUnderCursor()) {
updateSelected();
}
}
}, lifetime());
Auth().data().animationPlayInlineRequest( Auth().data().animationPlayInlineRequest(
) | rpl::start_with_next([this](auto item) { ) | rpl::start_with_next([this](auto item) {
if (const auto view = viewForItem(item)) { if (const auto view = viewForItem(item)) {

View File

@ -561,10 +561,10 @@ void ListWidget::start() {
ObservableViewer( ObservableViewer(
Auth().downloader().taskFinished() Auth().downloader().taskFinished()
) | rpl::start_with_next([this] { update(); }, lifetime()); ) | rpl::start_with_next([this] { update(); }, lifetime());
//Auth().data().itemLayoutChanged( // #TODO Auth().data().itemLayoutChanged(
//) | rpl::start_with_next([this](auto item) { ) | rpl::start_with_next([this](auto item) {
// itemLayoutChanged(item); itemLayoutChanged(item);
//}, lifetime()); }, lifetime());
Auth().data().itemRemoved( Auth().data().itemRemoved(
) | rpl::start_with_next([this](auto item) { ) | rpl::start_with_next([this](auto item) {
itemRemoved(item); itemRemoved(item);

View File

@ -50,12 +50,7 @@ Float::Float(
prepareShadow(); prepareShadow();
rpl::merge( Auth().data().itemRepaintRequest(
//Auth().data().viewLayoutChanged( // #TODO not needed?
//) | rpl::map(
// [](auto view) { return view->data(); }
//),
Auth().data().itemRepaintRequest()
) | rpl::start_with_next([this](auto item) { ) | rpl::start_with_next([this](auto item) {
if (_item == item) { if (_item == item) {
repaintItem(); repaintItem();