 aa155db0bd
			
		
	
	
		aa155db0bd
		
	
	
	
	
		
			
			If some row widgets of VerticalLayout are living in its own lifetime() we have a use-after-free. Because this lifetime() destroys this child row already after ~VerticalLayout which can call back into VerticalLayout::childHeightUpdated (the subscription lives inside the same lifetime() as well) which will access _rows member that was already destroyed. Now all subscriptions die before _rows are destroyed.
		
			
				
	
	
		
			89 lines
		
	
	
	
		
			2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			89 lines
		
	
	
	
		
			2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| // This file is part of Desktop App Toolkit,
 | |
| // a set of libraries for developing nice desktop applications.
 | |
| //
 | |
| // For license and copyright information please follow this link:
 | |
| // https://github.com/desktop-app/legal/blob/master/LEGAL
 | |
| //
 | |
| #pragma once
 | |
| 
 | |
| #include "ui/rp_widget.h"
 | |
| #include "base/object_ptr.h"
 | |
| 
 | |
| namespace Ui {
 | |
| 
 | |
| class VerticalLayout : public RpWidget {
 | |
| public:
 | |
| 	using RpWidget::RpWidget;
 | |
| 
 | |
| 	[[nodiscard]] int count() const {
 | |
| 		return _rows.size();
 | |
| 	}
 | |
| 	[[nodiscard]] not_null<RpWidget*> widgetAt(int index) const {
 | |
| 		Expects(index >= 0 && index < count());
 | |
| 
 | |
| 		return _rows[index].widget.data();
 | |
| 	}
 | |
| 
 | |
| 	template <
 | |
| 		typename Widget,
 | |
| 		typename = std::enable_if_t<
 | |
| 			std::is_base_of_v<RpWidget, Widget>>>
 | |
| 	Widget *insert(
 | |
| 			int atPosition,
 | |
| 			object_ptr<Widget> &&child,
 | |
| 			const style::margins &margin = style::margins()) {
 | |
| 		return static_cast<Widget*>(insertChild(
 | |
| 			atPosition,
 | |
| 			std::move(child),
 | |
| 			margin));
 | |
| 	}
 | |
| 
 | |
| 	template <
 | |
| 		typename Widget,
 | |
| 		typename = std::enable_if_t<
 | |
| 			std::is_base_of_v<RpWidget, Widget>>>
 | |
| 	Widget *add(
 | |
| 			object_ptr<Widget> &&child,
 | |
| 			const style::margins &margin = style::margins()) {
 | |
| 		return insert(count(), std::move(child), margin);
 | |
| 	}
 | |
| 
 | |
| 	QMargins getMargins() const override;
 | |
| 	int naturalWidth() const override;
 | |
| 
 | |
| 	void setVerticalShift(int index, int shift);
 | |
| 	void reorderRows(int oldIndex, int newIndex);
 | |
| 
 | |
| protected:
 | |
| 	int resizeGetHeight(int newWidth) override;
 | |
| 	void visibleTopBottomUpdated(
 | |
| 		int visibleTop,
 | |
| 		int visibleBottom) override;
 | |
| 
 | |
| private:
 | |
| 	RpWidget *insertChild(
 | |
| 		int addPosition,
 | |
| 		object_ptr<RpWidget> child,
 | |
| 		const style::margins &margin);
 | |
| 	void childHeightUpdated(RpWidget *child);
 | |
| 	void removeChild(RpWidget *child);
 | |
| 	void updateChildGeometry(
 | |
| 		const style::margins &margins,
 | |
| 		RpWidget *child,
 | |
| 		const style::margins &margin,
 | |
| 		int width,
 | |
| 		int top) const;
 | |
| 
 | |
| 	struct Row {
 | |
| 		object_ptr<RpWidget> widget;
 | |
| 		style::margins margin;
 | |
| 		int verticalShift = 0;
 | |
| 	};
 | |
| 	std::vector<Row> _rows;
 | |
| 	bool _inResize = false;
 | |
| 
 | |
| 	rpl::lifetime _rowsLifetime;
 | |
| 
 | |
| };
 | |
| 
 | |
| } // namespace Ui
 |