Fix crash on layer -> section migration.

This commit is contained in:
John Preston 2018-12-04 11:19:28 +04:00
parent 679347309e
commit 71efd10c83
3 changed files with 57 additions and 15 deletions

View File

@ -56,12 +56,16 @@ not_null<Window::Controller*> LayerWidget::floatPlayerController() {
not_null<Window::AbstractSectionWidget*> LayerWidget::floatPlayerGetSection( not_null<Window::AbstractSectionWidget*> LayerWidget::floatPlayerGetSection(
Window::Column column) { Window::Column column) {
Expects(_content != nullptr);
return _content; return _content;
} }
void LayerWidget::floatPlayerEnumerateSections(Fn<void( void LayerWidget::floatPlayerEnumerateSections(Fn<void(
not_null<Window::AbstractSectionWidget*> widget, not_null<Window::AbstractSectionWidget*> widget,
Window::Column widgetColumn)> callback) { Window::Column widgetColumn)> callback) {
Expects(_content != nullptr);
callback(_content, Window::Column::Second); callback(_content, Window::Column::Second);
} }
@ -70,6 +74,8 @@ bool LayerWidget::floatPlayerIsVisible(not_null<HistoryItem*> item) {
} }
void LayerWidget::setupHeightConsumers() { void LayerWidget::setupHeightConsumers() {
Expects(_content != nullptr);
_content->scrollTillBottomChanges( _content->scrollTillBottomChanges(
) | rpl::filter([this] { ) | rpl::filter([this] {
return !_inResize; return !_inResize;
@ -90,14 +96,33 @@ void LayerWidget::showFinished() {
} }
void LayerWidget::parentResized() { void LayerWidget::parentResized() {
if (!_content) {
return;
}
auto parentSize = parentWidget()->size(); auto parentSize = parentWidget()->size();
auto parentWidth = parentSize.width(); auto parentWidth = parentSize.width();
if (parentWidth < MinimalSupportedWidth()) { if (parentWidth < MinimalSupportedWidth()) {
Ui::FocusPersister persister(this); Ui::FocusPersister persister(this);
auto localCopy = _controller; restoreFloatPlayerDelegate();
auto memento = MoveMemento(std::move(_content)); auto memento = MoveMemento(std::move(_content));
localCopy->hideSpecialLayer(anim::type::instant);
localCopy->showSection( // We want to call hideSpecialLayer synchronously to avoid glitches,
// but we can't destroy LayerStackWidget from its' resizeEvent,
// because QWidget has such code for resizing:
//
// QResizeEvent e(r.size(), olds);
// QApplication::sendEvent(q, &e);
// if (q->windowHandle())
// q->update();
//
// So we call it queued. It would be cool to call it 'right after'
// the resize event handling was finished.
InvokeQueued(this, [=] {
_controller->hideSpecialLayer(anim::type::instant);
});
_controller->showSection(
std::move(memento), std::move(memento),
Window::SectionShow( Window::SectionShow(
Window::SectionShow::Way::Forward, Window::SectionShow::Way::Forward,
@ -153,7 +178,7 @@ bool LayerWidget::takeToThirdSection() {
bool LayerWidget::showSectionInternal( bool LayerWidget::showSectionInternal(
not_null<Window::SectionMemento*> memento, not_null<Window::SectionMemento*> memento,
const Window::SectionShow &params) { const Window::SectionShow &params) {
if (_content->showInternal(memento, params)) { if (_content && _content->showInternal(memento, params)) {
if (params.activation != anim::activation::background) { if (params.activation != anim::activation::background) {
Ui::hideLayer(); Ui::hideLayer();
} }
@ -163,11 +188,11 @@ bool LayerWidget::showSectionInternal(
} }
bool LayerWidget::closeByOutsideClick() const { bool LayerWidget::closeByOutsideClick() const {
return _content->closeByOutsideClick(); return _content ? _content->closeByOutsideClick() : true;
} }
int LayerWidget::MinimalSupportedWidth() { int LayerWidget::MinimalSupportedWidth() {
auto minimalMargins = 2 * st::infoMinimalLayerMargin; const auto minimalMargins = 2 * st::infoMinimalLayerMargin;
return st::infoMinimalWidth + minimalMargins; return st::infoMinimalWidth + minimalMargins;
} }
@ -225,7 +250,9 @@ int LayerWidget::resizeGetHeight(int newWidth) {
} }
void LayerWidget::doSetInnerFocus() { void LayerWidget::doSetInnerFocus() {
_content->setInnerFocus(); if (_content) {
_content->setInnerFocus();
}
} }
void LayerWidget::paintEvent(QPaintEvent *e) { void LayerWidget::paintEvent(QPaintEvent *e) {

View File

@ -164,6 +164,8 @@ Memento::~Memento() = default;
MoveMemento::MoveMemento(object_ptr<WrapWidget> content) MoveMemento::MoveMemento(object_ptr<WrapWidget> content)
: _content(std::move(content)) { : _content(std::move(content)) {
_content->hide();
_content->setParent(nullptr);
} }
object_ptr<Window::SectionWidget> MoveMemento::createWidget( object_ptr<Window::SectionWidget> MoveMemento::createWidget(

View File

@ -497,16 +497,16 @@ void LayerStackWidget::closeLayer(not_null<LayerWidget*> layer) {
} }
void LayerStackWidget::updateLayerBoxes() { void LayerStackWidget::updateLayerBoxes() {
auto getLayerBox = [this]() { const auto layerBox = [&] {
if (auto layer = currentLayer()) { if (const auto layer = currentLayer()) {
return layer->geometry(); return layer->geometry();
} }
return QRect(); return QRect();
}; }();
auto getSpecialLayerBox = [this]() { const auto specialLayerBox = _specialLayer
return _specialLayer ? _specialLayer->geometry() : QRect(); ? _specialLayer->geometry()
}; : QRect();
_background->setLayerBoxes(getSpecialLayerBox(), getLayerBox()); _background->setLayerBoxes(specialLayerBox, layerBox);
update(); update();
} }
@ -566,15 +566,28 @@ void LayerStackWidget::startAnimation(
} }
void LayerStackWidget::resizeEvent(QResizeEvent *e) { void LayerStackWidget::resizeEvent(QResizeEvent *e) {
const auto weak = make_weak(this);
_background->setGeometry(rect()); _background->setGeometry(rect());
if (!weak) {
return;
}
if (_specialLayer) { if (_specialLayer) {
_specialLayer->parentResized(); _specialLayer->parentResized();
if (!weak) {
return;
}
} }
if (auto layer = currentLayer()) { if (const auto layer = currentLayer()) {
layer->parentResized(); layer->parentResized();
if (!weak) {
return;
}
} }
if (_mainMenu) { if (_mainMenu) {
_mainMenu->resize(_mainMenu->width(), height()); _mainMenu->resize(_mainMenu->width(), height());
if (!weak) {
return;
}
} }
updateLayerBoxes(); updateLayerBoxes();
} }